/* * 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 onMakeFromData(sk_sp, int) const override { return nullptr; } sk_sp onMakeFromStreamIndex(std::unique_ptr, int) const override { return nullptr; } sk_sp onMakeFromStreamArgs(std::unique_ptr, const SkFontArguments&) const override { return nullptr; } sk_sp onMakeFromFontData(std::unique_ptr) const override { return nullptr; } sk_sp onMakeFromFile(const char[], int) const override { return nullptr; } sk_sp 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 SkFontMgr::makeFromData(sk_sp data, int ttcIndex) const { if (nullptr == data) { return nullptr; } return this->onMakeFromData(std::move(data), ttcIndex); } sk_sp SkFontMgr::makeFromStream(std::unique_ptr stream, int ttcIndex) const { if (nullptr == stream) { return nullptr; } return this->onMakeFromStreamIndex(std::move(stream), ttcIndex); } sk_sp SkFontMgr::makeFromStream(std::unique_ptr stream, const SkFontArguments& args) const { if (nullptr == stream) { return nullptr; } return this->onMakeFromStreamArgs(std::move(stream), args); } sk_sp SkFontMgr::makeFromFontData(std::unique_ptr data) const { if (nullptr == data) { return nullptr; } return this->onMakeFromFontData(std::move(data)); } sk_sp SkFontMgr::makeFromFile(const char path[], int ttcIndex) const { if (nullptr == path) { return nullptr; } return this->onMakeFromFile(path, ttcIndex); } sk_sp SkFontMgr::legacyMakeTypeface(const char familyName[], SkFontStyle style) const { return this->onLegacyMakeTypeface(familyName, style); } sk_sp SkFontMgr::onMakeFromStreamArgs(std::unique_ptr stream, const SkFontArguments& args) const { return this->makeFromStream(std::move(stream), args.getCollectionIndex()); } sk_sp SkFontMgr::onMakeFromFontData(std::unique_ptr 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 (*gSkFontMgr_DefaultFactory)() = nullptr; sk_sp SkFontMgr::RefDefault() { static SkOnce once; static sk_sp singleton; once([]{ sk_sp fm = gSkFontMgr_DefaultFactory ? gSkFontMgr_DefaultFactory() : SkFontMgr::Factory(); singleton = fm ? std::move(fm) : sk_make_sp(); }); 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); }