/* * Copyright (c) 2022-2023, Andreas Kling * Copyright (c) 2023, Linus Groh * * SPDX-License-Identifier: BSD-2-Clause */ #include "FontPlugin.h" #include #include #include #include #include #include #include #ifdef USE_FONTCONFIG # include #endif namespace Ladybird { FontPlugin::FontPlugin(bool is_layout_test_mode, Gfx::SystemFontProvider* font_provider) : m_is_layout_test_mode(is_layout_test_mode) { #ifdef USE_FONTCONFIG { auto fontconfig_initialized = FcInit(); VERIFY(fontconfig_initialized); } #endif if (!font_provider) font_provider = &static_cast(Gfx::FontDatabase::the().install_system_font_provider(make())); if (is(*font_provider)) { auto& path_font_provider = static_cast(*font_provider); // Load anything we can find in the system's font directories for (auto const& path : Core::StandardPaths::font_directories().release_value_but_fixme_should_propagate_errors()) path_font_provider.load_all_fonts_from_uri(MUST(String::formatted("file://{}", path))); } update_generic_fonts(); auto default_font_name = generic_font_name(Web::Platform::GenericFont::UiSansSerif); m_default_font = Gfx::FontDatabase::the().get(default_font_name, 12.0, 400, Gfx::FontWidth::Normal, 0); VERIFY(m_default_font); auto default_fixed_width_font_name = generic_font_name(Web::Platform::GenericFont::UiMonospace); m_default_fixed_width_font = Gfx::FontDatabase::the().get(default_fixed_width_font_name, 12.0, 400, Gfx::FontWidth::Normal, 0); VERIFY(m_default_fixed_width_font); } FontPlugin::~FontPlugin() = default; Gfx::Font& FontPlugin::default_font() { return *m_default_font; } Gfx::Font& FontPlugin::default_fixed_width_font() { return *m_default_fixed_width_font; } RefPtr FontPlugin::default_emoji_font(float point_size) { FlyString default_emoji_font_name; if (m_is_layout_test_mode) { default_emoji_font_name = "Noto Emoji"_fly_string; } else { #ifdef AK_OS_MACOS default_emoji_font_name = "Apple Color Emoji"_fly_string; #else default_emoji_font_name = "Noto Color Emoji"_fly_string; #endif } return Gfx::FontDatabase::the().get(default_emoji_font_name, point_size, 400, Gfx::FontWidth::Normal, 0); } #ifdef USE_FONTCONFIG static Optional query_fontconfig_for_generic_family(Web::Platform::GenericFont generic_font) { char const* pattern_string = nullptr; switch (generic_font) { case Web::Platform::GenericFont::Cursive: pattern_string = "cursive"; break; case Web::Platform::GenericFont::Fantasy: pattern_string = "fantasy"; break; case Web::Platform::GenericFont::Monospace: pattern_string = "monospace"; break; case Web::Platform::GenericFont::SansSerif: pattern_string = "sans-serif"; break; case Web::Platform::GenericFont::Serif: pattern_string = "serif"; break; case Web::Platform::GenericFont::UiMonospace: pattern_string = "monospace"; break; case Web::Platform::GenericFont::UiRounded: pattern_string = "sans-serif"; break; case Web::Platform::GenericFont::UiSansSerif: pattern_string = "sans-serif"; break; case Web::Platform::GenericFont::UiSerif: pattern_string = "serif"; break; default: VERIFY_NOT_REACHED(); } auto* config = FcConfigGetCurrent(); VERIFY(config); FcPattern* pattern = FcNameParse(reinterpret_cast(pattern_string)); VERIFY(pattern); auto success = FcConfigSubstitute(config, pattern, FcMatchPattern); VERIFY(success); FcDefaultSubstitute(pattern); // Never select bitmap fonts. success = FcPatternAddBool(pattern, FC_SCALABLE, FcTrue); VERIFY(success); // FIXME: Enable this once we can handle OpenType variable fonts. success = FcPatternAddBool(pattern, FC_VARIABLE, FcFalse); VERIFY(success); Optional name; FcResult result {}; if (auto* matched = FcFontMatch(config, pattern, &result)) { FcChar8* family = nullptr; if (FcPatternGetString(matched, FC_FAMILY, 0, &family) == FcResultMatch) { auto const* family_cstring = reinterpret_cast(family); if (auto string = String::from_utf8(StringView { family_cstring, strlen(family_cstring) }); !string.is_error()) { name = string.release_value(); } } FcPatternDestroy(matched); } FcPatternDestroy(pattern); return name; } #endif void FontPlugin::update_generic_fonts() { // How we choose which system font to use for each CSS font: // 1. Try a list of known-suitable fonts with their names hard-coded below. // This is rather weird, but it's how things work right now. // We should eventually have a way to query the system for the default font. // Furthermore, we should allow overriding via some kind of configuration mechanism. m_generic_font_names.resize(static_cast(Web::Platform::GenericFont::__Count)); auto update_mapping = [&](Web::Platform::GenericFont generic_font, ReadonlySpan fallbacks) { if (m_is_layout_test_mode) { m_generic_font_names[static_cast(generic_font)] = "SerenitySans"_fly_string; return; } RefPtr gfx_font; #ifdef USE_FONTCONFIG auto name = query_fontconfig_for_generic_family(generic_font); if (name.has_value()) { gfx_font = Gfx::FontDatabase::the().get(name.value(), 16, 400, Gfx::FontWidth::Normal, 0); } #endif if (!gfx_font) { for (auto const& fallback : fallbacks) { gfx_font = Gfx::FontDatabase::the().get(fallback, 16, 400, Gfx::FontWidth::Normal, 0); if (gfx_font) break; } } m_generic_font_names[static_cast(generic_font)] = gfx_font ? gfx_font->family() : String {}; }; // Fallback fonts to look for if Gfx::Font can't load expected font // The lists are basically arbitrary, taken from https://www.w3.org/Style/Examples/007/fonts.en.html // (We also add Android-specific font names to the list from W3 where required.) Vector cursive_fallbacks { "Comic Sans MS"_fly_string, "Comic Sans"_fly_string, "Apple Chancery"_fly_string, "Bradley Hand"_fly_string, "Brush Script MT"_fly_string, "Snell Roundhand"_fly_string, "URW Chancery L"_fly_string, "Dancing Script"_fly_string }; Vector fantasy_fallbacks { "Impact"_fly_string, "Luminari"_fly_string, "Chalkduster"_fly_string, "Jazz LET"_fly_string, "Blippo"_fly_string, "Stencil Std"_fly_string, "Marker Felt"_fly_string, "Trattatello"_fly_string, "Coming Soon"_fly_string }; Vector monospace_fallbacks { "Andale Mono"_fly_string, "Courier New"_fly_string, "Courier"_fly_string, "FreeMono"_fly_string, "OCR A Std"_fly_string, "DejaVu Sans Mono"_fly_string, "Droid Sans Mono"_fly_string, "Liberation Mono"_fly_string }; Vector sans_serif_fallbacks { "Arial"_fly_string, "Helvetica"_fly_string, "Verdana"_fly_string, "Trebuchet MS"_fly_string, "Gill Sans"_fly_string, "Noto Sans"_fly_string, "Avantgarde"_fly_string, "Optima"_fly_string, "Arial Narrow"_fly_string, "Liberation Sans"_fly_string, "Roboto"_fly_string }; Vector serif_fallbacks { "Times"_fly_string, "Times New Roman"_fly_string, "Didot"_fly_string, "Georgia"_fly_string, "Palatino"_fly_string, "Bookman"_fly_string, "New Century Schoolbook"_fly_string, "American Typewriter"_fly_string, "Liberation Serif"_fly_string, "Roman"_fly_string, "Noto Serif"_fly_string }; update_mapping(Web::Platform::GenericFont::Cursive, cursive_fallbacks); update_mapping(Web::Platform::GenericFont::Fantasy, fantasy_fallbacks); update_mapping(Web::Platform::GenericFont::Monospace, monospace_fallbacks); update_mapping(Web::Platform::GenericFont::SansSerif, sans_serif_fallbacks); update_mapping(Web::Platform::GenericFont::Serif, serif_fallbacks); update_mapping(Web::Platform::GenericFont::UiMonospace, monospace_fallbacks); update_mapping(Web::Platform::GenericFont::UiRounded, sans_serif_fallbacks); update_mapping(Web::Platform::GenericFont::UiSansSerif, sans_serif_fallbacks); update_mapping(Web::Platform::GenericFont::UiSerif, serif_fallbacks); } FlyString FontPlugin::generic_font_name(Web::Platform::GenericFont generic_font) { return m_generic_font_names[static_cast(generic_font)]; } }