/* * Copyright (c) 2018-2020, Andreas Kling * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include namespace Gfx { static FontDatabase* s_the; FontDatabase& FontDatabase::the() { if (!s_the) s_the = new FontDatabase; return *s_the; } struct FontDatabase::Private { HashMap> full_name_to_font_map; }; FontDatabase::FontDatabase() : m_private(make()) { Core::DirIterator di("/res/fonts", Core::DirIterator::SkipDots); if (di.has_error()) { fprintf(stderr, "DirIterator: %s\n", di.error_string()); exit(1); } while (di.has_next()) { String name = di.next_path(); if (!name.ends_with(".font")) continue; auto path = String::format("/res/fonts/%s", name.characters()); if (auto font = Gfx::Font::load_from_file(path)) { m_private->full_name_to_font_map.set(font->qualified_name(), font); } } } FontDatabase::~FontDatabase() { } void FontDatabase::for_each_font(Function callback) { Vector> fonts; fonts.ensure_capacity(m_private->full_name_to_font_map.size()); for (auto& it : m_private->full_name_to_font_map) fonts.append(it.value); quick_sort(fonts, [](auto& a, auto& b) { return a->qualified_name() < b->qualified_name(); }); for (auto& font : fonts) callback(*font); } void FontDatabase::for_each_fixed_width_font(Function callback) { Vector> fonts; fonts.ensure_capacity(m_private->full_name_to_font_map.size()); for (auto& it : m_private->full_name_to_font_map) { if (it.value->is_fixed_width()) fonts.append(it.value); } quick_sort(fonts, [](auto& a, auto& b) { return a->qualified_name() < b->qualified_name(); }); for (auto& font : fonts) callback(*font); } RefPtr FontDatabase::get_by_name(const StringView& name) { auto it = m_private->full_name_to_font_map.find(name); if (it == m_private->full_name_to_font_map.end()) { dbgln("Font lookup failed: '{}'", name); return nullptr; } return it->value; } }