Improve use of NeverDestroyed
[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 <wtf/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     // FIXME: See comment in FontCascadeDescription::effectiveFamilyAt() in FontDescriptionCocoa.cpp
81     if (equalLettersIgnoringASCIICase(family, "-webkit-system-font") || equalLettersIgnoringASCIICase(family, "-apple-system") || equalLettersIgnoringASCIICase(family, "-apple-system-font") || equalLettersIgnoringASCIICase(family, "system-ui")) {
82         RetainPtr<CTFontRef> result = toCTFont([NSFont systemFontOfSize:size weight:toNSFontWeight(request.weight)]);
83         if (isItalic(request.slope)) {
84             CTFontSymbolicTraits desiredTraits = kCTFontItalicTrait;
85             if (isFontWeightBold(request.weight))
86                 desiredTraits |= kCTFontBoldTrait;
87             if (auto italicizedFont = adoptCF(CTFontCreateCopyWithSymbolicTraits(result.get(), size, nullptr, desiredTraits, desiredTraits)))
88                 result = italicizedFont;
89         }
90         return result;
91     }
92
93     if (equalLettersIgnoringASCIICase(family, "-apple-system-monospaced-numbers")) {
94         int numberSpacingType = kNumberSpacingType;
95         int monospacedNumbersSelector = kMonospacedNumbersSelector;
96         RetainPtr<CFNumberRef> numberSpacingNumber = adoptCF(CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &numberSpacingType));
97         RetainPtr<CFNumberRef> monospacedNumbersNumber = adoptCF(CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &monospacedNumbersSelector));
98         CFTypeRef featureKeys[] = { kCTFontFeatureTypeIdentifierKey, kCTFontFeatureSelectorIdentifierKey };
99         CFTypeRef featureValues[] = { numberSpacingNumber.get(), monospacedNumbersNumber.get() };
100         RetainPtr<CFDictionaryRef> featureIdentifier = adoptCF(CFDictionaryCreate(kCFAllocatorDefault, featureKeys, featureValues, WTF_ARRAY_LENGTH(featureKeys), &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
101         CFTypeRef featureIdentifiers[] = { featureIdentifier.get() };
102         RetainPtr<CFArrayRef> featureArray = adoptCF(CFArrayCreate(kCFAllocatorDefault, featureIdentifiers, 1, &kCFTypeArrayCallBacks));
103         CFTypeRef attributesKeys[] = { kCTFontFeatureSettingsAttribute };
104         CFTypeRef attributesValues[] = { featureArray.get() };
105         RetainPtr<CFDictionaryRef> attributes = adoptCF(CFDictionaryCreate(kCFAllocatorDefault, attributesKeys, attributesValues, WTF_ARRAY_LENGTH(attributesKeys), &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
106
107         RetainPtr<CTFontRef> result = toCTFont([NSFont systemFontOfSize:size]);
108         return adoptCF(CTFontCreateCopyWithAttributes(result.get(), size, nullptr, adoptCF(CTFontDescriptorCreateWithAttributes(attributes.get())).get()));
109     }
110
111     if (equalLettersIgnoringASCIICase(family, "-apple-menu"))
112         return toCTFont([NSFont menuFontOfSize:size]);
113
114     if (equalLettersIgnoringASCIICase(family, "-apple-status-bar"))
115         return toCTFont([NSFont labelFontOfSize:size]);
116
117     if (equalLettersIgnoringASCIICase(family, "lastresort")) {
118 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 101300
119         static const CTFontDescriptorRef lastResort = CTFontDescriptorCreateLastResort();
120         return adoptCF(CTFontCreateWithFontDescriptor(lastResort, size, nullptr));
121 #else
122         // LastResort is special, so it's important to look this exact string up, and not some case-folded version.
123         // We handle this here so any caching and case folding we do in our general text codepath is bypassed.
124         return adoptCF(CTFontCreateWithName(CFSTR("LastResort"), size, nullptr));
125 #endif
126     }
127
128     return nullptr;
129 }
130
131 Ref<Font> FontCache::lastResortFallbackFont(const FontDescription& fontDescription)
132 {
133     // FIXME: Would be even better to somehow get the user's default font here.  For now we'll pick
134     // the default that the user would get without changing any prefs.
135     if (RefPtr<Font> font = fontForFamily(fontDescription, AtomicString("Times", AtomicString::ConstructFromLiteral)))
136         return *font;
137
138     // The Times fallback will almost always work, but in the highly unusual case where
139     // the user doesn't have it, we fall back on Lucida Grande because that's
140     // guaranteed to be there, according to Nathan Taylor. This is good enough
141     // to avoid a crash at least.
142     return *fontForFamily(fontDescription, AtomicString("Lucida Grande", AtomicString::ConstructFromLiteral), nullptr, nullptr, { }, false);
143 }
144
145 #endif // PLATFORM(MAC)
146
147 } // namespace WebCore