1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/cocoa/font.mm
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 static wxNativeFontInfo MakeNativeFontInfo(int size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
286 wxNativeFontInfo m_info; // NOTE: not an i-var despite name
287 m_info.pointSize = size;
288 m_info.family = static_cast<wxFontFamily>(family);
289 m_info.style = static_cast<wxFontStyle>(style);
290 m_info.weight = static_cast<wxFontWeight>(weight);
291 m_info.underlined = underlined;
292 m_info.faceName = faceName;
293 m_info.encoding = encoding;
297 void wxFontRefData::Init(int size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
299 m_info = MakeNativeFontInfo(size, family, style, weight, underlined, faceName, encoding);
302 wxFontRefData::~wxFontRefData()
304 [m_cocoaNSFont release];
306 // TODO: delete font data
309 #define M_FONTDATA ((wxFontRefData*)m_refData)
311 wxFont wxCocoaFontFactory::InstanceForNSFont(WX_NSFont cocoaFont, bool underlined)
313 return wxFont(new wxFontRefData(cocoaFont, underlined));
316 bool wxFont::Create(wxFontRefData *refData)
321 return m_refData != NULL;
324 bool wxFont::Create(const wxNativeFontInfo& nativeFontInfo)
327 m_refData = new wxFontRefData(nativeFontInfo);
332 wxGDIRefData *wxFont::CreateGDIRefData() const
334 return new wxFontRefData;
337 wxGDIRefData *wxFont::CloneGDIRefData(const wxGDIRefData *data) const
339 return new wxFontRefData(*static_cast<const wxFontRefData *>(data));
342 void wxFont::SetEncoding(wxFontEncoding)
346 wxFontEncoding wxFont::GetEncoding() const
348 return wxFontEncoding();
351 int wxFont::GetPointSize() const
353 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
354 return M_FONTDATA->m_info.pointSize;
357 bool wxFont::GetUnderlined() const
360 return M_FONTDATA->m_info.underlined;
365 wxFontStyle wxFont::GetStyle() const
367 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
368 return M_FONTDATA->m_info.style;
371 wxFontFamily wxFont::DoGetFamily() const
373 return M_FONTDATA->m_info.family;
376 wxFontWeight wxFont::GetWeight() const
378 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
379 return M_FONTDATA->m_info.weight;
382 const wxNativeFontInfo *wxFont::GetNativeFontInfo() const
384 wxCHECK_MSG( Ok(), 0, wxT("invalid font") );
385 return &M_FONTDATA->m_info;
388 bool wxFont::Create(int pointSize, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underlined, const wxString& faceName, wxFontEncoding encoding)
391 m_refData = new wxFontRefData(pointSize, family, style, weight, underlined, faceName, encoding);
402 bool wxFont::RealizeResource()
404 // TODO: create the font (if there is a native font object)
408 void wxFont::SetPointSize(int pointSize)
412 M_FONTDATA->m_info.pointSize = pointSize;
417 void wxFont::SetFamily(wxFontFamily family)
421 M_FONTDATA->m_info.family = static_cast<wxFontFamily>(family);
426 void wxFont::SetStyle(wxFontStyle style)
430 M_FONTDATA->m_info.style = static_cast<wxFontStyle>(style);
435 void wxFont::SetWeight(wxFontWeight weight)
439 M_FONTDATA->m_info.weight = static_cast<wxFontWeight>(weight);
444 bool wxFont::SetFaceName(const wxString& faceName)
448 M_FONTDATA->m_info.faceName = faceName;
452 return wxFontBase::SetFaceName(faceName);
455 void wxFont::SetUnderlined(bool underlined)
459 M_FONTDATA->m_info.underlined = underlined;
464 /* New font system */
465 wxString wxFont::GetFaceName() const
469 str = M_FONTDATA->m_info.faceName;