]> git.saurik.com Git - wxWidgets.git/blame - src/osx/carbon/font.cpp
remove "item.cchTextMax=0" line from SetItem() as it doesn't seem to make any sense...
[wxWidgets.git] / src / osx / carbon / font.cpp
CommitLineData
489468fe 1/////////////////////////////////////////////////////////////////////////////
524c47aa 2// Name: src/osx/carbon/font.cpp
489468fe
SC
3// Purpose: wxFont class
4// Author: Stefan Csomor
5// Modified by:
6// Created: 1998-01-01
7// RCS-ID: $Id$
8// Copyright: (c) Stefan Csomor
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#include "wx/wxprec.h"
13
14#include "wx/font.h"
15
16#ifndef WX_PRECOMP
17 #include "wx/string.h"
18 #include "wx/utils.h"
19 #include "wx/intl.h"
20 #include "wx/gdicmn.h"
21 #include "wx/log.h"
22#endif
23
24#include "wx/fontutil.h"
25#include "wx/graphics.h"
26#include "wx/settings.h"
f1c40652 27#include "wx/tokenzr.h"
489468fe 28
b2680ced 29#include "wx/osx/private.h"
29e32b7f
SC
30
31#if wxOSX_USE_ATSU_TEXT && !wxOSX_USE_CARBON
32// include themeing support
33#include <Carbon/Carbon.h>
b2680ced 34#endif
489468fe 35
489468fe
SC
36#include <map>
37#include <string>
38
39IMPLEMENT_DYNAMIC_CLASS(wxFont, wxGDIObject)
40
489468fe
SC
41class WXDLLEXPORT wxFontRefData: public wxGDIRefData
42{
43public:
489468fe 44
f1c40652 45 wxFontRefData()
489468fe 46 {
f1c40652
SC
47 Init();
48 m_info.Init(10, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL,
49 false, wxEmptyString, wxFONTENCODING_DEFAULT);
489468fe
SC
50 }
51
f1c40652
SC
52 wxFontRefData(const wxFontRefData& data);
53
54 wxFontRefData( const wxNativeFontInfo& info ) : m_info(info)
489468fe 55 {
f1c40652 56 Init();
489468fe
SC
57 }
58
f1c40652
SC
59 wxFontRefData(wxOSXSystemFont font, int size);
60
292e5e1f 61#if wxOSX_USE_CORE_TEXT
489468fe
SC
62 wxFontRefData( wxUint32 coreTextFontType );
63 wxFontRefData( CTFontRef font );
64 wxFontRefData( CTFontDescriptorRef fontdescriptor, int size );
65#endif
66
67 virtual ~wxFontRefData();
68
69 void SetNoAntiAliasing( bool no = true ) { m_noAA = no; }
70
71 bool GetNoAntiAliasing() const { return m_noAA; }
72
73 void SetPointSize( int size )
74 {
f1c40652
SC
75 if( GetPointSize() != size )
76 {
77 m_info.SetPointSize(size);
78 Free();
79 }
489468fe
SC
80 }
81
f1c40652 82 int GetPointSize() const { return m_info.GetPointSize(); }
489468fe 83
0c14b6c3 84 void SetFamily( wxFontFamily family )
489468fe 85 {
f1c40652
SC
86 if ( m_info.m_family != family )
87 {
88 m_info.SetFamily( family );
89 Free();
90 }
489468fe
SC
91 }
92
f1c40652 93 wxFontFamily GetFamily() const { return m_info.GetFamily(); }
489468fe 94
0c14b6c3 95 void SetStyle( wxFontStyle style )
489468fe 96 {
f1c40652
SC
97 if ( m_info.m_style != style )
98 {
99 m_info.SetStyle( style );
100 Free();
101 }
489468fe
SC
102 }
103
104
f1c40652 105 wxFontStyle GetStyle() const { return m_info.GetStyle(); }
489468fe 106
0c14b6c3 107 void SetWeight( wxFontWeight weight )
489468fe 108 {
f1c40652
SC
109 if ( m_info.m_weight != weight )
110 {
111 m_info.SetWeight( weight );
112 Free();
113 }
489468fe
SC
114 }
115
116
f1c40652 117 wxFontWeight GetWeight() const { return m_info.GetWeight(); }
489468fe
SC
118
119 void SetUnderlined( bool u )
120 {
f1c40652
SC
121 if ( m_info.m_underlined != u )
122 {
123 m_info.SetUnderlined( u );
124 Free();
125 }
489468fe
SC
126 }
127
f1c40652 128 bool GetUnderlined() const { return m_info.GetUnderlined(); }
489468fe
SC
129
130 void SetFaceName( const wxString& facename )
131 {
f1c40652
SC
132 if ( m_info.m_faceName != facename )
133 {
134 m_info.SetFaceName( facename );
135 Free();
136 }
489468fe
SC
137 }
138
f1c40652 139 wxString GetFaceName() const { return m_info.GetFaceName(); }
489468fe
SC
140
141 void SetEncoding( wxFontEncoding encoding )
142 {
f1c40652
SC
143 if ( m_info.m_encoding != encoding )
144 {
145 m_info.SetEncoding( encoding );
146 Free();
147 }
489468fe
SC
148 }
149
f1c40652 150 wxFontEncoding GetEncoding() const { return m_info.GetEncoding(); }
489468fe 151
f1c40652 152 void Free();
489468fe
SC
153
154 void MacFindFont();
155
156protected:
157 // common part of all ctors
f1c40652 158 void Init();
292e5e1f 159#if wxOSX_USE_CORE_TEXT
f1c40652 160 // void Init( CTFontRef font );
489468fe 161#endif
489468fe 162 bool m_noAA; // No anti-aliasing
489468fe 163public:
f1c40652
SC
164 bool m_fontValid;
165#if wxOSX_USE_CARBON && wxOSX_USE_ATSU_TEXT
489468fe
SC
166 // for true themeing support we must store the correct font
167 // information here, as this speeds up and optimizes rendering
168 ThemeFontID m_macThemeFontID ;
169#endif
292e5e1f 170#if wxOSX_USE_CORE_TEXT
489468fe 171 wxCFRef<CTFontRef> m_ctFont;
489468fe 172#endif
beb2638a 173#if wxOSX_USE_ATSU_TEXT
489468fe 174 ATSUStyle m_macATSUStyle ;
f1c40652 175#endif
9445de1e 176 wxCFRef<CGFontRef> m_cgFont;
f1c40652
SC
177#if wxOSX_USE_COCOA
178 WX_NSFont m_nsFont;
179#endif
180#if wxOSX_USE_IPHONE
181 WX_UIFont m_uiFont;
489468fe
SC
182#endif
183 wxNativeFontInfo m_info;
184};
185
186#define M_FONTDATA ((wxFontRefData*)m_refData)
187
f1c40652
SC
188wxFontRefData::wxFontRefData(const wxFontRefData& data)
189{
190 Init();
191 m_info = data.m_info;
192 m_noAA = data.m_noAA;
193 m_fontValid = data.m_fontValid;
194#if wxOSX_USE_CARBON && wxOSX_USE_ATSU_TEXT
195 m_macThemeFontID = data.m_macThemeFontID;
196#endif
197#if wxOSX_USE_CORE_TEXT
198 m_ctFont = data.m_ctFont;
beb2638a 199#endif
9445de1e 200 m_cgFont = data.m_cgFont;
beb2638a 201#if wxOSX_USE_ATSU_TEXT
f1c40652
SC
202 if ( data.m_macATSUStyle != NULL )
203 {
204 ATSUCreateStyle(&m_macATSUStyle) ;
205 ATSUCopyAttributes(data.m_macATSUStyle, m_macATSUStyle);
206 }
207#endif
208#if wxOSX_USE_COCOA
209 m_nsFont = (NSFont*) wxMacCocoaRetain(data.m_nsFont);
210#endif
211#if wxOSX_USE_IPHONE
b4ff8b3e 212 m_uiFont = (UIFont*) wxMacCocoaRetain(data.m_uiFont);
f1c40652
SC
213#endif
214
215}
489468fe
SC
216
217// ============================================================================
218// implementation
219// ============================================================================
220
221// ----------------------------------------------------------------------------
222// wxFontRefData
223// ----------------------------------------------------------------------------
224
f1c40652 225void wxFontRefData::Init()
489468fe 226{
489468fe 227 m_noAA = false;
f1c40652 228#if wxOSX_USE_CARBON && wxOSX_USE_ATSU_TEXT
489468fe
SC
229 m_macThemeFontID = kThemeCurrentPortFont ;
230#endif
beb2638a 231#if wxOSX_USE_ATSU_TEXT
489468fe
SC
232 m_macATSUStyle = NULL ;
233#endif
f1c40652
SC
234#if wxOSX_USE_COCOA
235 m_nsFont = NULL;
236#endif
237#if wxOSX_USE_IPHONE
238 m_uiFont = NULL;
239#endif
240 m_fontValid = false;
489468fe
SC
241}
242
243wxFontRefData::~wxFontRefData()
244{
f1c40652 245 Free();
489468fe
SC
246}
247
f1c40652 248void wxFontRefData::Free()
489468fe 249{
292e5e1f 250#if wxOSX_USE_CORE_TEXT
489468fe 251 m_ctFont.reset();
489468fe 252#endif
9445de1e 253 m_cgFont.reset();
beb2638a 254#if wxOSX_USE_ATSU_TEXT
489468fe
SC
255 if ( m_macATSUStyle )
256 {
257 ::ATSUDisposeStyle((ATSUStyle)m_macATSUStyle);
258 m_macATSUStyle = NULL ;
259 }
260#endif
f1c40652
SC
261#if wxOSX_USE_COCOA
262 if (m_nsFont != NULL)
263 {
264 wxMacCocoaRelease(m_nsFont);
265 m_nsFont = NULL;
489468fe 266 }
f1c40652
SC
267#endif
268#if wxOSX_USE_IPHONE
269 if (m_uiFont != NULL)
489468fe 270 {
f1c40652
SC
271 wxMacCocoaRelease(m_uiFont);
272 m_uiFont = NULL;
489468fe 273 }
f1c40652
SC
274#endif
275 m_fontValid = false;
489468fe
SC
276}
277
f1c40652 278wxFontRefData::wxFontRefData(wxOSXSystemFont font, int size)
489468fe 279{
f1c40652
SC
280 wxASSERT( font != wxOSX_SYSTEM_FONT_NONE );
281 Init();
282
283#if wxOSX_USE_CORE_TEXT
284 if ( UMAGetSystemVersion() >= 0x1050 )
489468fe 285 {
de0d2095 286 CTFontUIFontType uifont = kCTFontSystemFontType;
f1c40652 287 switch( font )
489468fe 288 {
f1c40652
SC
289 case wxOSX_SYSTEM_FONT_NORMAL:
290 uifont = kCTFontSystemFontType;
291 break;
292 case wxOSX_SYSTEM_FONT_BOLD:
293 uifont = kCTFontEmphasizedSystemFontType;
294 break;
295 case wxOSX_SYSTEM_FONT_SMALL:
296 uifont = kCTFontSmallSystemFontType;
297 break;
298 case wxOSX_SYSTEM_FONT_SMALL_BOLD:
299 uifont = kCTFontSmallEmphasizedSystemFontType;
300 break;
301 case wxOSX_SYSTEM_FONT_MINI:
302 uifont = kCTFontMiniSystemFontType;
303 break;
304 case wxOSX_SYSTEM_FONT_MINI_BOLD:
305 uifont = kCTFontMiniEmphasizedSystemFontType;
306 break;
307 case wxOSX_SYSTEM_FONT_LABELS:
308 uifont = kCTFontLabelFontType;
309 break;
310 case wxOSX_SYSTEM_FONT_VIEWS:
311 uifont = kCTFontViewsFontType;
312 break;
313 default:
314 break;
489468fe 315 }
f1c40652
SC
316 m_ctFont.reset(CTFontCreateUIFontForLanguage( uifont, (CGFloat) size, NULL ));
317 wxCFRef<CTFontDescriptorRef> descr;
318 descr.reset( CTFontCopyFontDescriptor( m_ctFont ) );
319 m_info.Init(descr);
320 }
321#endif
322#if wxOSX_USE_ATSU_TEXT
323 {
9a672c75
SC
324#if !wxOSX_USE_CARBON
325 // not needed outside
de0d2095 326 ThemeFontID m_macThemeFontID = kThemeSystemFont;
9a672c75 327#endif
f1c40652
SC
328 switch( font )
329 {
330 case wxOSX_SYSTEM_FONT_NORMAL:
331 m_macThemeFontID = kThemeSystemFont;
332 break;
333 case wxOSX_SYSTEM_FONT_BOLD:
334 m_macThemeFontID = kThemeEmphasizedSystemFont;
335 break;
336 case wxOSX_SYSTEM_FONT_SMALL:
337 m_macThemeFontID = kThemeSmallSystemFont;
338 break;
339 case wxOSX_SYSTEM_FONT_SMALL_BOLD:
340 m_macThemeFontID = kThemeSmallEmphasizedSystemFont;
341 break;
342 case wxOSX_SYSTEM_FONT_MINI:
343 m_macThemeFontID = kThemeMiniSystemFont;
344 break;
345 case wxOSX_SYSTEM_FONT_MINI_BOLD:
346 // bold not available under themeing
347 m_macThemeFontID = kThemeMiniSystemFont;
348 break;
349 case wxOSX_SYSTEM_FONT_LABELS:
350 m_macThemeFontID = kThemeLabelFont;
351 break;
352 case wxOSX_SYSTEM_FONT_VIEWS:
353 m_macThemeFontID = kThemeViewsFont;
354 break;
355 default:
356 break;
357 }
358 if ( m_info.m_faceName.empty() )
359 {
360 Style style ;
361 FMFontSize fontSize;
362 Str255 qdFontName ;
363
364 GetThemeFont( m_macThemeFontID, GetApplicationScript(), qdFontName, &fontSize, &style );
365 if ( size != 0 )
366 fontSize = size;
367
368 wxFontStyle fontstyle = wxFONTSTYLE_NORMAL;
369 wxFontWeight fontweight = wxFONTWEIGHT_NORMAL;
370 bool underlined = false;
371
372 if ( style & bold )
373 fontweight = wxFONTWEIGHT_BOLD ;
374 else
375 fontweight = wxFONTWEIGHT_NORMAL ;
376 if ( style & italic )
377 fontstyle = wxFONTSTYLE_ITALIC ;
378 if ( style & underline )
379 underlined = true ;
380
381 m_info.Init(size,wxFONTFAMILY_DEFAULT,fontstyle,fontweight,underlined,
382 wxMacMakeStringFromPascal( qdFontName ), wxFONTENCODING_DEFAULT);
383 }
489468fe 384 }
489468fe 385#endif
f1c40652 386#if wxOSX_USE_COCOA
aa6208d9 387 m_nsFont = wxFont::OSXCreateNSFont( font, &m_info );
f1c40652
SC
388#endif
389#if wxOSX_USE_IPHONE
aa6208d9 390 m_uiFont = wxFont::OSXCreateUIFont( font, &m_info );
f1c40652
SC
391#endif
392}
489468fe
SC
393
394void wxFontRefData::MacFindFont()
395{
f1c40652
SC
396 if ( m_fontValid )
397 return;
398
399 m_info.EnsureValid();
400
292e5e1f 401#if wxOSX_USE_CORE_TEXT
489468fe
SC
402 if ( UMAGetSystemVersion() >= 0x1050 )
403 {
f1c40652 404 CTFontSymbolicTraits traits = 0;
0c14b6c3 405
f1c40652
SC
406 if (m_info.m_weight == wxFONTWEIGHT_BOLD)
407 traits |= kCTFontBoldTrait;
408 if (m_info.m_style == wxFONTSTYLE_ITALIC || m_info.m_style == wxFONTSTYLE_SLANT)
409 traits |= kCTFontItalicTrait;
489468fe 410
f1c40652
SC
411 // use font caching
412 wxString lookupnameWithSize = wxString::Format( "%s_%ld_%ld", m_info.m_faceName.c_str(), traits, m_info.m_pointSize );
489468fe 413
f1c40652
SC
414 static std::map< std::wstring , wxCFRef< CTFontRef > > fontcache ;
415 m_ctFont = fontcache[ std::wstring(lookupnameWithSize.wc_str()) ];
416 if ( !m_ctFont )
489468fe 417 {
f1c40652 418 m_ctFont.reset( CTFontCreateWithFontDescriptor( m_info.m_ctFontDescriptor, 0/*m_pointSize */, NULL ) );
489468fe 419 }
9445de1e 420 m_cgFont.reset(CTFontCopyGraphicsFont(m_ctFont, NULL));
489468fe 421 }
f1c40652 422
489468fe 423#endif
292e5e1f 424#if wxOSX_USE_ATSU_TEXT
489468fe 425 {
489468fe
SC
426 // we try to get as much styles as possible into ATSU
427
f1c40652 428 OSStatus status = ::ATSUCreateStyle(&m_macATSUStyle);
489468fe
SC
429 wxASSERT_MSG( status == noErr , wxT("couldn't create ATSU style") );
430
431 ATSUAttributeTag atsuTags[] =
432 {
433 kATSUFontTag ,
434 kATSUSizeTag ,
435 kATSUVerticalCharacterTag,
436 kATSUQDBoldfaceTag ,
437 kATSUQDItalicTag ,
438 kATSUQDUnderlineTag ,
439 kATSUQDCondensedTag ,
440 kATSUQDExtendedTag ,
441 };
442 ByteCount atsuSizes[sizeof(atsuTags) / sizeof(ATSUAttributeTag)] =
443 {
444 sizeof( ATSUFontID ) ,
445 sizeof( Fixed ) ,
446 sizeof( ATSUVerticalCharacterType),
447 sizeof( Boolean ) ,
448 sizeof( Boolean ) ,
449 sizeof( Boolean ) ,
450 sizeof( Boolean ) ,
451 sizeof( Boolean ) ,
452 };
453
454 Boolean kTrue = true ;
455 Boolean kFalse = false ;
456
f1c40652 457 Fixed atsuSize = IntToFixed( m_info.m_pointSize );
489468fe 458 ATSUVerticalCharacterType kHorizontal = kATSUStronglyHorizontal;
f1c40652 459 FMFontStyle addQDStyle = m_info.m_atsuAdditionalQDStyles;
489468fe
SC
460 ATSUAttributeValuePtr atsuValues[sizeof(atsuTags) / sizeof(ATSUAttributeTag)] =
461 {
f1c40652 462 &m_info.m_atsuFontID ,
489468fe
SC
463 &atsuSize ,
464 &kHorizontal,
f1c40652
SC
465 (addQDStyle & bold) ? &kTrue : &kFalse ,
466 (addQDStyle & italic) ? &kTrue : &kFalse ,
467 (addQDStyle & underline) ? &kTrue : &kFalse ,
468 (addQDStyle & condense) ? &kTrue : &kFalse ,
469 (addQDStyle & extend) ? &kTrue : &kFalse ,
489468fe
SC
470 };
471
472 status = ::ATSUSetAttributes(
473 (ATSUStyle)m_macATSUStyle,
474 sizeof(atsuTags) / sizeof(ATSUAttributeTag) ,
475 atsuTags, atsuSizes, atsuValues);
476
477 wxASSERT_MSG( status == noErr , wxT("couldn't modify ATSU style") );
9445de1e
SC
478 if ( m_cgFont.get() == NULL )
479 {
480 ATSFontRef fontRef = FMGetATSFontRefFromFont(m_info.m_atsuFontID);
481 m_cgFont.reset( CGFontCreateWithPlatformFont( &fontRef ) );
482 }
489468fe
SC
483 }
484#endif
f1c40652 485#if wxOSX_USE_COCOA
aa6208d9 486 m_nsFont = wxFont::OSXCreateNSFont( &m_info );
f1c40652
SC
487#endif
488#if wxOSX_USE_IPHONE
aa6208d9 489 m_uiFont = wxFont::OSXCreateUIFont( &m_info );
f1c40652
SC
490#endif
491 m_fontValid = true;
489468fe
SC
492}
493
494// ----------------------------------------------------------------------------
495// wxFont
496// ----------------------------------------------------------------------------
497
498bool wxFont::Create(const wxNativeFontInfo& info)
499{
f1c40652
SC
500 UnRef();
501
502 m_refData = new wxFontRefData( info );
503 RealizeResource();
504
505 return true;
489468fe
SC
506}
507
508wxFont::wxFont(const wxString& fontdesc)
509{
510 wxNativeFontInfo info;
511 if ( info.FromString(fontdesc) )
512 (void)Create(info);
513}
514
515bool wxFont::Create(int pointSize,
0c14b6c3
FM
516 wxFontFamily family,
517 wxFontStyle style,
518 wxFontWeight weight,
519 bool underlined,
f1c40652 520 const wxString& faceNameParam,
0c14b6c3 521 wxFontEncoding encoding)
489468fe
SC
522{
523 UnRef();
f1c40652
SC
524
525 wxString faceName = faceNameParam;
489468fe 526
f1c40652
SC
527 if ( faceName.empty() )
528 {
529 switch ( family )
530 {
531 case wxFONTFAMILY_DEFAULT :
532 faceName = wxT("Lucida Grande");
533 break;
534
535 case wxFONTFAMILY_SCRIPT :
536 case wxFONTFAMILY_ROMAN :
537 case wxFONTFAMILY_DECORATIVE :
538 faceName = wxT("Times");
539 break ;
540
541 case wxFONTFAMILY_SWISS :
542 faceName = wxT("Helvetica");
543 break ;
544
545 case wxFONTFAMILY_MODERN :
546 case wxFONTFAMILY_TELETYPE:
547 faceName = wxT("Courier");
548 break ;
549
550 default:
551 faceName = wxT("Times");
552 break ;
553 }
554 }
555
556 wxNativeFontInfo info;
557
558 info.Init(pointSize, family, style, weight,
489468fe
SC
559 underlined, faceName, encoding);
560
f1c40652 561 m_refData = new wxFontRefData(info);
489468fe
SC
562
563 return true;
564}
565
f1c40652 566bool wxFont::CreateSystemFont(wxOSXSystemFont font)
489468fe
SC
567{
568 UnRef();
f1c40652
SC
569
570 m_refData = new wxFontRefData( font, 0 );
571
489468fe
SC
572 return true;
573}
574
f1c40652 575wxFont::~wxFont()
489468fe 576{
f1c40652 577}
489468fe 578
f1c40652
SC
579bool wxFont::RealizeResource()
580{
581 M_FONTDATA->MacFindFont();
489468fe
SC
582
583 return true;
584}
585
f1c40652
SC
586void wxFont::SetEncoding(wxFontEncoding encoding)
587{
588 AllocExclusive();
489468fe
SC
589
590 M_FONTDATA->SetEncoding( encoding );
489468fe
SC
591}
592
593wxGDIRefData *wxFont::CreateGDIRefData() const
594{
595 return new wxFontRefData;
596}
597
598wxGDIRefData *wxFont::CloneGDIRefData(const wxGDIRefData *data) const
599{
5c33522f 600 return new wxFontRefData(*static_cast<const wxFontRefData *>(data));
489468fe
SC
601}
602
603void wxFont::SetPointSize(int pointSize)
604{
605 if ( M_FONTDATA->GetPointSize() == pointSize )
606 return;
607
f1c40652 608 AllocExclusive();
489468fe
SC
609
610 M_FONTDATA->SetPointSize( pointSize );
489468fe
SC
611}
612
0c14b6c3 613void wxFont::SetFamily(wxFontFamily family)
489468fe 614{
f1c40652 615 AllocExclusive();
489468fe
SC
616
617 M_FONTDATA->SetFamily( family );
489468fe
SC
618}
619
0c14b6c3 620void wxFont::SetStyle(wxFontStyle style)
489468fe 621{
f1c40652 622 AllocExclusive();
489468fe
SC
623
624 M_FONTDATA->SetStyle( style );
489468fe
SC
625}
626
0c14b6c3 627void wxFont::SetWeight(wxFontWeight weight)
489468fe 628{
f1c40652 629 AllocExclusive();
489468fe
SC
630
631 M_FONTDATA->SetWeight( weight );
489468fe
SC
632}
633
634bool wxFont::SetFaceName(const wxString& faceName)
635{
f1c40652 636 AllocExclusive();
489468fe
SC
637
638 M_FONTDATA->SetFaceName( faceName );
639
489468fe
SC
640 return wxFontBase::SetFaceName(faceName);
641}
642
643void wxFont::SetUnderlined(bool underlined)
644{
f1c40652 645 AllocExclusive();
489468fe
SC
646
647 M_FONTDATA->SetUnderlined( underlined );
489468fe
SC
648}
649
650void wxFont::SetNoAntiAliasing( bool no )
651{
f1c40652 652 AllocExclusive();
489468fe
SC
653
654 M_FONTDATA->SetNoAntiAliasing( no );
489468fe
SC
655}
656
657// ----------------------------------------------------------------------------
658// accessors
659// ----------------------------------------------------------------------------
660
661// TODO: insert checks everywhere for M_FONTDATA == NULL!
662
663int wxFont::GetPointSize() const
664{
665 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
666
667 return M_FONTDATA->GetPointSize();
668}
669
670wxSize wxFont::GetPixelSize() const
671{
672#if wxUSE_GRAPHICS_CONTEXT
673 // TODO: consider caching the value
674 wxGraphicsContext* dc = wxGraphicsContext::CreateFromNative((CGContextRef) NULL);
675 dc->SetFont(*(wxFont *)this,*wxBLACK);
676 wxDouble width, height = 0;
677 dc->GetTextExtent( wxT("g"), &width, &height, NULL, NULL);
678 delete dc;
679 return wxSize((int)width, (int)height);
680#else
681 return wxFontBase::GetPixelSize();
682#endif
683}
684
0c14b6c3 685wxFontFamily wxFont::GetFamily() const
489468fe 686{
0c14b6c3 687 wxCHECK_MSG( M_FONTDATA != NULL , wxFONTFAMILY_MAX, wxT("invalid font") );
489468fe
SC
688
689 return M_FONTDATA->GetFamily();
690}
691
0c14b6c3 692wxFontStyle wxFont::GetStyle() const
489468fe 693{
0c14b6c3 694 wxCHECK_MSG( M_FONTDATA != NULL , wxFONTSTYLE_MAX, wxT("invalid font") );
489468fe
SC
695
696 return M_FONTDATA->GetStyle() ;
697}
698
0c14b6c3 699wxFontWeight wxFont::GetWeight() const
489468fe 700{
0c14b6c3 701 wxCHECK_MSG( M_FONTDATA != NULL , wxFONTWEIGHT_MAX, wxT("invalid font") );
489468fe
SC
702
703 return M_FONTDATA->GetWeight();
704}
705
706bool wxFont::GetUnderlined() const
707{
708 wxCHECK_MSG( M_FONTDATA != NULL , false, wxT("invalid font") );
709
710 return M_FONTDATA->GetUnderlined();
711}
712
713wxString wxFont::GetFaceName() const
714{
715 wxCHECK_MSG( M_FONTDATA != NULL , wxEmptyString , wxT("invalid font") );
716
717 return M_FONTDATA->GetFaceName() ;
718}
719
720wxFontEncoding wxFont::GetEncoding() const
721{
722 wxCHECK_MSG( M_FONTDATA != NULL , wxFONTENCODING_DEFAULT , wxT("invalid font") );
723
724 return M_FONTDATA->GetEncoding() ;
725}
726
727bool wxFont::GetNoAntiAliasing() const
728{
729 wxCHECK_MSG( M_FONTDATA != NULL , false, wxT("invalid font") );
730
731 return M_FONTDATA->GetNoAntiAliasing();
732}
733
f1c40652 734#if wxOSX_USE_ATSU_TEXT && wxOSX_USE_CARBON
489468fe
SC
735
736short wxFont::MacGetFontNum() const
737{
738 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
739
f1c40652
SC
740 // cast away constness otherwise lazy font resolution is not possible
741 const_cast<wxFont *>(this)->RealizeResource();
742
743 return M_FONTDATA->m_info.m_qdFontFamily;
489468fe
SC
744}
745
f1c40652 746wxByte wxFont::MacGetFontStyle() const
489468fe
SC
747{
748 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
749
f1c40652
SC
750 // cast away constness otherwise lazy font resolution is not possible
751 const_cast<wxFont *>(this)->RealizeResource();
752
753 return M_FONTDATA->m_info.m_qdFontStyle;
489468fe
SC
754}
755
f1c40652 756wxUint16 wxFont::MacGetThemeFontID() const
489468fe
SC
757{
758 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
759
f1c40652 760 return M_FONTDATA->m_macThemeFontID;
489468fe
SC
761}
762
f1c40652
SC
763#endif
764
beb2638a 765#if wxOSX_USE_ATSU_TEXT
f1c40652 766void * wxFont::MacGetATSUStyle() const
489468fe 767{
f1c40652 768 wxCHECK_MSG( M_FONTDATA != NULL , NULL, wxT("invalid font") );
489468fe 769
f1c40652
SC
770 // cast away constness otherwise lazy font resolution is not possible
771 const_cast<wxFont *>(this)->RealizeResource();
772
773 return M_FONTDATA->m_macATSUStyle;
489468fe 774}
6f283573
SC
775
776#if WXWIN_COMPATIBILITY_2_8
777wxUint32 wxFont::MacGetATSUFontID() const
778{
779 wxCHECK_MSG( M_FONTDATA != NULL , NULL, wxT("invalid font") );
780
781 // cast away constness otherwise lazy font resolution is not possible
782 const_cast<wxFont *>(this)->RealizeResource();
783
784 return M_FONTDATA->m_info.m_atsuFontID;
785}
786
787wxUint32 wxFont::MacGetATSUAdditionalQDStyles() const
788{
789 wxCHECK_MSG( M_FONTDATA != NULL , NULL, wxT("invalid font") );
790
791 // cast away constness otherwise lazy font resolution is not possible
792 const_cast<wxFont *>(this)->RealizeResource();
793
794 return M_FONTDATA->m_info.m_atsuAdditionalQDStyles;
795}
796#endif
797
f1c40652
SC
798#endif
799
800#if wxOSX_USE_CORE_TEXT
489468fe 801
aa6208d9 802CTFontRef wxFont::OSXGetCTFont() const
489468fe
SC
803{
804 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
805
f1c40652
SC
806 // cast away constness otherwise lazy font resolution is not possible
807 const_cast<wxFont *>(this)->RealizeResource();
808
809 return (CTFontRef)(M_FONTDATA->m_ctFont);
489468fe
SC
810}
811
f1c40652
SC
812#endif
813
9445de1e
SC
814#if wxOSX_USE_COCOA_OR_CARBON
815
aa6208d9 816CGFontRef wxFont::OSXGetCGFont() const
9445de1e
SC
817{
818 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
819
820 // cast away constness otherwise lazy font resolution is not possible
821 const_cast<wxFont *>(this)->RealizeResource();
822
823 return (M_FONTDATA->m_cgFont);
824}
825
826#endif
827
828
f1c40652
SC
829#if wxOSX_USE_COCOA
830
aa6208d9 831NSFont* wxFont::OSXGetNSFont() const
489468fe
SC
832{
833 wxCHECK_MSG( M_FONTDATA != NULL , 0, wxT("invalid font") );
834
f1c40652
SC
835 // cast away constness otherwise lazy font resolution is not possible
836 const_cast<wxFont *>(this)->RealizeResource();
837
838 return (M_FONTDATA->m_nsFont);
489468fe 839}
f1c40652 840
489468fe
SC
841#endif
842
f1c40652 843const wxNativeFontInfo * wxFont::GetNativeFontInfo() const
489468fe
SC
844{
845 wxCHECK_MSG( M_FONTDATA != NULL , NULL, wxT("invalid font") );
f1c40652 846 wxCHECK_MSG( Ok(), NULL, wxT("invalid font") );
489468fe 847
f1c40652
SC
848 // cast away constness otherwise lazy font resolution is not possible
849 const_cast<wxFont *>(this)->RealizeResource();
850
851 // M_FONTDATA->m_info.InitFromFont(*this);
852
853 return &(M_FONTDATA->m_info);
854}
855
856// ----------------------------------------------------------------------------
857// wxNativeFontInfo
858// ----------------------------------------------------------------------------
859
860#if wxOSX_USE_CORE_TEXT
861
862/* from Core Text Manual Common Operations */
863
864static CTFontDescriptorRef wxMacCreateCTFontDescriptor(CFStringRef iFamilyName, CTFontSymbolicTraits iTraits )
865{
866 CTFontDescriptorRef descriptor = NULL;
867 CFMutableDictionaryRef attributes;
868
869 assert(iFamilyName != NULL);
870 // Create a mutable dictionary to hold our attributes.
871 attributes = CFDictionaryCreateMutable(kCFAllocatorDefault, 0,
872 &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
873 check(attributes != NULL);
874
875 if (attributes != NULL) {
876 // Add a family name to our attributes.
877 CFDictionaryAddValue(attributes, kCTFontFamilyNameAttribute, iFamilyName);
878
879
880 if ( iTraits ) {
881 CFMutableDictionaryRef traits;
882 CFNumberRef symTraits;
883
884 // Create the traits dictionary.
885 symTraits = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type,
886 &iTraits);
887 check(symTraits != NULL);
888
889 if (symTraits != NULL) {
890 // Create a dictionary to hold our traits values.
891 traits = CFDictionaryCreateMutable(kCFAllocatorDefault, 0,
892 &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
893 check(traits != NULL);
894
895 if (traits != NULL) {
896 // Add the symbolic traits value to the traits dictionary.
897 CFDictionaryAddValue(traits, kCTFontSymbolicTrait, symTraits);
898
899 // Add the traits attribute to our attributes.
900 CFDictionaryAddValue(attributes, kCTFontTraitsAttribute, traits);
901 CFRelease(traits);
902 }
903 CFRelease(symTraits);
904 }
905 }
906 // Create the font descriptor with our attributes
907 descriptor = CTFontDescriptorCreateWithAttributes(attributes);
908 check(descriptor != NULL);
909
910 CFRelease(attributes);
911 }
912 // Return our font descriptor.
913 return descriptor ;
914}
915
916#endif
917
918void wxNativeFontInfo::Init()
919{
920#if wxOSX_USE_CORE_TEXT
921 m_ctFontDescriptor = NULL;
922#endif
923#if wxOSX_USE_ATSU_TEXT
924 m_atsuFontID = 0 ;
925 m_atsuAdditionalQDStyles = 0;
926 m_atsuFontValid = false;
927#if wxOSX_USE_CARBON
928 m_qdFontStyle = 0;
929 m_qdFontFamily = 0;
930#endif
931#endif
932#if wxOSX_USE_COCOA
933 m_nsFontDescriptor = NULL;
f1c40652
SC
934#endif
935 m_pointSize = 0;
936 m_family = wxFONTFAMILY_DEFAULT;
937 m_style = wxFONTSTYLE_NORMAL;
938 m_weight = wxFONTWEIGHT_NORMAL;
939 m_underlined = false;
940 m_faceName.clear();
941 m_encoding = wxFONTENCODING_DEFAULT;
942 m_descriptorValid = false;
943}
944
945#if wxOSX_USE_CORE_TEXT
946void wxNativeFontInfo::Init(CTFontDescriptorRef descr)
947{
948 Init();
949 m_ctFontDescriptor = wxCFRetain(descr);
950
951 wxCFRef< CFNumberRef > sizevalue( (CFNumberRef) CTFontDescriptorCopyAttribute( m_ctFontDescriptor, kCTFontSizeAttribute ) );
952 float fsize;
953 if ( CFNumberGetValue( sizevalue , kCFNumberFloatType , &fsize ) )
954 m_pointSize = (int)( fsize + 0.5 );
955
956 wxCFRef< CFDictionaryRef > traitsvalue( (CFDictionaryRef) CTFontDescriptorCopyAttribute( m_ctFontDescriptor, kCTFontTraitsAttribute ) );
957 CTFontSymbolicTraits traits;
958 if ( CFNumberGetValue((CFNumberRef) CFDictionaryGetValue(traitsvalue,kCTFontSymbolicTrait),kCFNumberIntType,&traits) )
959 {
960 if ( traits & kCTFontItalicTrait )
961 m_style = wxFONTSTYLE_ITALIC;
962 if ( traits & kCTFontBoldTrait )
963 m_weight = wxFONTWEIGHT_BOLD ;
964 }
965
966 wxCFStringRef familyName( (CFStringRef) CTFontDescriptorCopyAttribute(m_ctFontDescriptor, kCTFontFamilyNameAttribute));
967 m_faceName = familyName.AsString();
489468fe
SC
968}
969#endif
970
f1c40652
SC
971void wxNativeFontInfo::EnsureValid()
972{
973 if ( m_descriptorValid )
974 return;
975
292e5e1f 976#if wxOSX_USE_CORE_TEXT
f1c40652
SC
977 if ( m_ctFontDescriptor == NULL && UMAGetSystemVersion() >= 0x1050 )
978 {
979 CTFontSymbolicTraits traits = 0;
980
981 if (m_weight == wxFONTWEIGHT_BOLD)
982 traits |= kCTFontBoldTrait;
983 if (m_style == wxFONTSTYLE_ITALIC || m_style == wxFONTSTYLE_SLANT)
984 traits |= kCTFontItalicTrait;
985
986 // use font caching
987 wxString lookupnameWithSize = wxString::Format( "%s_%ld_%ld", m_faceName.c_str(), traits, m_pointSize );
988
989 static std::map< std::wstring , wxCFRef< CTFontDescriptorRef > > fontdescriptorcache ;
990 m_ctFontDescriptor = wxCFRetain((CTFontDescriptorRef)fontdescriptorcache[ std::wstring(lookupnameWithSize.wc_str()) ]);
991 if ( !m_ctFontDescriptor )
992 {
993 // QD selection algorithm is the fastest by orders of magnitude on 10.5
994 if ( m_faceName.IsAscii() )
995 {
996 uint8_t qdstyle = 0;
997 if (m_weight == wxFONTWEIGHT_BOLD)
998 qdstyle |= bold;
999 if (m_style == wxFONTSTYLE_ITALIC || m_style == wxFONTSTYLE_SLANT)
1000 qdstyle |= italic;
1001
1002 Str255 qdFontName ;
1003 wxMacStringToPascal( m_faceName , qdFontName );
1004 wxCFRef< CTFontRef > font;
1005 font.reset( CTFontCreateWithQuickdrawInstance(qdFontName, 0 , qdstyle, m_pointSize) );
1006 m_ctFontDescriptor = CTFontCopyFontDescriptor(font);
1007 }
1008 else
1009 {
1010 m_ctFontDescriptor = wxMacCreateCTFontDescriptor( wxCFStringRef(m_faceName),traits );
1011 }
1012 fontdescriptorcache[ std::wstring(lookupnameWithSize.wc_str()) ].reset(wxCFRetain(m_ctFontDescriptor));
1013 }
1014 }
1015#endif
1016#if wxOSX_USE_ATSU_TEXT
1017 if ( !m_atsuFontValid )
1018 {
9a672c75
SC
1019#if !wxOSX_USE_CARBON
1020 // not needed outside
1021 wxInt16 m_qdFontFamily;
1022 wxInt16 m_qdFontStyle;
1023#endif
f1c40652
SC
1024 wxCFStringRef cf( m_faceName, wxLocale::GetSystemEncoding() );
1025 ATSFontFamilyRef atsfamily = ATSFontFamilyFindFromName( cf , kATSOptionFlagsDefault );
1026 if ( atsfamily == (ATSFontFamilyRef) -1 )
1027 {
1028 wxLogDebug( wxT("ATSFontFamilyFindFromName failed for ") + m_faceName );
1029 m_qdFontFamily = GetAppFont();
1030 }
1031 else
1032 {
1033 m_qdFontFamily = FMGetFontFamilyFromATSFontFamilyRef( atsfamily );
1034 }
1035
1036 m_qdFontStyle = 0;
1037 if (m_weight == wxFONTWEIGHT_BOLD)
1038 m_qdFontStyle |= bold;
1039 if (m_style == wxFONTSTYLE_ITALIC || m_style == wxFONTSTYLE_SLANT)
1040 m_qdFontStyle |= italic;
1041 if (m_underlined)
1042 m_qdFontStyle |= underline;
1043
489468fe 1044
f1c40652
SC
1045 // we try to get as much styles as possible into ATSU
1046
1047 // ATSUFontID and FMFont are equivalent
1048 FMFontStyle intrinsicStyle = 0 ;
1049 OSStatus status = FMGetFontFromFontFamilyInstance( m_qdFontFamily , m_qdFontStyle , (FMFont*)&m_atsuFontID , &intrinsicStyle);
1050 wxASSERT_MSG( status == noErr , wxT("couldn't get an ATSUFont from font family") );
1051 m_atsuAdditionalQDStyles = m_qdFontStyle & (~intrinsicStyle );
1052 m_atsuFontValid = true;
1053 }
1054#endif
1055#if wxOSX_USE_COCOA
1056 if ( m_nsFontDescriptor == NULL )
aa6208d9 1057 OSXValidateNSFontDescriptor();
f1c40652
SC
1058#endif
1059#if wxOSX_USE_IPHONE
1060 // TODO
1061#endif
1062 m_descriptorValid = true;
1063}
1064
1065void wxNativeFontInfo::Init(const wxNativeFontInfo& info)
489468fe 1066{
f1c40652
SC
1067 Init();
1068#if wxOSX_USE_CORE_TEXT
1069 m_ctFontDescriptor = wxCFRetain(info.m_ctFontDescriptor);
1070#endif
1071#if wxOSX_USE_ATSU_TEXT
1072 m_atsuFontValid = info.m_atsuFontValid;
1073 m_atsuFontID = info.m_atsuFontID ;
1074 m_atsuAdditionalQDStyles = info.m_atsuAdditionalQDStyles;
1075#if wxOSX_USE_CARBON
1076 m_qdFontFamily = info.m_qdFontFamily;
1077 m_qdFontStyle = info.m_qdFontStyle;
1078#endif
1079#endif
1080#if wxOSX_USE_COCOA
1081 m_nsFontDescriptor = (NSFontDescriptor*) wxMacCocoaRetain(info.m_nsFontDescriptor);
f1c40652
SC
1082#endif
1083 m_pointSize = info.m_pointSize;
1084 m_family = info.m_family;
1085 m_style = info.m_style;
1086 m_weight = info.m_weight;
1087 m_underlined = info.m_underlined;
1088 m_faceName = info.m_faceName;
1089 m_encoding = info.m_encoding;
1090 m_descriptorValid = info.m_descriptorValid;
1091}
1092
1093void wxNativeFontInfo::Init(int size,
1094 wxFontFamily family,
1095 wxFontStyle style,
1096 wxFontWeight weight,
1097 bool underlined,
1098 const wxString& faceName,
1099 wxFontEncoding encoding)
1100{
1101 Init();
1102 m_pointSize = size;
1103 m_family = family;
1104 m_style = style;
1105 m_weight = weight;
1106 m_underlined = underlined;
1107 m_faceName = faceName;
1108 m_encoding = encoding;
489468fe 1109
489468fe
SC
1110}
1111
f1c40652
SC
1112void wxNativeFontInfo::Free()
1113{
1114#if wxOSX_USE_CORE_TEXT
1115 wxCFRelease(m_ctFontDescriptor);
1116 m_ctFontDescriptor = NULL;
1117#endif
1118#if wxOSX_USE_ATSU_TEXT
1119 m_atsuFontID = 0 ;
1120 m_atsuAdditionalQDStyles = 0;
1121 m_atsuFontValid = false;
1122#endif
1123#if wxOSX_USE_COCOA
1124 wxMacCocoaRelease(m_nsFontDescriptor);
1125 m_nsFontDescriptor = NULL;
489468fe 1126#endif
f1c40652
SC
1127 m_descriptorValid = false;
1128}
489468fe 1129
f1c40652 1130bool wxNativeFontInfo::FromString(const wxString& s)
489468fe 1131{
f1c40652 1132 long l;
489468fe 1133
f1c40652 1134 wxStringTokenizer tokenizer(s, _T(";"));
489468fe 1135
f1c40652
SC
1136 wxString token = tokenizer.GetNextToken();
1137 //
1138 // Ignore the version for now
1139 //
1140
1141 token = tokenizer.GetNextToken();
1142 if ( !token.ToLong(&l) )
1143 return false;
1144 m_pointSize = (int)l;
1145
1146 token = tokenizer.GetNextToken();
1147 if ( !token.ToLong(&l) )
1148 return false;
1149 m_family = (wxFontFamily)l;
1150
1151 token = tokenizer.GetNextToken();
1152 if ( !token.ToLong(&l) )
1153 return false;
1154 m_style = (wxFontStyle)l;
1155
1156 token = tokenizer.GetNextToken();
1157 if ( !token.ToLong(&l) )
1158 return false;
1159 m_weight = (wxFontWeight)l;
1160
1161 token = tokenizer.GetNextToken();
1162 if ( !token.ToLong(&l) )
1163 return false;
1164 m_underlined = l != 0;
1165
1166 m_faceName = tokenizer.GetNextToken();
1167
1168#ifndef __WXMAC__
1169 if( !faceName )
1170 return false;
1171#endif
1172
1173 token = tokenizer.GetNextToken();
1174 if ( !token.ToLong(&l) )
1175 return false;
1176 m_encoding = (wxFontEncoding)l;
1177
1178 return true;
1179}
1180
1181wxString wxNativeFontInfo::ToString() const
1182{
1183 wxString s;
1184
1185 s.Printf(_T("%d;%d;%d;%d;%d;%d;%s;%d"),
1186 0, // version
1187 m_pointSize,
1188 m_family,
1189 (int)m_style,
1190 (int)m_weight,
1191 m_underlined,
1192 m_faceName.GetData(),
1193 (int)m_encoding);
1194
1195 return s;
1196}
1197
1198int wxNativeFontInfo::GetPointSize() const
1199{
1200 return m_pointSize;
1201}
1202
1203wxFontStyle wxNativeFontInfo::GetStyle() const
1204{
1205 return m_style;
1206}
1207
1208wxFontWeight wxNativeFontInfo::GetWeight() const
1209{
1210 return m_weight;
1211}
1212
1213bool wxNativeFontInfo::GetUnderlined() const
1214{
1215 return m_underlined;
1216}
1217
1218wxString wxNativeFontInfo::GetFaceName() const
1219{
1220 return m_faceName;
1221}
1222
1223wxFontFamily wxNativeFontInfo::GetFamily() const
1224{
1225 return m_family;
489468fe 1226}
f1c40652
SC
1227
1228wxFontEncoding wxNativeFontInfo::GetEncoding() const
1229{
1230 return m_encoding;
1231}
1232
1233// changing the font descriptor
1234
1235void wxNativeFontInfo::SetPointSize(int pointsize)
1236{
1237 if ( m_pointSize != pointsize )
1238 {
1239 m_pointSize = pointsize;
1240 Free();
1241 }
1242}
1243
1244void wxNativeFontInfo::SetStyle(wxFontStyle style_)
1245{
1246 if ( m_style != style_ )
1247 {
1248 m_style = style_;
1249 Free();
1250 }
1251}
1252
1253void wxNativeFontInfo::SetWeight(wxFontWeight weight_)
1254{
1255 if ( m_weight != weight_ )
1256 {
1257 m_weight = weight_;
1258 Free();
1259 }
1260}
1261
1262void wxNativeFontInfo::SetUnderlined(bool underlined_)
1263{
1264 if ( m_underlined != underlined_ )
1265 {
1266 m_underlined = underlined_;
1267 Free();
1268 }
1269}
1270
1271bool wxNativeFontInfo::SetFaceName(const wxString& facename_)
1272{
1273 if ( m_faceName != facename_ )
1274 {
1275 m_faceName = facename_;
1276 Free();
1277 }
1278 return true;
1279}
1280
1281void wxNativeFontInfo::SetFamily(wxFontFamily family_)
1282{
1283 if ( m_family != family_ )
1284 {
1285 m_family = family_;
1286 Free();
1287 }
1288}
1289
1290void wxNativeFontInfo::SetEncoding(wxFontEncoding encoding_)
1291{
1292 m_encoding = encoding_;
1293 // not reflected in native descriptors
1294}