Index: src/ports/SkFontMgr_android.cpp |
diff --git a/src/ports/SkFontMgr_android.cpp b/src/ports/SkFontMgr_android.cpp |
new file mode 100644 |
index 0000000000000000000000000000000000000000..ba3846c412c5d8067601e545071914199db65518 |
--- /dev/null |
+++ b/src/ports/SkFontMgr_android.cpp |
@@ -0,0 +1,406 @@ |
+/* |
+ * Copyright 2014 Google Inc. |
+ * |
+ * Use of this source code is governed by a BSD-style license that can be |
+ * found in the LICENSE file. |
+ */ |
+ |
+#include "SkFontConfigParser_android.h" |
+#include "SkFontDescriptor.h" |
+#include "SkFontHost_FreeType_common.h" |
+#include "SkFontMgr.h" |
+#include "SkFontStyle.h" |
+#include "SkStream.h" |
+#include "SkTDArray.h" |
+#include "SkTSearch.h" |
+#include "SkTypeface.h" |
+#include "SkTypefaceCache.h" |
+ |
+#include <limits> |
+#include <stdlib.h> |
+ |
+#ifndef SK_FONT_FILE_PREFIX |
+# define SK_FONT_FILE_PREFIX "/fonts/" |
+#endif |
+ |
+#ifndef SK_DEBUG_FONTS |
+ #define SK_DEBUG_FONTS 0 |
+#endif |
+ |
+#if SK_DEBUG_FONTS |
+# define DEBUG_FONT(args) SkDebugf args |
+#else |
+# define DEBUG_FONT(args) |
+#endif |
+ |
+class SkTypeface_Android : public SkTypeface_FreeType { |
+public: |
+ SkTypeface_Android(int index, |
+ Style style, |
+ bool isFixedPitch, |
+ const SkString familyName) |
+ : INHERITED(style, SkTypefaceCache::NewFontID(), isFixedPitch) |
+ , fIndex(index) |
+ , fFamilyName(familyName) |
+ { } |
+ |
+ const SkString& name() const { return fFamilyName; } |
+ |
+protected: |
+ int fIndex; |
+ SkString fFamilyName; |
+ |
+private: |
+ typedef SkTypeface_FreeType INHERITED; |
+}; |
+ |
+class SkTypeface_AndroidSystem : public SkTypeface_Android { |
+public: |
+ SkTypeface_AndroidSystem(const SkString pathName, |
+ int index, |
+ Style style, |
+ bool isFixedPitch, |
+ const SkString familyName) |
+ : INHERITED(index, style, isFixedPitch, familyName) |
+ , fPathName(pathName) |
+ { } |
+ |
+ virtual void onGetFontDescriptor(SkFontDescriptor* desc, |
+ bool* serialize) const SK_OVERRIDE |
+ { |
+ SkASSERT(desc); |
+ SkASSERT(serialize); |
+ desc->setFamilyName(fFamilyName.c_str()); |
+ desc->setFontFileName(fPathName.c_str()); |
+ *serialize = false; |
+ } |
+ virtual SkStream* onOpenStream(int* ttcIndex) const SK_OVERRIDE { |
+ *ttcIndex = fIndex; |
+ return SkStream::NewFromFile(fPathName.c_str()); |
+ } |
+ |
+private: |
+ SkString fPathName; |
+ |
+ typedef SkTypeface_Android INHERITED; |
+}; |
+ |
+class SkTypeface_AndroidStream : public SkTypeface_Android { |
+public: |
+ SkTypeface_AndroidStream(SkStream* stream, |
+ int index, |
+ Style style, |
+ bool isFixedPitch, |
+ const SkString familyName) |
+ : INHERITED(index, style, isFixedPitch, familyName) |
+ , fStream(stream) |
+ { } |
+ |
+ virtual void onGetFontDescriptor(SkFontDescriptor* desc, |
+ bool* serialize) const SK_OVERRIDE { |
+ SkASSERT(desc); |
+ SkASSERT(serialize); |
+ desc->setFamilyName(fFamilyName.c_str()); |
+ desc->setFontFileName(NULL); |
+ *serialize = true; |
+ } |
+ |
+ virtual SkStream* onOpenStream(int* ttcIndex) const SK_OVERRIDE { |
+ *ttcIndex = fIndex; |
+ return fStream->duplicate(); |
+ } |
+ |
+private: |
+ SkAutoTUnref<SkStream> fStream; |
+ |
+ typedef SkTypeface_Android INHERITED; |
+}; |
+ |
+void get_path_for_sys_fonts(SkString* full, const SkString& name) { |
+ full->set(getenv("ANDROID_ROOT")); |
+ full->append(SK_FONT_FILE_PREFIX); |
+ full->append(name); |
+} |
+ |
+class SkFontStyleSet_Android : public SkFontStyleSet { |
+public: |
+ explicit SkFontStyleSet_Android(FontFamily* family) : fFontFamily(family) { |
+ // TODO? make this lazy |
+ for (int i = 0; i < family->fFontFiles.count(); ++i) { |
+ const SkString& fileName = family->fFontFiles[i].fFileName; |
+ |
+ SkString pathName; |
+ get_path_for_sys_fonts(&pathName, fileName); |
+ |
+ SkAutoTUnref<SkStream> stream(SkStream::NewFromFile(pathName.c_str())); |
+ if (!stream.get()) { |
+ DEBUG_FONT(("---- SystemFonts[%d] file=%s (NOT EXIST)", i, filename.c_str())); |
+ continue; |
+ } |
+ |
+ SkString fontName; |
+ SkTypeface::Style style; |
+ bool isFixedWidth; |
+ if (!SkTypeface_FreeType::ScanFont(stream.get(), family->fFontFiles[i].fIndex, |
+ &fontName, &style, &isFixedWidth)) |
+ { |
+ DEBUG_FONT(("---- SystemFonts[%d] file=%s (INVALID)", i, filename.c_str())); |
+ continue; |
+ } |
+ |
+ fStyles.push_back().reset(SkNEW_ARGS(SkTypeface_AndroidSystem, |
+ (pathName, 0, |
+ style, isFixedWidth, fontName))); |
+ } |
+ } |
+ |
+ virtual int count() SK_OVERRIDE { |
+ return fStyles.count(); |
+ } |
+ virtual void getStyle(int index, SkFontStyle* style, SkString* name) SK_OVERRIDE { |
+ if (index < 0 || fStyles.count() <= index) { |
+ return; |
+ } |
+ if (style) { |
+ *style = this->style(index); |
+ } |
+ if (name) { |
+ name->reset(); |
+ } |
+ } |
+ virtual SkTypeface* createTypeface(int index) SK_OVERRIDE { |
+ if (index < 0 || fStyles.count() <= index) { |
+ return NULL; |
+ } |
+ return SkRef(fStyles[index].get()); |
+ } |
+ |
+ /** Find the typeface in this style set that most closely matches the given pattern. |
+ * TODO: consider replacing with SkStyleSet_Indirect::matchStyle(); |
+ * this simpler version using match_score() passes all our tests. |
+ */ |
+ virtual SkTypeface* matchStyle(const SkFontStyle& pattern) SK_OVERRIDE { |
+ if (0 == fStyles.count()) { |
+ return NULL; |
+ } |
+ SkTypeface* closest = fStyles[0]; |
+ int minScore = std::numeric_limits<int>::max(); |
+ for (int i = 0; i < fStyles.count(); ++i) { |
+ SkFontStyle style = this->style(i); |
+ int score = match_score(pattern, style); |
+ if (score < minScore) { |
+ closest = fStyles[i]; |
+ minScore = score; |
+ } |
+ } |
+ return SkRef(closest); |
+ } |
+ |
+private: |
+ SkFontStyle style(int index) { |
+ return SkFontStyle(this->weight(index), SkFontStyle::kNormal_Width, |
+ this->slant(index)); |
+ } |
+ SkFontStyle::Weight weight(int index) { |
+ if (fStyles[index]->isBold()) return SkFontStyle::kBold_Weight; |
+ return SkFontStyle::kNormal_Weight; |
+ } |
+ SkFontStyle::Slant slant(int index) { |
+ if (fStyles[index]->isItalic()) return SkFontStyle::kItalic_Slant; |
+ return SkFontStyle::kUpright_Slant; |
+ } |
+ static int match_score(const SkFontStyle& pattern, const SkFontStyle& candidate) { |
+ int score = 0; |
+ score += abs((pattern.width() - candidate.width()) * 100); |
+ score += abs((pattern.isItalic() == candidate.isItalic()) ? 0 : 1000); |
+ score += abs(pattern.weight() - candidate.weight()); |
+ return score; |
+ } |
+ |
+ |
+ FontFamily* fFontFamily; |
+ SkTArray<SkAutoTUnref<SkTypeface>, true> fStyles; |
+ |
+ friend struct NameToFamily; |
+ friend class SkFontMgr_Android; |
+ |
+ typedef SkFontStyleSet INHERITED; |
+}; |
+ |
+/** On Android a single family can have many names, but our API assumes unique names. |
+ * Map names to the back end so that all names for a given family refer to the same |
+ * (non-replicated) set of typefaces. |
+ * SkTDict<> doesn't let us do index-based lookup, so we write our own mapping. |
+ */ |
+struct NameToFamily { |
+ SkString name; |
+ SkFontStyleSet_Android* styleSet; |
+}; |
+ |
+class SkFontMgr_Android : public SkFontMgr { |
+public: |
+ SkFontMgr_Android() { |
+ SkTDArray<FontFamily*> fontFamilies; |
+ SkFontConfigParser::GetFontFamilies(fontFamilies); |
+ this->buildNameToFamilyMap(fontFamilies); |
+ this->findDefaultFont(); |
+ } |
+ |
+protected: |
+ /** Returns not how many families we have, but how many unique names |
+ * exist among the families. |
+ */ |
+ virtual int onCountFamilies() const SK_OVERRIDE { |
+ return fNameToFamilyMap.count(); |
+ } |
+ |
+ virtual void onGetFamilyName(int index, SkString* familyName) const SK_OVERRIDE { |
+ if (index < 0 || fNameToFamilyMap.count() <= index) { |
+ familyName->reset(); |
+ return; |
+ } |
+ familyName->set(fNameToFamilyMap[index].name); |
+ } |
+ |
+ virtual SkFontStyleSet* onCreateStyleSet(int index) const SK_OVERRIDE { |
+ if (index < 0 || fNameToFamilyMap.count() <= index) { |
+ return NULL; |
+ } |
+ return SkRef(fNameToFamilyMap[index].styleSet); |
+ } |
+ |
+ virtual SkFontStyleSet* onMatchFamily(const char familyName[]) const SK_OVERRIDE { |
+ if (!familyName) { |
+ return NULL; |
+ } |
+ SkAutoAsciiToLC tolc(familyName); |
+ for (int i = 0; i < fNameToFamilyMap.count(); ++i) { |
+ if (fNameToFamilyMap[i].name.equals(tolc.lc())) { |
+ return SkRef(fNameToFamilyMap[i].styleSet); |
+ } |
+ } |
+ return NULL; |
+ } |
+ |
+ virtual SkTypeface* onMatchFamilyStyle(const char familyName[], |
+ const SkFontStyle& style) const SK_OVERRIDE { |
+ SkAutoTUnref<SkFontStyleSet> sset(this->matchFamily(familyName)); |
+ return sset->matchStyle(style); |
+ } |
+ |
+ virtual SkTypeface* onMatchFaceStyle(const SkTypeface* typeface, |
+ const SkFontStyle& style) const SK_OVERRIDE { |
+ for (int i = 0; i < fFontStyleSets.count(); ++i) { |
+ for (int j = 0; j < fFontStyleSets[i]->fStyles.count(); ++j) { |
+ if (fFontStyleSets[i]->fStyles[j] == typeface) { |
+ return fFontStyleSets[i]->matchStyle(style); |
+ } |
+ } |
+ } |
+ return NULL; |
+ } |
+ |
+ virtual SkTypeface* onCreateFromData(SkData* data, int ttcIndex) const SK_OVERRIDE { |
+ SkAutoTUnref<SkStream> stream(new SkMemoryStream(data)); |
+ return this->createFromStream(stream, ttcIndex); |
+ } |
+ |
+ virtual SkTypeface* onCreateFromFile(const char path[], int ttcIndex) const SK_OVERRIDE { |
+ SkAutoTUnref<SkStream> stream(SkStream::NewFromFile(path)); |
+ return stream.get() ? this->createFromStream(stream, ttcIndex) : NULL; |
+ } |
+ |
+ virtual SkTypeface* onCreateFromStream(SkStream* s, int ttcIndex) const SK_OVERRIDE { |
+ SkAutoTUnref<SkStream> stream(s); |
+ |
+ bool isFixedPitch; |
+ SkTypeface::Style style; |
+ SkString name; |
+ if (!SkTypeface_FreeType::ScanFont(stream, ttcIndex, &name, &style, &isFixedPitch)) { |
+ return NULL; |
+ } |
+ return SkNEW_ARGS(SkTypeface_AndroidStream, (stream.detach(), ttcIndex, |
+ style, isFixedPitch, name)); |
+ } |
+ |
+ |
+ virtual SkTypeface* onLegacyCreateTypeface(const char familyName[], |
+ unsigned styleBits) const SK_OVERRIDE { |
+ SkTypeface::Style oldStyle = (SkTypeface::Style)styleBits; |
+ SkFontStyle style = SkFontStyle(oldStyle & SkTypeface::kBold |
+ ? SkFontStyle::kBold_Weight |
+ : SkFontStyle::kNormal_Weight, |
+ SkFontStyle::kNormal_Width, |
+ oldStyle & SkTypeface::kItalic |
+ ? SkFontStyle::kItalic_Slant |
+ : SkFontStyle::kUpright_Slant); |
+ SkTypeface* tf = NULL; |
+ |
+ if (NULL != familyName) { |
+ // On Android, we must return NULL when we can't find the requested |
+ // named typeface so that the system/app can provide their own recovery |
+ // mechanism. On other platforms we'd provide a typeface from the |
+ // default family instead. |
+ tf = this->onMatchFamilyStyle(familyName, style); |
+ } else { |
+ tf = fDefaultFamily->matchStyle(style); |
+ } |
+ |
+ // TODO: double ref? qv matchStyle() |
+ return SkSafeRef(tf); |
+ } |
+ |
+ |
+private: |
+ |
+ SkTArray<SkAutoTUnref<SkFontStyleSet_Android>, true> fFontStyleSets; |
+ SkFontStyleSet* fDefaultFamily; |
+ SkTypeface* fDefaultTypeface; |
+ |
+ SkTDArray<NameToFamily> fNameToFamilyMap; |
+ |
+ void buildNameToFamilyMap(SkTDArray<FontFamily*> families) { |
+ for (int i = 0; i < families.count(); i++) { |
+ fFontStyleSets.push_back().reset( |
+ SkNEW_ARGS(SkFontStyleSet_Android, (families[i]))); |
+ for (int j = 0; j < families[i]->fNames.count(); j++) { |
+ NameToFamily* nextEntry = fNameToFamilyMap.append(); |
+ SkNEW_PLACEMENT_ARGS(&nextEntry->name, SkString, (families[i]->fNames[j])); |
+ nextEntry->styleSet = fFontStyleSets.back().get(); |
+ } |
+ } |
+ } |
+ |
+ void findDefaultFont() { |
+ SkASSERT(!fFontStyleSets.empty()); |
+ |
+ static const char* gDefaultNames[] = { "sans-serif" }; |
+ for (size_t i = 0; i < SK_ARRAY_COUNT(gDefaultNames); ++i) { |
+ SkFontStyleSet* set = this->onMatchFamily(gDefaultNames[i]); |
+ if (NULL == set) { |
+ continue; |
+ } |
+ SkTypeface* tf = set->matchStyle(SkFontStyle()); |
+ if (NULL == tf) { |
+ continue; |
+ } |
+ fDefaultFamily = set; |
+ fDefaultTypeface = tf; |
+ break; |
+ } |
+ if (NULL == fDefaultTypeface) { |
+ fDefaultFamily = fFontStyleSets[0]; |
+ fDefaultTypeface = fDefaultFamily->createTypeface(0); |
+ } |
+ SkASSERT(fDefaultFamily); |
+ SkASSERT(fDefaultTypeface); |
+ } |
+ |
+ typedef SkFontMgr INHERITED; |
+}; |
+ |
+/////////////////////////////////////////////////////////////////////////////// |
+ |
+SkFontMgr* SkFontMgr::Factory() { |
+ return SkNEW(SkFontMgr_Android); |
+} |