WebCore:
[WebKit-https.git] / WebCore / platform / graphics / mac / SimpleFontDataMac.mm
1 /*
2  * Copyright (C) 2005, 2006 Apple Computer, Inc.  All rights reserved.
3  * Copyright (C) 2006 Alexey Proskuryakov
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *
9  * 1.  Redistributions of source code must retain the above copyright
10  *     notice, this list of conditions and the following disclaimer. 
11  * 2.  Redistributions in binary form must reproduce the above copyright
12  *     notice, this list of conditions and the following disclaimer in the
13  *     documentation and/or other materials provided with the distribution. 
14  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
15  *     its contributors may be used to endorse or promote products derived
16  *     from this software without specific prior written permission. 
17  *
18  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
19  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
22  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29
30 #import "config.h"
31 #import "SimpleFontData.h"
32
33 #import "BlockExceptions.h"
34 #import "Color.h"
35 #import "FloatRect.h"
36 #import "Font.h"
37 #import "FontCache.h"
38 #import "FontDescription.h"
39 #import "SharedBuffer.h"
40 #import "WebCoreSystemInterface.h"
41 #import <ApplicationServices/ApplicationServices.h>
42 #import <float.h>
43 #import <unicode/uchar.h>
44 #import <wtf/Assertions.h>
45 #import <wtf/RetainPtr.h>
46
47 @interface NSFont (WebAppKitSecretAPI)
48 - (BOOL)_isFakeFixedPitch;
49 @end
50
51 namespace WebCore {
52   
53 const float smallCapsFontSizeMultiplier = 0.7f;
54 const float contextDPI = 72.0f;
55 static inline float scaleEmToUnits(float x, unsigned unitsPerEm) { return x * (contextDPI / (contextDPI * unitsPerEm)); }
56
57 bool initFontData(SimpleFontData* fontData)
58 {
59     if (!fontData->m_font.m_cgFont)
60         return false;
61
62     ATSUStyle fontStyle;
63     if (ATSUCreateStyle(&fontStyle) != noErr)
64         return false;
65     
66     ATSUFontID fontId = fontData->m_font.m_atsuFontID;
67     if (!fontId) {
68         ATSUDisposeStyle(fontStyle);
69         return false;
70     }
71
72     ATSUAttributeTag tag = kATSUFontTag;
73     ByteCount size = sizeof(ATSUFontID);
74     ATSUFontID *valueArray[1] = {&fontId};
75     OSStatus status = ATSUSetAttributes(fontStyle, 1, &tag, &size, (void* const*)valueArray);
76     if (status != noErr) {
77         ATSUDisposeStyle(fontStyle);
78         return false;
79     }
80
81     if (wkGetATSStyleGroup(fontStyle, &fontData->m_styleGroup) != noErr) {
82         ATSUDisposeStyle(fontStyle);
83         return false;
84     }
85
86     ATSUDisposeStyle(fontStyle);
87
88     return true;
89 }
90
91 static NSString *webFallbackFontFamily(void)
92 {
93     static RetainPtr<NSString> webFallbackFontFamily = nil;
94     if (!webFallbackFontFamily)
95         webFallbackFontFamily = [[NSFont systemFontOfSize:16.0f] familyName];
96     return webFallbackFontFamily.get();
97 }
98
99 void SimpleFontData::platformInit()
100 {
101     m_styleGroup = 0;
102     m_ATSUStyleInitialized = false;
103     m_ATSUMirrors = false;
104     m_checkedShapesArabic = false;
105     m_shapesArabic = false;
106
107     m_syntheticBoldOffset = m_font.m_syntheticBold ? 1.0f : 0.f;
108     
109     bool failedSetup = false;
110     if (!initFontData(this)) {
111         // Ack! Something very bad happened, like a corrupt font.
112         // Try looking for an alternate 'base' font for this renderer.
113
114         // Special case hack to use "Times New Roman" in place of "Times".
115         // "Times RO" is a common font whose family name is "Times".
116         // It overrides the normal "Times" family font.
117         // It also appears to have a corrupt regular variant.
118         NSString *fallbackFontFamily;
119         if ([[m_font.font() familyName] isEqual:@"Times"])
120             fallbackFontFamily = @"Times New Roman";
121         else
122             fallbackFontFamily = webFallbackFontFamily();
123         
124         // Try setting up the alternate font.
125         // This is a last ditch effort to use a substitute font when something has gone wrong.
126 #if !ERROR_DISABLED
127         RetainPtr<NSFont> initialFont = m_font.font();
128 #endif
129         m_font.setFont([[NSFontManager sharedFontManager] convertFont:m_font.font() toFamily:fallbackFontFamily]);
130 #if !ERROR_DISABLED
131         NSString *filePath = wkPathFromFont(initialFont.get());
132         if (!filePath)
133             filePath = @"not known";
134 #endif
135         if (!initFontData(this)) {
136             if ([fallbackFontFamily isEqual:@"Times New Roman"]) {
137                 // OK, couldn't setup Times New Roman as an alternate to Times, fallback
138                 // on the system font.  If this fails we have no alternative left.
139                 m_font.setFont([[NSFontManager sharedFontManager] convertFont:m_font.font() toFamily:webFallbackFontFamily()]);
140                 if (!initFontData(this)) {
141                     // We tried, Times, Times New Roman, and the system font. No joy. We have to give up.
142                     LOG_ERROR("unable to initialize with font %@ at %@", initialFont.get(), filePath);
143                     failedSetup = true;
144                 }
145             } else {
146                 // We tried the requested font and the system font. No joy. We have to give up.
147                 LOG_ERROR("unable to initialize with font %@ at %@", initialFont.get(), filePath);
148                 failedSetup = true;
149             }
150         }
151
152         // Report the problem.
153         LOG_ERROR("Corrupt font detected, using %@ in place of %@ located at \"%@\".",
154             [m_font.font() familyName], [initialFont.get() familyName], filePath);
155     }
156
157     // If all else fails, try to set up using the system font.
158     // This is probably because Times and Times New Roman are both unavailable.
159     if (failedSetup) {
160         m_font.setFont([NSFont systemFontOfSize:[m_font.font() pointSize]]);
161         LOG_ERROR("failed to set up font, using system font %s", m_font.font());
162         initFontData(this);
163     }
164     
165     int iAscent;
166     int iDescent;
167     int iLineGap;
168     wkGetFontMetrics(m_font.m_cgFont, &iAscent, &iDescent, &iLineGap, &m_unitsPerEm); 
169     float pointSize = m_font.m_size;
170     float fAscent = scaleEmToUnits(iAscent, m_unitsPerEm) * pointSize;
171     float fDescent = -scaleEmToUnits(iDescent, m_unitsPerEm) * pointSize;
172     float fLineGap = scaleEmToUnits(iLineGap, m_unitsPerEm) * pointSize;
173
174     // We need to adjust Times, Helvetica, and Courier to closely match the
175     // vertical metrics of their Microsoft counterparts that are the de facto
176     // web standard. The AppKit adjustment of 20% is too big and is
177     // incorrectly added to line spacing, so we use a 15% adjustment instead
178     // and add it to the ascent.
179     NSString *familyName = [m_font.font() familyName];
180     if ([familyName isEqualToString:@"Times"] || [familyName isEqualToString:@"Helvetica"] || [familyName isEqualToString:@"Courier"])
181         fAscent += floorf(((fAscent + fDescent) * 0.15f) + 0.5f);
182
183     m_ascent = lroundf(fAscent);
184     m_descent = lroundf(fDescent);
185     m_lineGap = lroundf(fLineGap);
186     m_lineSpacing = m_ascent + m_descent + m_lineGap;
187     
188     // Hack Hiragino line metrics to allow room for marked text underlines.
189     // <rdar://problem/5386183>
190     if (m_descent < 3 && m_lineGap >= 3 && [familyName hasPrefix:@"Hiragino"]) {
191         m_lineGap -= 3 - m_descent;
192         m_descent = 3;
193     }
194     
195     // Measure the actual character "x", because AppKit synthesizes X height rather than getting it from the font.
196     // Unfortunately, NSFont will round this for us so we don't quite get the right value.
197     GlyphPage* glyphPageZero = GlyphPageTreeNode::getRootChild(this, 0)->page();
198     NSGlyph xGlyph = glyphPageZero ? glyphPageZero->glyphDataForCharacter('x').glyph : 0;
199     if (xGlyph) {
200         NSRect xBox = [m_font.font() boundingRectForGlyph:xGlyph];
201         // Use the maximum of either width or height because "x" is nearly square
202         // and web pages that foolishly use this metric for width will be laid out
203         // poorly if we return an accurate height. Classic case is Times 13 point,
204         // which has an "x" that is 7x6 pixels.
205         m_xHeight = MAX(NSMaxX(xBox), NSMaxY(xBox));
206     } else
207         m_xHeight = [m_font.font() xHeight];
208 }
209
210 void SimpleFontData::platformDestroy()
211 {
212     if (m_styleGroup)
213         wkReleaseStyleGroup(m_styleGroup);
214
215     if (m_ATSUStyleInitialized)
216         ATSUDisposeStyle(m_ATSUStyle);
217 }
218
219 SimpleFontData* SimpleFontData::smallCapsFontData(const FontDescription& fontDescription) const
220 {
221     if (!m_smallCapsFontData) {
222         if (isCustomFont()) {
223             FontPlatformData smallCapsFontData(m_font);
224             smallCapsFontData.m_size = smallCapsFontData.m_size * smallCapsFontSizeMultiplier;
225             m_smallCapsFontData = new SimpleFontData(smallCapsFontData, true, false);
226         } else {
227             BEGIN_BLOCK_OBJC_EXCEPTIONS;
228             float size = [m_font.font() pointSize] * smallCapsFontSizeMultiplier;
229             FontPlatformData smallCapsFont([[NSFontManager sharedFontManager] convertFont:m_font.font() toSize:size]);
230             
231             // AppKit resets the type information (screen/printer) when you convert a font to a different size.
232             // We have to fix up the font that we're handed back.
233             smallCapsFont.setFont(fontDescription.usePrinterFont() ? [smallCapsFont.font() printerFont] : [smallCapsFont.font() screenFont]);
234
235             if (smallCapsFont.font()) {
236                 NSFontManager *fontManager = [NSFontManager sharedFontManager];
237                 NSFontTraitMask fontTraits = [fontManager traitsOfFont:m_font.font()];
238
239                 if (m_font.m_syntheticBold)
240                     fontTraits |= NSBoldFontMask;
241                 if (m_font.m_syntheticOblique)
242                     fontTraits |= NSItalicFontMask;
243
244                 NSFontTraitMask smallCapsFontTraits = [fontManager traitsOfFont:smallCapsFont.font()];
245                 smallCapsFont.m_syntheticBold = (fontTraits & NSBoldFontMask) && !(smallCapsFontTraits & NSBoldFontMask);
246                 smallCapsFont.m_syntheticOblique = (fontTraits & NSItalicFontMask) && !(smallCapsFontTraits & NSItalicFontMask);
247
248                 m_smallCapsFontData = FontCache::getCachedFontData(&smallCapsFont);
249             }
250             END_BLOCK_OBJC_EXCEPTIONS;
251         }
252     }
253     return m_smallCapsFontData;
254 }
255
256 bool SimpleFontData::containsCharacters(const UChar* characters, int length) const
257 {
258     NSString *string = [[NSString alloc] initWithCharactersNoCopy:(UniChar*)characters length:length freeWhenDone:NO];
259     NSCharacterSet *set = [[m_font.font() coveredCharacterSet] invertedSet];
260     bool result = set && [string rangeOfCharacterFromSet:set].location == NSNotFound;
261     [string release];
262     return result;
263 }
264
265 void SimpleFontData::determinePitch()
266 {
267     NSFont* f = m_font.font();
268     // Special case Osaka-Mono.
269     // According to <rdar://problem/3999467>, we should treat Osaka-Mono as fixed pitch.
270     // Note that the AppKit does not report Osaka-Mono as fixed pitch.
271
272     // Special case MS-PGothic.
273     // According to <rdar://problem/4032938>, we should not treat MS-PGothic as fixed pitch.
274     // Note that AppKit does report MS-PGothic as fixed pitch.
275
276     // Special case MonotypeCorsiva
277     // According to <rdar://problem/5454704>, we should not treat MonotypeCorsiva as fixed pitch.
278     // Note that AppKit does report MonotypeCorsiva as fixed pitch.
279
280     NSString *name = [f fontName];
281     m_treatAsFixedPitch = ([f isFixedPitch] || [f _isFakeFixedPitch] ||
282            [name caseInsensitiveCompare:@"Osaka-Mono"] == NSOrderedSame) &&
283            [name caseInsensitiveCompare:@"MS-PGothic"] != NSOrderedSame &&
284            [name caseInsensitiveCompare:@"MonotypeCorsiva"] != NSOrderedSame;
285 }
286
287 float SimpleFontData::platformWidthForGlyph(Glyph glyph) const
288 {
289     NSFont* font = m_font.font();
290     float pointSize = m_font.m_size;
291     CGAffineTransform m = CGAffineTransformMakeScale(pointSize, pointSize);
292     CGSize advance;
293     if (!wkGetGlyphTransformedAdvances(m_font.m_cgFont, font, &m, &glyph, &advance)) {
294         LOG_ERROR("Unable to cache glyph widths for %@ %f", [font displayName], pointSize);
295         advance.width = 0;
296     }
297     return advance.width + m_syntheticBoldOffset;
298 }
299
300 void SimpleFontData::checkShapesArabic() const
301 {
302     ASSERT(!m_checkedShapesArabic);
303
304     m_checkedShapesArabic = true;
305     
306     ATSUFontID fontID = m_font.m_atsuFontID;
307     if (!fontID) {
308         LOG_ERROR("unable to get ATSUFontID for %@", m_font.font());
309         return;
310     }
311
312     // This function is called only on fonts that contain Arabic glyphs. Our
313     // heuristic is that if such a font has a glyph metamorphosis table, then
314     // it includes shaping information for Arabic.
315     FourCharCode tables[] = { 'morx', 'mort' };
316     for (unsigned i = 0; i < sizeof(tables) / sizeof(tables[0]); ++i) {
317         ByteCount tableSize;
318         OSStatus status = ATSFontGetTable(fontID, tables[i], 0, 0, 0, &tableSize);
319         if (status == noErr) {
320             m_shapesArabic = true;
321             return;
322         }
323
324         if (status != kATSInvalidFontTableAccess)
325             LOG_ERROR("ATSFontGetTable failed (%d)", status);
326     }
327 }
328
329 }