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" 
  36 #include "wx/gdicmn.h" 
  38 #if defined(__WXMSW__) 
  39     #include  "wx/msw/private.h"  // includes windows.h for LOGFONT 
  40     #include  "wx/msw/winundef.h" 
  43 #include "wx/fontutil.h" // for wxNativeFontInfo 
  44 #include "wx/fontmap.h" 
  45 #include "wx/fontenum.h" 
  47 #include "wx/tokenzr.h" 
  49 // ============================================================================ 
  51 // ============================================================================ 
  53 // ---------------------------------------------------------------------------- 
  55 // ---------------------------------------------------------------------------- 
  57 static void AdjustFontSize(wxFont
& font
, wxDC
& dc
, const wxSize
& pixelSize
) 
  63     bool initialGoodFound 
= false; 
  64     bool initialBadFound 
= false; 
  66     // NB: this assignment was separated from the variable definition 
  67     // in order to fix a gcc v3.3.3 compiler crash 
  68     currentSize 
= font
.GetPointSize(); 
  69     while (currentSize 
> 0) 
  73         // if currentSize (in points) results in a font that is smaller 
  74         // than required by pixelSize it is considered a good size 
  75         if (dc
.GetCharHeight() <= pixelSize
.GetHeight() && 
  76                 (!pixelSize
.GetWidth() || 
  77                  dc
.GetCharWidth() <= pixelSize
.GetWidth())) 
  79             largestGood 
= currentSize
; 
  80             initialGoodFound 
= true; 
  84             smallestBad 
= currentSize
; 
  85             initialBadFound 
= true; 
  87         if (!initialGoodFound
) 
  91         else if (!initialBadFound
) 
  97             int distance 
= smallestBad 
- largestGood
; 
 101             currentSize 
= largestGood 
+ distance 
/ 2; 
 104         font
.SetPointSize(currentSize
); 
 107     if (currentSize 
!= largestGood
) 
 108         font
.SetPointSize(largestGood
); 
 111 // ---------------------------------------------------------------------------- 
 113 // ---------------------------------------------------------------------------- 
 115 wxFontEncoding 
wxFontBase::ms_encodingDefault 
= wxFONTENCODING_SYSTEM
; 
 118 void wxFontBase::SetDefaultEncoding(wxFontEncoding encoding
) 
 120     // GetDefaultEncoding() should return something != wxFONTENCODING_DEFAULT 
 121     // and, besides, using this value here doesn't make any sense 
 122     wxCHECK_RET( encoding 
!= wxFONTENCODING_DEFAULT
, 
 123                  _T("can't set default encoding to wxFONTENCODING_DEFAULT") ); 
 125     ms_encodingDefault 
= encoding
; 
 128 wxFontBase::~wxFontBase() 
 130     // this destructor is required for Darwin 
 134 wxFont 
*wxFontBase::New(int size
, 
 139                         const wxString
& face
, 
 140                         wxFontEncoding encoding
) 
 142     return new wxFont(size
, family
, style
, weight
, underlined
, face
, encoding
); 
 145 static inline int flags2Style(int flags
) 
 147     return flags 
& wxFONTFLAG_ITALIC
 
 149                     : flags 
& wxFONTFLAG_SLANT
 
 151                         : wxFONTSTYLE_NORMAL
; 
 154 static inline int flags2Weight(int flags
) 
 156     return flags 
& wxFONTFLAG_LIGHT
 
 158                     : flags 
& wxFONTFLAG_BOLD
 
 160                         : wxFONTWEIGHT_NORMAL
; 
 163 static inline bool flags2Underlined(int flags
) 
 165     return (flags 
& wxFONTFLAG_UNDERLINED
) != 0; 
 169 wxFont 
*wxFontBase::New(int pointSize
, 
 172                         const wxString
& face
, 
 173                         wxFontEncoding encoding
) 
 175     return New(pointSize
, family
, flags2Style(flags
), flags2Weight(flags
), 
 176                flags2Underlined(flags
), face
, encoding
); 
 180 wxFont 
*wxFontBase::New(const wxSize
& pixelSize
, 
 185                         const wxString
& face
, 
 186                         wxFontEncoding encoding
) 
 188 #if defined(__WXMSW__) 
 189     return new wxFont(pixelSize
, family
, style
, weight
, underlined
, 
 192     wxFont 
*self 
= New(10, family
, style
, weight
, underlined
, face
, encoding
); 
 194     AdjustFontSize(*(wxFont 
*)self
, dc
, pixelSize
); 
 200 wxFont 
*wxFontBase::New(const wxSize
& pixelSize
, 
 203                         const wxString
& face
, 
 204                         wxFontEncoding encoding
) 
 206     return New(pixelSize
, family
, flags2Style(flags
), flags2Weight(flags
), 
 207                flags2Underlined(flags
), face
, encoding
); 
 210 wxSize 
wxFontBase::GetPixelSize() const 
 213     dc
.SetFont(*(wxFont 
*)this); 
 214     return wxSize(dc
.GetCharWidth(), dc
.GetCharHeight()); 
 217 bool wxFontBase::IsUsingSizeInPixels() const 
 222 void wxFontBase::SetPixelSize( const wxSize
& pixelSize 
) 
 225     AdjustFontSize(*(wxFont 
*)this, dc
, pixelSize
); 
 229 wxFont 
*wxFontBase::New(const wxNativeFontInfo
& info
) 
 231     return new wxFont(info
); 
 235 wxFont 
*wxFontBase::New(const wxString
& strNativeFontDesc
) 
 237     wxNativeFontInfo fontInfo
; 
 238     if ( !fontInfo
.FromString(strNativeFontDesc
) ) 
 239         return new wxFont(*wxNORMAL_FONT
); 
 241     return New(fontInfo
); 
 244 bool wxFontBase::IsFixedWidth() const 
 246     return GetFamily() == wxFONTFAMILY_TELETYPE
; 
 249 void wxFontBase::DoSetNativeFontInfo(const wxNativeFontInfo
& info
) 
 251 #ifdef wxNO_NATIVE_FONTINFO 
 252     SetPointSize(info
.pointSize
); 
 253     SetFamily(info
.family
); 
 254     SetStyle(info
.style
); 
 255     SetWeight(info
.weight
); 
 256     SetUnderlined(info
.underlined
); 
 257     SetFaceName(info
.faceName
); 
 258     SetEncoding(info
.encoding
); 
 264 wxString 
wxFontBase::GetNativeFontInfoDesc() const 
 267     const wxNativeFontInfo 
*fontInfo 
= GetNativeFontInfo(); 
 270         fontDesc 
= fontInfo
->ToString(); 
 271         wxASSERT_MSG(!fontDesc
.IsEmpty(), wxT("This should be a non-empty string!")); 
 275         wxFAIL_MSG(wxT("Derived class should have created the wxNativeFontInfo!")); 
 281 wxString 
wxFontBase::GetNativeFontInfoUserDesc() const 
 284     const wxNativeFontInfo 
*fontInfo 
= GetNativeFontInfo(); 
 287         fontDesc 
= fontInfo
->ToUserString(); 
 288         wxASSERT_MSG(!fontDesc
.IsEmpty(), wxT("This should be a non-empty string!")); 
 292         wxFAIL_MSG(wxT("Derived class should have created the wxNativeFontInfo!")); 
 298 bool wxFontBase::SetNativeFontInfo(const wxString
& info
) 
 300     wxNativeFontInfo fontInfo
; 
 301     if ( !info
.empty() && fontInfo
.FromString(info
) ) 
 303         SetNativeFontInfo(fontInfo
); 
 311 bool wxFontBase::SetNativeFontInfoUserDesc(const wxString
& info
) 
 313     wxNativeFontInfo fontInfo
; 
 314     if ( !info
.empty() && fontInfo
.FromUserString(info
) ) 
 316         SetNativeFontInfo(fontInfo
); 
 324 bool wxFontBase::operator==(const wxFont
& font
) const 
 326     // either it is the same font, i.e. they share the same common data or they 
 327     // have different ref datas but still describe the same font 
 328     return GetFontData() == font
.GetFontData() || 
 331             GetPointSize() == font
.GetPointSize() && 
 332             GetFamily() == font
.GetFamily() && 
 333             GetStyle() == font
.GetStyle() && 
 334             GetWeight() == font
.GetWeight() && 
 335             GetUnderlined() == font
.GetUnderlined() && 
 336             GetFaceName().IsSameAs(font
.GetFaceName(), false) && 
 337             GetEncoding() == font
.GetEncoding() 
 341 bool wxFontBase::operator!=(const wxFont
& font
) const 
 343     return !(*this == font
); 
 346 wxString 
wxFontBase::GetFamilyString() const 
 348     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 350     switch ( GetFamily() ) 
 352         case wxDECORATIVE
:   return wxT("wxDECORATIVE"); 
 353         case wxROMAN
:        return wxT("wxROMAN"); 
 354         case wxSCRIPT
:       return wxT("wxSCRIPT"); 
 355         case wxSWISS
:        return wxT("wxSWISS"); 
 356         case wxMODERN
:       return wxT("wxMODERN"); 
 357         case wxTELETYPE
:     return wxT("wxTELETYPE"); 
 358         default:             return wxT("wxDEFAULT"); 
 362 wxString 
wxFontBase::GetStyleString() const 
 364     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 366     switch ( GetStyle() ) 
 368         case wxNORMAL
:   return wxT("wxNORMAL"); 
 369         case wxSLANT
:    return wxT("wxSLANT"); 
 370         case wxITALIC
:   return wxT("wxITALIC"); 
 371         default:         return wxT("wxDEFAULT"); 
 375 wxString 
wxFontBase::GetWeightString() const 
 377     wxCHECK_MSG( Ok(), wxT("wxDEFAULT"), wxT("invalid font") ); 
 379     switch ( GetWeight() ) 
 381         case wxNORMAL
:   return wxT("wxNORMAL"); 
 382         case wxBOLD
:     return wxT("wxBOLD"); 
 383         case wxLIGHT
:    return wxT("wxLIGHT"); 
 384         default:         return wxT("wxDEFAULT"); 
 388 bool wxFontBase::SetFaceName(const wxString 
&facename
) 
 390     if (!wxFontEnumerator::IsValidFacename(facename
)) 
 392         UnRef();        // make Ok() return false 
 400 // ---------------------------------------------------------------------------- 
 402 // ---------------------------------------------------------------------------- 
 404 // Up to now, there are no native implementations of this function: 
 405 void wxNativeFontInfo::SetFaceName(const wxArrayString 
&facenames
) 
 407     for (size_t i
=0; i 
< facenames
.GetCount(); i
++) 
 409         if (wxFontEnumerator::IsValidFacename(facenames
[i
])) 
 411             SetFaceName(facenames
[i
]); 
 416     // set the first valid facename we can find on this system 
 417     wxString validfacename 
= wxFontEnumerator::GetFacenames().Item(0); 
 418     wxLogTrace(wxT("font"), wxT("Falling back to '%s'"), validfacename
.c_str()); 
 419     SetFaceName(validfacename
); 
 423 #ifdef wxNO_NATIVE_FONTINFO 
 425 // These are the generic forms of FromString()/ToString. 
 427 // convert to/from the string representation: format is 
 428 //      version;pointsize;family;style;weight;underlined;facename;encoding 
 430 bool wxNativeFontInfo::FromString(const wxString
& s
) 
 434     wxStringTokenizer 
tokenizer(s
, _T(";")); 
 436     wxString token 
= tokenizer
.GetNextToken(); 
 438     //  Ignore the version for now 
 441     token 
= tokenizer
.GetNextToken(); 
 442     if ( !token
.ToLong(&l
) ) 
 446     token 
= tokenizer
.GetNextToken(); 
 447     if ( !token
.ToLong(&l
) ) 
 449     family 
= (wxFontFamily
)l
; 
 451     token 
= tokenizer
.GetNextToken(); 
 452     if ( !token
.ToLong(&l
) ) 
 454     style 
= (wxFontStyle
)l
; 
 456     token 
= tokenizer
.GetNextToken(); 
 457     if ( !token
.ToLong(&l
) ) 
 459     weight 
= (wxFontWeight
)l
; 
 461     token 
= tokenizer
.GetNextToken(); 
 462     if ( !token
.ToLong(&l
) ) 
 466     faceName 
= tokenizer
.GetNextToken(); 
 473     token 
= tokenizer
.GetNextToken(); 
 474     if ( !token
.ToLong(&l
) ) 
 476     encoding 
= (wxFontEncoding
)l
; 
 481 wxString 
wxNativeFontInfo::ToString() const 
 485     s
.Printf(_T("%d;%d;%d;%d;%d;%d;%s;%d"), 
 498 void wxNativeFontInfo::Init() 
 501     family 
= wxFONTFAMILY_DEFAULT
; 
 502     style 
= wxFONTSTYLE_NORMAL
; 
 503     weight 
= wxFONTWEIGHT_NORMAL
; 
 506     encoding 
= wxFONTENCODING_DEFAULT
; 
 509 int wxNativeFontInfo::GetPointSize() const 
 514 wxFontStyle 
wxNativeFontInfo::GetStyle() const 
 519 wxFontWeight 
wxNativeFontInfo::GetWeight() const 
 524 bool wxNativeFontInfo::GetUnderlined() const 
 529 wxString 
wxNativeFontInfo::GetFaceName() const 
 534 wxFontFamily 
wxNativeFontInfo::GetFamily() const 
 539 wxFontEncoding 
wxNativeFontInfo::GetEncoding() const 
 544 void wxNativeFontInfo::SetPointSize(int pointsize
) 
 546     pointSize 
= pointsize
; 
 549 void wxNativeFontInfo::SetStyle(wxFontStyle style_
) 
 554 void wxNativeFontInfo::SetWeight(wxFontWeight weight_
) 
 559 void wxNativeFontInfo::SetUnderlined(bool underlined_
) 
 561     underlined 
= underlined_
; 
 564 bool wxNativeFontInfo::SetFaceName(const wxString
& facename_
) 
 566     faceName 
= facename_
; 
 570 void wxNativeFontInfo::SetFamily(wxFontFamily family_
) 
 575 void wxNativeFontInfo::SetEncoding(wxFontEncoding encoding_
) 
 577     encoding 
= encoding_
; 
 580 #endif // generic wxNativeFontInfo implementation 
 582 // conversion to/from user-readable string: this is used in the generic 
 583 // versions and under MSW as well because there is no standard font description 
 584 // format there anyhow (but there is a well-defined standard for X11 fonts used 
 585 // by wxGTK and wxMotif) 
 587 #if defined(wxNO_NATIVE_FONTINFO) || defined(__WXMSW__) || defined (__WXPM__) 
 589 wxString 
wxNativeFontInfo::ToUserString() const 
 593     // first put the adjectives, if any - this is English-centric, of course, 
 594     // but what else can we do? 
 595     if ( GetUnderlined() ) 
 597         desc 
<< _("underlined"); 
 600     switch ( GetWeight() ) 
 603             wxFAIL_MSG( _T("unknown font weight") ); 
 606         case wxFONTWEIGHT_NORMAL
: 
 609         case wxFONTWEIGHT_LIGHT
: 
 613         case wxFONTWEIGHT_BOLD
: 
 618     switch ( GetStyle() ) 
 621             wxFAIL_MSG( _T("unknown font style") ); 
 624         case wxFONTSTYLE_NORMAL
: 
 627             // we don't distinguish between the two for now anyhow... 
 628         case wxFONTSTYLE_ITALIC
: 
 629         case wxFONTSTYLE_SLANT
: 
 630             desc 
<< _(" italic"); 
 634     wxString face 
= GetFaceName(); 
 637         desc 
<< _T(' ') << face
; 
 640     int size 
= GetPointSize(); 
 641     if ( size 
!= wxNORMAL_FONT
->GetPointSize() ) 
 643         desc 
<< _T(' ') << size
; 
 647     wxFontEncoding enc 
= GetEncoding(); 
 648     if ( enc 
!= wxFONTENCODING_DEFAULT 
&& enc 
!= wxFONTENCODING_SYSTEM 
) 
 650         desc 
<< _T(' ') << wxFontMapper::GetEncodingName(enc
); 
 652 #endif // wxUSE_FONTMAP 
 654     return desc
.Strip(wxString::both
).MakeLower(); 
 657 bool wxNativeFontInfo::FromUserString(const wxString
& s
) 
 659     // reset to the default state 
 662     // parse a more or less free form string 
 664     // TODO: we should handle at least the quoted facenames 
 665     wxStringTokenizer 
tokenizer(s
, _T(";, "), wxTOKEN_STRTOK
); 
 669     bool weightfound 
= false, pointsizefound 
= false, encodingfound 
= false; 
 671     while ( tokenizer
.HasMoreTokens() ) 
 673         wxString token 
= tokenizer
.GetNextToken(); 
 676         token
.Trim(true).Trim(false).MakeLower(); 
 678         // look for the known tokens 
 679         if ( token 
== _T("underlined") || token 
== _("underlined") ) 
 683         else if ( token 
== _T("light") || token 
== _("light") ) 
 685             SetWeight(wxFONTWEIGHT_LIGHT
); 
 688         else if ( token 
== _T("bold") || token 
== _("bold") ) 
 690             SetWeight(wxFONTWEIGHT_BOLD
); 
 693         else if ( token 
== _T("italic") || token 
== _("italic") ) 
 695             SetStyle(wxFONTSTYLE_ITALIC
); 
 697         else if ( token
.ToULong(&size
) ) 
 700             pointsizefound 
= true; 
 705             // try to interpret this as an encoding 
 706             wxFontEncoding encoding 
= wxFontMapper::Get()->CharsetToEncoding(token
, false); 
 707             if ( encoding 
!= wxFONTENCODING_DEFAULT 
&& 
 708                  encoding 
!= wxFONTENCODING_SYSTEM 
)    // returned when the recognition failed 
 710             SetEncoding(encoding
); 
 711                 encodingfound 
= true; 
 715 #endif // wxUSE_FONTMAP 
 717                 // assume it is the face name 
 725             // skip the code which resets face below 
 730 #endif // wxUSE_FONTMAP 
 733         // if we had had the facename, we shouldn't continue appending tokens 
 734         // to it (i.e. "foo bold bar" shouldn't result in the facename "foo 
 738             // NB: the check on the facename is implemented in wxFontBase::SetFaceName 
 739             //     and not in wxNativeFontInfo::SetFaceName thus we need to explicitely 
 740             //     call here wxFontEnumerator::IsValidFacename 
 741             if (!wxFontEnumerator::IsValidFacename(face
) || 
 743                 SetFaceName(wxNORMAL_FONT
->GetFaceName()); 
 748     // we might not have flushed it inside the loop 
 751         // NB: the check on the facename is implemented in wxFontBase::SetFaceName 
 752         //     and not in wxNativeFontInfo::SetFaceName thus we need to explicitely 
 753         //     call here wxFontEnumerator::IsValidFacename 
 754         if (!wxFontEnumerator::IsValidFacename(face
) || 
 756             SetFaceName(wxNORMAL_FONT
->GetFaceName()); 
 759     // set point size to default value if size was not given 
 760     if ( !pointsizefound 
) 
 761         SetPointSize(wxNORMAL_FONT
->GetPointSize()); 
 763     // set font weight to default value if weight was not given 
 765         SetWeight(wxFONTWEIGHT_NORMAL
); 
 768     // set font encoding to default value if encoding was not given 
 769     if ( !encodingfound 
) 
 770         SetEncoding(wxFONTENCODING_SYSTEM
); 
 771 #endif // wxUSE_FONTMAP 
 776 #endif // generic or wxMSW or wxOS2