// Copyright (c) 2012, Thomas Goyne // // Permission to use, copy, modify, and distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice appear in all copies. // // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. // // Aegisub Project http://www.aegisub.org/ /// @file font_file_lister_fontconfig.cpp /// @brief Font Config-based font collector /// @ingroup font_collector /// #include "font_file_lister_fontconfig.h" #include #include #ifdef __APPLE__ #include #endif #include #include #include #include namespace { FcConfig *init_fontconfig() { #ifdef __APPLE__ FcConfig *config = FcConfigCreate(); std::string conf_path = agi::util::GetBundleResourcesDirectory() + "/etc/fonts/fonts.conf"; if (FcConfigParseAndLoad(config, (unsigned char *)conf_path.c_str(), FcTrue)) return config; LOG_E("font_collector/fontconfig") << "Loading fontconfig configuration file failed"; FcConfigDestroy(config); #endif return FcInitLoadConfig(); } void find_font(FcFontSet *src, FcFontSet *dst, std::string const& family) { if (!src) return; for (FcPattern *pat : boost::make_iterator_range(&src->fonts[0], &src->fonts[src->nfont])) { int val; if (FcPatternGetBool(pat, FC_OUTLINE, 0, &val) != FcResultMatch || val != FcTrue) continue; FcChar8 *str; for (int i = 0; FcPatternGetString(pat, FC_FULLNAME, i, &str) == FcResultMatch; ++i) { if (boost::to_lower_copy(std::string((char *)str)) == family) { FcFontSetAdd(dst, FcPatternDuplicate(pat)); goto skip_family; } } for (int i = 0; FcPatternGetString(pat, FC_FAMILY, i, &str) == FcResultMatch; ++i) { if (boost::to_lower_copy(std::string((char *)str)) == family) { FcFontSetAdd(dst, FcPatternDuplicate(pat)); break; } } skip_family:; } } } FontConfigFontFileLister::FontConfigFontFileLister(FontCollectorStatusCallback cb) : config(init_fontconfig(), FcConfigDestroy) { cb(_("Updating font cache\n"), 0); FcConfigBuildFonts(config); } FontFileLister::CollectionResult FontConfigFontFileLister::GetFontPaths(std::string const& facename, int bold, bool italic, std::set const& characters) { CollectionResult ret; std::string family(facename); if (family[0] == '@') family.erase(0, 1); boost::to_lower(family); int weight = bold == 0 ? 80 : bold == 1 ? 200 : bold; int slant = italic ? 110 : 0; // Create a fontconfig pattern to match the desired weight/slant agi::scoped_holder pat(FcPatternCreate(), FcPatternDestroy); if (!pat) return ret; FcPatternAddBool(pat, FC_OUTLINE, true); FcPatternAddInteger(pat, FC_SLANT, slant); FcPatternAddInteger(pat, FC_WEIGHT, weight); FcDefaultSubstitute(pat); if (!FcConfigSubstitute(config, pat, FcMatchPattern)) return ret; // Create a font set with only correctly named fonts // This is needed because the patterns returned by font matching only // include the first family and fullname, so we can't always verify that // we got the actual font we were asking for after the fact agi::scoped_holder fset(FcFontSetCreate(), FcFontSetDestroy); find_font(FcConfigGetFonts(config, FcSetApplication), fset, family); find_font(FcConfigGetFonts(config, FcSetSystem), fset, family); // Get the best match from fontconfig FcResult result; FcFontSet *sets[] = { (FcFontSet*)fset }; agi::scoped_holder match(FcFontSetMatch(config, sets, 1, pat, &result), FcPatternDestroy); if (!match) return ret; FcChar8 *file; if(FcPatternGetString(match, FC_FILE, 0, &file) != FcResultMatch) return ret; FcCharSet *charset; if (FcPatternGetCharSet(match, FC_CHARSET, 0, &charset) == FcResultMatch) { for (wxUniChar chr : characters) { if (!FcCharSetHasChar(charset, chr)) ret.missing += chr; } } int actual_weight = weight; if (FcPatternGetInteger(match, FC_WEIGHT, 0, &actual_weight) == FcResultMatch) ret.fake_bold = actual_weight < weight; int actual_slant = slant; if (FcPatternGetInteger(match, FC_SLANT, 0, &actual_slant) == FcResultMatch) ret.fake_italic = italic && !actual_slant; ret.paths.emplace_back((const char *)file); return ret; }