• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2015 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #include "include/core/SkFontMgr.h"
9 #include "include/core/SkStream.h"
10 #include "include/core/SkTypes.h"
11 #include "include/private/SkOnce.h"
12 #include "src/core/SkFontDescriptor.h"
13 
14 class SkFontStyle;
15 class SkTypeface;
16 
17 class SkEmptyFontStyleSet : public SkFontStyleSet {
18 public:
count()19     int count() override { return 0; }
getStyle(int,SkFontStyle *,SkString *)20     void getStyle(int, SkFontStyle*, SkString*) override {
21         SkDEBUGFAIL("SkFontStyleSet::getStyle called on empty set");
22     }
createTypeface(int index)23     SkTypeface* createTypeface(int index) override {
24         SkDEBUGFAIL("SkFontStyleSet::createTypeface called on empty set");
25         return nullptr;
26     }
matchStyle(const SkFontStyle &)27     SkTypeface* matchStyle(const SkFontStyle&) override {
28         return nullptr;
29     }
30 };
31 
CreateEmpty()32 SkFontStyleSet* SkFontStyleSet::CreateEmpty() { return new SkEmptyFontStyleSet; }
33 
34 ///////////////////////////////////////////////////////////////////////////////
35 
36 class SkEmptyFontMgr : public SkFontMgr {
37 protected:
onCountFamilies() const38     int onCountFamilies() const override {
39         return 0;
40     }
onGetFamilyName(int index,SkString * familyName) const41     void onGetFamilyName(int index, SkString* familyName) const override {
42         SkDEBUGFAIL("onGetFamilyName called with bad index");
43     }
onCreateStyleSet(int index) const44     SkFontStyleSet* onCreateStyleSet(int index) const override {
45         SkDEBUGFAIL("onCreateStyleSet called with bad index");
46         return nullptr;
47     }
onMatchFamily(const char[]) const48     SkFontStyleSet* onMatchFamily(const char[]) const override {
49         return SkFontStyleSet::CreateEmpty();
50     }
51 
onMatchFamilyStyle(const char[],const SkFontStyle &) const52     SkTypeface* onMatchFamilyStyle(const char[], const SkFontStyle&) const override {
53         return nullptr;
54     }
onMatchFamilyStyleCharacter(const char familyName[],const SkFontStyle & style,const char * bcp47[],int bcp47Count,SkUnichar character) const55     SkTypeface* onMatchFamilyStyleCharacter(const char familyName[],
56                                             const SkFontStyle& style,
57                                             const char* bcp47[],
58                                             int bcp47Count,
59                                             SkUnichar character) const override {
60         return nullptr;
61     }
onMatchFaceStyle(const SkTypeface *,const SkFontStyle &) const62     SkTypeface* onMatchFaceStyle(const SkTypeface*, const SkFontStyle&) const override {
63         return nullptr;
64     }
65 
onMakeFromData(sk_sp<SkData>,int) const66     sk_sp<SkTypeface> onMakeFromData(sk_sp<SkData>, int) const override {
67         return nullptr;
68     }
onMakeFromStreamIndex(std::unique_ptr<SkStreamAsset>,int) const69     sk_sp<SkTypeface> onMakeFromStreamIndex(std::unique_ptr<SkStreamAsset>, int) const override {
70         return nullptr;
71     }
onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset>,const SkFontArguments &) const72     sk_sp<SkTypeface> onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset>,
73                                            const SkFontArguments&) const override {
74         return nullptr;
75     }
onMakeFromFontData(std::unique_ptr<SkFontData>) const76     sk_sp<SkTypeface> onMakeFromFontData(std::unique_ptr<SkFontData>) const override {
77         return nullptr;
78     }
onMakeFromFile(const char[],int) const79     sk_sp<SkTypeface> onMakeFromFile(const char[], int) const override {
80         return nullptr;
81     }
onLegacyMakeTypeface(const char[],SkFontStyle) const82     sk_sp<SkTypeface> onLegacyMakeTypeface(const char [], SkFontStyle) const override {
83         return nullptr;
84     }
85 };
86 
emptyOnNull(SkFontStyleSet * fsset)87 static SkFontStyleSet* emptyOnNull(SkFontStyleSet* fsset) {
88     if (nullptr == fsset) {
89         fsset = SkFontStyleSet::CreateEmpty();
90     }
91     return fsset;
92 }
93 
countFamilies() const94 int SkFontMgr::countFamilies() const {
95     return this->onCountFamilies();
96 }
97 
getFamilyName(int index,SkString * familyName) const98 void SkFontMgr::getFamilyName(int index, SkString* familyName) const {
99     this->onGetFamilyName(index, familyName);
100 }
101 
createStyleSet(int index) const102 SkFontStyleSet* SkFontMgr::createStyleSet(int index) const {
103     return emptyOnNull(this->onCreateStyleSet(index));
104 }
105 
matchFamily(const char familyName[]) const106 SkFontStyleSet* SkFontMgr::matchFamily(const char familyName[]) const {
107     return emptyOnNull(this->onMatchFamily(familyName));
108 }
109 
matchFamilyStyle(const char familyName[],const SkFontStyle & fs) const110 SkTypeface* SkFontMgr::matchFamilyStyle(const char familyName[],
111                                         const SkFontStyle& fs) const {
112     return this->onMatchFamilyStyle(familyName, fs);
113 }
114 
matchFamilyStyleCharacter(const char familyName[],const SkFontStyle & style,const char * bcp47[],int bcp47Count,SkUnichar character) const115 SkTypeface* SkFontMgr::matchFamilyStyleCharacter(const char familyName[], const SkFontStyle& style,
116                                                  const char* bcp47[], int bcp47Count,
117                                                  SkUnichar character) const {
118     return this->onMatchFamilyStyleCharacter(familyName, style, bcp47, bcp47Count, character);
119 }
120 
matchFaceStyle(const SkTypeface * face,const SkFontStyle & fs) const121 SkTypeface* SkFontMgr::matchFaceStyle(const SkTypeface* face,
122                                       const SkFontStyle& fs) const {
123     return this->onMatchFaceStyle(face, fs);
124 }
125 
makeFromData(sk_sp<SkData> data,int ttcIndex) const126 sk_sp<SkTypeface> SkFontMgr::makeFromData(sk_sp<SkData> data, int ttcIndex) const {
127     if (nullptr == data) {
128         return nullptr;
129     }
130     return this->onMakeFromData(std::move(data), ttcIndex);
131 }
132 
makeFromStream(std::unique_ptr<SkStreamAsset> stream,int ttcIndex) const133 sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream,
134                                             int ttcIndex) const {
135     if (nullptr == stream) {
136         return nullptr;
137     }
138     return this->onMakeFromStreamIndex(std::move(stream), ttcIndex);
139 }
140 
makeFromStream(std::unique_ptr<SkStreamAsset> stream,const SkFontArguments & args) const141 sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream,
142                                             const SkFontArguments& args) const {
143     if (nullptr == stream) {
144         return nullptr;
145     }
146     return this->onMakeFromStreamArgs(std::move(stream), args);
147 }
148 
makeFromFontData(std::unique_ptr<SkFontData> data) const149 sk_sp<SkTypeface> SkFontMgr::makeFromFontData(std::unique_ptr<SkFontData> data) const {
150     if (nullptr == data) {
151         return nullptr;
152     }
153     return this->onMakeFromFontData(std::move(data));
154 }
155 
makeFromFile(const char path[],int ttcIndex) const156 sk_sp<SkTypeface> SkFontMgr::makeFromFile(const char path[], int ttcIndex) const {
157     if (nullptr == path) {
158         return nullptr;
159     }
160     return this->onMakeFromFile(path, ttcIndex);
161 }
162 
legacyMakeTypeface(const char familyName[],SkFontStyle style) const163 sk_sp<SkTypeface> SkFontMgr::legacyMakeTypeface(const char familyName[], SkFontStyle style) const {
164     return this->onLegacyMakeTypeface(familyName, style);
165 }
166 
onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset> stream,const SkFontArguments & args) const167 sk_sp<SkTypeface> SkFontMgr::onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset> stream,
168                                                   const SkFontArguments& args) const {
169     return this->makeFromStream(std::move(stream), args.getCollectionIndex());
170 }
onMakeFromFontData(std::unique_ptr<SkFontData> data) const171 sk_sp<SkTypeface> SkFontMgr::onMakeFromFontData(std::unique_ptr<SkFontData> data) const {
172     return this->makeFromStream(data->detachStream(), data->getIndex());
173 }
174 
175 // A global function pointer that's not declared, but can be overriden at startup by test tools.
176 sk_sp<SkFontMgr> (*gSkFontMgr_DefaultFactory)() = nullptr;
177 
RefDefault()178 sk_sp<SkFontMgr> SkFontMgr::RefDefault() {
179     static SkOnce once;
180     static sk_sp<SkFontMgr> singleton;
181 
182     once([]{
183         sk_sp<SkFontMgr> fm = gSkFontMgr_DefaultFactory ? gSkFontMgr_DefaultFactory()
184                                                         : SkFontMgr::Factory();
185         singleton = fm ? std::move(fm) : sk_make_sp<SkEmptyFontMgr>();
186     });
187     return singleton;
188 }
189 
190 /**
191 * Width has the greatest priority.
192 * If the value of pattern.width is 5 (normal) or less,
193 *    narrower width values are checked first, then wider values.
194 * If the value of pattern.width is greater than 5 (normal),
195 *    wider values are checked first, followed by narrower values.
196 *
197 * Italic/Oblique has the next highest priority.
198 * If italic requested and there is some italic font, use it.
199 * If oblique requested and there is some oblique font, use it.
200 * If italic requested and there is some oblique font, use it.
201 * If oblique requested and there is some italic font, use it.
202 *
203 * Exact match.
204 * If pattern.weight < 400, weights below pattern.weight are checked
205 *   in descending order followed by weights above pattern.weight
206 *   in ascending order until a match is found.
207 * If pattern.weight > 500, weights above pattern.weight are checked
208 *   in ascending order followed by weights below pattern.weight
209 *   in descending order until a match is found.
210 * If pattern.weight is 400, 500 is checked first
211 *   and then the rule for pattern.weight < 400 is used.
212 * If pattern.weight is 500, 400 is checked first
213 *   and then the rule for pattern.weight < 400 is used.
214 */
matchStyleCSS3(const SkFontStyle & pattern)215 SkTypeface* SkFontStyleSet::matchStyleCSS3(const SkFontStyle& pattern) {
216     int count = this->count();
217     if (0 == count) {
218         return nullptr;
219     }
220 
221     struct Score {
222         int score;
223         int index;
224         Score& operator +=(int rhs) { this->score += rhs; return *this; }
225         Score& operator <<=(int rhs) { this->score <<= rhs; return *this; }
226         bool operator <(const Score& that) { return this->score < that.score; }
227     };
228 
229     Score maxScore = { 0, 0 };
230     for (int i = 0; i < count; ++i) {
231         SkFontStyle current;
232         this->getStyle(i, &current, nullptr);
233         Score currentScore = { 0, i };
234 
235         // CSS stretch / SkFontStyle::Width
236         // Takes priority over everything else.
237         if (pattern.width() <= SkFontStyle::kNormal_Width) {
238             if (current.width() <= pattern.width()) {
239                 currentScore += 10 - pattern.width() + current.width();
240             } else {
241                 currentScore += 10 - current.width();
242             }
243         } else {
244             if (current.width() > pattern.width()) {
245                 currentScore += 10 + pattern.width() - current.width();
246             } else {
247                 currentScore += current.width();
248             }
249         }
250         currentScore <<= 8;
251 
252         // CSS style (normal, italic, oblique) / SkFontStyle::Slant (upright, italic, oblique)
253         // Takes priority over all valid weights.
254         static_assert(SkFontStyle::kUpright_Slant == 0 &&
255                       SkFontStyle::kItalic_Slant  == 1 &&
256                       SkFontStyle::kOblique_Slant == 2,
257                       "SkFontStyle::Slant values not as required.");
258         SkASSERT(0 <= pattern.slant() && pattern.slant() <= 2 &&
259                  0 <= current.slant() && current.slant() <= 2);
260         static const int score[3][3] = {
261             /*               Upright Italic Oblique  [current]*/
262             /*   Upright */ {   3   ,  1   ,   2   },
263             /*   Italic  */ {   1   ,  3   ,   2   },
264             /*   Oblique */ {   1   ,  2   ,   3   },
265             /* [pattern] */
266         };
267         currentScore += score[pattern.slant()][current.slant()];
268         currentScore <<= 8;
269 
270         // Synthetics (weight, style) [no stretch synthetic?]
271 
272         // CSS weight / SkFontStyle::Weight
273         // The 'closer' to the target weight, the higher the score.
274         // 1000 is the 'heaviest' recognized weight
275         if (pattern.weight() == current.weight()) {
276             currentScore += 1000;
277         // less than 400 prefer lighter weights
278         } else if (pattern.weight() < 400) {
279             if (current.weight() <= pattern.weight()) {
280                 currentScore += 1000 - pattern.weight() + current.weight();
281             } else {
282                 currentScore += 1000 - current.weight();
283             }
284         // between 400 and 500 prefer heavier up to 500, then lighter weights
285         } else if (pattern.weight() <= 500) {
286             if (current.weight() >= pattern.weight() && current.weight() <= 500) {
287                 currentScore += 1000 + pattern.weight() - current.weight();
288             } else if (current.weight() <= pattern.weight()) {
289                 currentScore += 500 + current.weight();
290             } else {
291                 currentScore += 1000 - current.weight();
292             }
293         // greater than 500 prefer heavier weights
294         } else if (pattern.weight() > 500) {
295             if (current.weight() > pattern.weight()) {
296                 currentScore += 1000 + pattern.weight() - current.weight();
297             } else {
298                 currentScore += current.weight();
299             }
300         }
301 
302         if (maxScore < currentScore) {
303             maxScore = currentScore;
304         }
305     }
306 
307     return this->createTypeface(maxScore.index);
308 }
309