]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/font.cpp
Fix crash when drawing bitmaps with mask in wxGTK with GTK+ < 2.20.
[wxWidgets.git] / src / gtk / font.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/gtk/font.cpp
3// Purpose: wxFont for wxGTK
4// Author: Robert Roebling
5// Id: $Id$
6// Copyright: (c) 1998 Robert Roebling and Julian Smart
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10// ============================================================================
11// declarations
12// ============================================================================
13
14// ----------------------------------------------------------------------------
15// headers
16// ----------------------------------------------------------------------------
17
18// For compilers that support precompilation, includes "wx.h".
19#include "wx/wxprec.h"
20
21#include "wx/font.h"
22
23#ifndef WX_PRECOMP
24 #include "wx/log.h"
25 #include "wx/utils.h"
26 #include "wx/settings.h"
27 #include "wx/gdicmn.h"
28#endif
29
30#include "wx/fontutil.h"
31#include "wx/tokenzr.h"
32
33#include "wx/gtk/private.h"
34
35// ----------------------------------------------------------------------------
36// constants
37// ----------------------------------------------------------------------------
38
39// the default size (in points) for the fonts
40static const int wxDEFAULT_FONT_SIZE = 12;
41
42// ----------------------------------------------------------------------------
43// wxFontRefData
44// ----------------------------------------------------------------------------
45
46class wxFontRefData : public wxGDIRefData
47{
48public:
49 // from broken down font parameters, also default ctor
50 wxFontRefData(int size = -1,
51 wxFontFamily family = wxFONTFAMILY_DEFAULT,
52 wxFontStyle style = wxFONTSTYLE_NORMAL,
53 wxFontWeight weight = wxFONTWEIGHT_NORMAL,
54 bool underlined = false,
55 bool strikethrough = false,
56 const wxString& faceName = wxEmptyString,
57 wxFontEncoding encoding = wxFONTENCODING_DEFAULT);
58
59 wxFontRefData(const wxString& nativeFontInfoString);
60
61 // copy ctor
62 wxFontRefData( const wxFontRefData& data );
63
64 virtual ~wxFontRefData();
65
66 // setters: all of them also take care to modify m_nativeFontInfo if we
67 // have it so as to not lose the information not carried by our fields
68 void SetPointSize(int pointSize);
69 void SetFamily(wxFontFamily family);
70 void SetStyle(wxFontStyle style);
71 void SetWeight(wxFontWeight weight);
72 void SetUnderlined(bool underlined);
73 void SetStrikethrough(bool strikethrough);
74 bool SetFaceName(const wxString& facename);
75 void SetEncoding(wxFontEncoding encoding);
76
77 // and this one also modifies all the other font data fields
78 void SetNativeFontInfo(const wxNativeFontInfo& info);
79
80protected:
81 // common part of all ctors
82 void Init(int pointSize,
83 wxFontFamily family,
84 wxFontStyle style,
85 wxFontWeight weight,
86 bool underlined,
87 bool strikethrough,
88 const wxString& faceName,
89 wxFontEncoding encoding);
90
91 // set all fields from (already initialized and valid) m_nativeFontInfo
92 void InitFromNative();
93
94private:
95 // The native font info: basically a PangoFontDescription, plus
96 // 'underlined' and 'strikethrough' attributes not supported by Pango.
97 wxNativeFontInfo m_nativeFontInfo;
98
99 friend class wxFont;
100};
101
102#define M_FONTDATA ((wxFontRefData*)m_refData)
103
104// ----------------------------------------------------------------------------
105// wxFontRefData
106// ----------------------------------------------------------------------------
107
108void wxFontRefData::Init(int pointSize,
109 wxFontFamily family,
110 wxFontStyle style,
111 wxFontWeight weight,
112 bool underlined,
113 bool strikethrough,
114 const wxString& faceName,
115 wxFontEncoding WXUNUSED(encoding))
116{
117 if (family == wxFONTFAMILY_DEFAULT)
118 family = wxFONTFAMILY_SWISS;
119
120 // Create native font info
121 m_nativeFontInfo.description = pango_font_description_new();
122
123 // And set its values
124 if (!faceName.empty())
125 {
126 pango_font_description_set_family( m_nativeFontInfo.description,
127 wxGTK_CONV_SYS(faceName) );
128 }
129 else
130 {
131 SetFamily(family);
132 }
133
134 SetStyle( style == wxDEFAULT ? wxFONTSTYLE_NORMAL : style );
135 SetPointSize( (pointSize == wxDEFAULT || pointSize == -1)
136 ? wxDEFAULT_FONT_SIZE
137 : pointSize );
138 SetWeight( weight == wxDEFAULT ? wxFONTWEIGHT_NORMAL : weight );
139 SetUnderlined( underlined );
140 SetStrikethrough( strikethrough );
141}
142
143void wxFontRefData::InitFromNative()
144{
145 // Get native info
146 PangoFontDescription *desc = m_nativeFontInfo.description;
147
148 // Pango sometimes needs to have a size
149 int pango_size = pango_font_description_get_size( desc );
150 if (pango_size == 0)
151 m_nativeFontInfo.SetPointSize(wxDEFAULT_FONT_SIZE);
152}
153
154wxFontRefData::wxFontRefData( const wxFontRefData& data )
155 : wxGDIRefData()
156{
157 // Forces a copy of the internal data. wxNativeFontInfo should probably
158 // have a copy ctor and assignment operator to fix this properly but that
159 // would break binary compatibility...
160 m_nativeFontInfo.FromString(data.m_nativeFontInfo.ToString());
161}
162
163wxFontRefData::wxFontRefData(int size, wxFontFamily family, wxFontStyle style,
164 wxFontWeight weight, bool underlined, bool strikethrough,
165 const wxString& faceName,
166 wxFontEncoding encoding)
167{
168 Init(size, family, style, weight, underlined, strikethrough, faceName, encoding);
169}
170
171wxFontRefData::wxFontRefData(const wxString& nativeFontInfoString)
172{
173 m_nativeFontInfo.FromString( nativeFontInfoString );
174
175 InitFromNative();
176}
177
178wxFontRefData::~wxFontRefData()
179{
180}
181
182// ----------------------------------------------------------------------------
183// wxFontRefData SetXXX()
184// ----------------------------------------------------------------------------
185
186void wxFontRefData::SetPointSize(int pointSize)
187{
188 m_nativeFontInfo.SetPointSize(pointSize);
189}
190
191/*
192 NOTE: disabled because pango_font_description_set_absolute_size() and
193 wxDC::GetCharHeight() do not mix well: setting with the former a pixel
194 size of "30" makes the latter return 36...
195 Besides, we need to return GetPointSize() a point size value even if
196 SetPixelSize() was used and this would require further changes
197 (and use of pango_font_description_get_size_is_absolute in some places).
198
199bool wxFontRefData::SetPixelSize(const wxSize& pixelSize)
200{
201 wxCHECK_MSG( pixelSize.GetWidth() >= 0 && pixelSize.GetHeight() > 0, false,
202 "Negative values for the pixel size or zero pixel height are not allowed" );
203
204 if (wx_pango_version_check(1,8,0) != NULL ||
205 pixelSize.GetWidth() != 0)
206 {
207 // NOTE: pango_font_description_set_absolute_size() only sets the font height;
208 // if the user set the pixel width of the font explicitly or the pango
209 // library is too old, we cannot proceed
210 return false;
211 }
212
213 pango_font_description_set_absolute_size( m_nativeFontInfo.description,
214 pixelSize.GetHeight() * PANGO_SCALE );
215
216 return true;
217}
218*/
219
220void wxFontRefData::SetFamily(wxFontFamily family)
221{
222 m_nativeFontInfo.SetFamily(family);
223}
224
225void wxFontRefData::SetStyle(wxFontStyle style)
226{
227 m_nativeFontInfo.SetStyle(style);
228}
229
230void wxFontRefData::SetWeight(wxFontWeight weight)
231{
232 m_nativeFontInfo.SetWeight(weight);
233}
234
235void wxFontRefData::SetUnderlined(bool underlined)
236{
237 m_nativeFontInfo.SetUnderlined(underlined);
238}
239
240void wxFontRefData::SetStrikethrough(bool strikethrough)
241{
242 m_nativeFontInfo.SetStrikethrough(strikethrough);
243}
244
245bool wxFontRefData::SetFaceName(const wxString& facename)
246{
247 return m_nativeFontInfo.SetFaceName(facename);
248}
249
250void wxFontRefData::SetEncoding(wxFontEncoding WXUNUSED(encoding))
251{
252 // with GTK+ 2 Pango always uses UTF8 internally, we cannot change it
253}
254
255void wxFontRefData::SetNativeFontInfo(const wxNativeFontInfo& info)
256{
257 m_nativeFontInfo = info;
258
259 // set all the other font parameters from the native font info
260 InitFromNative();
261}
262
263// ----------------------------------------------------------------------------
264// wxFont creation
265// ----------------------------------------------------------------------------
266
267wxFont::wxFont(const wxNativeFontInfo& info)
268{
269 Create( info.GetPointSize(),
270 info.GetFamily(),
271 info.GetStyle(),
272 info.GetWeight(),
273 info.GetUnderlined(),
274 info.GetFaceName(),
275 info.GetEncoding() );
276
277 if ( info.GetStrikethrough() )
278 SetStrikethrough(true);
279}
280
281wxFont::wxFont(int pointSize,
282 wxFontFamily family,
283 int flags,
284 const wxString& face,
285 wxFontEncoding encoding)
286{
287 m_refData = new wxFontRefData(pointSize, family,
288 GetStyleFromFlags(flags),
289 GetWeightFromFlags(flags),
290 GetUnderlinedFromFlags(flags),
291 GetStrikethroughFromFlags(flags),
292 face, encoding);
293}
294
295bool wxFont::Create( int pointSize,
296 wxFontFamily family,
297 wxFontStyle style,
298 wxFontWeight weight,
299 bool underlined,
300 const wxString& face,
301 wxFontEncoding encoding )
302{
303 UnRef();
304
305 m_refData = new wxFontRefData(pointSize, family, style, weight,
306 underlined, false, face, encoding);
307
308 return true;
309}
310
311bool wxFont::Create(const wxString& fontname)
312{
313 // VZ: does this really happen?
314 if ( fontname.empty() )
315 {
316 *this = wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
317
318 return true;
319 }
320
321 m_refData = new wxFontRefData(fontname);
322
323 return true;
324}
325
326wxFont::~wxFont()
327{
328}
329
330// ----------------------------------------------------------------------------
331// accessors
332// ----------------------------------------------------------------------------
333
334int wxFont::GetPointSize() const
335{
336 wxCHECK_MSG( IsOk(), 0, wxT("invalid font") );
337
338 return M_FONTDATA->m_nativeFontInfo.GetPointSize();
339}
340
341wxString wxFont::GetFaceName() const
342{
343 wxCHECK_MSG( IsOk(), wxEmptyString, wxT("invalid font") );
344
345 return M_FONTDATA->m_nativeFontInfo.GetFaceName();
346}
347
348wxFontFamily wxFont::DoGetFamily() const
349{
350 return M_FONTDATA->m_nativeFontInfo.GetFamily();
351}
352
353wxFontStyle wxFont::GetStyle() const
354{
355 wxCHECK_MSG( IsOk(), wxFONTSTYLE_MAX, wxT("invalid font") );
356
357 return M_FONTDATA->m_nativeFontInfo.GetStyle();
358}
359
360wxFontWeight wxFont::GetWeight() const
361{
362 wxCHECK_MSG( IsOk(), wxFONTWEIGHT_MAX, wxT("invalid font") );
363
364 return M_FONTDATA->m_nativeFontInfo.GetWeight();
365}
366
367bool wxFont::GetUnderlined() const
368{
369 wxCHECK_MSG( IsOk(), false, wxT("invalid font") );
370
371 return M_FONTDATA->m_nativeFontInfo.GetUnderlined();
372}
373
374bool wxFont::GetStrikethrough() const
375{
376 wxCHECK_MSG( IsOk(), false, wxT("invalid font") );
377
378 return M_FONTDATA->m_nativeFontInfo.GetStrikethrough();
379}
380
381wxFontEncoding wxFont::GetEncoding() const
382{
383 wxCHECK_MSG( IsOk(), wxFONTENCODING_SYSTEM, wxT("invalid font") );
384
385 return wxFONTENCODING_UTF8;
386 // Pango always uses UTF8... see also SetEncoding()
387}
388
389const wxNativeFontInfo *wxFont::GetNativeFontInfo() const
390{
391 wxCHECK_MSG( IsOk(), NULL, wxT("invalid font") );
392
393 return &(M_FONTDATA->m_nativeFontInfo);
394}
395
396bool wxFont::IsFixedWidth() const
397{
398 wxCHECK_MSG( IsOk(), false, wxT("invalid font") );
399
400 return wxFontBase::IsFixedWidth();
401}
402
403// ----------------------------------------------------------------------------
404// change font attributes
405// ----------------------------------------------------------------------------
406
407void wxFont::SetPointSize(int pointSize)
408{
409 AllocExclusive();
410
411 M_FONTDATA->SetPointSize(pointSize);
412}
413
414void wxFont::SetFamily(wxFontFamily family)
415{
416 AllocExclusive();
417
418 M_FONTDATA->SetFamily(family);
419}
420
421void wxFont::SetStyle(wxFontStyle style)
422{
423 AllocExclusive();
424
425 M_FONTDATA->SetStyle(style);
426}
427
428void wxFont::SetWeight(wxFontWeight weight)
429{
430 AllocExclusive();
431
432 M_FONTDATA->SetWeight(weight);
433}
434
435bool wxFont::SetFaceName(const wxString& faceName)
436{
437 AllocExclusive();
438
439 return M_FONTDATA->SetFaceName(faceName) &&
440 wxFontBase::SetFaceName(faceName);
441}
442
443void wxFont::SetUnderlined(bool underlined)
444{
445 AllocExclusive();
446
447 M_FONTDATA->SetUnderlined(underlined);
448}
449
450void wxFont::SetStrikethrough(bool strikethrough)
451{
452 AllocExclusive();
453
454 M_FONTDATA->SetStrikethrough(strikethrough);
455}
456
457void wxFont::SetEncoding(wxFontEncoding encoding)
458{
459 AllocExclusive();
460
461 M_FONTDATA->SetEncoding(encoding);
462}
463
464void wxFont::DoSetNativeFontInfo( const wxNativeFontInfo& info )
465{
466 AllocExclusive();
467
468 M_FONTDATA->SetNativeFontInfo( info );
469}
470
471wxGDIRefData* wxFont::CreateGDIRefData() const
472{
473 return new wxFontRefData;
474}
475
476wxGDIRefData* wxFont::CloneGDIRefData(const wxGDIRefData* data) const
477{
478 return new wxFontRefData(*static_cast<const wxFontRefData*>(data));
479}
480
481bool wxFont::GTKSetPangoAttrs(PangoLayout* layout) const
482{
483 if (!IsOk() || !(GetUnderlined() || GetStrikethrough()))
484 return false;
485
486 PangoAttrList* attrs = pango_attr_list_new();
487 PangoAttribute* a;
488
489 if (wx_pango_version_check(1,16,0))
490 {
491 // a PangoLayout which has leading/trailing spaces with underlined font
492 // is not correctly drawn by this pango version: Pango won't underline the spaces.
493 // This can be a problem; e.g. wxHTML rendering of underlined text relies on
494 // this behaviour. To workaround this problem, we use a special hack here
495 // suggested by pango maintainer Behdad Esfahbod: we prepend and append two
496 // empty space characters and give them a dummy colour attribute.
497 // This will force Pango to underline the leading/trailing spaces, too.
498
499 const char* text = pango_layout_get_text(layout);
500 const size_t n = strlen(text);
501 if ((n > 0 && text[0] == ' ') || (n > 1 && text[n - 1] == ' '))
502 {
503 wxCharBuffer buf(n + 6);
504 // copy the leading U+200C ZERO WIDTH NON-JOINER encoded in UTF8 format
505 memcpy(buf.data(), "\342\200\214", 3);
506 // copy the user string
507 memcpy(buf.data() + 3, text, n);
508 // copy the trailing U+200C ZERO WIDTH NON-JOINER encoded in UTF8 format
509 memcpy(buf.data() + 3 + n, "\342\200\214", 3);
510
511 pango_layout_set_text(layout, buf, n + 6);
512
513 // Add dummy attributes (use colour as it's invisible anyhow for 0
514 // width spaces) to ensure that the spaces in the beginning/end of the
515 // string are underlined too.
516 a = pango_attr_foreground_new(0x0057, 0x52A9, 0xD614);
517 a->start_index = 0;
518 a->end_index = 3;
519 pango_attr_list_insert(attrs, a);
520
521 a = pango_attr_foreground_new(0x0057, 0x52A9, 0xD614);
522 a->start_index = n + 3;
523 a->end_index = n + 6;
524 pango_attr_list_insert(attrs, a);
525 }
526 }
527 if (GetUnderlined())
528 {
529 a = pango_attr_underline_new(PANGO_UNDERLINE_SINGLE);
530 pango_attr_list_insert(attrs, a);
531 }
532 if (GetStrikethrough())
533 {
534 a = pango_attr_strikethrough_new(true);
535 pango_attr_list_insert(attrs, a);
536 }
537
538 pango_layout_set_attributes(layout, attrs);
539 pango_attr_list_unref(attrs);
540
541 return true;
542}