| /* | 
 |  * Copyright 2017 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include "include/core/SkRefCnt.h" | 
 | #include "include/core/SkStream.h" | 
 | #include "include/ports/SkFontMgr_directory.h" | 
 | #include "src/core/SkFontScanner.h" | 
 | #include "src/core/SkOSFile.h" | 
 | #include "src/ports/SkFontMgr_custom.h" | 
 | #include "src/ports/SkTypeface_FreeType.h" | 
 | #include "src/utils/SkOSPath.h" | 
 |  | 
 | class DirectorySystemFontLoader : public SkFontMgr_Custom::SystemFontLoader { | 
 | public: | 
 |     DirectorySystemFontLoader(const char* dir) : fBaseDirectory(dir) { } | 
 |  | 
 |     void loadSystemFonts(const SkFontScanner* scanner, | 
 |                          SkFontMgr_Custom::Families* families) const override | 
 |     { | 
 |         load_directory_fonts(scanner, fBaseDirectory, ".ttf", families); | 
 |         load_directory_fonts(scanner, fBaseDirectory, ".ttc", families); | 
 |         load_directory_fonts(scanner, fBaseDirectory, ".otf", families); | 
 |         load_directory_fonts(scanner, fBaseDirectory, ".pfb", families); | 
 |  | 
 |         if (families->empty()) { | 
 |             SkFontStyleSet_Custom* family = new SkFontStyleSet_Custom(SkString()); | 
 |             families->push_back().reset(family); | 
 |             family->appendTypeface(sk_make_sp<SkTypeface_Empty>()); | 
 |         } | 
 |     } | 
 |  | 
 | private: | 
 |     static SkFontStyleSet_Custom* find_family(SkFontMgr_Custom::Families& families, | 
 |                                               const char familyName[]) | 
 |     { | 
 |        for (int i = 0; i < families.size(); ++i) { | 
 |             if (families[i]->getFamilyName().equals(familyName)) { | 
 |                 return families[i].get(); | 
 |             } | 
 |         } | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     static void load_directory_fonts(const SkFontScanner* scanner, | 
 |                                      const SkString& directory, const char* suffix, | 
 |                                      SkFontMgr_Custom::Families* families) | 
 |     { | 
 |         SkOSFile::Iter iter(directory.c_str(), suffix); | 
 |         SkString name; | 
 |  | 
 |         while (iter.next(&name, false)) { | 
 |             SkString filename(SkOSPath::Join(directory.c_str(), name.c_str())); | 
 |             std::unique_ptr<SkStreamAsset> stream = SkStream::MakeFromFile(filename.c_str()); | 
 |             if (!stream) { | 
 |                 // SkDebugf("---- failed to open <%s>\n", filename.c_str()); | 
 |                 continue; | 
 |             } | 
 |  | 
 |             int numFaces; | 
 |             if (!scanner->scanFile(stream.get(), &numFaces)) { | 
 |                 // SkDebugf("---- failed to open <%s> as a font\n", filename.c_str()); | 
 |                 continue; | 
 |             } | 
 |  | 
 |             for (int faceIndex = 0; faceIndex < numFaces; ++faceIndex) { | 
 |                 int numInstances; | 
 |                 if (!scanner->scanFace(stream.get(), faceIndex, &numInstances)) { | 
 |                     // SkDebugf("---- failed to open <%s> as a font\n", filename.c_str()); | 
 |                     continue; | 
 |                 } | 
 |                 for (int instanceIndex = 0; instanceIndex <= numInstances; ++instanceIndex) { | 
 |                     bool isFixedPitch; | 
 |                     SkString realname; | 
 |                     SkFontStyle style = SkFontStyle(); // avoid uninitialized warning | 
 |                     if (!scanner->scanInstance(stream.get(), | 
 |                                                faceIndex, | 
 |                                                instanceIndex, | 
 |                                                &realname, | 
 |                                                &style, | 
 |                                                &isFixedPitch, | 
 |                                                nullptr)) { | 
 |                         // SkDebugf("---- failed to open <%s> <%d> as a font\n", | 
 |                         //          filename.c_str(), faceIndex); | 
 |                         continue; | 
 |                     } | 
 |  | 
 |                     SkFontStyleSet_Custom* addTo = find_family(*families, realname.c_str()); | 
 |                     if (nullptr == addTo) { | 
 |                         addTo = new SkFontStyleSet_Custom(realname); | 
 |                         families->push_back().reset(addTo); | 
 |                     } | 
 |                     addTo->appendTypeface(sk_make_sp<SkTypeface_File>( | 
 |                             style, isFixedPitch, true, realname, filename.c_str(), | 
 |                             (instanceIndex << 16) + faceIndex)); | 
 |                 } | 
 |             } | 
 |         } | 
 |  | 
 |         SkOSFile::Iter dirIter(directory.c_str()); | 
 |         while (dirIter.next(&name, true)) { | 
 |             if (name.startsWith(".")) { | 
 |                 continue; | 
 |             } | 
 |             SkString dirname(SkOSPath::Join(directory.c_str(), name.c_str())); | 
 |             load_directory_fonts(scanner, dirname, suffix, families); | 
 |         } | 
 |     } | 
 |  | 
 |     SkString fBaseDirectory; | 
 | }; | 
 |  | 
 | sk_sp<SkFontMgr> SkFontMgr_New_Custom_Directory(const char* dir) { | 
 |     return sk_make_sp<SkFontMgr_Custom>(DirectorySystemFontLoader(dir)); | 
 | } |