f0779b0646b5a7c9baf3a252efb95c7c5c94446b
[WebKit-https.git] / Source / WebCore / platform / graphics / mac / FontCacheMac.mm
1 /*
2  * Copyright (C) 2006, 2007, 2008, 2009 Apple Inc. All rights reserved.
3  * Copyright (C) 2007 Nicholas Shanks <webkit@nickshanks.com>
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 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 "FontCache.h"
32
33 #import "CoreGraphicsSPI.h"
34 #import "CoreTextSPI.h"
35 #import "Font.h"
36 #import "FontCascade.h"
37 #import "FontPlatformData.h"
38
39 #if PLATFORM(MAC)
40 #import "NSFontSPI.h"
41 #import "WebCoreNSStringExtras.h"
42 #import "WebCoreSystemInterface.h"
43 #import <AppKit/AppKit.h>
44 #import <wtf/MainThread.h>
45 #import <wtf/NeverDestroyed.h>
46 #import <wtf/StdLibExtras.h>
47 #import <wtf/Threading.h>
48 #import <wtf/text/AtomicStringHash.h>
49 #endif
50
51 #import "SoftLinking.h"
52
53 namespace WebCore {
54
55 #if PLATFORM(MAC)
56
57 static CGFloat toNSFontWeight(FontSelectionValue fontWeight)
58 {
59     if (fontWeight < FontSelectionValue(150))
60         return NSFontWeightUltraLight;
61     if (fontWeight < FontSelectionValue(250))
62         return NSFontWeightThin;
63     if (fontWeight < FontSelectionValue(350))
64         return NSFontWeightLight;
65     if (fontWeight < FontSelectionValue(450))
66         return NSFontWeightRegular;
67     if (fontWeight < FontSelectionValue(550))
68         return NSFontWeightMedium;
69     if (fontWeight < FontSelectionValue(650))
70         return NSFontWeightSemibold;
71     if (fontWeight < FontSelectionValue(750))
72         return NSFontWeightBold;
73     if (fontWeight < FontSelectionValue(850))
74         return NSFontWeightHeavy;
75     return NSFontWeightBlack;
76 }
77
78 RetainPtr<CTFontRef> platformFontWithFamilySpecialCase(const AtomicString& family, FontSelectionRequest request, float size)
79 {
80     if (equalLettersIgnoringASCIICase(family, "-webkit-system-font") || equalLettersIgnoringASCIICase(family, "-apple-system") || equalLettersIgnoringASCIICase(family, "-apple-system-font") || equalLettersIgnoringASCIICase(family, "system-ui")) {
81         RetainPtr<CTFontRef> result = toCTFont([NSFont systemFontOfSize:size weight:toNSFontWeight(request.weight)]);
82         if (isItalic(request.slope)) {
83             CTFontSymbolicTraits desiredTraits = kCTFontItalicTrait;
84             if (isFontWeightBold(request.weight))
85                 desiredTraits |= kCTFontBoldTrait;
86             if (auto italicizedFont = adoptCF(CTFontCreateCopyWithSymbolicTraits(result.get(), size, nullptr, desiredTraits, desiredTraits)))
87                 result = italicizedFont;
88         }
89         return result;
90     }
91
92     if (equalLettersIgnoringASCIICase(family, "-apple-system-monospaced-numbers")) {
93         int numberSpacingType = kNumberSpacingType;
94         int monospacedNumbersSelector = kMonospacedNumbersSelector;
95         RetainPtr<CFNumberRef> numberSpacingNumber = adoptCF(CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &numberSpacingType));
96         RetainPtr<CFNumberRef> monospacedNumbersNumber = adoptCF(CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &monospacedNumbersSelector));
97         CFTypeRef featureKeys[] = { kCTFontFeatureTypeIdentifierKey, kCTFontFeatureSelectorIdentifierKey };
98         CFTypeRef featureValues[] = { numberSpacingNumber.get(), monospacedNumbersNumber.get() };
99         RetainPtr<CFDictionaryRef> featureIdentifier = adoptCF(CFDictionaryCreate(kCFAllocatorDefault, featureKeys, featureValues, WTF_ARRAY_LENGTH(featureKeys), &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
100         CFTypeRef featureIdentifiers[] = { featureIdentifier.get() };
101         RetainPtr<CFArrayRef> featureArray = adoptCF(CFArrayCreate(kCFAllocatorDefault, featureIdentifiers, 1, &kCFTypeArrayCallBacks));
102         CFTypeRef attributesKeys[] = { kCTFontFeatureSettingsAttribute };
103         CFTypeRef attributesValues[] = { featureArray.get() };
104         RetainPtr<CFDictionaryRef> attributes = adoptCF(CFDictionaryCreate(kCFAllocatorDefault, attributesKeys, attributesValues, WTF_ARRAY_LENGTH(attributesKeys), &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
105
106         RetainPtr<CTFontRef> result = toCTFont([NSFont systemFontOfSize:size]);
107         return adoptCF(CTFontCreateCopyWithAttributes(result.get(), size, nullptr, adoptCF(CTFontDescriptorCreateWithAttributes(attributes.get())).get()));
108     }
109
110     if (equalLettersIgnoringASCIICase(family, "-apple-menu"))
111         return toCTFont([NSFont menuFontOfSize:size]);
112
113     if (equalLettersIgnoringASCIICase(family, "-apple-status-bar"))
114         return toCTFont([NSFont labelFontOfSize:size]);
115
116     if (equalLettersIgnoringASCIICase(family, "lastresort")) {
117 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 101300
118         static NeverDestroyed<RetainPtr<CTFontDescriptorRef>> lastResort = adoptCF(CTFontDescriptorCreateLastResort());
119         return adoptCF(CTFontCreateWithFontDescriptor(lastResort.get().get(), size, nullptr));
120 #else
121         // LastResort is special, so it's important to look this exact string up, and not some case-folded version.
122         // We handle this here so any caching and case folding we do in our general text codepath is bypassed.
123         return adoptCF(CTFontCreateWithName(CFSTR("LastResort"), size, nullptr));
124 #endif
125     }
126
127     return nullptr;
128 }
129
130 Ref<Font> FontCache::lastResortFallbackFont(const FontDescription& fontDescription)
131 {
132     // FIXME: Would be even better to somehow get the user's default font here.  For now we'll pick
133     // the default that the user would get without changing any prefs.
134     if (RefPtr<Font> font = fontForFamily(fontDescription, AtomicString("Times", AtomicString::ConstructFromLiteral)))
135         return *font;
136
137     // The Times fallback will almost always work, but in the highly unusual case where
138     // the user doesn't have it, we fall back on Lucida Grande because that's
139     // guaranteed to be there, according to Nathan Taylor. This is good enough
140     // to avoid a crash at least.
141     return *fontForFamily(fontDescription, AtomicString("Lucida Grande", AtomicString::ConstructFromLiteral), nullptr, nullptr, { }, false);
142 }
143
144 #endif // PLATFORM(MAC)
145
146 } // namespace WebCore