| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 |
- /*
- * This source file is part of RmlUi, the HTML/CSS Interface Middleware
- *
- * For the latest information, see http://github.com/mikke89/RmlUi
- *
- * Copyright (c) 2008-2010 CodePoint Ltd, Shift Technology Ltd
- * Copyright (c) 2019-2023 The RmlUi Team, and contributors
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- *
- */
- #include "FontProvider.h"
- #include "../../../Include/RmlUi/Core/Core.h"
- #include "../../../Include/RmlUi/Core/FileInterface.h"
- #include "../../../Include/RmlUi/Core/Log.h"
- #include "../../../Include/RmlUi/Core/Math.h"
- #include "../../../Include/RmlUi/Core/StringUtilities.h"
- #include "../ComputeProperty.h"
- #include "FontFace.h"
- #include "FontFamily.h"
- #include "FreeTypeInterface.h"
- #include <algorithm>
- namespace Rml {
- static FontProvider* g_font_provider = nullptr;
- FontProvider::FontProvider()
- {
- RMLUI_ASSERT(!g_font_provider);
- }
- FontProvider::~FontProvider()
- {
- RMLUI_ASSERT(g_font_provider == this);
- }
- bool FontProvider::Initialise()
- {
- RMLUI_ASSERT(!g_font_provider);
- if (!FreeType::Initialise())
- return false;
- g_font_provider = new FontProvider;
- return true;
- }
- void FontProvider::Shutdown()
- {
- RMLUI_ASSERT(g_font_provider);
- delete g_font_provider;
- g_font_provider = nullptr;
- FreeType::Shutdown();
- }
- FontProvider& FontProvider::Get()
- {
- RMLUI_ASSERT(g_font_provider);
- return *g_font_provider;
- }
- FontFaceHandleDefault* FontProvider::GetFontFaceHandle(const String& family, Style::FontStyle style, Style::FontWeight weight, int size)
- {
- RMLUI_ASSERTMSG(family == StringUtilities::ToLower(family), "Font family name must be converted to lowercase before entering here.");
- FontFamilyMap& families = Get().font_families;
- auto it = families.find(family);
- if (it == families.end())
- return nullptr;
- return it->second->GetFaceHandle(style, weight, size);
- }
- int FontProvider::CountFallbackFontFaces()
- {
- return (int)Get().fallback_font_faces.size();
- }
- FontFaceHandleDefault* FontProvider::GetFallbackFontFace(int index, int font_size)
- {
- auto& faces = FontProvider::Get().fallback_font_faces;
- if (index >= 0 && index < (int)faces.size())
- return faces[index]->GetHandle(font_size, false);
- return nullptr;
- }
- void FontProvider::ReleaseFontResources()
- {
- RMLUI_ASSERT(g_font_provider);
- for (auto& name_family : g_font_provider->font_families)
- name_family.second->ReleaseFontResources();
- }
- bool FontProvider::LoadFontFace(const String& file_name, bool fallback_face, Style::FontWeight weight)
- {
- FileInterface* file_interface = GetFileInterface();
- FileHandle handle = file_interface->Open(file_name);
- if (!handle)
- {
- Log::Message(Log::LT_ERROR, "Failed to load font face from %s, could not open file.", file_name.c_str());
- return false;
- }
- size_t length = file_interface->Length(handle);
- auto buffer_ptr = UniquePtr<byte[]>(new byte[length]);
- byte* buffer = buffer_ptr.get();
- file_interface->Read(buffer, length, handle);
- file_interface->Close(handle);
- bool result = Get().LoadFontFace(buffer, (int)length, fallback_face, std::move(buffer_ptr), file_name, {}, Style::FontStyle::Normal, weight);
- return result;
- }
- bool FontProvider::LoadFontFace(const byte* data, int data_size, const String& font_family, Style::FontStyle style, Style::FontWeight weight,
- bool fallback_face)
- {
- const String source = "memory";
- bool result = Get().LoadFontFace(data, data_size, fallback_face, nullptr, source, font_family, style, weight);
- return result;
- }
- bool FontProvider::LoadFontFace(const byte* data, int data_size, bool fallback_face, UniquePtr<byte[]> face_memory, const String& source,
- String font_family, Style::FontStyle style, Style::FontWeight weight)
- {
- using Style::FontWeight;
- Vector<FaceVariation> face_variations;
- if (!FreeType::GetFaceVariations(data, data_size, face_variations))
- {
- Log::Message(Log::LT_ERROR, "Failed to load font face from '%s': Invalid or unsupported font face file format.", source.c_str());
- return false;
- }
- Vector<FaceVariation> load_variations;
- if (face_variations.empty())
- {
- load_variations.push_back(FaceVariation{Style::FontWeight::Auto, 0, 0});
- }
- else
- {
- // Iterate through all the face variations and pick the ones to load. The list is already sorted by (weight, width). When weight is set to
- // 'auto' we load all the weights of the face. However, we only want to load one width for each weight.
- for (auto it = face_variations.begin(); it != face_variations.end();)
- {
- if (weight != FontWeight::Auto && it->weight != weight)
- {
- ++it;
- continue;
- }
- // We don't currently have any way for users to select widths, so we search for a regular (medium) value here.
- constexpr int search_width = 100;
- const FontWeight current_weight = it->weight;
- int best_width_distance = Math::Absolute((int)it->width - search_width);
- auto it_best_width = it;
- // Search forward to find the best 'width' with the same weight.
- for (++it; it != face_variations.end(); ++it)
- {
- if (it->weight != current_weight)
- break;
- const int width_distance = Math::Absolute((int)it->width - search_width);
- if (width_distance < best_width_distance)
- {
- best_width_distance = width_distance;
- it_best_width = it;
- }
- }
- load_variations.push_back(*it_best_width);
- }
- }
- if (load_variations.empty())
- {
- Log::Message(Log::LT_ERROR, "Failed to load font face from '%s': Could not locate face with weight %d.", source.c_str(), (int)weight);
- return false;
- }
- for (const FaceVariation& variation : load_variations)
- {
- FontFaceHandleFreetype ft_face = FreeType::LoadFace(data, data_size, source, variation.named_instance_index);
- if (!ft_face)
- return false;
- if (font_family.empty())
- FreeType::GetFaceStyle(ft_face, &font_family, &style, nullptr);
- if (weight == FontWeight::Auto)
- FreeType::GetFaceStyle(ft_face, nullptr, nullptr, &weight);
- const FontWeight variation_weight = (variation.weight == FontWeight::Auto ? weight : variation.weight);
- const String font_face_description = GetFontFaceDescription(font_family, style, variation_weight);
- if (!AddFace(ft_face, font_family, style, variation_weight, fallback_face, std::move(face_memory)))
- {
- Log::Message(Log::LT_ERROR, "Failed to load font face %s from '%s'.", font_face_description.c_str(), source.c_str());
- return false;
- }
- Log::Message(Log::LT_INFO, "Loaded font face %s from '%s'.", font_face_description.c_str(), source.c_str());
- }
- return true;
- }
- bool FontProvider::AddFace(FontFaceHandleFreetype face, const String& family, Style::FontStyle style, Style::FontWeight weight, bool fallback_face,
- UniquePtr<byte[]> face_memory)
- {
- if (family.empty() || weight == Style::FontWeight::Auto)
- return false;
- String family_lower = StringUtilities::ToLower(family);
- FontFamily* font_family = nullptr;
- auto it = font_families.find(family_lower);
- if (it != font_families.end())
- {
- font_family = (FontFamily*)it->second.get();
- }
- else
- {
- auto font_family_ptr = MakeUnique<FontFamily>(family_lower);
- font_family = font_family_ptr.get();
- font_families[family_lower] = std::move(font_family_ptr);
- }
- FontFace* font_face_result = font_family->AddFace(face, style, weight, std::move(face_memory));
- if (font_face_result && fallback_face)
- {
- auto it_fallback_face = std::find(fallback_font_faces.begin(), fallback_font_faces.end(), font_face_result);
- if (it_fallback_face == fallback_font_faces.end())
- {
- fallback_font_faces.push_back(font_face_result);
- }
- }
- return static_cast<bool>(font_face_result);
- }
- } // namespace Rml
|