1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/fontcmn.cpp 
   3 // Purpose:     implementation of wxFontBase methods 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) wxWidgets team 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  32     #include "wx/dcscreen.h" 
  34     #include "wx/gdicmn.h" 
  37 #if defined(__WXMSW__) 
  38     #include  "wx/msw/private.h"  // includes windows.h for LOGFONT 
  39     #include  "wx/msw/winundef.h" 
  42 #include "wx/fontutil.h" // for wxNativeFontInfo 
  43 #include "wx/fontmap.h" 
  44 #include "wx/fontenum.h" 
  46 #include "wx/tokenzr.h" 
  48 // ============================================================================ 
  50 // ============================================================================ 
  52 // ---------------------------------------------------------------------------- 
  54 // ---------------------------------------------------------------------------- 
  56 static void AdjustFontSize(wxFont
& font
, wxDC
& dc
, const wxSize
& pixelSize
) 
  62     bool initialGoodFound 
= false; 
  63     bool initialBadFound 
= false; 
  65     // NB: this assignment was separated from the variable definition 
  66     // in order to fix a gcc v3.3.3 compiler crash 
  67     currentSize 
= font
.GetPointSize(); 
  68     while (currentSize 
> 0) 
  72         // if currentSize (in points) results in a font that is smaller 
  73         // than required by pixelSize it is considered a good size 
  74         if (dc
.GetCharHeight() <= pixelSize
.GetHeight() && 
  75                 (!pixelSize
.GetWidth() || 
  76                  dc
.GetCharWidth() <= pixelSize
.GetWidth())) 
  78             largestGood 
= currentSize
; 
  79             initialGoodFound 
= true; 
  83             smallestBad 
= currentSize
; 
  84             initialBadFound 
= true; 
  86         if (!initialGoodFound
) 
  90         else if (!initialBadFound
) 
  96             int distance 
= smallestBad 
- largestGood
; 
 100             currentSize 
= largestGood 
+ distance 
/ 2; 
 103         font
.SetPointSize(currentSize
); 
 106     if (currentSize 
!= largestGood
) 
 107         font
.SetPointSize(largestGood
); 
 110 // ---------------------------------------------------------------------------- 
 112 // ---------------------------------------------------------------------------- 
 114 wxFontEncoding 
wxFontBase::ms_encodingDefault 
= wxFONTENCODING_SYSTEM
; 
 117 void wxFontBase::SetDefaultEncoding(wxFontEncoding encoding
) 
 119     // GetDefaultEncoding() should return something != wxFONTENCODING_DEFAULT 
 120     // and, besides, using this value here doesn't make any sense 
 121     wxCHECK_RET( encoding 
!= wxFONTENCODING_DEFAULT
, 
 122                  _T("can't set default encoding to wxFONTENCODING_DEFAULT") ); 
 124     ms_encodingDefault 
= encoding
; 
 127 wxFontBase::~wxFontBase() 
 129     // this destructor is required for Darwin 
 133 wxFont 
*wxFontBase::New(int size
, 
 138                         const wxString
& face
, 
 139                         wxFontEncoding encoding
) 
 141     return new wxFont(size
, family
, style
, weight
, underlined
, face
, encoding
); 
 144 static inline int flags2Style(int flags
) 
 146     return flags 
& wxFONTFLAG_ITALIC
 
 148                     : flags 
& wxFONTFLAG_SLANT
 
 150                         : wxFONTSTYLE_NORMAL
; 
 153 static inline int flags2Weight(int flags
) 
 155     return flags 
& wxFONTFLAG_LIGHT
 
 157                     : flags 
& wxFONTFLAG_BOLD
 
 159                         : wxFONTWEIGHT_NORMAL
; 
 162 static inline bool flags2Underlined(int flags
) 
 164     return (flags 
& wxFONTFLAG_UNDERLINED
) != 0; 
 168 wxFont 
*wxFontBase::New(int pointSize
, 
 171                         const wxString
& face
, 
 172                         wxFontEncoding encoding
) 
 174     return New(pointSize
, family
, flags2Style(flags
), flags2Weight(flags
), 
 175                flags2Underlined(flags
), face
, encoding
); 
 179 wxFont 
*wxFontBase::New(const wxSize
& pixelSize
, 
 184                         const wxString
& face
, 
 185                         wxFontEncoding encoding
) 
 187 #if defined(__WXMSW__) 
 188     return new wxFont(pixelSize
, family
, style
, weight
, underlined
, 
 191     wxFont 
*self 
= New(10, family
, style
, weight
, underlined
, face
, encoding
); 
 193     AdjustFontSize(*(wxFont 
*)self
, dc
, pixelSize
); 
 199 wxFont 
*wxFontBase::New(const wxSize
& pixelSize
, 
 202                         const wxString
& face
, 
 203                         wxFontEncoding encoding
) 
 205     return New(pixelSize
, family
, flags2Style(flags
), flags2Weight(flags
), 
 206                flags2Underlined(flags
), face
, encoding
); 
 209 wxSize 
wxFontBase::GetPixelSize() const 
 212     dc
.SetFont(*(wxFont 
*)this); 
 213     return wxSize(dc
.GetCharWidth(), dc
.GetCharHeight()); 
 216 bool wxFontBase::IsUsingSizeInPixels() const 
 221 void wxFontBase::SetPixelSize( const wxSize
& pixelSize 
) 
 224     AdjustFontSize(*(wxFont 
*)this, dc
, pixelSize
); 
 228 wxFont 
*wxFontBase::New(const wxNativeFontInfo
& info
) 
 230     return new wxFont(info
); 
 234 wxFont 
*wxFontBase::New(const wxString
& strNativeFontDesc
) 
 236     wxNativeFontInfo fontInfo
; 
 237     if ( !fontInfo
.FromString(strNativeFontDesc
) ) 
 238         return new wxFont(*wxNORMAL_FONT
); 
 240     return New(fontInfo
); 
 243 bool wxFontBase::IsFixedWidth() const 
 245     return GetFamily() == wxFONTFAMILY_TELETYPE
; 
 248 void wxFontBase::DoSetNativeFontInfo(const wxNativeFontInfo
& info
) 
 250 #ifdef wxNO_NATIVE_FONTINFO 
 251     SetPointSize(info
.pointSize
); 
 252     SetFamily(info
.family
); 
 253     SetStyle(info
.style
); 
 254     SetWeight(info
.weight
); 
 255     SetUnderlined(info
.underlined
); 
 256     SetFaceName(info
.faceName
); 
 257     SetEncoding(info
.encoding
); 
 263 wxString 
wxFontBase::GetNativeFontInfoDesc() const 
 266     const wxNativeFontInfo 
*fontInfo 
= GetNativeFontInfo(); 
 269         fontDesc 
= fontInfo
->ToString(); 
 270         wxASSERT_MSG(!fontDesc
.empty(), wxT("This should be a non-empty string!")); 
 274         wxFAIL_MSG(wxT("Derived class should have created the wxNativeFontInfo!")); 
 280 wxString 
wxFontBase::GetNativeFontInfoUserDesc() const 
 283     const wxNativeFontInfo 
*fontInfo 
= GetNativeFontInfo(); 
 286         fontDesc 
= fontInfo
->ToUserString(); 
 287         wxASSERT_MSG(!fontDesc
.empty(), wxT("This should be a non-empty string!")); 
 291         wxFAIL_MSG(wxT("Derived class should have created the wxNativeFontInfo!")); 
 297 bool wxFontBase::SetNativeFontInfo(const wxString
& info
) 
 299     wxNativeFontInfo fontInfo
; 
 300     if ( !info
.empty() && fontInfo
.FromString(info
) ) 
 302         SetNativeFontInfo(fontInfo
); 
 309 bool wxFontBase::SetNativeFontInfoUserDesc(const wxString
& info
) 
 311     wxNativeFontInfo fontInfo
; 
 312     if ( !info
.empty() && fontInfo
.FromUserString(info
) ) 
 314         SetNativeFontInfo(fontInfo
); 
 321 bool wxFontBase::operator==(const wxFont
& font
) const 
 323     // either it is the same font, i.e. they share the same common data or they 
 324     // have different ref datas but still describe the same font 
 325     return IsSameAs(font
) || 
 328             GetPointSize() == font
.GetPointSize() && 
 329             // in wxGTK1 GetPixelSize() calls GetInternalFont() which uses 
 330             // operator==() resulting in infinite recursion so we can't use it 
 332 #if !defined(__WXGTK__) || defined(__WXGTK20__) 
 333             GetPixelSize() == font
.GetPixelSize() && 
 335             GetFamily() == font
.GetFamily() && 
 336             GetStyle() == font
.GetStyle() && 
 337             GetWeight() == font
.GetWeight() && 
 338             GetUnderlined() == font
.GetUnderlined() && 
 339             GetFaceName().IsSameAs(font
.GetFaceName(), false) && 
 340             GetEncoding() == font
.GetEncoding() 
 344 wxString 
wxFontBase::GetFamilyString() const 
 346     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 348     switch ( GetFamily() ) 
 350         case wxDECORATIVE
:   return wxT("wxDECORATIVE"); 
 351         case wxROMAN
:        return wxT("wxROMAN"); 
 352         case wxSCRIPT
:       return wxT("wxSCRIPT"); 
 353         case wxSWISS
:        return wxT("wxSWISS"); 
 354         case wxMODERN
:       return wxT("wxMODERN"); 
 355         case wxTELETYPE
:     return wxT("wxTELETYPE"); 
 356         default:             return wxT("wxDEFAULT"); 
 360 wxString 
wxFontBase::GetStyleString() const 
 362     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 364     switch ( GetStyle() ) 
 366         case wxNORMAL
:   return wxT("wxNORMAL"); 
 367         case wxSLANT
:    return wxT("wxSLANT"); 
 368         case wxITALIC
:   return wxT("wxITALIC"); 
 369         default:         return wxT("wxDEFAULT"); 
 373 wxString 
wxFontBase::GetWeightString() const 
 375     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 377     switch ( GetWeight() ) 
 379         case wxNORMAL
:   return wxT("wxNORMAL"); 
 380         case wxBOLD
:     return wxT("wxBOLD"); 
 381         case wxLIGHT
:    return wxT("wxLIGHT"); 
 382         default:         return wxT("wxDEFAULT"); 
 386 bool wxFontBase::SetFaceName(const wxString
& facename
) 
 389     if (!wxFontEnumerator::IsValidFacename(facename
)) 
 391         UnRef();        // make Ok() return false 
 394 #else // !wxUSE_FONTENUM 
 395     wxUnusedVar(facename
); 
 396 #endif // wxUSE_FONTENUM/!wxUSE_FONTENUM 
 402 // ---------------------------------------------------------------------------- 
 404 // ---------------------------------------------------------------------------- 
 406 // Up to now, there are no native implementations of this function: 
 407 void wxNativeFontInfo::SetFaceName(const wxArrayString
& facenames
) 
 410     for (size_t i
=0; i 
< facenames
.GetCount(); i
++) 
 412         if (wxFontEnumerator::IsValidFacename(facenames
[i
])) 
 414             SetFaceName(facenames
[i
]); 
 419     // set the first valid facename we can find on this system 
 420     wxString validfacename 
= wxFontEnumerator::GetFacenames().Item(0); 
 421     wxLogTrace(wxT("font"), wxT("Falling back to '%s'"), validfacename
.c_str()); 
 422     SetFaceName(validfacename
); 
 423 #else // !wxUSE_FONTENUM 
 424     SetFaceName(facenames
[0]); 
 425 #endif // wxUSE_FONTENUM/!wxUSE_FONTENUM 
 429 #ifdef wxNO_NATIVE_FONTINFO 
 431 // These are the generic forms of FromString()/ToString. 
 433 // convert to/from the string representation: format is 
 434 //      version;pointsize;family;style;weight;underlined;facename;encoding 
 436 bool wxNativeFontInfo::FromString(const wxString
& s
) 
 440     wxStringTokenizer 
tokenizer(s
, _T(";")); 
 442     wxString token 
= tokenizer
.GetNextToken(); 
 444     //  Ignore the version for now 
 447     token 
= tokenizer
.GetNextToken(); 
 448     if ( !token
.ToLong(&l
) ) 
 452     token 
= tokenizer
.GetNextToken(); 
 453     if ( !token
.ToLong(&l
) ) 
 455     family 
= (wxFontFamily
)l
; 
 457     token 
= tokenizer
.GetNextToken(); 
 458     if ( !token
.ToLong(&l
) ) 
 460     style 
= (wxFontStyle
)l
; 
 462     token 
= tokenizer
.GetNextToken(); 
 463     if ( !token
.ToLong(&l
) ) 
 465     weight 
= (wxFontWeight
)l
; 
 467     token 
= tokenizer
.GetNextToken(); 
 468     if ( !token
.ToLong(&l
) ) 
 472     faceName 
= tokenizer
.GetNextToken(); 
 479     token 
= tokenizer
.GetNextToken(); 
 480     if ( !token
.ToLong(&l
) ) 
 482     encoding 
= (wxFontEncoding
)l
; 
 487 wxString 
wxNativeFontInfo::ToString() const 
 491     s
.Printf(_T("%d;%d;%d;%d;%d;%d;%s;%d"), 
 504 void wxNativeFontInfo::Init() 
 507     family 
= wxFONTFAMILY_DEFAULT
; 
 508     style 
= wxFONTSTYLE_NORMAL
; 
 509     weight 
= wxFONTWEIGHT_NORMAL
; 
 512     encoding 
= wxFONTENCODING_DEFAULT
; 
 515 int wxNativeFontInfo::GetPointSize() const 
 520 wxFontStyle 
wxNativeFontInfo::GetStyle() const 
 525 wxFontWeight 
wxNativeFontInfo::GetWeight() const 
 530 bool wxNativeFontInfo::GetUnderlined() const 
 535 wxString 
wxNativeFontInfo::GetFaceName() const 
 540 wxFontFamily 
wxNativeFontInfo::GetFamily() const 
 545 wxFontEncoding 
wxNativeFontInfo::GetEncoding() const 
 550 void wxNativeFontInfo::SetPointSize(int pointsize
) 
 552     pointSize 
= pointsize
; 
 555 void wxNativeFontInfo::SetStyle(wxFontStyle style_
) 
 560 void wxNativeFontInfo::SetWeight(wxFontWeight weight_
) 
 565 void wxNativeFontInfo::SetUnderlined(bool underlined_
) 
 567     underlined 
= underlined_
; 
 570 bool wxNativeFontInfo::SetFaceName(const wxString
& facename_
) 
 572     faceName 
= facename_
; 
 576 void wxNativeFontInfo::SetFamily(wxFontFamily family_
) 
 581 void wxNativeFontInfo::SetEncoding(wxFontEncoding encoding_
) 
 583     encoding 
= encoding_
; 
 586 #endif // generic wxNativeFontInfo implementation 
 588 // conversion to/from user-readable string: this is used in the generic 
 589 // versions and under MSW as well because there is no standard font description 
 590 // format there anyhow (but there is a well-defined standard for X11 fonts used 
 591 // by wxGTK and wxMotif) 
 593 #if defined(wxNO_NATIVE_FONTINFO) || defined(__WXMSW__) || defined (__WXPM__) 
 595 wxString 
wxNativeFontInfo::ToUserString() const 
 599     // first put the adjectives, if any - this is English-centric, of course, 
 600     // but what else can we do? 
 601     if ( GetUnderlined() ) 
 603         desc 
<< _("underlined"); 
 606     switch ( GetWeight() ) 
 609             wxFAIL_MSG( _T("unknown font weight") ); 
 612         case wxFONTWEIGHT_NORMAL
: 
 615         case wxFONTWEIGHT_LIGHT
: 
 619         case wxFONTWEIGHT_BOLD
: 
 624     switch ( GetStyle() ) 
 627             wxFAIL_MSG( _T("unknown font style") ); 
 630         case wxFONTSTYLE_NORMAL
: 
 633             // we don't distinguish between the two for now anyhow... 
 634         case wxFONTSTYLE_ITALIC
: 
 635         case wxFONTSTYLE_SLANT
: 
 636             desc 
<< _(" italic"); 
 640     wxString face 
= GetFaceName(); 
 643         desc 
<< _T(' ') << face
; 
 646     int size 
= GetPointSize(); 
 647     if ( size 
!= wxNORMAL_FONT
->GetPointSize() ) 
 649         desc 
<< _T(' ') << size
; 
 653     wxFontEncoding enc 
= GetEncoding(); 
 654     if ( enc 
!= wxFONTENCODING_DEFAULT 
&& enc 
!= wxFONTENCODING_SYSTEM 
) 
 656         desc 
<< _T(' ') << wxFontMapper::GetEncodingName(enc
); 
 658 #endif // wxUSE_FONTMAP 
 660     return desc
.Strip(wxString::both
).MakeLower(); 
 663 bool wxNativeFontInfo::FromUserString(const wxString
& s
) 
 665     // reset to the default state 
 668     // parse a more or less free form string 
 670     // TODO: we should handle at least the quoted facenames 
 671     wxStringTokenizer 
tokenizer(s
, _T(";, "), wxTOKEN_STRTOK
); 
 675     bool weightfound 
= false, pointsizefound 
= false; 
 677     bool encodingfound 
= false; 
 680     while ( tokenizer
.HasMoreTokens() ) 
 682         wxString token 
= tokenizer
.GetNextToken(); 
 685         token
.Trim(true).Trim(false).MakeLower(); 
 687         // look for the known tokens 
 688         if ( token 
== _T("underlined") || token 
== _("underlined") ) 
 692         else if ( token 
== _T("light") || token 
== _("light") ) 
 694             SetWeight(wxFONTWEIGHT_LIGHT
); 
 697         else if ( token 
== _T("bold") || token 
== _("bold") ) 
 699             SetWeight(wxFONTWEIGHT_BOLD
); 
 702         else if ( token 
== _T("italic") || token 
== _("italic") ) 
 704             SetStyle(wxFONTSTYLE_ITALIC
); 
 706         else if ( token
.ToULong(&size
) ) 
 709             pointsizefound 
= true; 
 714             // try to interpret this as an encoding 
 715             wxFontEncoding encoding 
= wxFontMapper::Get()->CharsetToEncoding(token
, false); 
 716             if ( encoding 
!= wxFONTENCODING_DEFAULT 
&& 
 717                  encoding 
!= wxFONTENCODING_SYSTEM 
)    // returned when the recognition failed 
 719             SetEncoding(encoding
); 
 720                 encodingfound 
= true; 
 724 #endif // wxUSE_FONTMAP 
 726                 // assume it is the face name 
 734             // skip the code which resets face below 
 739 #endif // wxUSE_FONTMAP 
 742         // if we had had the facename, we shouldn't continue appending tokens 
 743         // to it (i.e. "foo bold bar" shouldn't result in the facename "foo 
 747             // NB: the check on the facename is implemented in wxFontBase::SetFaceName 
 748             //     and not in wxNativeFontInfo::SetFaceName thus we need to explicitely 
 749             //     call here wxFontEnumerator::IsValidFacename 
 752                     !wxFontEnumerator::IsValidFacename(face
) || 
 753 #endif // wxUSE_FONTENUM 
 756                 SetFaceName(wxNORMAL_FONT
->GetFaceName()); 
 763     // we might not have flushed it inside the loop 
 766         // NB: the check on the facename is implemented in wxFontBase::SetFaceName 
 767         //     and not in wxNativeFontInfo::SetFaceName thus we need to explicitely 
 768         //     call here wxFontEnumerator::IsValidFacename 
 771                 !wxFontEnumerator::IsValidFacename(face
) || 
 772 #endif // wxUSE_FONTENUM 
 775                 SetFaceName(wxNORMAL_FONT
->GetFaceName()); 
 779     // set point size to default value if size was not given 
 780     if ( !pointsizefound 
) 
 781         SetPointSize(wxNORMAL_FONT
->GetPointSize()); 
 783     // set font weight to default value if weight was not given 
 785         SetWeight(wxFONTWEIGHT_NORMAL
); 
 788     // set font encoding to default value if encoding was not given 
 789     if ( !encodingfound 
) 
 790         SetEncoding(wxFONTENCODING_SYSTEM
); 
 791 #endif // wxUSE_FONTMAP 
 796 #endif // generic or wxMSW or wxOS2 
 799 // wxFont <-> wxString utilities, used by wxConfig 
 800 wxString 
wxToString(const wxFontBase
& font
) 
 802     return font
.IsOk() ? font
.GetNativeFontInfoDesc() 
 806 bool wxFromString(const wxString
& str
, wxFontBase 
*font
) 
 808     wxCHECK_MSG( font
, false, _T("NULL output parameter") ); 
 816     return font
->SetNativeFontInfo(str
);