]> git.saurik.com Git - wxWidgets.git/blame - src/x11/font.cpp
wxX11:
[wxWidgets.git] / src / x11 / font.cpp
CommitLineData
83df96d6
JS
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#ifdef __GNUG__
21 #pragma implementation "font.h"
22#endif
23
24#ifdef __VMS
25#pragma message disable nosimpint
26#include "wx/vms_x_fix.h"
27#endif
bc797f4c 28
83df96d6
JS
29#ifdef __VMS
30#pragma message enable nosimpint
31#endif
32
33#include "wx/defs.h"
34#include "wx/string.h"
35#include "wx/font.h"
36#include "wx/gdicmn.h"
37#include "wx/utils.h" // for wxGetDisplay()
38#include "wx/fontutil.h" // for wxNativeFontInfo
39#include "wx/tokenzr.h"
40#include "wx/settings.h"
41
42IMPLEMENT_DYNAMIC_CLASS(wxFont, wxGDIObject)
43
44// ----------------------------------------------------------------------------
45// private classes
46// ----------------------------------------------------------------------------
47
48// For every wxFont, there must be a font for each display and scale requested.
49// So these objects are stored in wxFontRefData::m_fonts
50class wxXFont : public wxObject
51{
52public:
53 wxXFont();
54 ~wxXFont();
55
56 WXFontStructPtr m_fontStruct; // XFontStruct
83df96d6
JS
57 WXDisplay* m_display; // XDisplay
58 int m_scale; // Scale * 100
59};
60
61class wxFontRefData: public wxGDIRefData
62{
63friend class wxFont;
64
65public:
66 wxFontRefData(int size = wxDEFAULT,
67 int family = wxDEFAULT,
68 int style = wxDEFAULT,
69 int weight = wxDEFAULT,
70 bool underlined = FALSE,
71 const wxString& faceName = wxEmptyString,
72 wxFontEncoding encoding = wxFONTENCODING_DEFAULT)
73 {
74 Init(size, family, style, weight, underlined, faceName, encoding);
75 }
76
77 wxFontRefData(const wxFontRefData& data)
78 {
79 Init(data.m_pointSize, data.m_family, data.m_style, data.m_weight,
80 data.m_underlined, data.m_faceName, data.m_encoding);
81 }
82
83 ~wxFontRefData();
84
85protected:
86 // common part of all ctors
87 void Init(int size,
88 int family,
89 int style,
90 int weight,
91 bool underlined,
92 const wxString& faceName,
93 wxFontEncoding encoding);
94
95 // font attributes
96 int m_pointSize;
97 int m_family;
98 int m_style;
99 int m_weight;
100 bool m_underlined;
101 wxString m_faceName;
102 wxFontEncoding m_encoding;
103
104 wxNativeFontInfo m_nativeFontInfo;
105
106 // A list of wxXFonts
107 wxList m_fonts;
108};
109
110// ============================================================================
111// implementation
112// ============================================================================
113
114// ----------------------------------------------------------------------------
115// wxXFont
116// ----------------------------------------------------------------------------
117
118wxXFont::wxXFont()
119{
120 m_fontStruct = (WXFontStructPtr) 0;
83df96d6
JS
121 m_display = (WXDisplay*) 0;
122 m_scale = 100;
123}
124
125wxXFont::~wxXFont()
126{
83df96d6
JS
127 // TODO: why does freeing the font produce a segv???
128 // Note that XFreeFont wasn't called in wxWin 1.68 either.
129 // XFontStruct* fontStruct = (XFontStruct*) m_fontStruct;
130 // XFreeFont((Display*) m_display, fontStruct);
131}
132
133// ----------------------------------------------------------------------------
134// wxFontRefData
135// ----------------------------------------------------------------------------
136
137void wxFontRefData::Init(int pointSize,
138 int family,
139 int style,
140 int weight,
141 bool underlined,
142 const wxString& faceName,
143 wxFontEncoding encoding)
144{
145 if (family == wxDEFAULT)
146 m_family = wxSWISS;
147 else
148 m_family = family;
149
150 m_faceName = faceName;
151
152 if (style == wxDEFAULT)
153 m_style = wxNORMAL;
154 else
155 m_style = style;
156
157 if (weight == wxDEFAULT)
158 m_weight = wxNORMAL;
159 else
160 m_weight = weight;
161
162 if (pointSize == wxDEFAULT)
163 m_pointSize = 12;
164 else
165 m_pointSize = pointSize;
166
167 m_underlined = underlined;
168 m_encoding = encoding;
169}
170
171wxFontRefData::~wxFontRefData()
172{
173 wxNode* node = m_fonts.First();
174 while (node)
175 {
176 wxXFont* f = (wxXFont*) node->Data();
177 delete f;
178 node = node->Next();
179 }
180 m_fonts.Clear();
181}
182
183// ----------------------------------------------------------------------------
184// wxFont
185// ----------------------------------------------------------------------------
186
187wxFont::wxFont(const wxNativeFontInfo& info)
188{
189 Init();
190
191 (void)Create(info.GetXFontName());
192}
193
194void wxFont::Init()
195{
196}
197
198bool wxFont::Create(int pointSize,
199 int family,
200 int style,
201 int weight,
202 bool underlined,
203 const wxString& faceName,
204 wxFontEncoding encoding)
205{
206 UnRef();
207 m_refData = new wxFontRefData(pointSize, family, style, weight,
208 underlined, faceName, encoding);
209
210 RealizeResource();
211
212 return TRUE;
213}
214
215bool wxFont::Create(const wxString& fontname, wxFontEncoding enc)
216{
217 if( !fontname )
218 {
219 *this = wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT);
220 return TRUE;
221 }
222
223 m_refData = new wxFontRefData();
224
225 M_FONTDATA->m_nativeFontInfo.SetXFontName(fontname); // X font name
226
227 wxString tmp;
228
229 wxStringTokenizer tn( fontname, wxT("-") );
230
231 tn.GetNextToken(); // skip initial empty token
232 tn.GetNextToken(); // foundry
233
234
235 M_FONTDATA->m_faceName = tn.GetNextToken(); // family
236
237 tmp = tn.GetNextToken().MakeUpper(); // weight
238 if (tmp == wxT("BOLD")) M_FONTDATA->m_weight = wxBOLD;
239 if (tmp == wxT("BLACK")) M_FONTDATA->m_weight = wxBOLD;
240 if (tmp == wxT("EXTRABOLD")) M_FONTDATA->m_weight = wxBOLD;
241 if (tmp == wxT("DEMIBOLD")) M_FONTDATA->m_weight = wxBOLD;
242 if (tmp == wxT("ULTRABOLD")) M_FONTDATA->m_weight = wxBOLD;
243
244 if (tmp == wxT("LIGHT")) M_FONTDATA->m_weight = wxLIGHT;
245 if (tmp == wxT("THIN")) M_FONTDATA->m_weight = wxLIGHT;
246
247 tmp = tn.GetNextToken().MakeUpper(); // slant
248 if (tmp == wxT("I")) M_FONTDATA->m_style = wxITALIC;
249 if (tmp == wxT("O")) M_FONTDATA->m_style = wxITALIC;
250
251 tn.GetNextToken(); // set width
252 tn.GetNextToken(); // add. style
253 tn.GetNextToken(); // pixel size
254
255 tmp = tn.GetNextToken(); // pointsize
256 if (tmp != wxT("*"))
257 {
258 long num = wxStrtol (tmp.c_str(), (wxChar **) NULL, 10);
259 M_FONTDATA->m_pointSize = (int)(num / 10);
260 }
261
262 tn.GetNextToken(); // x-res
263 tn.GetNextToken(); // y-res
264
265 tmp = tn.GetNextToken().MakeUpper(); // spacing
266
267 if (tmp == wxT("M"))
268 M_FONTDATA->m_family = wxMODERN;
269 else if (M_FONTDATA->m_faceName == wxT("TIMES"))
270 M_FONTDATA->m_family = wxROMAN;
271 else if (M_FONTDATA->m_faceName == wxT("HELVETICA"))
272 M_FONTDATA->m_family = wxSWISS;
273 else if (M_FONTDATA->m_faceName == wxT("LUCIDATYPEWRITER"))
274 M_FONTDATA->m_family = wxTELETYPE;
275 else if (M_FONTDATA->m_faceName == wxT("LUCIDA"))
276 M_FONTDATA->m_family = wxDECORATIVE;
277 else if (M_FONTDATA->m_faceName == wxT("UTOPIA"))
278 M_FONTDATA->m_family = wxSCRIPT;
279
280 tn.GetNextToken(); // avg width
281
282 // deal with font encoding
283 M_FONTDATA->m_encoding = enc;
284 if ( M_FONTDATA->m_encoding == wxFONTENCODING_SYSTEM )
285 {
286 wxString registry = tn.GetNextToken().MakeUpper(),
287 encoding = tn.GetNextToken().MakeUpper();
288
289 if ( registry == _T("ISO8859") )
290 {
291 int cp;
292 if ( wxSscanf(encoding, wxT("%d"), &cp) == 1 )
293 {
294 M_FONTDATA->m_encoding =
295 (wxFontEncoding)(wxFONTENCODING_ISO8859_1 + cp - 1);
296 }
297 }
298 else if ( registry == _T("MICROSOFT") )
299 {
300 int cp;
301 if ( wxSscanf(encoding, wxT("cp125%d"), &cp) == 1 )
302 {
303 M_FONTDATA->m_encoding =
304 (wxFontEncoding)(wxFONTENCODING_CP1250 + cp);
305 }
306 }
307 else if ( registry == _T("KOI8") )
308 {
309 M_FONTDATA->m_encoding = wxFONTENCODING_KOI8;
310 }
311 //else: unknown encoding - may be give a warning here?
312 else
313 return FALSE;
314 }
315 return TRUE;
316}
317
318wxFont::~wxFont()
319{
320}
321
322// ----------------------------------------------------------------------------
323// change the font attributes
324// ----------------------------------------------------------------------------
325
326void wxFont::Unshare()
327{
328 // Don't change shared data
329 if (!m_refData)
330 {
331 m_refData = new wxFontRefData();
332 }
333 else
334 {
335 wxFontRefData* ref = new wxFontRefData(*(wxFontRefData*)m_refData);
336 UnRef();
337 m_refData = ref;
338 }
339}
340
341void wxFont::SetPointSize(int pointSize)
342{
343 Unshare();
344
345 M_FONTDATA->m_pointSize = pointSize;
346 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
347
348 RealizeResource();
349}
350
351void wxFont::SetFamily(int family)
352{
353 Unshare();
354
355 M_FONTDATA->m_family = family;
356 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
357
358 RealizeResource();
359}
360
361void wxFont::SetStyle(int style)
362{
363 Unshare();
364
365 M_FONTDATA->m_style = style;
366 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
367
368 RealizeResource();
369}
370
371void wxFont::SetWeight(int weight)
372{
373 Unshare();
374
375 M_FONTDATA->m_weight = weight;
376 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
377
378 RealizeResource();
379}
380
381void wxFont::SetFaceName(const wxString& faceName)
382{
383 Unshare();
384
385 M_FONTDATA->m_faceName = faceName;
386 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
387
388 RealizeResource();
389}
390
391void wxFont::SetUnderlined(bool underlined)
392{
393 Unshare();
394
395 M_FONTDATA->m_underlined = underlined;
396
397 RealizeResource();
398}
399
400void wxFont::SetEncoding(wxFontEncoding encoding)
401{
402 Unshare();
403
404 M_FONTDATA->m_encoding = encoding;
405 M_FONTDATA->m_nativeFontInfo.GetXFontName().Clear(); // invalid now
406
407 RealizeResource();
408}
409
410void wxFont::SetNativeFontInfo(const wxNativeFontInfo& info)
411{
412 Unshare();
413
414 M_FONTDATA->m_nativeFontInfo = info;
415}
416
417// ----------------------------------------------------------------------------
418// query font attributes
419// ----------------------------------------------------------------------------
420
421int wxFont::GetPointSize() const
422{
423 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
424
425 return M_FONTDATA->m_pointSize;
426}
427
428wxString wxFont::GetFaceName() const
429{
430 wxCHECK_MSG( Ok(), wxT(""), wxT("invalid font") );
431
432 return M_FONTDATA->m_faceName ;
433}
434
435int wxFont::GetFamily() const
436{
437 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
438
439 return M_FONTDATA->m_family;
440}
441
442int wxFont::GetStyle() const
443{
444 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
445
446 return M_FONTDATA->m_style;
447}
448
449int wxFont::GetWeight() const
450{
451 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
452
453 return M_FONTDATA->m_weight;
454}
455
456bool wxFont::GetUnderlined() const
457{
458 wxCHECK_MSG( Ok(), FALSE, wxT("invalid font") );
459
460 return M_FONTDATA->m_underlined;
461}
462
463wxFontEncoding wxFont::GetEncoding() const
464{
465 wxCHECK_MSG( Ok(), wxFONTENCODING_DEFAULT, wxT("invalid font") );
466
467 return M_FONTDATA->m_encoding;
468}
469
470wxNativeFontInfo *wxFont::GetNativeFontInfo() const
471{
472 wxCHECK_MSG( Ok(), (wxNativeFontInfo *)NULL, wxT("invalid font") );
473
474 if(M_FONTDATA->m_nativeFontInfo.GetXFontName().IsEmpty())
475 GetInternalFont();
476
477 return new wxNativeFontInfo(M_FONTDATA->m_nativeFontInfo);
478}
479
480// ----------------------------------------------------------------------------
481// real implementation
482// ----------------------------------------------------------------------------
483
484// Find an existing, or create a new, XFontStruct
485// based on this wxFont and the given scale. Append the
486// font to list in the private data for future reference.
487wxXFont* wxFont::GetInternalFont(double scale, WXDisplay* display) const
488{
489 if ( !Ok() )
490 return (wxXFont *)NULL;
491
492 long intScale = long(scale * 100.0 + 0.5); // key for wxXFont
493 int pointSize = (M_FONTDATA->m_pointSize * 10 * intScale) / 100;
494
495 // search existing fonts first
496 wxNode* node = M_FONTDATA->m_fonts.First();
497 while (node)
498 {
499 wxXFont* f = (wxXFont*) node->Data();
500 if ((!display || (f->m_display == display)) && (f->m_scale == intScale))
501 return f;
502 node = node->Next();
503 }
504
505 // not found, create a new one
506 XFontStruct *font = (XFontStruct *)
507 wxLoadQueryNearestFont(pointSize,
508 M_FONTDATA->m_family,
509 M_FONTDATA->m_style,
510 M_FONTDATA->m_weight,
511 M_FONTDATA->m_underlined,
512 wxT(""),
513 M_FONTDATA->m_encoding);
514
515 if ( !font )
516 {
517 wxFAIL_MSG( wxT("Could not allocate even a default font -- something is wrong.") );
518
519 return (wxXFont*) NULL;
520 }
521
522 wxXFont* f = new wxXFont;
523 f->m_fontStruct = (WXFontStructPtr)font;
524 f->m_display = ( display ? display : wxGetDisplay() );
525 f->m_scale = intScale;
83df96d6
JS
526 M_FONTDATA->m_fonts.Append(f);
527
528 return f;
529}
530
531WXFontStructPtr wxFont::GetFontStruct(double scale, WXDisplay* display) const
532{
533 wxXFont* f = GetInternalFont(scale, display);
534
535 return (f ? f->m_fontStruct : (WXFontStructPtr) 0);
536}
537