]>
Commit | Line | Data |
---|---|---|
c801d85f KB |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: font.cpp | |
3 | // Purpose: | |
4 | // Author: Robert Roebling | |
a81258be | 5 | // Id: $Id$ |
c801d85f | 6 | // Copyright: (c) 1998 Robert Roebling, Julian Smart and Markus Holzem |
8bbe427f | 7 | // Licence: wxWindows licence |
c801d85f KB |
8 | ///////////////////////////////////////////////////////////////////////////// |
9 | ||
10 | #ifdef __GNUG__ | |
11 | #pragma implementation "font.h" | |
12 | #endif | |
13 | ||
14 | #include "wx/font.h" | |
15 | #include "wx/utils.h" | |
5705323e | 16 | #include "wx/log.h" |
c801d85f KB |
17 | #include <strings.h> |
18 | ||
83624f79 RR |
19 | #include "gdk/gdk.h" |
20 | ||
c801d85f KB |
21 | //----------------------------------------------------------------------------- |
22 | // local data | |
23 | //----------------------------------------------------------------------------- | |
24 | ||
36b3b54a | 25 | /* |
a3622daa | 26 | extern wxFontNameDirectory *wxTheFontNameDirectory; |
36b3b54a | 27 | */ |
c801d85f KB |
28 | |
29 | //----------------------------------------------------------------------------- | |
30 | // wxFont | |
31 | //----------------------------------------------------------------------------- | |
32 | ||
33 | class wxFontRefData: public wxObjectRefData | |
34 | { | |
8bbe427f VZ |
35 | public: |
36 | ||
37 | wxFontRefData(); | |
358fc25c | 38 | wxFontRefData( const wxFontRefData& data ); |
8bbe427f VZ |
39 | ~wxFontRefData(); |
40 | ||
358fc25c RR |
41 | wxList m_scaled_xfonts; |
42 | int m_pointSize; | |
43 | int m_family, m_style, m_weight; | |
44 | bool m_underlined; | |
358fc25c | 45 | wxString m_faceName; |
8bbe427f | 46 | |
358fc25c RR |
47 | bool m_byXFontName; |
48 | GdkFont *m_font; | |
8bbe427f | 49 | |
c801d85f KB |
50 | friend wxFont; |
51 | }; | |
52 | ||
8bbe427f VZ |
53 | wxFontRefData::wxFontRefData() : m_scaled_xfonts(wxKEY_INTEGER) |
54 | { | |
55 | m_byXFontName = FALSE; | |
56 | m_pointSize = 12; | |
57 | m_family = wxSWISS; | |
58 | m_style = wxNORMAL; | |
59 | m_weight = wxNORMAL; | |
60 | m_underlined = FALSE; | |
8bbe427f VZ |
61 | m_font = (GdkFont *) NULL; |
62 | } | |
63 | ||
358fc25c RR |
64 | wxFontRefData::wxFontRefData( const wxFontRefData& data ) : m_scaled_xfonts(wxKEY_INTEGER) |
65 | { | |
66 | m_byXFontName = FALSE; | |
67 | m_pointSize = data.m_pointSize; | |
68 | m_family = data.m_family; | |
69 | m_style = data.m_style; | |
70 | m_weight = data.m_weight; | |
71 | m_underlined = data.m_underlined; | |
358fc25c RR |
72 | m_faceName = data.m_faceName; |
73 | m_font = (GdkFont *) NULL; | |
74 | if (data.m_font) m_font = gdk_font_ref( data.m_font ); | |
75 | } | |
76 | ||
8bbe427f VZ |
77 | wxFontRefData::~wxFontRefData() |
78 | { | |
79 | wxNode *node = m_scaled_xfonts.First(); | |
80 | while (node) | |
81 | { | |
82 | GdkFont *font = (GdkFont*)node->Data(); | |
83 | wxNode *next = node->Next(); | |
84 | gdk_font_unref( font ); | |
85 | node = next; | |
86 | } | |
8bbe427f | 87 | if (m_font) gdk_font_unref( m_font ); |
ff7b1510 | 88 | } |
c801d85f KB |
89 | |
90 | //----------------------------------------------------------------------------- | |
91 | ||
92 | #define M_FONTDATA ((wxFontRefData *)m_refData) | |
93 | ||
94 | IMPLEMENT_DYNAMIC_CLASS(wxFont, wxGDIObject) | |
95 | ||
8bbe427f | 96 | wxFont::wxFont() |
c801d85f | 97 | { |
8bbe427f | 98 | if (wxTheFontList) wxTheFontList->Append( this ); |
ff7b1510 | 99 | } |
c801d85f KB |
100 | |
101 | wxFont::wxFont( char *xFontName ) | |
102 | { | |
8bbe427f VZ |
103 | if (!xFontName) return; |
104 | ||
105 | m_refData = new wxFontRefData(); | |
106 | ||
107 | M_FONTDATA->m_byXFontName = TRUE; | |
108 | M_FONTDATA->m_font = gdk_font_load( xFontName ); | |
ff7b1510 | 109 | } |
c801d85f | 110 | |
8fc613f1 | 111 | wxFont::wxFont( int pointSize, int family, int style, int weight, bool underlined, const wxString& face ) |
8bbe427f VZ |
112 | { |
113 | m_refData = new wxFontRefData(); | |
114 | ||
36b3b54a RR |
115 | if (family == wxDEFAULT) |
116 | M_FONTDATA->m_family = wxSWISS; | |
8bbe427f | 117 | else |
36b3b54a RR |
118 | M_FONTDATA->m_family = family; |
119 | ||
120 | if (!face.IsEmpty()) M_FONTDATA->m_faceName = face; | |
8bbe427f | 121 | |
36b3b54a RR |
122 | if (style == wxDEFAULT) |
123 | M_FONTDATA->m_style = wxNORMAL; | |
124 | else | |
125 | M_FONTDATA->m_style = style; | |
126 | ||
127 | if (weight == wxDEFAULT) | |
128 | M_FONTDATA->m_weight = wxNORMAL; | |
129 | else | |
130 | M_FONTDATA->m_weight = weight; | |
131 | ||
132 | if (pointSize == wxDEFAULT) | |
133 | M_FONTDATA->m_pointSize = 12; | |
134 | else | |
135 | M_FONTDATA->m_pointSize = pointSize; | |
136 | ||
358fc25c | 137 | M_FONTDATA->m_underlined = underlined; |
8bbe427f VZ |
138 | |
139 | if (wxTheFontList) wxTheFontList->Append( this ); | |
358fc25c | 140 | |
ff7b1510 | 141 | } |
c801d85f | 142 | |
8bbe427f VZ |
143 | wxFont::wxFont( const wxFont& font ) |
144 | { | |
145 | Ref( font ); | |
146 | ||
147 | if (wxTheFontList) wxTheFontList->Append( this ); | |
ff7b1510 | 148 | } |
c801d85f | 149 | |
8bbe427f | 150 | wxFont::~wxFont() |
c801d85f | 151 | { |
8bbe427f | 152 | if (wxTheFontList) wxTheFontList->DeleteObject( this ); |
ff7b1510 | 153 | } |
c801d85f | 154 | |
8bbe427f VZ |
155 | wxFont& wxFont::operator = ( const wxFont& font ) |
156 | { | |
157 | if (*this == font) return (*this); | |
158 | Ref( font ); | |
159 | return *this; | |
ff7b1510 | 160 | } |
c801d85f | 161 | |
f03fc89f | 162 | bool wxFont::operator == ( const wxFont& font ) const |
8bbe427f VZ |
163 | { |
164 | return m_refData == font.m_refData; | |
ff7b1510 | 165 | } |
c801d85f | 166 | |
f03fc89f | 167 | bool wxFont::operator != ( const wxFont& font ) const |
8bbe427f VZ |
168 | { |
169 | return m_refData != font.m_refData; | |
ff7b1510 | 170 | } |
c801d85f | 171 | |
d84eb083 | 172 | bool wxFont::Ok() const |
c801d85f | 173 | { |
8bbe427f | 174 | return (m_refData != NULL); |
ff7b1510 | 175 | } |
c801d85f | 176 | |
8bbe427f | 177 | int wxFont::GetPointSize() const |
c801d85f | 178 | { |
ed9b9841 | 179 | wxCHECK_MSG( Ok(), 0, _T("invalid font") ); |
8bbe427f VZ |
180 | |
181 | return M_FONTDATA->m_pointSize; | |
ff7b1510 | 182 | } |
c801d85f | 183 | |
8bbe427f | 184 | wxString wxFont::GetFaceName() const |
c801d85f | 185 | { |
ed9b9841 | 186 | wxCHECK_MSG( Ok(), _T(""), _T("invalid font") ); |
8bbe427f | 187 | |
36b3b54a | 188 | return M_FONTDATA->m_faceName; |
ff7b1510 | 189 | } |
c801d85f | 190 | |
8bbe427f | 191 | int wxFont::GetFamily() const |
c801d85f | 192 | { |
ed9b9841 | 193 | wxCHECK_MSG( Ok(), 0, _T("invalid font") ); |
8bbe427f VZ |
194 | |
195 | return M_FONTDATA->m_family; | |
ff7b1510 | 196 | } |
c801d85f | 197 | |
8bbe427f | 198 | wxString wxFont::GetFamilyString() const |
c801d85f | 199 | { |
ed9b9841 | 200 | wxCHECK_MSG( Ok(), _T("wxDEFAULT"), _T("invalid font") ); |
8bbe427f VZ |
201 | |
202 | switch (M_FONTDATA->m_family) | |
203 | { | |
ed9b9841 OK |
204 | case wxDECORATIVE: return wxString(_T("wxDECORATIVE")); |
205 | case wxROMAN: return wxString(_T("wxROMAN")); | |
206 | case wxSCRIPT: return wxString(_T("wxSCRIPT")); | |
207 | case wxSWISS: return wxString(_T("wxSWISS")); | |
208 | case wxMODERN: return wxString(_T("wxMODERN")); | |
209 | case wxTELETYPE: return wxString(_T("wxTELETYPE")); | |
210 | default: return _T("wxDEFAULT"); | |
8bbe427f VZ |
211 | } |
212 | ||
e55ad60e | 213 | return "wxDEFAULT"; |
ff7b1510 | 214 | } |
c801d85f | 215 | |
8bbe427f | 216 | int wxFont::GetStyle() const |
c801d85f | 217 | { |
ed9b9841 | 218 | wxCHECK_MSG( Ok(), 0, _T("invalid font") ); |
d84eb083 | 219 | |
8bbe427f | 220 | return M_FONTDATA->m_style; |
ff7b1510 | 221 | } |
c801d85f | 222 | |
8bbe427f | 223 | wxString wxFont::GetStyleString() const |
c801d85f | 224 | { |
ed9b9841 | 225 | wxCHECK_MSG( Ok(), _T("wxDEFAULT"), _T("invalid font") ); |
8bbe427f VZ |
226 | |
227 | switch (M_FONTDATA->m_style) | |
228 | { | |
ed9b9841 OK |
229 | case wxNORMAL: return wxString(_T("wxNORMAL")); |
230 | case wxSLANT: return wxString(_T("wxSLANT")); | |
231 | case wxITALIC: return wxString(_T("wxITALIC")); | |
232 | default: return wxString(_T("wxDEFAULT")); | |
8bbe427f | 233 | } |
d84eb083 | 234 | |
ed9b9841 | 235 | return wxString(_T("wxDEFAULT")); |
ff7b1510 | 236 | } |
c801d85f | 237 | |
8bbe427f | 238 | int wxFont::GetWeight() const |
c801d85f | 239 | { |
ed9b9841 | 240 | wxCHECK_MSG( Ok(), 0, _T("invalid font") ); |
8bbe427f VZ |
241 | |
242 | return M_FONTDATA->m_weight; | |
243 | } | |
244 | ||
245 | wxString wxFont::GetWeightString() const | |
246 | { | |
ed9b9841 | 247 | wxCHECK_MSG( Ok(), _T("wxDEFAULT"), _T("invalid font") ); |
8bbe427f VZ |
248 | |
249 | switch (M_FONTDATA->m_weight) | |
250 | { | |
ed9b9841 OK |
251 | case wxNORMAL: return wxString(_T("wxNORMAL")); |
252 | case wxBOLD: return wxString(_T("wxBOLD")); | |
253 | case wxLIGHT: return wxString(_T("wxLIGHT")); | |
254 | default: return wxString(_T("wxDEFAULT")); | |
8bbe427f VZ |
255 | } |
256 | ||
ed9b9841 | 257 | return wxString(_T("wxDEFAULT")); |
8bbe427f VZ |
258 | } |
259 | ||
260 | bool wxFont::GetUnderlined() const | |
261 | { | |
ed9b9841 | 262 | wxCHECK_MSG( Ok(), FALSE, _T("invalid font") ); |
8bbe427f VZ |
263 | |
264 | return M_FONTDATA->m_underlined; | |
ff7b1510 | 265 | } |
c801d85f | 266 | |
358fc25c RR |
267 | void wxFont::Unshare() |
268 | { | |
269 | if (!m_refData) | |
270 | { | |
271 | m_refData = new wxFontRefData(); | |
272 | } | |
273 | else | |
274 | { | |
275 | wxFontRefData* ref = new wxFontRefData(*(wxFontRefData*)m_refData); | |
276 | UnRef(); | |
277 | m_refData = ref; | |
278 | } | |
279 | } | |
280 | ||
281 | void wxFont::SetPointSize(int pointSize) | |
282 | { | |
283 | Unshare(); | |
284 | ||
285 | M_FONTDATA->m_pointSize = pointSize; | |
286 | } | |
287 | ||
288 | void wxFont::SetFamily(int family) | |
289 | { | |
290 | Unshare(); | |
291 | ||
292 | M_FONTDATA->m_family = family; | |
293 | } | |
294 | ||
295 | void wxFont::SetStyle(int style) | |
296 | { | |
297 | Unshare(); | |
298 | ||
299 | M_FONTDATA->m_style = style; | |
300 | } | |
301 | ||
302 | void wxFont::SetWeight(int weight) | |
303 | { | |
304 | Unshare(); | |
305 | ||
306 | M_FONTDATA->m_weight = weight; | |
307 | } | |
308 | ||
309 | void wxFont::SetFaceName(const wxString& faceName) | |
310 | { | |
311 | Unshare(); | |
312 | ||
313 | M_FONTDATA->m_faceName = faceName; | |
314 | } | |
315 | ||
316 | void wxFont::SetUnderlined(bool underlined) | |
317 | { | |
318 | Unshare(); | |
319 | ||
320 | M_FONTDATA->m_underlined = underlined; | |
321 | } | |
322 | ||
c801d85f KB |
323 | //----------------------------------------------------------------------------- |
324 | // get internal representation of font | |
325 | //----------------------------------------------------------------------------- | |
326 | ||
36b3b54a RR |
327 | static GdkFont *wxLoadQueryNearestFont( int point_size, int family, int style, int weight, |
328 | bool underlined, const wxString &facename ); | |
c801d85f | 329 | |
36b3b54a | 330 | GdkFont *wxFont::GetInternalFont( float scale ) const |
c801d85f | 331 | { |
8bbe427f VZ |
332 | if (!Ok()) |
333 | { | |
ed9b9841 | 334 | wxFAIL_MSG( _T("invalid font") ); |
8bbe427f VZ |
335 | return (GdkFont*) NULL; |
336 | } | |
337 | ||
36b3b54a | 338 | /* short cut if the special X font constructor has been used */ |
8bbe427f VZ |
339 | if (M_FONTDATA->m_byXFontName) return M_FONTDATA->m_font; |
340 | ||
36b3b54a | 341 | long int_scale = long(scale * 100.0 + 0.5); /* key for fontlist */ |
8bbe427f VZ |
342 | int point_scale = (M_FONTDATA->m_pointSize * 10 * int_scale) / 100; |
343 | GdkFont *font = (GdkFont *) NULL; | |
344 | ||
345 | wxNode *node = M_FONTDATA->m_scaled_xfonts.Find(int_scale); | |
346 | if (node) | |
347 | { | |
348 | font = (GdkFont*)node->Data(); | |
349 | } | |
350 | else | |
351 | { | |
36b3b54a | 352 | /* |
8bbe427f VZ |
353 | if ((int_scale == 100) && |
354 | (M_FONTDATA->m_family == wxSWISS) && | |
355 | (M_FONTDATA->m_style == wxNORMAL) && | |
356 | (M_FONTDATA->m_pointSize == 12) && | |
357 | (M_FONTDATA->m_weight == wxNORMAL) && | |
358 | (M_FONTDATA->m_underlined == FALSE)) | |
359 | { | |
360 | font = gdk_font_load( "-adobe-helvetica-medium-r-normal--*-120-*-*-*-*-*-*" ); | |
361 | } | |
362 | else | |
36b3b54a | 363 | */ |
8bbe427f | 364 | { |
36b3b54a RR |
365 | font = wxLoadQueryNearestFont( point_scale, M_FONTDATA->m_family, M_FONTDATA->m_style, |
366 | M_FONTDATA->m_weight, M_FONTDATA->m_underlined, M_FONTDATA->m_faceName ); | |
8bbe427f VZ |
367 | } |
368 | M_FONTDATA->m_scaled_xfonts.Append( int_scale, (wxObject*)font ); | |
369 | } | |
36b3b54a | 370 | |
8bbe427f | 371 | if (!font) |
36b3b54a | 372 | { |
ed9b9841 | 373 | wxLogError(_T("could not load any font")); |
36b3b54a | 374 | } |
5705323e | 375 | |
8bbe427f | 376 | return font; |
ff7b1510 | 377 | } |
c801d85f KB |
378 | |
379 | //----------------------------------------------------------------------------- | |
380 | // local utilities to find a X font | |
381 | //----------------------------------------------------------------------------- | |
382 | ||
36b3b54a RR |
383 | static GdkFont*wxLoadQueryFont( int pointSize, int family, int style, int weight, |
384 | bool underlined, const wxString &facename ) | |
c801d85f | 385 | { |
ed9b9841 OK |
386 | wxChar *xfamily = (wxChar*) NULL; |
387 | wxChar *xstyle = (wxChar*) NULL; | |
388 | wxChar *xweight = (wxChar*) NULL; | |
36b3b54a RR |
389 | |
390 | switch (family) | |
391 | { | |
ed9b9841 OK |
392 | case wxDECORATIVE: xfamily = _T("lucida"); break; |
393 | case wxROMAN: xfamily = _T("times"); break; | |
394 | case wxMODERN: xfamily = _T("courier"); break; | |
395 | case wxSWISS: xfamily = _T("helvetica"); break; | |
396 | case wxTELETYPE: xfamily = _T("lucidatypewriter"); break; | |
397 | case wxSCRIPT: xfamily = _T("utopia"); break; | |
398 | default: xfamily = _T("*"); | |
36b3b54a RR |
399 | } |
400 | ||
401 | if (!facename.IsEmpty()) | |
402 | { | |
ed9b9841 OK |
403 | wxSprintf( wxBuffer, _T("-*-%s-*-*-normal-*-*-*-*-*-*-*-*-*"), facename.c_str() ); |
404 | GdkFont *test = gdk_font_load( wxConv_libc.cWX2MB(wxBuffer) ); | |
36b3b54a RR |
405 | if (test) |
406 | { | |
407 | gdk_font_unref( test ); | |
408 | xfamily = WXSTRINGCAST facename; | |
409 | } | |
410 | } | |
411 | ||
412 | switch (style) | |
413 | { | |
ed9b9841 OK |
414 | case wxITALIC: xstyle = _T("i"); break; |
415 | case wxSLANT: xstyle = _T("o"); break; | |
416 | case wxNORMAL: xstyle = _T("r"); break; | |
417 | default: xstyle = _T("*"); break; | |
36b3b54a RR |
418 | } |
419 | switch (weight) | |
420 | { | |
ed9b9841 | 421 | case wxBOLD: xweight = _T("bold"); break; |
36b3b54a | 422 | case wxLIGHT: |
ed9b9841 OK |
423 | case wxNORMAL: xweight = _T("medium"); break; |
424 | default: xweight = _T("*"); break; | |
36b3b54a RR |
425 | } |
426 | ||
ed9b9841 | 427 | wxSprintf( wxBuffer, _T("-*-%s-%s-%s-normal-*-*-%d-*-*-*-*-*-*"), |
36b3b54a RR |
428 | xfamily, xweight, xstyle, pointSize); |
429 | ||
ed9b9841 | 430 | return gdk_font_load( wxConv_libc.cWX2MB(wxBuffer) ); |
c801d85f KB |
431 | } |
432 | ||
36b3b54a RR |
433 | static GdkFont *wxLoadQueryNearestFont( int point_size, int family, int style, int weight, |
434 | bool underlined, const wxString &facename ) | |
c801d85f | 435 | { |
36b3b54a | 436 | GdkFont *font = wxLoadQueryFont( point_size, family, style, weight, underlined, facename ); |
c801d85f | 437 | |
36b3b54a RR |
438 | if (!font) |
439 | { | |
440 | /* search up and down by stepsize 10 */ | |
8bbe427f VZ |
441 | int max_size = point_size + 20 * (1 + (point_size/180)); |
442 | int min_size = point_size - 20 * (1 + (point_size/180)); | |
36b3b54a | 443 | |
8fc613f1 RR |
444 | int i; |
445 | ||
36b3b54a | 446 | /* Search for smaller size (approx.) */ |
8fc613f1 | 447 | for (i=point_size-10; !font && i >= 10 && i >= min_size; i -= 10) |
36b3b54a RR |
448 | font = wxLoadQueryFont(i, family, style, weight, underlined, facename ); |
449 | ||
450 | /* Search for larger size (approx.) */ | |
8fc613f1 | 451 | for (i=point_size+10; !font && i <= max_size; i += 10) |
36b3b54a RR |
452 | font = wxLoadQueryFont( i, family, style, weight, underlined, facename ); |
453 | ||
454 | /* Try default family */ | |
455 | if (!font && family != wxDEFAULT) | |
456 | font = wxLoadQueryFont( point_size, wxDEFAULT, style, weight, underlined, facename ); | |
457 | ||
458 | /* Bogus font */ | |
8bbe427f | 459 | if (!font) |
36b3b54a | 460 | font = wxLoadQueryFont(120, wxDEFAULT, wxNORMAL, wxNORMAL, underlined, facename ); |
c801d85f | 461 | } |
36b3b54a | 462 | |
c801d85f KB |
463 | return font; |
464 | } | |
465 | ||
36b3b54a RR |
466 | /* |
467 | ||
c801d85f KB |
468 | //----------------------------------------------------------------------------- |
469 | // face names and index functions | |
470 | //----------------------------------------------------------------------------- | |
471 | ||
472 | static char *font_defaults[] = { | |
473 | "FamilyDefault", "Default", | |
474 | "FamilyRoman", "Roman", | |
475 | "FamilyDecorative", "Decorative", | |
476 | "FamilyModern", "Modern", | |
477 | "FamilyTeletype", "Teletype", | |
478 | "FamilySwiss", "Swiss", | |
479 | "FamilyScript", "Script", | |
480 | ||
481 | "AfmMedium", "", | |
482 | "AfmBold", "Bo", | |
483 | "AfmLight", "", | |
484 | "AfmStraight", "", | |
485 | "AfmItalic", "${AfmSlant}", | |
486 | "AfmSlant", "O", | |
487 | "AfmRoman", "Ro", | |
488 | "AfmTimes", "Times", | |
489 | "AfmHelvetica", "Helv", | |
490 | "AfmCourier", "Cour", | |
8bbe427f | 491 | |
c801d85f KB |
492 | "Afm___", "${AfmTimes,$[weight],$[style]}", |
493 | ||
494 | "AfmTimes__", "${AfmTimes}${Afm$[weight]}${Afm$[style]}", | |
495 | "AfmTimesMediumStraight", "${AfmTimes}${AfmRoman}", | |
496 | "AfmTimesLightStraight", "${AfmTimes}${AfmRoman}", | |
497 | "AfmTimes_Italic", "${AfmTimes}$[weight]${AfmItalic}", | |
498 | "AfmTimes_Slant", "${AfmTimes}$[weight]${AfmItalic}", | |
499 | ||
500 | "AfmSwiss__", "${AfmHelvetica}${Afm$[weight]}${Afm$[style]}", | |
501 | "AfmModern__", "${AfmCourier}${Afm$[weight]}${Afm$[style]}", | |
502 | ||
503 | "AfmTeletype__", "${AfmModern,$[weight],$[style]}", | |
504 | ||
505 | "PostScriptMediumStraight", "", | |
506 | "PostScriptMediumItalic", "-Oblique", | |
507 | "PostScriptMediumSlant", "-Oblique", | |
508 | "PostScriptLightStraight", "", | |
509 | "PostScriptLightItalic", "-Oblique", | |
510 | "PostScriptLightSlant", "-Oblique", | |
511 | "PostScriptBoldStraight", "-Bold", | |
512 | "PostScriptBoldItalic", "-BoldOblique", | |
513 | "PostScriptBoldSlant", "-BoldOblique", | |
8bbe427f | 514 | |
c801d85f KB |
515 | #if WX_NORMALIZED_PS_FONTS |
516 | "PostScript___", "${PostScriptTimes,$[weight],$[style]}", | |
517 | #else | |
518 | "PostScriptRoman__", "${PostScriptTimes,$[weight],$[style]}", | |
519 | "PostScript___", "LucidaSans${PostScript$[weight]$[style]}", | |
520 | #endif | |
521 | ||
522 | "PostScriptTimesMedium", "", | |
523 | "PostScriptTimesLight", "", | |
524 | "PostScriptTimesBold", "Bold", | |
525 | ||
526 | "PostScriptTimes__", "Times${PostScript$[weight]$[style]}", | |
527 | "PostScriptTimesMediumStraight", "Times-Roman", | |
528 | "PostScriptTimesLightStraight", "Times-Roman", | |
529 | "PostScriptTimes_Slant", "Times-${PostScriptTimes$[weight]}Italic", | |
530 | "PostScriptTimes_Italic", "Times-${PostScriptTimes$[weight]}Italic", | |
531 | ||
532 | "PostScriptSwiss__", "Helvetica${PostScript$[weight]$[style]}", | |
533 | "PostScriptModern__", "Courier${PostScript$[weight]$[style]}", | |
534 | ||
535 | "PostScriptTeletype__", "${PostScriptModern,$[weight],$[style]}", | |
536 | ||
537 | #if !WX_NORMALIZED_PS_FONTS | |
538 | "PostScriptScript__", "Zapf-Chancery-MediumItalic", | |
539 | #endif | |
540 | ||
541 | "ScreenMedium", "medium", | |
542 | "ScreenBold", "bold", | |
543 | "ScreenLight", "light", | |
544 | "ScreenStraight", "r", | |
545 | "ScreenItalic", "i", | |
546 | "ScreenSlant", "o", | |
547 | ||
b1c71cd0 RR |
548 | "ScreenDefaultBase", "*-times", |
549 | ||
c801d85f KB |
550 | "ScreenRomanBase", "*-times", |
551 | "ScreenDecorativeBase", "*-helvetica", | |
552 | "ScreenModernBase", "*-courier", | |
553 | "ScreenTeletypeBase", "*-lucidatypewriter", | |
554 | "ScreenSwissBase", "*-lucida", | |
555 | "ScreenScriptBase", "*-zapfchancery", | |
556 | ||
557 | "ScreenStdSuffix", "-${Screen$[weight]}-${Screen$[style]}" | |
8bbe427f | 558 | "-normal-*-*-%d-*-*-*-*-*-*", |
c801d85f KB |
559 | |
560 | "Screen___", | |
561 | "-${ScreenDefaultBase}${ScreenStdSuffix}", | |
562 | "ScreenRoman__", | |
563 | "-${ScreenRomanBase}${ScreenStdSuffix}", | |
564 | "ScreenDecorative__", | |
565 | "-${ScreenDecorativeBase}${ScreenStdSuffix}", | |
566 | "ScreenModern__", | |
567 | "-${ScreenModernBase}${ScreenStdSuffix}", | |
568 | "ScreenTeletype__", | |
569 | "-${ScreenTeletypeBase}${ScreenStdSuffix}", | |
570 | "ScreenSwiss__", | |
571 | "-${ScreenSwissBase}${ScreenStdSuffix}", | |
572 | "ScreenScript__", | |
573 | "-${ScreenScriptBase}${ScreenStdSuffix}", | |
c67daf87 | 574 | (char *) NULL |
c801d85f KB |
575 | }; |
576 | ||
577 | enum {wxWEIGHT_NORMAL, wxWEIGHT_BOLD, wxWEIGHT_LIGHT, wxNUM_WEIGHTS}; | |
578 | enum {wxSTYLE_NORMAL, wxSTYLE_ITALIC, wxSTYLE_SLANT, wxNUM_STYLES}; | |
579 | ||
580 | static int WCoordinate(int w) | |
581 | { | |
5705323e RR |
582 | switch (w) |
583 | { | |
8bbe427f VZ |
584 | case wxBOLD: return wxWEIGHT_BOLD; |
585 | case wxLIGHT: return wxWEIGHT_LIGHT; | |
586 | case wxNORMAL: | |
587 | default: return wxWEIGHT_NORMAL; | |
c801d85f | 588 | } |
ff7b1510 | 589 | }; |
c801d85f KB |
590 | |
591 | static int SCoordinate(int s) | |
592 | { | |
5705323e RR |
593 | switch (s) |
594 | { | |
8bbe427f VZ |
595 | case wxITALIC: return wxSTYLE_ITALIC; |
596 | case wxSLANT: return wxSTYLE_SLANT; | |
597 | case wxNORMAL: | |
598 | default: return wxSTYLE_NORMAL; | |
c801d85f | 599 | } |
ff7b1510 | 600 | }; |
c801d85f KB |
601 | |
602 | //----------------------------------------------------------------------------- | |
603 | // wxSuffixMap | |
604 | //----------------------------------------------------------------------------- | |
605 | ||
5705323e RR |
606 | class wxSuffixMap |
607 | { | |
c801d85f | 608 | public: |
8bbe427f | 609 | ~wxSuffixMap(); |
c801d85f KB |
610 | |
611 | inline char *GetName(int weight, int style) | |
612 | { | |
8bbe427f | 613 | return ( map [WCoordinate(weight)] [SCoordinate(style)] ); |
c801d85f KB |
614 | } |
615 | ||
616 | char *map[wxNUM_WEIGHTS][wxNUM_STYLES]; | |
617 | void Initialize(const char *, const char *); | |
618 | }; | |
619 | ||
c801d85f KB |
620 | static void SearchResource(const char *prefix, const char **names, int count, char **v) |
621 | { | |
622 | int k, i, j; | |
623 | char resource[1024], **defaults, *internal; | |
624 | ||
625 | k = 1 << count; | |
8bbe427f | 626 | |
c67daf87 UR |
627 | *v = (char *) NULL; |
628 | internal = (char *) NULL; | |
c801d85f | 629 | |
5705323e RR |
630 | for (i = 0; i < k; i++) |
631 | { | |
8bbe427f | 632 | strcpy(resource, prefix); |
5705323e RR |
633 | for (j = 0; j < count; j++) |
634 | { | |
36b3b54a RR |
635 | // upon failure to find a matching fontname |
636 | // in the default fonts above, we substitute more | |
637 | // and more values by _ so that at last ScreenMyFontBoldNormal | |
638 | // would turn into Screen___ and this will then get | |
639 | // converted to -${ScreenDefaultBase}${ScreenStdSuffix} | |
5705323e | 640 | |
8bbe427f VZ |
641 | if (!(i & (1 << j))) |
642 | strcat(resource, names[j]); | |
643 | else | |
644 | strcat(resource, "_"); | |
645 | } | |
5705323e | 646 | |
36b3b54a | 647 | // we previously search the Xt-resources here |
5705323e RR |
648 | |
649 | if (!internal) | |
650 | { | |
8bbe427f | 651 | defaults = font_defaults; |
5705323e RR |
652 | while (*defaults) |
653 | { | |
654 | if (!strcmp(*defaults, resource)) | |
655 | { | |
8bbe427f VZ |
656 | internal = defaults[1]; |
657 | break; | |
658 | } | |
659 | defaults += 2; | |
660 | } | |
661 | } | |
c801d85f | 662 | } |
5705323e | 663 | |
c801d85f | 664 | if (internal) |
5705323e | 665 | { |
b1c71cd0 RR |
666 | if ((strcmp(internal,"-${ScreenDefaultBase}${ScreenStdSuffix}") == 0) && |
667 | (strcmp(names[0], "Default") != 0)) | |
5705323e | 668 | { |
36b3b54a RR |
669 | // we did not find any font name in the standard list. |
670 | // this can (hopefully does) mean that someone supplied | |
671 | // the facename in the wxFont constructor so we insert | |
672 | // it here | |
5705323e | 673 | |
36b3b54a RR |
674 | strcpy( resource,"-*-" ); // any producer |
675 | strcat( resource, names[0] ); // facename | |
676 | strcat( resource, "${ScreenStdSuffix}" ); // add size params later on | |
5705323e RR |
677 | *v = copystring(resource); |
678 | } | |
679 | else | |
680 | { | |
681 | *v = copystring(internal); | |
682 | } | |
683 | } | |
c801d85f KB |
684 | } |
685 | ||
8bbe427f | 686 | wxSuffixMap::~wxSuffixMap() |
c801d85f KB |
687 | { |
688 | int k, j; | |
689 | ||
690 | for (k = 0; k < wxNUM_WEIGHTS; ++k) | |
8bbe427f | 691 | for (j = 0; j < wxNUM_STYLES; ++j) |
5705323e RR |
692 | if (map[k][j]) |
693 | { | |
8bbe427f VZ |
694 | delete[] map[k][j]; |
695 | map[k][j] = (char *) NULL; | |
696 | } | |
c801d85f KB |
697 | } |
698 | ||
699 | void wxSuffixMap::Initialize(const char *resname, const char *devresname) | |
700 | { | |
701 | const char *weight, *style; | |
702 | char *v; | |
703 | int i, j, k; | |
704 | const char *names[3]; | |
705 | ||
5705323e RR |
706 | for (k = 0; k < wxNUM_WEIGHTS; k++) |
707 | { | |
708 | switch (k) | |
709 | { | |
8bbe427f VZ |
710 | case wxWEIGHT_NORMAL: weight = "Medium"; break; |
711 | case wxWEIGHT_LIGHT: weight = "Light"; break; | |
712 | case wxWEIGHT_BOLD: | |
5705323e | 713 | default: weight = "Bold"; |
8bbe427f | 714 | } |
5705323e RR |
715 | for (j = 0; j < wxNUM_STYLES; j++) |
716 | { | |
717 | switch (j) | |
718 | { | |
8bbe427f VZ |
719 | case wxSTYLE_NORMAL: style = "Straight"; break; |
720 | case wxSTYLE_ITALIC: style = "Italic"; break; | |
721 | case wxSTYLE_SLANT: | |
722 | default: style = "Slant"; | |
723 | } | |
724 | names[0] = resname; | |
725 | names[1] = weight; | |
726 | names[2] = style; | |
727 | ||
728 | SearchResource(devresname, names, 3, &v); | |
5705323e | 729 | |
36b3b54a | 730 | // Expand macros in the found string: |
8bbe427f VZ |
731 | found: |
732 | int len, closer = 0, startpos = 0; | |
733 | ||
734 | len = (v ? strlen(v) : 0); | |
5705323e RR |
735 | for (i = 0; i < len; i++) |
736 | { | |
737 | if (v[i] == '$' && ((v[i+1] == '[') || (v[i+1] == '{'))) | |
738 | { | |
8bbe427f VZ |
739 | startpos = i; |
740 | closer = (v[i+1] == '[') ? ']' : '}'; | |
741 | ++i; | |
5705323e RR |
742 | } |
743 | else if (v[i] == closer) | |
744 | { | |
8bbe427f VZ |
745 | int newstrlen; |
746 | const char *r = (char *) NULL; bool delete_r = FALSE; | |
747 | char *name; | |
748 | ||
749 | name = v + startpos + 2; | |
750 | v[i] = 0; | |
751 | ||
5705323e RR |
752 | if (closer == '}') |
753 | { | |
8bbe427f VZ |
754 | int i, count, len; |
755 | char **names; | |
756 | ||
757 | for (i = 0, count = 1; name[i]; i++) | |
758 | if (name[i] == ',') | |
759 | count++; | |
760 | ||
761 | len = i; | |
762 | ||
763 | names = new char*[count]; | |
764 | names[0] = name; | |
765 | for (i = 0, count = 1; i < len; i++) | |
5705323e RR |
766 | if (name[i] == ',') |
767 | { | |
8bbe427f VZ |
768 | names[count++] = name + i + 1; |
769 | name[i] = 0; | |
770 | } | |
771 | ||
772 | SearchResource("", (const char **)names, count, (char **)&r); | |
773 | delete_r = (r != 0); | |
774 | delete[] names; | |
775 | ||
5705323e RR |
776 | if (!r) |
777 | { | |
8bbe427f VZ |
778 | for (i = 0; i < len; i++) |
779 | if (!name[i]) | |
780 | name[i] = ','; | |
781 | r = ""; | |
5705323e | 782 | wxLogError( "Bad resource name in font lookup." ); |
8bbe427f VZ |
783 | } |
784 | } else if (!strcmp(name, "weight")) { | |
785 | r = weight; | |
786 | } else if (!strcmp(name, "style")) { | |
787 | r = style; | |
788 | } else if (!strcmp(name, "family")) { | |
789 | r = resname; | |
790 | } else { | |
791 | r = ""; | |
5705323e | 792 | wxLogError( "Bad font macro name." ); |
8bbe427f VZ |
793 | } |
794 | ||
795 | // add r to v | |
796 | newstrlen = strlen(r); | |
797 | char *naya = new char[startpos + newstrlen + len - i]; | |
798 | memcpy(naya, v, startpos); | |
799 | memcpy(naya + startpos, r, newstrlen); | |
800 | memcpy(naya + startpos + newstrlen, v + i + 1, len - i); | |
801 | if (delete_r) | |
802 | delete[] (char*)r; | |
803 | delete[] v; | |
804 | v = naya; | |
805 | ||
806 | goto found; | |
807 | } | |
808 | } | |
36b3b54a | 809 | // We have a final value: |
8bbe427f VZ |
810 | map[k][j] = v; |
811 | } | |
c801d85f KB |
812 | } |
813 | } | |
814 | ||
815 | //----------------------------------------------------------------------------- | |
816 | // wxFontNameItem | |
817 | //----------------------------------------------------------------------------- | |
818 | ||
358fc25c RR |
819 | class wxFontNameItem : public wxObject |
820 | { | |
8bbe427f | 821 | DECLARE_DYNAMIC_CLASS(wxFontNameItem) |
c801d85f | 822 | public: |
8bbe427f VZ |
823 | wxFontNameItem(const char *name, int id, int family); |
824 | ~wxFontNameItem(); | |
825 | ||
826 | inline char* GetScreenName(int w, int s) {return screen.GetName(w, s);} | |
827 | inline char* GetPostScriptName(int w, int s) {return printing.GetName(w, s);} | |
828 | inline char* GetAFMName(int w, int s) {return afm.GetName(w, s);} | |
829 | inline char* GetName() {return name;} | |
830 | inline int GetFamily() {return family;} | |
831 | inline int GetId() {return id;} | |
832 | inline bool IsRoman() {return isroman;} | |
ea57084d | 833 | #if defined(__WXDEBUG__) |
8bbe427f | 834 | void Dump(ostream& str); |
c801d85f KB |
835 | #endif |
836 | ||
8bbe427f VZ |
837 | int id; |
838 | int family; | |
839 | char *name; | |
840 | wxSuffixMap screen, printing, afm; | |
841 | bool isroman; | |
c801d85f KB |
842 | }; |
843 | ||
844 | IMPLEMENT_ABSTRACT_CLASS(wxFontNameItem, wxObject) | |
845 | ||
846 | wxFontNameItem::wxFontNameItem(const char *Name, int Id, int Family) | |
847 | { | |
848 | name = copystring(Name); | |
849 | id = Id; | |
850 | family = Family; | |
851 | ||
852 | screen. Initialize(name, "Screen"); | |
853 | printing.Initialize(name, "PostScript"); | |
854 | afm. Initialize(name, "Afm"); | |
855 | } | |
856 | ||
8bbe427f | 857 | wxFontNameItem::~wxFontNameItem() |
c801d85f KB |
858 | { |
859 | if (name) | |
8bbe427f | 860 | delete[] name; |
c67daf87 | 861 | name = (char *) NULL; |
c801d85f KB |
862 | } |
863 | ||
ea57084d | 864 | #if defined(__WXDEBUG__) |
c801d85f KB |
865 | void wxFontNameItem::Dump(ostream& str) |
866 | { | |
867 | str << "wxFontNameItem(" << name << ")"; | |
868 | } | |
869 | #endif | |
870 | ||
871 | //----------------------------------------------------------------------------- | |
872 | // wxFontDirectory | |
873 | //----------------------------------------------------------------------------- | |
874 | ||
875 | IMPLEMENT_DYNAMIC_CLASS(wxFontNameDirectory, wxObject) | |
876 | ||
8bbe427f | 877 | wxFontNameDirectory::wxFontNameDirectory() |
c801d85f KB |
878 | { |
879 | table = new wxHashTable(wxKEY_INTEGER, 20); | |
880 | nextFontId = -1; | |
c801d85f KB |
881 | } |
882 | ||
883 | wxFontNameDirectory::~wxFontNameDirectory() | |
884 | { | |
885 | // Cleanup wxFontNameItems allocated | |
886 | table->BeginFind(); | |
887 | wxNode *node = table->Next(); | |
358fc25c RR |
888 | while (node) |
889 | { | |
8bbe427f VZ |
890 | wxFontNameItem *item = (wxFontNameItem*)node->Data(); |
891 | delete item; | |
892 | node = table->Next(); | |
c801d85f KB |
893 | } |
894 | delete table; | |
895 | } | |
896 | ||
8bbe427f | 897 | int wxFontNameDirectory::GetNewFontId() |
c801d85f KB |
898 | { |
899 | return (nextFontId--); | |
900 | } | |
901 | ||
902 | void wxFontNameDirectory::Initialize() | |
903 | { | |
904 | Initialize(wxDEFAULT, wxDEFAULT, "Default"); | |
905 | Initialize(wxDECORATIVE, wxDECORATIVE, "Decorative"); | |
906 | Initialize(wxROMAN, wxROMAN, "Roman"); | |
907 | Initialize(wxMODERN, wxMODERN, "Modern"); | |
908 | Initialize(wxTELETYPE, wxTELETYPE, "Teletype"); | |
909 | Initialize(wxSWISS, wxSWISS, "Swiss"); | |
910 | Initialize(wxSCRIPT, wxSCRIPT, "Script"); | |
911 | } | |
912 | ||
913 | void wxFontNameDirectory::Initialize(int fontid, int family, const char *resname) | |
914 | { | |
915 | char *fam, resource[256]; | |
8bbe427f | 916 | |
c801d85f | 917 | sprintf(resource, "Family%s", resname); |
c67daf87 | 918 | SearchResource((const char *)resource, (const char **) NULL, 0, (char **)&fam); |
5705323e | 919 | |
358fc25c RR |
920 | if (fam) |
921 | { | |
8bbe427f VZ |
922 | if (!strcmp(fam, "Default")) family = wxDEFAULT; |
923 | else if (!strcmp(fam, "Roman")) family = wxROMAN; | |
924 | else if (!strcmp(fam, "Decorative")) family = wxDECORATIVE; | |
925 | else if (!strcmp(fam, "Modern")) family = wxMODERN; | |
926 | else if (!strcmp(fam, "Teletype")) family = wxTELETYPE; | |
927 | else if (!strcmp(fam, "Swiss")) family = wxSWISS; | |
928 | else if (!strcmp(fam, "Script")) family = wxSCRIPT; | |
929 | delete[] fam; // free resource | |
c801d85f KB |
930 | } |
931 | table->Put(fontid, new wxFontNameItem(resname, fontid, family)); | |
932 | } | |
933 | ||
934 | int wxFontNameDirectory::FindOrCreateFontId(const char *name, int family) | |
935 | { | |
936 | int id; | |
5705323e | 937 | |
c801d85f KB |
938 | // font exists -> return id |
939 | if ( (id = GetFontId(name)) ) return id; | |
358fc25c | 940 | |
c801d85f KB |
941 | // create new font |
942 | Initialize(id=GetNewFontId(), family, name); | |
943 | return id; | |
944 | } | |
945 | ||
946 | char *wxFontNameDirectory::GetScreenName(int fontid, int weight, int style) | |
947 | { | |
948 | wxFontNameItem *item = (wxFontNameItem*)table->Get(fontid); // find font | |
949 | if (item) | |
8bbe427f | 950 | return item->GetScreenName(weight, style); |
358fc25c | 951 | |
c801d85f | 952 | // font does not exist |
c67daf87 | 953 | return (char *) NULL; |
c801d85f KB |
954 | } |
955 | ||
956 | char *wxFontNameDirectory::GetPostScriptName(int fontid, int weight, int style) | |
957 | { | |
958 | wxFontNameItem *item = (wxFontNameItem*)table->Get(fontid); // find font | |
959 | if (item) | |
8bbe427f | 960 | return item->GetPostScriptName(weight, style); |
358fc25c | 961 | |
c801d85f | 962 | // font does not exist |
c67daf87 | 963 | return (char *) NULL; |
c801d85f KB |
964 | } |
965 | ||
966 | char *wxFontNameDirectory::GetAFMName(int fontid, int weight, int style) | |
967 | { | |
968 | wxFontNameItem *item = (wxFontNameItem *)table->Get(fontid); // find font | |
969 | if (item) | |
8bbe427f | 970 | return item->GetAFMName(weight, style); |
c801d85f | 971 | // font does not exist |
c67daf87 | 972 | return (char *) NULL; |
c801d85f KB |
973 | } |
974 | ||
975 | char *wxFontNameDirectory::GetFontName(int fontid) | |
976 | { | |
977 | wxFontNameItem *item = (wxFontNameItem *)table->Get(fontid); // find font | |
978 | if (item) | |
8bbe427f | 979 | return item->GetName(); |
358fc25c | 980 | |
c801d85f | 981 | // font does not exist |
c67daf87 | 982 | return (char *) NULL; |
c801d85f KB |
983 | } |
984 | ||
985 | int wxFontNameDirectory::GetFontId(const char *name) | |
986 | { | |
987 | wxNode *node; | |
988 | ||
989 | table->BeginFind(); | |
990 | ||
358fc25c RR |
991 | while ( (node = table->Next()) ) |
992 | { | |
8bbe427f VZ |
993 | wxFontNameItem *item = (wxFontNameItem*)node->Data(); |
994 | if (!strcmp(name, item->name)) | |
995 | return item->id; | |
c801d85f | 996 | } |
358fc25c | 997 | |
c801d85f KB |
998 | // font does not exist |
999 | return 0; | |
1000 | } | |
1001 | ||
1002 | int wxFontNameDirectory::GetFamily(int fontid) | |
1003 | { | |
1004 | wxFontNameItem *item = (wxFontNameItem *)table->Get(fontid); | |
8bbe427f | 1005 | |
c801d85f | 1006 | if (item) |
8bbe427f | 1007 | return item->family; |
358fc25c | 1008 | |
c801d85f KB |
1009 | // font does not exist |
1010 | return wxDEFAULT; | |
1011 | } | |
36b3b54a RR |
1012 | |
1013 | */ |