1 /////////////////////////////////////////////////////////////////////////////
 
   2 // Name:        src/cocoa/font.cpp
 
   3 // Purpose:     wxFont class
 
   8 // Copyright:   (c) AUTHOR
 
   9 // Licence:     wxWindows licence
 
  10 /////////////////////////////////////////////////////////////////////////////
 
  14     Cocoa has three classes which interact to form the font system:
 
  16     NSFont: Represents a specific font (e.g. Times-Bold) at a specific size
 
  17     with specific attributes.  That is, it's basically like this class.
 
  18     Notably, it doesn't hold an underlined flag, so this class does.
 
  19     Available on all OS X versions.
 
  21     NSFontManager: Fairly broad controller class which ties together the
 
  22     model (NSFont) with the view (NSFontPanel).  We are ignoring NSFontPanel
 
  23     in this discussion.  NSFontManager is actually a little broader than just
 
  24     a controller though.  It's also the traditional way of mutating fonts
 
  25     and asking for a font with a certain family and certain attributes.
 
  26     For example, you can use NSFont's factor methods to create a font named
 
  27     @"Times-Roman" then use NSFontManager to imbue attributes like italic and
 
  28     bold.  You might do this if, for instance, you already have times at the
 
  29     Roman weight but want to make it bold.
 
  31     Alternatively, you can use NSFontManager to ask for a font in the @"Times"
 
  32     family with the bold attribute.
 
  34     NSFontManager is available on all OS X versions.
 
  36     NSFontDescriptor: Added in OS X 10.3.  Prior to this there was no specific
 
  37     class to represent all of the attributes of a font.  Instead, a regular
 
  38     NSDictionary was used with a set of well-defined keys.  Unfortunately,
 
  39     there was no method accepting the attributes dictionary, only a method
 
  40     to retrieve it from an NSFont.  That meant that in order to create
 
  41     a new font by imbueing certain attributes like Bold one would need
 
  42     to use the specific method in NSFontManager to do so.
 
  44     The NSFontDescriptor class, on the other hand, has factory methods which
 
  45     can create a new font descriptor with an attributes dictionary as well
 
  46     as mutate (by copy) an existing font descriptor using attributes from
 
  47     an attributes dictionary.
 
  49     In theory, most of what can be done using NSFontDescriptor can just as
 
  50     well be done without it.  NSFontDescriptor is basically just a shell
 
  51     around an NSMutableDictionary with well-defined keys.
 
  54     Getting back to the broad overview, font matching is one of the weaker points
 
  55     in Cocoa's API and NSFontDescriptor is the easier to use solution.
 
  57     That said, it's not impossible to implement font matching without it. For instance,
 
  58     if you have a family name and want to list available variants (e.g. Bold, italic,
 
  59     underlined) then you can ask NSFontManager for availableMembersOfFontFamily:.
 
  61     The problem is that you can either match on family name or on font attributes
 
  62     but not on both.  To match both you have to do one then the other.
 
  63     NSFontDescriptor allows you to get a list of fonts matching both a family name
 
  64     and a particular set of attributes.  Furthermore, the attributes instead of
 
  65     being flags as in NSFontManager are instead well-defined keys in a dictionary.
 
  67     The only way to get that behavior without NSFontManager is to pare down the
 
  68     list as much as possible using the classic NSFontManager methods and then
 
  69     to instantiate each font in the list and match on each font's afmDictionary.
 
  71     A reasonable guess is that that's probably more or less exactly what
 
  72     NSFontDescriptor does internally.
 
  74 #include "wx/wxprec.h"
 
  79     #include "wx/string.h"
 
  80     #include "wx/gdicmn.h"
 
  83 #include "wx/fontutil.h"
 
  84 #include "wx/encinfo.h"
 
  86 #include "wx/cocoa/string.h"
 
  87 #include "wx/cocoa/private/fontfactory.h"
 
  88 #include "wx/cocoa/autorelease.h"
 
  90 #include <AppKit/NSFont.h>
 
  91 #include <AppKit/NSFontManager.h>
 
  93 // Helper methods for NSFont/wxNativeFontInfo
 
  94 static NSFont* GetNSFontForNativeFontInfo(const wxNativeFontInfo &info);
 
  95 static void UpdateNativeFontInfoWithNSFont(wxNativeFontInfo &info, NSFont *cocoaNSFont);
 
  96 static wxNativeFontInfo MakeNativeFontInfoForNSFont(NSFont *cocoaNSFont, bool underlined = false);
 
  97 static wxNativeFontInfo MakeNativeFontInfo(int size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding);
 
 100     Due to 2.8 ABI compatibility concerns we probably don't want to change wxNativeFontInfo
 
 101     although this may be unfounded because this class is supposed to be internal as is
 
 102     wxNativeFontInfo so anyone who subclassed it or created one without going through
 
 103     wxFont should expect what they get (i.e. horrible breakage)
 
 104     There's a concern that wxFontRefData was in the public header when 2.8 shipped so
 
 105     it's possible that someone did subclass it to get better font behavior.
 
 107     For right now, the plan is to write it strictly ABI compatible with 2.8 and eventually
 
 108     to enhance it in trunk to accurately represent font attributes as Cocoa sees them.
 
 110     I'd like to let at least one 2.8 release go out the door and get feedback as to whether
 
 111     this is going to be a problem or not.  If so, we'll keep it strictly ABI compatible.
 
 112     If not, we'll update it.
 
 114 class WXDLLEXPORT wxFontRefData: public wxGDIRefData
 
 116     friend class WXDLLIMPEXP_FWD_CORE wxFont;
 
 120     ,   m_info(MakeNativeFontInfo(10, wxDEFAULT, wxNORMAL, wxNORMAL, FALSE,
 
 121              wxT("Geneva"), wxFONTENCODING_DEFAULT))
 
 123         CreateNSFontAndUpdateInfo();
 
 126     wxFontRefData(const wxFontRefData& data)
 
 128     ,   m_cocoaNSFont([data.m_cocoaNSFont retain])
 
 129     ,   m_info(data.m_info)
 
 133     wxFontRefData(NSFont *cocoaNSFont, bool underlined)
 
 135     ,   m_cocoaNSFont([cocoaNSFont retain])
 
 136     ,   m_info(MakeNativeFontInfoForNSFont(m_cocoaNSFont, underlined))
 
 140     wxFontRefData(const wxNativeFontInfo& info)
 
 145         CreateNSFontAndUpdateInfo();
 
 148     wxFontRefData(int size,
 
 153                   const wxString& faceName,
 
 154                   wxFontEncoding encoding)
 
 156     ,   m_info(MakeNativeFontInfo(size, family, style, weight, underlined, faceName, encoding))
 
 158         CreateNSFontAndUpdateInfo();
 
 161     virtual ~wxFontRefData();
 
 163     /*! @abstract   common part of some ctors
 
 165         This is a leftover of sorts from the old stub code.
 
 166         FIXME: Remove from trunk
 
 173               const wxString& faceName,
 
 174               wxFontEncoding encoding);
 
 177         Uses the native font info to create an NSFont and then updates that info with
 
 178         the attributes of the font.  This is necessary because, for example, a font
 
 179         can be created with an empty faceName in which case some concrete typeface must
 
 181         We choose to handle this by first initializing the wxNativeFontInfo with the
 
 182         properties as given by the user and then creating the NSFont and updating
 
 183         the wxNativeFontInfo to match the NSFont.
 
 185     void CreateNSFontAndUpdateInfo()
 
 187         wxAutoNSAutoreleasePool pool;
 
 188         [m_cocoaNSFont release];
 
 189         m_cocoaNSFont = [GetNSFontForNativeFontInfo(m_info) retain];
 
 190         UpdateNativeFontInfoWithNSFont(m_info, m_cocoaNSFont);
 
 193     // font characterstics
 
 194     NSFont *m_cocoaNSFont;
 
 195     wxNativeFontInfo m_info;
 
 200 NSString *GetFamilyName(wxFontFamily family)
 
 204     case wxFONTFAMILY_DEFAULT:
 
 207     case wxFONTFAMILY_DECORATIVE:
 
 208     case wxFONTFAMILY_ROMAN:
 
 209     case wxFONTFAMILY_SCRIPT:
 
 211     case wxFONTFAMILY_SWISS:
 
 212         return @"Lucida Grande";
 
 213     case wxFONTFAMILY_MODERN:
 
 214     case wxFONTFAMILY_TELETYPE:
 
 218 // Returns an NSFont given the native font info
 
 219 // NOTE: It is not considered alloc'd (i.e. this is a get method not an alloc/new method)
 
 220 static NSFont* GetNSFontForNativeFontInfo(const wxNativeFontInfo &info)
 
 222     if(!info.faceName.empty())
 
 224         NSFont *font = [NSFont fontWithName:wxNSStringWithWxString(info.faceName) size:info.pointSize];
 
 225         // TODO: use NSFontManager to mix in the weights and whatnot
 
 228         // To err or not to err?
 
 230     // No font with that face name or no face name
 
 232     NSFontTraitMask cocoaTraits = 0;
 
 234     NSFont *font = [[NSFontManager sharedFontManager] fontWithFamily:GetFamilyName(info.family) traits:cocoaTraits weight:cocoaWeight size:info.pointSize];
 
 239     Updates all fields of @a info except for underlined which is not a property of NSFont.
 
 241 static void UpdateNativeFontInfoWithNSFont(wxNativeFontInfo &info, NSFont *cocoaNSFont)
 
 243     info.pointSize = [cocoaNSFont pointSize];
 
 245     // FIXME: We could maybe improve on this?
 
 246     info.family = wxFONTFAMILY_DEFAULT;
 
 248     // FIXME: italicAngle might indicate a slanted rather than truly italic font?
 
 249     info.style = [cocoaNSFont italicAngle] == 0.0?wxFONTSTYLE_NORMAL:wxFONTSTYLE_ITALIC;
 
 251     int cocoaWeight = [[NSFontManager sharedFontManager] weightOfFont:cocoaNSFont];
 
 253         info.weight = wxFONTWEIGHT_LIGHT;
 
 254     else if(cocoaWeight < 9)
 
 255         info.weight = wxFONTWEIGHT_NORMAL;
 
 257         info.weight = wxFONTWEIGHT_BOLD;
 
 259     // FIXME: Is this right?  I think so.
 
 260     info.faceName = wxStringWithNSString([cocoaNSFont fontName]);
 
 262     // TODO: Translate NSStringEncoding to wxFontEncoding
 
 263     info.encoding = wxFONTENCODING_SYSTEM;
 
 267     Creates a new generic wxNativeFontInfo from an NSFont and an underlined flag.
 
 268     Uses UpdateNativeFontInfoWithNSFont to do the work and sets the underlined field
 
 269     of wxNativeFontInfo to the @a underlined argument.
 
 271 static wxNativeFontInfo MakeNativeFontInfoForNSFont(NSFont *cocoaNSFont, bool underlined)
 
 273     wxNativeFontInfo info;
 
 274     UpdateNativeFontInfoWithNSFont(info, cocoaNSFont);
 
 276     // NSFont are never underlined.. that's a function of the drawing system
 
 277     info.underlined = underlined;
 
 282 //#include "_font_test_2_8_abi_compat.h"
 
 284 IMPLEMENT_DYNAMIC_CLASS(wxFont, wxGDIObject)
 
 286 static wxNativeFontInfo MakeNativeFontInfo(int size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
 
 288     wxNativeFontInfo m_info; // NOTE: not an i-var despite name
 
 289     m_info.pointSize = size;
 
 290     m_info.family = static_cast<wxFontFamily>(family);
 
 291     m_info.style = static_cast<wxFontStyle>(style);
 
 292     m_info.weight = static_cast<wxFontWeight>(weight);
 
 293     m_info.underlined = underlined;
 
 294     m_info.faceName = faceName;
 
 295     m_info.encoding = encoding;
 
 299 void wxFontRefData::Init(int size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
 
 301     m_info = MakeNativeFontInfo(size, family, style, weight, underlined, faceName, encoding);
 
 304 wxFontRefData::~wxFontRefData()
 
 306     [m_cocoaNSFont release];
 
 308     // TODO: delete font data
 
 311 #define M_FONTDATA ((wxFontRefData*)m_refData)
 
 313 wxFont wxCocoaFontFactory::InstanceForNSFont(WX_NSFont cocoaFont, bool underlined)
 
 315     return wxFont(new wxFontRefData(cocoaFont, underlined));
 
 318 bool wxFont::Create(wxFontRefData *refData)
 
 323     return m_refData != NULL;
 
 326 bool wxFont::Create(const wxNativeFontInfo& nativeFontInfo)
 
 329     m_refData = new wxFontRefData(nativeFontInfo);
 
 334 wxGDIRefData *wxFont::CreateGDIRefData() const
 
 336     return new wxFontRefData;
 
 339 wxGDIRefData *wxFont::CloneGDIRefData(const wxGDIRefData *data) const
 
 341     return new wxFontRefData(*static_cast<const wxFontRefData *>(data));
 
 344 void wxFont::SetEncoding(wxFontEncoding)
 
 348 wxFontEncoding wxFont::GetEncoding() const
 
 350     return wxFontEncoding();
 
 353 int wxFont::GetPointSize() const
 
 355     wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
 
 356     return M_FONTDATA->m_info.pointSize;
 
 359 bool wxFont::GetUnderlined() const
 
 362         return M_FONTDATA->m_info.underlined;
 
 367 wxFontStyle wxFont::GetStyle() const
 
 369     wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
 
 370     return M_FONTDATA->m_info.style;
 
 373 wxFontFamily wxFont::GetFamily() const
 
 375     wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
 
 376     return M_FONTDATA->m_info.family;
 
 379 wxFontWeight wxFont::GetWeight() const
 
 381     wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
 
 382     return M_FONTDATA->m_info.weight;
 
 385 const wxNativeFontInfo *wxFont::GetNativeFontInfo() const
 
 387     wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
 
 388     return &M_FONTDATA->m_info;
 
 391 bool wxFont::Create(int pointSize, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
 
 394     m_refData = new wxFontRefData(pointSize, family, style, weight, underlined, faceName, encoding);
 
 405 bool wxFont::RealizeResource()
 
 407     // TODO: create the font (if there is a native font object)
 
 411 void wxFont::SetPointSize(int pointSize)
 
 415     M_FONTDATA->m_info.pointSize = pointSize;
 
 420 void wxFont::SetFamily(wxFontFamily family)
 
 424     M_FONTDATA->m_info.family = static_cast<wxFontFamily>(family);
 
 429 void wxFont::SetStyle(wxFontStyle style)
 
 433     M_FONTDATA->m_info.style = static_cast<wxFontStyle>(style);
 
 438 void wxFont::SetWeight(wxFontWeight weight)
 
 442     M_FONTDATA->m_info.weight = static_cast<wxFontWeight>(weight);
 
 447 bool wxFont::SetFaceName(const wxString& faceName)
 
 451     M_FONTDATA->m_info.faceName = faceName;
 
 455     return wxFontBase::SetFaceName(faceName);
 
 458 void wxFont::SetUnderlined(bool underlined)
 
 462     M_FONTDATA->m_info.underlined = underlined;
 
 467 /* New font system */
 
 468 wxString wxFont::GetFaceName() const
 
 472         str = M_FONTDATA->m_info.faceName;