| 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: | 
|---|
| 19 | int count() override { return 0; } | 
|---|
| 20 | void getStyle(int, SkFontStyle*, SkString*) override { | 
|---|
| 21 | SkDEBUGFAIL( "SkFontStyleSet::getStyle called on empty set"); | 
|---|
| 22 | } | 
|---|
| 23 | SkTypeface* createTypeface(int index) override { | 
|---|
| 24 | SkDEBUGFAIL( "SkFontStyleSet::createTypeface called on empty set"); | 
|---|
| 25 | return nullptr; | 
|---|
| 26 | } | 
|---|
| 27 | SkTypeface* matchStyle(const SkFontStyle&) override { | 
|---|
| 28 | return nullptr; | 
|---|
| 29 | } | 
|---|
| 30 | }; | 
|---|
| 31 |  | 
|---|
| 32 | SkFontStyleSet* SkFontStyleSet::CreateEmpty() { return new SkEmptyFontStyleSet; } | 
|---|
| 33 |  | 
|---|
| 34 | /////////////////////////////////////////////////////////////////////////////// | 
|---|
| 35 |  | 
|---|
| 36 | class SkEmptyFontMgr : public SkFontMgr { | 
|---|
| 37 | protected: | 
|---|
| 38 | int onCountFamilies() const override { | 
|---|
| 39 | return 0; | 
|---|
| 40 | } | 
|---|
| 41 | void onGetFamilyName(int index, SkString* familyName) const override { | 
|---|
| 42 | SkDEBUGFAIL( "onGetFamilyName called with bad index"); | 
|---|
| 43 | } | 
|---|
| 44 | SkFontStyleSet* onCreateStyleSet(int index) const override { | 
|---|
| 45 | SkDEBUGFAIL( "onCreateStyleSet called with bad index"); | 
|---|
| 46 | return nullptr; | 
|---|
| 47 | } | 
|---|
| 48 | SkFontStyleSet* onMatchFamily(const char[]) const override { | 
|---|
| 49 | return SkFontStyleSet::CreateEmpty(); | 
|---|
| 50 | } | 
|---|
| 51 |  | 
|---|
| 52 | SkTypeface* onMatchFamilyStyle(const char[], const SkFontStyle&) const override { | 
|---|
| 53 | return nullptr; | 
|---|
| 54 | } | 
|---|
| 55 | 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 | } | 
|---|
| 62 | SkTypeface* onMatchFaceStyle(const SkTypeface*, const SkFontStyle&) const override { | 
|---|
| 63 | return nullptr; | 
|---|
| 64 | } | 
|---|
| 65 |  | 
|---|
| 66 | sk_sp<SkTypeface> onMakeFromData(sk_sp<SkData>, int) const override { | 
|---|
| 67 | return nullptr; | 
|---|
| 68 | } | 
|---|
| 69 | sk_sp<SkTypeface> onMakeFromStreamIndex(std::unique_ptr<SkStreamAsset>, int) const override { | 
|---|
| 70 | return nullptr; | 
|---|
| 71 | } | 
|---|
| 72 | sk_sp<SkTypeface> onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset>, | 
|---|
| 73 | const SkFontArguments&) const override { | 
|---|
| 74 | return nullptr; | 
|---|
| 75 | } | 
|---|
| 76 | sk_sp<SkTypeface> onMakeFromFontData(std::unique_ptr<SkFontData>) const override { | 
|---|
| 77 | return nullptr; | 
|---|
| 78 | } | 
|---|
| 79 | sk_sp<SkTypeface> onMakeFromFile(const char[], int) const override { | 
|---|
| 80 | return nullptr; | 
|---|
| 81 | } | 
|---|
| 82 | sk_sp<SkTypeface> onLegacyMakeTypeface(const char [], SkFontStyle) const override { | 
|---|
| 83 | return nullptr; | 
|---|
| 84 | } | 
|---|
| 85 | }; | 
|---|
| 86 |  | 
|---|
| 87 | static SkFontStyleSet* emptyOnNull(SkFontStyleSet* fsset) { | 
|---|
| 88 | if (nullptr == fsset) { | 
|---|
| 89 | fsset = SkFontStyleSet::CreateEmpty(); | 
|---|
| 90 | } | 
|---|
| 91 | return fsset; | 
|---|
| 92 | } | 
|---|
| 93 |  | 
|---|
| 94 | int SkFontMgr::countFamilies() const { | 
|---|
| 95 | return this->onCountFamilies(); | 
|---|
| 96 | } | 
|---|
| 97 |  | 
|---|
| 98 | void SkFontMgr::getFamilyName(int index, SkString* familyName) const { | 
|---|
| 99 | this->onGetFamilyName(index, familyName); | 
|---|
| 100 | } | 
|---|
| 101 |  | 
|---|
| 102 | SkFontStyleSet* SkFontMgr::createStyleSet(int index) const { | 
|---|
| 103 | return emptyOnNull(this->onCreateStyleSet(index)); | 
|---|
| 104 | } | 
|---|
| 105 |  | 
|---|
| 106 | SkFontStyleSet* SkFontMgr::matchFamily(const char familyName[]) const { | 
|---|
| 107 | return emptyOnNull(this->onMatchFamily(familyName)); | 
|---|
| 108 | } | 
|---|
| 109 |  | 
|---|
| 110 | SkTypeface* SkFontMgr::matchFamilyStyle(const char familyName[], | 
|---|
| 111 | const SkFontStyle& fs) const { | 
|---|
| 112 | return this->onMatchFamilyStyle(familyName, fs); | 
|---|
| 113 | } | 
|---|
| 114 |  | 
|---|
| 115 | 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 |  | 
|---|
| 121 | sk_sp<SkTypeface> SkFontMgr::makeFromData(sk_sp<SkData> data, int ttcIndex) const { | 
|---|
| 122 | if (nullptr == data) { | 
|---|
| 123 | return nullptr; | 
|---|
| 124 | } | 
|---|
| 125 | return this->onMakeFromData(std::move(data), ttcIndex); | 
|---|
| 126 | } | 
|---|
| 127 |  | 
|---|
| 128 | sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream, | 
|---|
| 129 | int ttcIndex) const { | 
|---|
| 130 | if (nullptr == stream) { | 
|---|
| 131 | return nullptr; | 
|---|
| 132 | } | 
|---|
| 133 | return this->onMakeFromStreamIndex(std::move(stream), ttcIndex); | 
|---|
| 134 | } | 
|---|
| 135 |  | 
|---|
| 136 | sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream, | 
|---|
| 137 | const SkFontArguments& args) const { | 
|---|
| 138 | if (nullptr == stream) { | 
|---|
| 139 | return nullptr; | 
|---|
| 140 | } | 
|---|
| 141 | return this->onMakeFromStreamArgs(std::move(stream), args); | 
|---|
| 142 | } | 
|---|
| 143 |  | 
|---|
| 144 | sk_sp<SkTypeface> SkFontMgr::makeFromFontData(std::unique_ptr<SkFontData> data) const { | 
|---|
| 145 | if (nullptr == data) { | 
|---|
| 146 | return nullptr; | 
|---|
| 147 | } | 
|---|
| 148 | return this->onMakeFromFontData(std::move(data)); | 
|---|
| 149 | } | 
|---|
| 150 |  | 
|---|
| 151 | sk_sp<SkTypeface> SkFontMgr::makeFromFile(const char path[], int ttcIndex) const { | 
|---|
| 152 | if (nullptr == path) { | 
|---|
| 153 | return nullptr; | 
|---|
| 154 | } | 
|---|
| 155 | return this->onMakeFromFile(path, ttcIndex); | 
|---|
| 156 | } | 
|---|
| 157 |  | 
|---|
| 158 | sk_sp<SkTypeface> SkFontMgr::legacyMakeTypeface(const char familyName[], SkFontStyle style) const { | 
|---|
| 159 | return this->onLegacyMakeTypeface(familyName, style); | 
|---|
| 160 | } | 
|---|
| 161 |  | 
|---|
| 162 | sk_sp<SkTypeface> SkFontMgr::onMakeFromFontData(std::unique_ptr<SkFontData> data) const { | 
|---|
| 163 | return this->makeFromStream(data->detachStream(), data->getIndex()); | 
|---|
| 164 | } | 
|---|
| 165 |  | 
|---|
| 166 | // A global function pointer that's not declared, but can be overriden at startup by test tools. | 
|---|
| 167 | sk_sp<SkFontMgr> (*gSkFontMgr_DefaultFactory)() = nullptr; | 
|---|
| 168 |  | 
|---|
| 169 | sk_sp<SkFontMgr> SkFontMgr::RefDefault() { | 
|---|
| 170 | static SkOnce once; | 
|---|
| 171 | static sk_sp<SkFontMgr> singleton; | 
|---|
| 172 |  | 
|---|
| 173 | once([]{ | 
|---|
| 174 | sk_sp<SkFontMgr> fm = gSkFontMgr_DefaultFactory ? gSkFontMgr_DefaultFactory() | 
|---|
| 175 | : SkFontMgr::Factory(); | 
|---|
| 176 | singleton = fm ? std::move(fm) : sk_make_sp<SkEmptyFontMgr>(); | 
|---|
| 177 | }); | 
|---|
| 178 | return singleton; | 
|---|
| 179 | } | 
|---|
| 180 |  | 
|---|
| 181 | /** | 
|---|
| 182 | * Width has the greatest priority. | 
|---|
| 183 | * If the value of pattern.width is 5 (normal) or less, | 
|---|
| 184 | *    narrower width values are checked first, then wider values. | 
|---|
| 185 | * If the value of pattern.width is greater than 5 (normal), | 
|---|
| 186 | *    wider values are checked first, followed by narrower values. | 
|---|
| 187 | * | 
|---|
| 188 | * Italic/Oblique has the next highest priority. | 
|---|
| 189 | * If italic requested and there is some italic font, use it. | 
|---|
| 190 | * If oblique requested and there is some oblique font, use it. | 
|---|
| 191 | * If italic requested and there is some oblique font, use it. | 
|---|
| 192 | * If oblique requested and there is some italic font, use it. | 
|---|
| 193 | * | 
|---|
| 194 | * Exact match. | 
|---|
| 195 | * If pattern.weight < 400, weights below pattern.weight are checked | 
|---|
| 196 | *   in descending order followed by weights above pattern.weight | 
|---|
| 197 | *   in ascending order until a match is found. | 
|---|
| 198 | * If pattern.weight > 500, weights above pattern.weight are checked | 
|---|
| 199 | *   in ascending order followed by weights below pattern.weight | 
|---|
| 200 | *   in descending order until a match is found. | 
|---|
| 201 | * If pattern.weight is 400, 500 is checked first | 
|---|
| 202 | *   and then the rule for pattern.weight < 400 is used. | 
|---|
| 203 | * If pattern.weight is 500, 400 is checked first | 
|---|
| 204 | *   and then the rule for pattern.weight < 400 is used. | 
|---|
| 205 | */ | 
|---|
| 206 | SkTypeface* SkFontStyleSet::matchStyleCSS3(const SkFontStyle& pattern) { | 
|---|
| 207 | int count = this->count(); | 
|---|
| 208 | if (0 == count) { | 
|---|
| 209 | return nullptr; | 
|---|
| 210 | } | 
|---|
| 211 |  | 
|---|
| 212 | struct Score { | 
|---|
| 213 | int score; | 
|---|
| 214 | int index; | 
|---|
| 215 | Score& operator +=(int rhs) { this->score += rhs; return *this; } | 
|---|
| 216 | Score& operator <<=(int rhs) { this->score <<= rhs; return *this; } | 
|---|
| 217 | bool operator <(const Score& that) { return this->score < that.score; } | 
|---|
| 218 | }; | 
|---|
| 219 |  | 
|---|
| 220 | Score maxScore = { 0, 0 }; | 
|---|
| 221 | for (int i = 0; i < count; ++i) { | 
|---|
| 222 | SkFontStyle current; | 
|---|
| 223 | this->getStyle(i, ¤t, nullptr); | 
|---|
| 224 | Score currentScore = { 0, i }; | 
|---|
| 225 |  | 
|---|
| 226 | // CSS stretch / SkFontStyle::Width | 
|---|
| 227 | // Takes priority over everything else. | 
|---|
| 228 | if (pattern.width() <= SkFontStyle::kNormal_Width) { | 
|---|
| 229 | if (current.width() <= pattern.width()) { | 
|---|
| 230 | currentScore += 10 - pattern.width() + current.width(); | 
|---|
| 231 | } else { | 
|---|
| 232 | currentScore += 10 - current.width(); | 
|---|
| 233 | } | 
|---|
| 234 | } else { | 
|---|
| 235 | if (current.width() > pattern.width()) { | 
|---|
| 236 | currentScore += 10 + pattern.width() - current.width(); | 
|---|
| 237 | } else { | 
|---|
| 238 | currentScore += current.width(); | 
|---|
| 239 | } | 
|---|
| 240 | } | 
|---|
| 241 | currentScore <<= 8; | 
|---|
| 242 |  | 
|---|
| 243 | // CSS style (normal, italic, oblique) / SkFontStyle::Slant (upright, italic, oblique) | 
|---|
| 244 | // Takes priority over all valid weights. | 
|---|
| 245 | static_assert(SkFontStyle::kUpright_Slant == 0 && | 
|---|
| 246 | SkFontStyle::kItalic_Slant  == 1 && | 
|---|
| 247 | SkFontStyle::kOblique_Slant == 2, | 
|---|
| 248 | "SkFontStyle::Slant values not as required."); | 
|---|
| 249 | SkASSERT(0 <= pattern.slant() && pattern.slant() <= 2 && | 
|---|
| 250 | 0 <= current.slant() && current.slant() <= 2); | 
|---|
| 251 | static const int score[3][3] = { | 
|---|
| 252 | /*               Upright Italic Oblique  [current]*/ | 
|---|
| 253 | /*   Upright */ {   3   ,  1   ,   2   }, | 
|---|
| 254 | /*   Italic  */ {   1   ,  3   ,   2   }, | 
|---|
| 255 | /*   Oblique */ {   1   ,  2   ,   3   }, | 
|---|
| 256 | /* [pattern] */ | 
|---|
| 257 | }; | 
|---|
| 258 | currentScore += score[pattern.slant()][current.slant()]; | 
|---|
| 259 | currentScore <<= 8; | 
|---|
| 260 |  | 
|---|
| 261 | // Synthetics (weight, style) [no stretch synthetic?] | 
|---|
| 262 |  | 
|---|
| 263 | // CSS weight / SkFontStyle::Weight | 
|---|
| 264 | // The 'closer' to the target weight, the higher the score. | 
|---|
| 265 | // 1000 is the 'heaviest' recognized weight | 
|---|
| 266 | if (pattern.weight() == current.weight()) { | 
|---|
| 267 | currentScore += 1000; | 
|---|
| 268 | // less than 400 prefer lighter weights | 
|---|
| 269 | } else if (pattern.weight() < 400) { | 
|---|
| 270 | if (current.weight() <= pattern.weight()) { | 
|---|
| 271 | currentScore += 1000 - pattern.weight() + current.weight(); | 
|---|
| 272 | } else { | 
|---|
| 273 | currentScore += 1000 - current.weight(); | 
|---|
| 274 | } | 
|---|
| 275 | // between 400 and 500 prefer heavier up to 500, then lighter weights | 
|---|
| 276 | } else if (pattern.weight() <= 500) { | 
|---|
| 277 | if (current.weight() >= pattern.weight() && current.weight() <= 500) { | 
|---|
| 278 | currentScore += 1000 + pattern.weight() - current.weight(); | 
|---|
| 279 | } else if (current.weight() <= pattern.weight()) { | 
|---|
| 280 | currentScore += 500 + current.weight(); | 
|---|
| 281 | } else { | 
|---|
| 282 | currentScore += 1000 - current.weight(); | 
|---|
| 283 | } | 
|---|
| 284 | // greater than 500 prefer heavier weights | 
|---|
| 285 | } else if (pattern.weight() > 500) { | 
|---|
| 286 | if (current.weight() > pattern.weight()) { | 
|---|
| 287 | currentScore += 1000 + pattern.weight() - current.weight(); | 
|---|
| 288 | } else { | 
|---|
| 289 | currentScore += current.weight(); | 
|---|
| 290 | } | 
|---|
| 291 | } | 
|---|
| 292 |  | 
|---|
| 293 | if (maxScore < currentScore) { | 
|---|
| 294 | maxScore = currentScore; | 
|---|
| 295 | } | 
|---|
| 296 | } | 
|---|
| 297 |  | 
|---|
| 298 | return this->createTypeface(maxScore.index); | 
|---|
| 299 | } | 
|---|
| 300 |  | 
|---|