1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/mac/carbon/font.cpp
3 // Purpose: wxFont class
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
17 #include "wx/string.h"
20 #include "wx/gdicmn.h"
24 #include "wx/fontutil.h"
25 #include "wx/graphics.h"
27 #include "wx/mac/private.h"
30 #include <ATSUnicode.h>
34 IMPLEMENT_DYNAMIC_CLASS(wxFont
, wxGDIObject
)
37 class WXDLLEXPORT wxFontRefData
: public wxGDIRefData
49 , m_faceName(wxT("applicationfont"))
50 , m_encoding(wxFONTENCODING_DEFAULT
)
60 Init(m_pointSize
, m_family
, m_style
, m_weight
,
61 m_underlined
, m_faceName
, m_encoding
);
64 wxFontRefData(const wxFontRefData
& data
)
66 , m_fontId(data
.m_fontId
)
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
)
72 , m_faceName(data
.m_faceName
)
73 , m_encoding(data
.m_encoding
)
76 , m_macFontFamily(data
.m_macFontFamily
)
77 , m_macFontSize(data
.m_macFontSize
)
78 , m_macFontStyle(data
.m_macFontStyle
)
79 , m_macATSUFontID(data
.m_macATSUFontID
)
83 Init(data
.m_pointSize
, data
.m_family
, data
.m_style
, data
.m_weight
,
84 data
.m_underlined
, data
.m_faceName
, data
.m_encoding
);
87 wxFontRefData(int size
,
92 const wxString
& faceName
,
93 wxFontEncoding encoding
)
99 , m_underlined(underlined
)
100 , m_faceName(faceName
)
101 , m_encoding(encoding
)
111 Init(size
, family
, style
, weight
, underlined
, faceName
, encoding
);
114 virtual ~wxFontRefData();
116 void SetNoAntiAliasing( bool no
= true )
119 bool GetNoAntiAliasing() const
125 // common part of all ctors
131 const wxString
& faceName
,
132 wxFontEncoding encoding
);
134 // font characterstics
142 wxFontEncoding m_encoding
;
143 bool m_noAA
; // No anti-aliasing
147 FMFontFamily m_macFontFamily
;
148 FMFontSize m_macFontSize
;
149 FMFontStyle m_macFontStyle
;
151 // ATSU Font Information
153 // this is split into an ATSU font id that may
154 // contain some styles (special bold fonts etc) and
155 // these are the additional qd styles that are not
156 // included in the ATSU font id
157 ATSUFontID m_macATSUFontID
;
158 FMFontStyle m_macATSUAdditionalQDStyles
;
160 // for true themeing support we must store the correct font
161 // information here, as this speeds up and optimizes rendering
162 ThemeFontID m_macThemeFontID
;
164 CTFontRef m_macFontRef
;
165 CTFontUIFontType m_macUIFontType
;
167 ATSUStyle m_macATSUStyle
;
168 wxNativeFontInfo m_info
;
171 #define M_FONTDATA ((wxFontRefData*)m_refData)
174 // ============================================================================
176 // ============================================================================
178 // ----------------------------------------------------------------------------
180 // ----------------------------------------------------------------------------
182 void wxFontRefData::Init(int pointSize
,
187 const wxString
& faceName
,
188 wxFontEncoding encoding
)
191 m_pointSize
= pointSize
;
195 m_underlined
= underlined
;
196 m_faceName
= faceName
;
197 m_encoding
= encoding
;
199 m_macUIFontType
= kCTFontNoFontType
;
202 m_macFontFamily
= 0 ;
206 m_macATSUAdditionalQDStyles
= 0 ;
208 m_macThemeFontID
= kThemeCurrentPortFont
;
210 m_macATSUStyle
= NULL
;
214 wxFontRefData::~wxFontRefData()
216 if ( m_macATSUStyle
)
218 ::ATSUDisposeStyle((ATSUStyle
)m_macATSUStyle
);
219 m_macATSUStyle
= NULL
;
223 void wxFontRefData::MacFindFont()
225 OSStatus status
= noErr
;
229 if ( m_faceName
.empty() && m_family
== wxDEFAULT
)
231 m_macUIFontType
= kCTFontSystemFontType
;
234 if ( m_macUIFontType
!= kCTFontNoFontType
)
236 m_macFontRef
= CTFontCreateUIFontForLanguage( m_macUIFontType
, 0.0, NULL
);
237 wxMacCFStringHolder
name( CTFontCopyFamilyName( m_macFontRef
) );
238 m_faceName
= name
.AsString();
242 if ( m_faceName
.empty() )
249 m_faceName
= wxT("Times");
253 m_faceName
= wxT("Lucida Grande");
258 m_faceName
= wxT("Monaco");
262 m_faceName
= wxT("Times");
267 wxMacCFStringHolder
cf( m_faceName
, wxLocale::GetSystemEncoding() );
268 m_macFontRef
= CTFontCreateWithName( cf
, m_pointSize
, NULL
);
271 if ( m_macATSUStyle
)
273 ::ATSUDisposeStyle((ATSUStyle
)m_macATSUStyle
);
274 m_macATSUStyle
= NULL
;
277 status
= ::ATSUCreateStyle((ATSUStyle
*)&m_macATSUStyle
);
278 wxASSERT_MSG( status
== noErr
, wxT("couldn't create ATSU style") );
280 ATSUAttributeTag atsuTags
[] =
283 kATSUVerticalCharacterTag
,
286 kATSUQDUnderlineTag
,
287 kATSUQDCondensedTag
,
291 ByteCount atsuSizes
[sizeof(atsuTags
) / sizeof(ATSUAttributeTag
)] =
294 sizeof( ATSUVerticalCharacterType
),
300 sizeof( ATSUFontID
) ,
303 Boolean kTrue
= true ;
304 Boolean kFalse
= false ;
306 Fixed atsuSize
= IntToFixed( m_pointSize
);
307 short m_macATSUAdditionalQDStyles
= 0;
308 ATSUVerticalCharacterType kHorizontal
= kATSUStronglyHorizontal
;
309 ATSUFontID atsuFontID
= 0;
310 int attributeCount
= sizeof(atsuTags
) / sizeof(ATSUAttributeTag
) ;
312 // attempt to add atsu font
314 status
= ATSUFindFontFromName(m_faceName
.c_str(), strlen(m_faceName
.c_str()), kFontFamilyName
, kFontNoPlatform
, kFontNoScript
, kFontNoLanguage
, &atsuFontID
);
315 if ( status
!= noErr
)
320 ATSUAttributeValuePtr atsuValues
[sizeof(atsuTags
) / sizeof(ATSUAttributeTag
)] =
324 (m_macATSUAdditionalQDStyles
& bold
) ? &kTrue
: &kFalse
,
325 (m_macATSUAdditionalQDStyles
& italic
) ? &kTrue
: &kFalse
,
326 (m_macATSUAdditionalQDStyles
& underline
) ? &kTrue
: &kFalse
,
327 (m_macATSUAdditionalQDStyles
& condense
) ? &kTrue
: &kFalse
,
328 (m_macATSUAdditionalQDStyles
& extend
) ? &kTrue
: &kFalse
,
332 status
= ::ATSUSetAttributes( (ATSUStyle
)m_macATSUStyle
, attributeCount
, atsuTags
, atsuSizes
, atsuValues
);
334 wxASSERT_MSG( status
== noErr
, wxT("couldn't modify ATSU style") );
336 if ( m_macThemeFontID
!= kThemeCurrentPortFont
)
339 GetThemeFont( m_macThemeFontID
, GetApplicationScript(), qdFontName
, &m_macFontSize
, &style
);
340 m_macFontStyle
= style
;
341 m_faceName
= wxMacMakeStringFromPascal( qdFontName
);
342 if ( m_macFontStyle
& bold
)
345 m_weight
= wxNORMAL
;
346 if ( m_macFontStyle
& italic
)
348 if ( m_macFontStyle
& underline
)
349 m_underlined
= true ;
350 m_pointSize
= m_macFontSize
;
351 m_macFontFamily
= FMGetFontFamilyFromName( qdFontName
);
355 if ( m_faceName
.empty() )
357 if ( m_family
== wxDEFAULT
)
359 m_macFontFamily
= GetAppFont();
360 FMGetFontFamilyName(m_macFontFamily
,qdFontName
);
361 m_faceName
= wxMacMakeStringFromPascal( qdFontName
);
370 m_faceName
= wxT("Times");
374 m_faceName
= wxT("Lucida Grande");
379 m_faceName
= wxT("Monaco");
383 m_faceName
= wxT("Times");
386 wxMacStringToPascal( m_faceName
, qdFontName
);
387 m_macFontFamily
= FMGetFontFamilyFromName( qdFontName
);
388 if ( m_macFontFamily
== kInvalidFontFamily
)
390 wxLogDebug( wxT("ATSFontFamilyFindFromName failed for %s"), m_faceName
.c_str() );
391 m_macFontFamily
= GetAppFont();
397 if ( m_faceName
== wxT("systemfont") )
398 m_macFontFamily
= GetSysFont();
399 else if ( m_faceName
== wxT("applicationfont") )
400 m_macFontFamily
= GetAppFont();
403 wxMacCFStringHolder
cf( m_faceName
, wxLocale::GetSystemEncoding() );
404 ATSFontFamilyRef atsfamily
= ATSFontFamilyFindFromName( cf
, kATSOptionFlagsDefault
);
405 if ( atsfamily
== (ATSFontFamilyRef
) -1 )
407 wxLogDebug( wxT("ATSFontFamilyFindFromName failed for ") + m_faceName
);
408 m_macFontFamily
= GetAppFont();
411 m_macFontFamily
= FMGetFontFamilyFromATSFontFamilyRef( atsfamily
);
416 if (m_weight
== wxBOLD
)
417 m_macFontStyle
|= bold
;
418 if (m_style
== wxITALIC
|| m_style
== wxSLANT
)
419 m_macFontStyle
|= italic
;
421 m_macFontStyle
|= underline
;
422 m_macFontSize
= m_pointSize
;
425 // we try to get as much styles as possible into ATSU
428 // ATSUFontID and FMFont are equivalent
429 FMFontStyle intrinsicStyle
= 0 ;
430 status
= FMGetFontFromFontFamilyInstance( m_macFontFamily
, m_macFontStyle
, &m_macATSUFontID
, &intrinsicStyle
);
431 wxASSERT_MSG( status
== noErr
, wxT("couldn't get an ATSUFont from font family") );
432 m_macATSUAdditionalQDStyles
= m_macFontStyle
& (~intrinsicStyle
);
434 if ( m_macATSUStyle
)
436 ::ATSUDisposeStyle((ATSUStyle
)m_macATSUStyle
);
437 m_macATSUStyle
= NULL
;
440 status
= ::ATSUCreateStyle((ATSUStyle
*)&m_macATSUStyle
);
441 wxASSERT_MSG( status
== noErr
, wxT("couldn't create ATSU style") );
443 ATSUAttributeTag atsuTags
[] =
447 kATSUVerticalCharacterTag
,
450 kATSUQDUnderlineTag
,
451 kATSUQDCondensedTag
,
454 ByteCount atsuSizes
[sizeof(atsuTags
) / sizeof(ATSUAttributeTag
)] =
456 sizeof( ATSUFontID
) ,
458 sizeof( ATSUVerticalCharacterType
),
466 Boolean kTrue
= true ;
467 Boolean kFalse
= false ;
469 Fixed atsuSize
= IntToFixed( m_macFontSize
);
470 ATSUVerticalCharacterType kHorizontal
= kATSUStronglyHorizontal
;
471 ATSUAttributeValuePtr atsuValues
[sizeof(atsuTags
) / sizeof(ATSUAttributeTag
)] =
476 (m_macATSUAdditionalQDStyles
& bold
) ? &kTrue
: &kFalse
,
477 (m_macATSUAdditionalQDStyles
& italic
) ? &kTrue
: &kFalse
,
478 (m_macATSUAdditionalQDStyles
& underline
) ? &kTrue
: &kFalse
,
479 (m_macATSUAdditionalQDStyles
& condense
) ? &kTrue
: &kFalse
,
480 (m_macATSUAdditionalQDStyles
& extend
) ? &kTrue
: &kFalse
,
483 status
= ::ATSUSetAttributes(
484 (ATSUStyle
)m_macATSUStyle
,
485 sizeof(atsuTags
) / sizeof(ATSUAttributeTag
) ,
486 atsuTags
, atsuSizes
, atsuValues
);
488 wxASSERT_MSG( status
== noErr
, wxT("couldn't modify ATSU style") );
492 // ----------------------------------------------------------------------------
494 // ----------------------------------------------------------------------------
496 bool wxFont::Create(const wxNativeFontInfo
& info
)
499 info
.pointSize
, info
.family
, info
.style
, info
.weight
,
500 info
.underlined
, info
.faceName
, info
.encoding
);
503 wxFont::wxFont(const wxString
& fontdesc
)
505 wxNativeFontInfo info
;
506 if ( info
.FromString(fontdesc
) )
510 bool wxFont::Create(int pointSize
,
515 const wxString
& faceName
,
516 wxFontEncoding encoding
)
520 m_refData
= new wxFontRefData(
521 pointSize
, family
, style
, weight
,
522 underlined
, faceName
, encoding
);
531 bool wxFont::MacCreateUIFont(wxUint32 ctFontType
)
535 m_refData
= new wxFontRefData(
536 12, wxDEFAULT
, wxFONTSTYLE_NORMAL
, wxFONTWEIGHT_NORMAL
,
537 false, wxEmptyString
, wxFONTENCODING_DEFAULT
);
539 M_FONTDATA
->m_macUIFontType
= ctFontType
;
547 bool wxFont::MacCreateThemeFont(wxUint16 themeFontID
)
550 return MacCreateUIFont(HIThemeGetUIFontType(themeFontID
));
554 m_refData
= new wxFontRefData(
555 12, wxDEFAULT
, wxFONTSTYLE_NORMAL
, wxFONTWEIGHT_NORMAL
,
556 false, wxEmptyString
, wxFONTENCODING_DEFAULT
);
558 M_FONTDATA
->m_macThemeFontID
= themeFontID
;
569 bool wxFont::RealizeResource()
571 M_FONTDATA
->MacFindFont();
576 void wxFont::SetEncoding(wxFontEncoding encoding
)
580 M_FONTDATA
->m_encoding
= encoding
;
585 void wxFont::Unshare()
587 // Don't change shared data
590 m_refData
= new wxFontRefData();
594 wxFontRefData
* ref
= new wxFontRefData(*(wxFontRefData
*)m_refData
);
600 void wxFont::SetPointSize(int pointSize
)
604 M_FONTDATA
->m_pointSize
= pointSize
;
609 void wxFont::SetFamily(int family
)
613 M_FONTDATA
->m_family
= family
;
618 void wxFont::SetStyle(int style
)
622 M_FONTDATA
->m_style
= style
;
627 void wxFont::SetWeight(int weight
)
631 M_FONTDATA
->m_weight
= weight
;
636 bool wxFont::SetFaceName(const wxString
& faceName
)
640 M_FONTDATA
->m_faceName
= faceName
;
644 return wxFontBase::SetFaceName(faceName
);
647 void wxFont::SetUnderlined(bool underlined
)
651 M_FONTDATA
->m_underlined
= underlined
;
656 void wxFont::SetNoAntiAliasing( bool no
)
660 M_FONTDATA
->SetNoAntiAliasing( no
);
665 // ----------------------------------------------------------------------------
667 // ----------------------------------------------------------------------------
669 // TODO: insert checks everywhere for M_FONTDATA == NULL!
671 int wxFont::GetPointSize() const
673 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
675 return M_FONTDATA
->m_pointSize
;
678 wxSize
wxFont::GetPixelSize() const
680 #if wxUSE_GRAPHICS_CONTEXT
681 // TODO: consider caching the value
682 wxGraphicsContext
* dc
= wxGraphicsContext::CreateFromNative((CGContextRef
) NULL
);
683 dc
->SetFont(*(wxFont
*)this,*wxBLACK
);
684 wxDouble width
, height
= 0;
685 dc
->GetTextExtent( wxT("g"), &width
, &height
, NULL
, NULL
);
687 return wxSize((int)width
, (int)height
);
689 return wxFontBase::GetPixelSize();
693 int wxFont::GetFamily() const
695 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
697 return M_FONTDATA
->m_family
;
700 int wxFont::GetStyle() const
702 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
704 return M_FONTDATA
->m_style
;
707 int wxFont::GetWeight() const
709 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
711 return M_FONTDATA
->m_weight
;
714 bool wxFont::GetUnderlined() const
716 wxCHECK_MSG( M_FONTDATA
!= NULL
, false, wxT("invalid font") );
718 return M_FONTDATA
->m_underlined
;
721 wxString
wxFont::GetFaceName() const
723 wxCHECK_MSG( M_FONTDATA
!= NULL
, wxEmptyString
, wxT("invalid font") );
725 return M_FONTDATA
->m_faceName
;
728 wxFontEncoding
wxFont::GetEncoding() const
730 wxCHECK_MSG( M_FONTDATA
!= NULL
, wxFONTENCODING_DEFAULT
, wxT("invalid font") );
732 return M_FONTDATA
->m_encoding
;
735 bool wxFont::GetNoAntiAliasing() const
737 wxCHECK_MSG( M_FONTDATA
!= NULL
, false, wxT("invalid font") );
739 return M_FONTDATA
->m_noAA
;
744 short wxFont::MacGetFontNum() const
746 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
748 return M_FONTDATA
->m_macFontFamily
;
751 short wxFont::MacGetFontSize() const
753 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
755 return M_FONTDATA
->m_macFontSize
;
758 wxByte
wxFont::MacGetFontStyle() const
760 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
762 return M_FONTDATA
->m_macFontStyle
;
765 wxUint32
wxFont::MacGetATSUFontID() const
767 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
769 return M_FONTDATA
->m_macATSUFontID
;
772 void * wxFont::MacGetATSUStyle() const
774 wxCHECK_MSG( M_FONTDATA
!= NULL
, NULL
, wxT("invalid font") );
776 return M_FONTDATA
->m_macATSUStyle
;
779 wxUint32
wxFont::MacGetATSUAdditionalQDStyles() const
781 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
783 return M_FONTDATA
->m_macATSUAdditionalQDStyles
;
786 wxUint16
wxFont::MacGetThemeFontID() const
788 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
790 return M_FONTDATA
->m_macThemeFontID
;
793 const void * wxFont::MacGetCTFont() const
795 wxCHECK_MSG( M_FONTDATA
!= NULL
, 0, wxT("invalid font") );
797 return M_FONTDATA
->m_macFontRef
;
801 void * wxFont::MacGetATSUStyle() const
803 wxCHECK_MSG( M_FONTDATA
!= NULL
, NULL
, wxT("invalid font") );
805 return M_FONTDATA
->m_macATSUStyle
;
810 const wxNativeFontInfo
* wxFont::GetNativeFontInfo() const
812 wxCHECK_MSG( M_FONTDATA
!= NULL
, NULL
, wxT("invalid font") );
813 wxCHECK_MSG( Ok(), NULL
, wxT("invalid font") );
815 M_FONTDATA
->m_info
.InitFromFont(*this);
817 return &(M_FONTDATA
->m_info
);