| /* | 
 |  * Copyright 2015 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include "SkFontDescriptor.h" | 
 | #include "SkFontMgr.h" | 
 | #include "SkOnce.h" | 
 | #include "SkStream.h" | 
 | #include "SkTypes.h" | 
 |  | 
 | class SkFontStyle; | 
 | class SkTypeface; | 
 |  | 
 | class SkEmptyFontStyleSet : public SkFontStyleSet { | 
 | public: | 
 |     int count() override { return 0; } | 
 |     void getStyle(int, SkFontStyle*, SkString*) override { | 
 |         SkDEBUGFAIL("SkFontStyleSet::getStyle called on empty set"); | 
 |     } | 
 |     SkTypeface* createTypeface(int index) override { | 
 |         SkDEBUGFAIL("SkFontStyleSet::createTypeface called on empty set"); | 
 |         return nullptr; | 
 |     } | 
 |     SkTypeface* matchStyle(const SkFontStyle&) override { | 
 |         return nullptr; | 
 |     } | 
 | }; | 
 |  | 
 | SkFontStyleSet* SkFontStyleSet::CreateEmpty() { return new SkEmptyFontStyleSet; } | 
 |  | 
 | /////////////////////////////////////////////////////////////////////////////// | 
 |  | 
 | class SkEmptyFontMgr : public SkFontMgr { | 
 | protected: | 
 |     int onCountFamilies() const override { | 
 |         return 0; | 
 |     } | 
 |     void onGetFamilyName(int index, SkString* familyName) const override { | 
 |         SkDEBUGFAIL("onGetFamilyName called with bad index"); | 
 |     } | 
 |     SkFontStyleSet* onCreateStyleSet(int index) const override { | 
 |         SkDEBUGFAIL("onCreateStyleSet called with bad index"); | 
 |         return nullptr; | 
 |     } | 
 |     SkFontStyleSet* onMatchFamily(const char[]) const override { | 
 |         return SkFontStyleSet::CreateEmpty(); | 
 |     } | 
 |  | 
 |     SkTypeface* onMatchFamilyStyle(const char[], const SkFontStyle&) const override { | 
 |         return nullptr; | 
 |     } | 
 |     SkTypeface* onMatchFamilyStyleCharacter(const char familyName[], | 
 |                                             const SkFontStyle& style, | 
 |                                             const char* bcp47[], | 
 |                                             int bcp47Count, | 
 |                                             SkUnichar character) const override { | 
 |         return nullptr; | 
 |     } | 
 |     SkTypeface* onMatchFaceStyle(const SkTypeface*, const SkFontStyle&) const override { | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     sk_sp<SkTypeface> onMakeFromData(sk_sp<SkData>, int) const override { | 
 |         return nullptr; | 
 |     } | 
 |     sk_sp<SkTypeface> onMakeFromStreamIndex(std::unique_ptr<SkStreamAsset>, int) const override { | 
 |         return nullptr; | 
 |     } | 
 |     sk_sp<SkTypeface> onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset>, | 
 |                                            const SkFontArguments&) const override { | 
 |         return nullptr; | 
 |     } | 
 |     sk_sp<SkTypeface> onMakeFromFontData(std::unique_ptr<SkFontData>) const override { | 
 |         return nullptr; | 
 |     } | 
 |     sk_sp<SkTypeface> onMakeFromFile(const char[], int) const override { | 
 |         return nullptr; | 
 |     } | 
 |     sk_sp<SkTypeface> onLegacyMakeTypeface(const char [], SkFontStyle) const override { | 
 |         return nullptr; | 
 |     } | 
 | }; | 
 |  | 
 | static SkFontStyleSet* emptyOnNull(SkFontStyleSet* fsset) { | 
 |     if (nullptr == fsset) { | 
 |         fsset = SkFontStyleSet::CreateEmpty(); | 
 |     } | 
 |     return fsset; | 
 | } | 
 |  | 
 | int SkFontMgr::countFamilies() const { | 
 |     return this->onCountFamilies(); | 
 | } | 
 |  | 
 | void SkFontMgr::getFamilyName(int index, SkString* familyName) const { | 
 |     this->onGetFamilyName(index, familyName); | 
 | } | 
 |  | 
 | SkFontStyleSet* SkFontMgr::createStyleSet(int index) const { | 
 |     return emptyOnNull(this->onCreateStyleSet(index)); | 
 | } | 
 |  | 
 | SkFontStyleSet* SkFontMgr::matchFamily(const char familyName[]) const { | 
 |     return emptyOnNull(this->onMatchFamily(familyName)); | 
 | } | 
 |  | 
 | SkTypeface* SkFontMgr::matchFamilyStyle(const char familyName[], | 
 |                                         const SkFontStyle& fs) const { | 
 |     return this->onMatchFamilyStyle(familyName, fs); | 
 | } | 
 |  | 
 | SkTypeface* SkFontMgr::matchFamilyStyleCharacter(const char familyName[], const SkFontStyle& style, | 
 |                                                  const char* bcp47[], int bcp47Count, | 
 |                                                  SkUnichar character) const { | 
 |     return this->onMatchFamilyStyleCharacter(familyName, style, bcp47, bcp47Count, character); | 
 | } | 
 |  | 
 | SkTypeface* SkFontMgr::matchFaceStyle(const SkTypeface* face, | 
 |                                       const SkFontStyle& fs) const { | 
 |     return this->onMatchFaceStyle(face, fs); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::makeFromData(sk_sp<SkData> data, int ttcIndex) const { | 
 |     if (nullptr == data) { | 
 |         return nullptr; | 
 |     } | 
 |     return this->onMakeFromData(std::move(data), ttcIndex); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream, | 
 |                                             int ttcIndex) const { | 
 |     if (nullptr == stream) { | 
 |         return nullptr; | 
 |     } | 
 |     return this->onMakeFromStreamIndex(std::move(stream), ttcIndex); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::makeFromStream(std::unique_ptr<SkStreamAsset> stream, | 
 |                                             const SkFontArguments& args) const { | 
 |     if (nullptr == stream) { | 
 |         return nullptr; | 
 |     } | 
 |     return this->onMakeFromStreamArgs(std::move(stream), args); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::makeFromFontData(std::unique_ptr<SkFontData> data) const { | 
 |     if (nullptr == data) { | 
 |         return nullptr; | 
 |     } | 
 |     return this->onMakeFromFontData(std::move(data)); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::makeFromFile(const char path[], int ttcIndex) const { | 
 |     if (nullptr == path) { | 
 |         return nullptr; | 
 |     } | 
 |     return this->onMakeFromFile(path, ttcIndex); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::legacyMakeTypeface(const char familyName[], SkFontStyle style) const { | 
 |     return this->onLegacyMakeTypeface(familyName, style); | 
 | } | 
 |  | 
 | sk_sp<SkTypeface> SkFontMgr::onMakeFromStreamArgs(std::unique_ptr<SkStreamAsset> stream, | 
 |                                                   const SkFontArguments& args) const { | 
 |     return this->makeFromStream(std::move(stream), args.getCollectionIndex()); | 
 | } | 
 | sk_sp<SkTypeface> SkFontMgr::onMakeFromFontData(std::unique_ptr<SkFontData> data) const { | 
 |     return this->makeFromStream(data->detachStream(), data->getIndex()); | 
 | } | 
 |  | 
 | // A global function pointer that's not declared, but can be overriden at startup by test tools. | 
 | sk_sp<SkFontMgr> (*gSkFontMgr_DefaultFactory)() = nullptr; | 
 |  | 
 | sk_sp<SkFontMgr> SkFontMgr::RefDefault() { | 
 |     static SkOnce once; | 
 |     static sk_sp<SkFontMgr> singleton; | 
 |  | 
 |     once([]{ | 
 |         sk_sp<SkFontMgr> fm = gSkFontMgr_DefaultFactory ? gSkFontMgr_DefaultFactory() | 
 |                                                         : SkFontMgr::Factory(); | 
 |         singleton = fm ? std::move(fm) : sk_make_sp<SkEmptyFontMgr>(); | 
 |     }); | 
 |     return singleton; | 
 | } | 
 |  | 
 | /** | 
 | * Width has the greatest priority. | 
 | * If the value of pattern.width is 5 (normal) or less, | 
 | *    narrower width values are checked first, then wider values. | 
 | * If the value of pattern.width is greater than 5 (normal), | 
 | *    wider values are checked first, followed by narrower values. | 
 | * | 
 | * Italic/Oblique has the next highest priority. | 
 | * If italic requested and there is some italic font, use it. | 
 | * If oblique requested and there is some oblique font, use it. | 
 | * If italic requested and there is some oblique font, use it. | 
 | * If oblique requested and there is some italic font, use it. | 
 | * | 
 | * Exact match. | 
 | * If pattern.weight < 400, weights below pattern.weight are checked | 
 | *   in descending order followed by weights above pattern.weight | 
 | *   in ascending order until a match is found. | 
 | * If pattern.weight > 500, weights above pattern.weight are checked | 
 | *   in ascending order followed by weights below pattern.weight | 
 | *   in descending order until a match is found. | 
 | * If pattern.weight is 400, 500 is checked first | 
 | *   and then the rule for pattern.weight < 400 is used. | 
 | * If pattern.weight is 500, 400 is checked first | 
 | *   and then the rule for pattern.weight < 400 is used. | 
 | */ | 
 | SkTypeface* SkFontStyleSet::matchStyleCSS3(const SkFontStyle& pattern) { | 
 |     int count = this->count(); | 
 |     if (0 == count) { | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     struct Score { | 
 |         int score; | 
 |         int index; | 
 |         Score& operator +=(int rhs) { this->score += rhs; return *this; } | 
 |         Score& operator <<=(int rhs) { this->score <<= rhs; return *this; } | 
 |         bool operator <(const Score& that) { return this->score < that.score; } | 
 |     }; | 
 |  | 
 |     Score maxScore = { 0, 0 }; | 
 |     for (int i = 0; i < count; ++i) { | 
 |         SkFontStyle current; | 
 |         this->getStyle(i, ¤t, nullptr); | 
 |         Score currentScore = { 0, i }; | 
 |  | 
 |         // CSS stretch / SkFontStyle::Width | 
 |         // Takes priority over everything else. | 
 |         if (pattern.width() <= SkFontStyle::kNormal_Width) { | 
 |             if (current.width() <= pattern.width()) { | 
 |                 currentScore += 10 - pattern.width() + current.width(); | 
 |             } else { | 
 |                 currentScore += 10 - current.width(); | 
 |             } | 
 |         } else { | 
 |             if (current.width() > pattern.width()) { | 
 |                 currentScore += 10 + pattern.width() - current.width(); | 
 |             } else { | 
 |                 currentScore += current.width(); | 
 |             } | 
 |         } | 
 |         currentScore <<= 8; | 
 |  | 
 |         // CSS style (normal, italic, oblique) / SkFontStyle::Slant (upright, italic, oblique) | 
 |         // Takes priority over all valid weights. | 
 |         static_assert(SkFontStyle::kUpright_Slant == 0 && | 
 |                       SkFontStyle::kItalic_Slant  == 1 && | 
 |                       SkFontStyle::kOblique_Slant == 2, | 
 |                       "SkFontStyle::Slant values not as required."); | 
 |         SkASSERT(0 <= pattern.slant() && pattern.slant() <= 2 && | 
 |                  0 <= current.slant() && current.slant() <= 2); | 
 |         static const int score[3][3] = { | 
 |             /*               Upright Italic Oblique  [current]*/ | 
 |             /*   Upright */ {   3   ,  1   ,   2   }, | 
 |             /*   Italic  */ {   1   ,  3   ,   2   }, | 
 |             /*   Oblique */ {   1   ,  2   ,   3   }, | 
 |             /* [pattern] */ | 
 |         }; | 
 |         currentScore += score[pattern.slant()][current.slant()]; | 
 |         currentScore <<= 8; | 
 |  | 
 |         // Synthetics (weight, style) [no stretch synthetic?] | 
 |  | 
 |         // CSS weight / SkFontStyle::Weight | 
 |         // The 'closer' to the target weight, the higher the score. | 
 |         // 1000 is the 'heaviest' recognized weight | 
 |         if (pattern.weight() == current.weight()) { | 
 |             currentScore += 1000; | 
 |         } else if (pattern.weight() <= 500) { | 
 |             if (400 <= pattern.weight() && pattern.weight() < 450) { | 
 |                 if (450 <= current.weight() && current.weight() <= 500) { | 
 |                     // Artificially boost the 500 weight. | 
 |                     // TODO: determine correct number to use. | 
 |                     currentScore += 500; | 
 |                 } | 
 |             } | 
 |             if (current.weight() <= pattern.weight()) { | 
 |                 currentScore += 1000 - pattern.weight() + current.weight(); | 
 |             } else { | 
 |                 currentScore += 1000 - current.weight(); | 
 |             } | 
 |         } else if (pattern.weight() > 500) { | 
 |             if (current.weight() > pattern.weight()) { | 
 |                 currentScore += 1000 + pattern.weight() - current.weight(); | 
 |             } else { | 
 |                 currentScore += current.weight(); | 
 |             } | 
 |         } | 
 |  | 
 |         if (maxScore < currentScore) { | 
 |             maxScore = currentScore; | 
 |         } | 
 |     } | 
 |  | 
 |     return this->createTypeface(maxScore.index); | 
 | } |