]> git.saurik.com Git - wxWidgets.git/blob - src/motif/font.cpp
adding new files for xti merge
[wxWidgets.git] / src / motif / font.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/motif/font.cpp
3 // Purpose: wxFont class
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 17/09/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
22
23 #ifdef __VMS
24 #pragma message disable nosimpint
25 #include "wx/vms_x_fix.h"
26 #endif
27 #include <Xm/Xm.h>
28 #ifdef __VMS
29 #pragma message enable nosimpint
30 #endif
31
32 #include "wx/font.h"
33
34 #ifndef WX_PRECOMP
35 #include "wx/string.h"
36 #include "wx/utils.h" // for wxGetDisplay()
37 #include "wx/settings.h"
38 #include "wx/gdicmn.h"
39 #endif
40
41 #include "wx/fontutil.h" // for wxNativeFontInfo
42 #include "wx/tokenzr.h"
43 #include "wx/motif/private.h"
44
45 IMPLEMENT_DYNAMIC_CLASS(wxFont, wxGDIObject)
46
47 // ----------------------------------------------------------------------------
48 // private classes
49 // ----------------------------------------------------------------------------
50
51 // For every wxFont, there must be a font for each display and scale requested.
52 // So these objects are stored in wxFontRefData::m_fonts
53 class wxXFont : public wxObject
54 {
55 public:
56 wxXFont();
57 virtual ~wxXFont();
58
59 #if !wxMOTIF_NEW_FONT_HANDLING
60 WXFontStructPtr m_fontStruct; // XFontStruct
61 #endif
62 #if !wxMOTIF_USE_RENDER_TABLE
63 WXFontList m_fontList; // Motif XmFontList
64 #else // if wxMOTIF_USE_RENDER_TABLE
65 WXRenderTable m_renderTable; // Motif XmRenderTable
66 WXRendition m_rendition; // Motif XmRendition
67 #endif
68 WXDisplay* m_display; // XDisplay
69 int m_scale; // Scale * 100
70 };
71
72 class wxFontRefData: public wxGDIRefData
73 {
74 friend class wxFont;
75
76 public:
77 wxFontRefData(int size = wxDEFAULT,
78 wxFontFamily family = wxFONTFAMILY_DEFAULT,
79 wxFontStyle style = wxFONTSTYLE_NORMAL,
80 wxFontWeight weight = wxFONTWEIGHT_NORMAL,
81 bool underlined = false,
82 const wxString& faceName = wxEmptyString,
83 wxFontEncoding encoding = wxFONTENCODING_DEFAULT)
84 {
85 Init(size, family, style, weight, underlined, faceName, encoding);
86 }
87
88 wxFontRefData(const wxFontRefData& data)
89 {
90 Init(data.m_pointSize, data.m_family, data.m_style, data.m_weight,
91 data.m_underlined, data.m_faceName, data.m_encoding);
92 }
93
94 virtual ~wxFontRefData();
95
96 protected:
97 // common part of all ctors
98 void Init(int size,
99 wxFontFamily family,
100 wxFontStyle style,
101 wxFontWeight weight,
102 bool underlined,
103 const wxString& faceName,
104 wxFontEncoding encoding);
105
106 // font attributes
107 int m_pointSize;
108 wxFontFamily m_family;
109 wxFontStyle m_style;
110 wxFontWeight m_weight;
111 bool m_underlined;
112 wxString m_faceName;
113 wxFontEncoding m_encoding;
114
115 wxNativeFontInfo m_nativeFontInfo;
116
117 // A list of wxXFonts
118 wxList m_fonts;
119 };
120
121 // ============================================================================
122 // implementation
123 // ============================================================================
124
125 // ----------------------------------------------------------------------------
126 // wxXFont
127 // ----------------------------------------------------------------------------
128
129 wxXFont::wxXFont()
130 {
131 #if !wxMOTIF_NEW_FONT_HANDLING
132 m_fontStruct = (WXFontStructPtr) 0;
133 #endif
134 #if !wxMOTIF_USE_RENDER_TABLE
135 m_fontList = (WXFontList) 0;
136 #else // if wxMOTIF_USE_RENDER_TABLE
137 m_renderTable = (WXRenderTable) 0;
138 m_rendition = (WXRendition) 0;
139 #endif
140 m_display = (WXDisplay*) 0;
141 m_scale = 100;
142 }
143
144 wxXFont::~wxXFont()
145 {
146 #if !wxMOTIF_USE_RENDER_TABLE
147 if (m_fontList)
148 XmFontListFree ((XmFontList) m_fontList);
149 m_fontList = NULL;
150 #else // if wxMOTIF_USE_RENDER_TABLE
151 if (m_renderTable)
152 XmRenderTableFree ((XmRenderTable) m_renderTable);
153 m_renderTable = NULL;
154 #endif
155
156 // TODO: why does freeing the font produce a segv???
157 // Note that XFreeFont wasn't called in wxWin 1.68 either.
158 // MBN: probably some interaction with fonts being still
159 // in use in some widgets...
160 // XFontStruct* fontStruct = (XFontStruct*) m_fontStruct;
161 // XFreeFont((Display*) m_display, fontStruct);
162 }
163
164 // ----------------------------------------------------------------------------
165 // wxFontRefData
166 // ----------------------------------------------------------------------------
167
168 void wxFontRefData::Init(int pointSize,
169 wxFontFamily family,
170 wxFontStyle style,
171 wxFontWeight weight,
172 bool underlined,
173 const wxString& faceName,
174 wxFontEncoding encoding)
175 {
176 if (family == wxDEFAULT)
177 m_family = wxFONTFAMILY_SWISS;
178 else
179 m_family = family;
180
181 m_faceName = faceName;
182
183 if (style == wxDEFAULT)
184 m_style = wxFONTSTYLE_NORMAL;
185 else
186 m_style = style;
187
188 if (weight == wxDEFAULT)
189 m_weight = wxFONTWEIGHT_NORMAL;
190 else
191 m_weight = weight;
192
193 if (pointSize == wxDEFAULT)
194 m_pointSize = 12;
195 else
196 m_pointSize = pointSize;
197
198 m_underlined = underlined;
199 m_encoding = encoding;
200 }
201
202 wxFontRefData::~wxFontRefData()
203 {
204 wxList::compatibility_iterator node = m_fonts.GetFirst();
205 while (node)
206 {
207 wxXFont* f = (wxXFont*) node->GetData();
208 delete f;
209 node = node->GetNext();
210 }
211 m_fonts.Clear();
212 }
213
214 #define M_FONTDATA ((wxFontRefData*)m_refData)
215
216 // ----------------------------------------------------------------------------
217 // wxFont
218 // ----------------------------------------------------------------------------
219
220 wxFont::wxFont(const wxNativeFontInfo& info)
221 {
222 (void)Create(info.GetXFontName());
223 }
224
225 bool wxFont::Create(int pointSize,
226 wxFontFamily family,
227 wxFontStyle style,
228 wxFontWeight weight,
229 bool underlined,
230 const wxString& faceName,
231 wxFontEncoding encoding)
232 {
233 UnRef();
234 m_refData = new wxFontRefData(pointSize, family, style, weight,
235 underlined, faceName, encoding);
236
237 return true;
238 }
239
240 bool wxFont::Create(const wxString& fontname, wxFontEncoding enc)
241 {
242 if( !fontname )
243 {
244 *this = wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT);
245 return true;
246 }
247
248 m_refData = new wxFontRefData();
249
250 M_FONTDATA->m_nativeFontInfo.SetXFontName(fontname); // X font name
251
252 wxString tmp;
253
254 wxStringTokenizer tn( fontname, wxT("-") );
255
256 tn.GetNextToken(); // skip initial empty token
257 tn.GetNextToken(); // foundry
258
259
260 M_FONTDATA->m_faceName = tn.GetNextToken(); // family
261
262 tmp = tn.GetNextToken().MakeUpper(); // weight
263 if (tmp == wxT("BOLD")) M_FONTDATA->m_weight = wxFONTWEIGHT_BOLD;
264 if (tmp == wxT("BLACK")) M_FONTDATA->m_weight = wxFONTWEIGHT_BOLD;
265 if (tmp == wxT("EXTRABOLD")) M_FONTDATA->m_weight = wxFONTWEIGHT_BOLD;
266 if (tmp == wxT("DEMIBOLD")) M_FONTDATA->m_weight = wxFONTWEIGHT_BOLD;
267 if (tmp == wxT("ULTRABOLD")) M_FONTDATA->m_weight = wxFONTWEIGHT_BOLD;
268
269 if (tmp == wxT("LIGHT")) M_FONTDATA->m_weight = wxFONTWEIGHT_LIGHT;
270 if (tmp == wxT("THIN")) M_FONTDATA->m_weight = wxFONTWEIGHT_LIGHT;
271
272 tmp = tn.GetNextToken().MakeUpper(); // slant
273 if (tmp == wxT("I")) M_FONTDATA->m_style = wxFONTSTYLE_ITALIC;
274 if (tmp == wxT("O")) M_FONTDATA->m_style = wxFONTSTYLE_ITALIC;
275
276 tn.GetNextToken(); // set width
277 tn.GetNextToken(); // add. style
278 tn.GetNextToken(); // pixel size
279
280 tmp = tn.GetNextToken(); // pointsize
281 if (tmp != wxT("*"))
282 {
283 long num = wxStrtol (tmp.mb_str(), (wxChar **) NULL, 10);
284 M_FONTDATA->m_pointSize = (int)(num / 10);
285 }
286
287 tn.GetNextToken(); // x-res
288 tn.GetNextToken(); // y-res
289
290 tmp = tn.GetNextToken().MakeUpper(); // spacing
291
292 if (tmp == wxT("M"))
293 M_FONTDATA->m_family = wxFONTFAMILY_MODERN;
294 else if (M_FONTDATA->m_faceName == wxT("TIMES"))
295 M_FONTDATA->m_family = wxFONTFAMILY_ROMAN;
296 else if (M_FONTDATA->m_faceName == wxT("HELVETICA"))
297 M_FONTDATA->m_family = wxFONTFAMILY_SWISS;
298 else if (M_FONTDATA->m_faceName == wxT("LUCIDATYPEWRITER"))
299 M_FONTDATA->m_family = wxFONTFAMILY_TELETYPE;
300 else if (M_FONTDATA->m_faceName == wxT("LUCIDA"))
301 M_FONTDATA->m_family = wxFONTFAMILY_DECORATIVE;
302 else if (M_FONTDATA->m_faceName == wxT("UTOPIA"))
303 M_FONTDATA->m_family = wxFONTFAMILY_SCRIPT;
304
305 tn.GetNextToken(); // avg width
306
307 // deal with font encoding
308 M_FONTDATA->m_encoding = enc;
309 if ( M_FONTDATA->m_encoding == wxFONTENCODING_SYSTEM )
310 {
311 wxString registry = tn.GetNextToken().MakeUpper(),
312 encoding = tn.GetNextToken().MakeUpper();
313
314 if ( registry == wxT("ISO8859") )
315 {
316 int cp;
317 if ( wxSscanf(encoding, wxT("%d"), &cp) == 1 )
318 {
319 M_FONTDATA->m_encoding =
320 (wxFontEncoding)(wxFONTENCODING_ISO8859_1 + cp - 1);
321 }
322 }
323 else if ( registry == wxT("MICROSOFT") )
324 {
325 int cp;
326 if ( wxSscanf(encoding, wxT("cp125%d"), &cp) == 1 )
327 {
328 M_FONTDATA->m_encoding =
329 (wxFontEncoding)(wxFONTENCODING_CP1250 + cp);
330 }
331 }
332 else if ( registry == wxT("KOI8") )
333 {
334 M_FONTDATA->m_encoding = wxFONTENCODING_KOI8;
335 }
336 //else: unknown encoding - may be give a warning here?
337 else
338 return false;
339 }
340 return true;
341 }
342
343 wxFont::~wxFont()
344 {
345 }
346
347 wxGDIRefData *wxFont::CreateGDIRefData() const
348 {
349 return new wxFontRefData;
350 }
351
352 wxGDIRefData *wxFont::CloneGDIRefData(const wxGDIRefData *data) const
353 {
354 return new wxFontRefData(*static_cast<const wxFontRefData *>(data));
355 }
356
357 // ----------------------------------------------------------------------------
358 // change the font attributes
359 // ----------------------------------------------------------------------------
360
361 void wxFont::Unshare()
362 {
363 // Don't change shared data
364 if (!m_refData)
365 {
366 m_refData = new wxFontRefData();
367 }
368 else
369 {
370 wxFontRefData* ref = new wxFontRefData(*(wxFontRefData*)m_refData);
371 UnRef();
372 m_refData = ref;
373 }
374 }
375
376 void wxFont::SetPointSize(int pointSize)
377 {
378 Unshare();
379
380 M_FONTDATA->m_pointSize = pointSize;
381 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
382 }
383
384 void wxFont::SetFamily(wxFontFamily family)
385 {
386 Unshare();
387
388 M_FONTDATA->m_family = family;
389 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
390 }
391
392 void wxFont::SetStyle(wxFontStyle style)
393 {
394 Unshare();
395
396 M_FONTDATA->m_style = style;
397 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
398 }
399
400 void wxFont::SetWeight(wxFontWeight weight)
401 {
402 Unshare();
403
404 M_FONTDATA->m_weight = weight;
405 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
406 }
407
408 bool wxFont::SetFaceName(const wxString& faceName)
409 {
410 Unshare();
411
412 M_FONTDATA->m_faceName = faceName;
413 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
414
415 return wxFontBase::SetFaceName(faceName);
416 }
417
418 void wxFont::SetUnderlined(bool underlined)
419 {
420 Unshare();
421
422 M_FONTDATA->m_underlined = underlined;
423 }
424
425 void wxFont::SetEncoding(wxFontEncoding encoding)
426 {
427 Unshare();
428
429 M_FONTDATA->m_encoding = encoding;
430 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
431 }
432
433 void wxFont::DoSetNativeFontInfo(const wxNativeFontInfo& info)
434 {
435 Unshare();
436
437 M_FONTDATA->m_nativeFontInfo = info;
438 }
439
440 // ----------------------------------------------------------------------------
441 // query font attributes
442 // ----------------------------------------------------------------------------
443
444 int wxFont::GetPointSize() const
445 {
446 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
447
448 return M_FONTDATA->m_pointSize;
449 }
450
451 wxString wxFont::GetFaceName() const
452 {
453 wxCHECK_MSG( Ok(), wxEmptyString, wxT("invalid font") );
454
455 return M_FONTDATA->m_faceName ;
456 }
457
458 wxFontFamily wxFont::DoGetFamily() const
459 {
460 return M_FONTDATA->m_family;
461 }
462
463 wxFontStyle wxFont::GetStyle() const
464 {
465 wxCHECK_MSG( Ok(), wxFONTSTYLE_MAX, wxT("invalid font") );
466
467 return M_FONTDATA->m_style;
468 }
469
470 wxFontWeight wxFont::GetWeight() const
471 {
472 wxCHECK_MSG( Ok(), wxFONTWEIGHT_MAX, wxT("invalid font") );
473
474 return M_FONTDATA->m_weight;
475 }
476
477 bool wxFont::GetUnderlined() const
478 {
479 wxCHECK_MSG( Ok(), false, wxT("invalid font") );
480
481 return M_FONTDATA->m_underlined;
482 }
483
484 wxFontEncoding wxFont::GetEncoding() const
485 {
486 wxCHECK_MSG( Ok(), wxFONTENCODING_DEFAULT, wxT("invalid font") );
487
488 return M_FONTDATA->m_encoding;
489 }
490
491 const wxNativeFontInfo *wxFont::GetNativeFontInfo() const
492 {
493 wxCHECK_MSG( Ok(), NULL, wxT("invalid font") );
494
495 if(M_FONTDATA->m_nativeFontInfo.GetXFontName().empty())
496 GetInternalFont();
497
498 return &(M_FONTDATA->m_nativeFontInfo);
499 }
500
501 // ----------------------------------------------------------------------------
502 // real implementation
503 // ----------------------------------------------------------------------------
504
505 // Find an existing, or create a new, XFontStruct
506 // based on this wxFont and the given scale. Append the
507 // font to list in the private data for future reference.
508 wxXFont* wxFont::GetInternalFont(double scale, WXDisplay* display) const
509 {
510 if ( !Ok() )
511 return NULL;
512
513 long intScale = long(scale * 100.0 + 0.5); // key for wxXFont
514 int pointSize = (M_FONTDATA->m_pointSize * 10 * intScale) / 100;
515
516 // search existing fonts first
517 wxList::compatibility_iterator node = M_FONTDATA->m_fonts.GetFirst();
518 while (node)
519 {
520 wxXFont* f = (wxXFont*) node->GetData();
521 if ((!display || (f->m_display == display)) && (f->m_scale == intScale))
522 return f;
523 node = node->GetNext();
524 }
525
526 // not found, create a new one
527 wxString xFontSpec;
528 XFontStruct *font = (XFontStruct *)
529 wxLoadQueryNearestFont(pointSize,
530 M_FONTDATA->m_family,
531 M_FONTDATA->m_style,
532 M_FONTDATA->m_weight,
533 M_FONTDATA->m_underlined,
534 wxT(""),
535 M_FONTDATA->m_encoding,
536 &xFontSpec);
537
538 if ( !font )
539 {
540 wxFAIL_MSG( wxT("Could not allocate even a default font -- something is wrong.") );
541
542 return NULL;
543 }
544
545 wxXFont* f = new wxXFont;
546 #if wxMOTIF_NEW_FONT_HANDLING
547 XFreeFont( (Display*) display, font );
548 #else
549 f->m_fontStruct = (WXFontStructPtr)font;
550 #endif
551 f->m_display = ( display ? display : wxGetDisplay() );
552 f->m_scale = intScale;
553
554 #if wxMOTIF_USE_RENDER_TABLE
555 XmRendition rendition;
556 XmRenderTable renderTable;
557 Arg args[5];
558 int count = 0;
559
560 #if wxMOTIF_NEW_FONT_HANDLING
561 char* fontSpec = wxStrdup(xFontSpec.mb_str());
562 XtSetArg( args[count], XmNfontName, fontSpec ); ++count;
563 XtSetArg( args[count], XmNfontType, XmFONT_IS_FONTSET ); ++count;
564 #else
565 XtSetArg( args[count], XmNfont, font ); ++count;
566 #endif
567 XtSetArg( args[count], XmNunderlineType,
568 GetUnderlined() ? XmSINGLE_LINE : XmNO_LINE ); ++count;
569 rendition = XmRenditionCreate( XmGetXmDisplay( (Display*)f->m_display ),
570 (XmStringTag)"",
571 args, count );
572 renderTable = XmRenderTableAddRenditions( NULL, &rendition, 1,
573 XmMERGE_REPLACE );
574
575 f->m_renderTable = (WXRenderTable)renderTable;
576 f->m_rendition = (WXRendition)rendition;
577 wxASSERT( f->m_renderTable != NULL );
578 #else // if !wxMOTIF_USE_RENDER_TABLE
579 f->m_fontList = XmFontListCreate ((XFontStruct*) font, XmSTRING_DEFAULT_CHARSET);
580 wxASSERT( f->m_fontList != NULL );
581 #endif
582
583 M_FONTDATA->m_fonts.Append(f);
584
585 return f;
586 }
587
588 #if !wxMOTIF_NEW_FONT_HANDLING
589
590 WXFontStructPtr wxFont::GetFontStruct(double scale, WXDisplay* display) const
591 {
592 wxXFont* f = GetInternalFont(scale, display);
593
594 return (f ? f->m_fontStruct : (WXFontStructPtr) 0);
595 }
596
597 #endif
598
599 #if !wxMOTIF_USE_RENDER_TABLE
600
601 WXFontList wxFont::GetFontList(double scale, WXDisplay* display) const
602 {
603 wxXFont* f = GetInternalFont(scale, display);
604
605 return (f ? f->m_fontList : (WXFontList) 0);
606 }
607
608 #else // if wxMOTIF_USE_RENDER_TABLE
609
610 WXRenderTable wxFont::GetRenderTable(WXDisplay* display) const
611 {
612 wxXFont* f = GetInternalFont(1.0, display);
613
614 return (f ? f->m_renderTable : (WXRenderTable) 0);
615 }
616
617 #endif // wxMOTIF_USE_RENDER_TABLE
618
619 WXFontType wxFont::GetFontType(WXDisplay* display) const
620 {
621 #if wxMOTIF_USE_RENDER_TABLE
622 return Ok() ? GetRenderTable(display) : NULL;
623 #else
624 return Ok() ? GetFontList(1.0, display) : NULL;
625 #endif
626 }
627
628 WXFontType wxFont::GetFontTypeC(WXDisplay* display) const
629 {
630 #if wxMOTIF_USE_RENDER_TABLE
631 return Ok() ? GetRenderTable(display) : NULL;
632 #else
633 return Ok() ? XmFontListCopy( (XmFontList)GetFontList(1.0, display) ) : NULL;
634 #endif
635 }
636
637 /*static*/ WXString wxFont::GetFontTag()
638 {
639 #if wxMOTIF_USE_RENDER_TABLE
640 return (WXString)XmNrenderTable;
641 #else
642 return (WXString)XmNfontList;
643 #endif
644 }
645
646 #if wxMOTIF_USE_RENDER_TABLE
647
648 WXFontSet wxFont::GetFontSet(double scale, WXDisplay* display) const
649 {
650 wxXFont* f = GetInternalFont(scale, display);
651
652 if( !f ) return (WXFontSet) 0;
653
654 Arg args[2];
655 int count = 0;
656
657 XtSetArg( args[count], XmNfont, 0 ); ++count;
658 XmRenditionRetrieve( (XmRendition) f->m_rendition, args, count );
659
660 return (WXFontSet) args[0].value;
661 }
662
663 void wxGetTextExtent(WXDisplay* display, const wxFont& font, double scale,
664 const wxString& str,
665 int* width, int* height, int* ascent, int* descent)
666 {
667 XRectangle ink, logical;
668 WXFontSet fset = font.GetFontSet(scale, display);
669
670 XmbTextExtents( (XFontSet)fset, str.mb_str(), str.length(), &ink, &logical);
671
672 if( width ) *width = logical.width;
673 if( height ) *height = logical.height;
674 if( ascent ) *ascent = -logical.y;
675 if( descent ) *descent = logical.height + logical.y;
676 }
677
678 #else // if !wxMOTIF_USE_RENDER_TABLE
679
680 void wxGetTextExtent(WXDisplay* display, const wxFont& font,
681 double scale, const wxString& str,
682 int* width, int* height, int* ascent, int* descent)
683 {
684 WXFontStructPtr pFontStruct = font.GetFontStruct(scale, display);
685
686 int direction, ascent2, descent2;
687 XCharStruct overall;
688 int slen = str.length();
689
690 XTextExtents((XFontStruct*) pFontStruct,
691 const_cast<char*>((const char *)str.mb_str()), slen,
692 &direction, &ascent2, &descent2, &overall);
693
694 if ( width )
695 *width = (overall.width);
696 if ( height )
697 *height = (ascent2 + descent2);
698 if ( descent )
699 *descent = descent2;
700 if ( ascent )
701 *ascent = ascent2;
702 }
703
704 #endif // !wxMOTIF_USE_RENDER_TABLE