From c117d9b3e545107b008baf6b2198ad1efa3db5e5 Mon Sep 17 00:00:00 2001 From: WerWolv Date: Sun, 19 Jan 2025 23:07:53 +0100 Subject: [PATCH] fix: Crash when changing DPI --- main/gui/source/window/window.cpp | 7 +++ plugins/fonts/include/font_atlas.hpp | 69 +++++++++---------------- plugins/fonts/include/font_settings.hpp | 2 +- plugins/fonts/source/library_fonts.cpp | 46 +++++------------ 4 files changed, 44 insertions(+), 80 deletions(-) diff --git a/main/gui/source/window/window.cpp b/main/gui/source/window/window.cpp index fa04d3d31255b..0045b3b3af4c1 100644 --- a/main/gui/source/window/window.cpp +++ b/main/gui/source/window/window.cpp @@ -901,6 +901,11 @@ namespace hex { if (!isMainWindow(window)) return; ImHexApi::System::impl::setMainWindowPosition(x, y); + + int width = 0, height = 0; + glfwGetWindowSize(window, &width, &height); + ImHexApi::System::impl::setMainWindowPosition(x, y); + ImHexApi::System::impl::setMainWindowSize(width, height); }); // Register window resize callback @@ -1091,6 +1096,8 @@ namespace hex { io.ConfigWindowsMoveFromTitleBarOnly = true; io.FontGlobalScale = 1.0F; + ImGui::GetCurrentContext()->FontAtlasOwnedByContext = false; + if (glfwGetPrimaryMonitor() != nullptr) { if (ImHexApi::System::isMutliWindowModeEnabled()) io.ConfigFlags |= ImGuiConfigFlags_ViewportsEnable; diff --git a/plugins/fonts/include/font_atlas.hpp b/plugins/fonts/include/font_atlas.hpp index 2f877e1216b31..ec2867dc09757 100644 --- a/plugins/fonts/include/font_atlas.hpp +++ b/plugins/fonts/include/font_atlas.hpp @@ -40,9 +40,9 @@ namespace hex::fonts { enableUnicodeCharacters(false); // Set the default configuration for the font atlas - m_config.OversampleH = m_config.OversampleV = 1; - m_config.PixelSnapH = true; - m_config.MergeMode = false; + m_defaultConfig.OversampleH = m_defaultConfig.OversampleV = 1; + m_defaultConfig.PixelSnapH = true; + m_defaultConfig.MergeMode = false; // Make sure the font atlas doesn't get too large, otherwise weaker GPUs might reject it m_fontAtlas->Flags |= ImFontAtlasFlags_NoPowerOfTwoHeight; @@ -52,42 +52,23 @@ namespace hex::fonts { FontAtlas(const FontAtlas &) = delete; FontAtlas &operator=(const FontAtlas &) = delete; - FontAtlas(FontAtlas &&other) noexcept { - this->m_fontAtlas = other.m_fontAtlas; - other.m_fontAtlas = nullptr; - - this->m_fontSizes = std::move(other.m_fontSizes); - this->m_config = other.m_config; - this->m_glyphRange = std::move(other.m_glyphRange); - this->m_fontData = std::move(other.m_fontData); - } - - FontAtlas &operator=(FontAtlas &&other) noexcept { - this->m_fontAtlas = other.m_fontAtlas; - other.m_fontAtlas = nullptr; - - this->m_fontSizes = std::move(other.m_fontSizes); - this->m_config = other.m_config; - this->m_glyphRange = std::move(other.m_glyphRange); - this->m_fontData = std::move(other.m_fontData); - - return *this; - } - ~FontAtlas() { - if (m_fontAtlas != nullptr) + if (m_fontAtlas != nullptr) { + m_fontAtlas->Locked = false; IM_DELETE(m_fontAtlas); + m_fontAtlas = nullptr; + } } Font addDefaultFont() { - ImFontConfig config = m_config; + auto &config = m_fontConfigs.emplace_back(m_defaultConfig); config.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Monochrome | ImGuiFreeTypeBuilderFlags_MonoHinting; config.SizePixels = std::floor(getAdjustedFontSize(ImHexApi::System::getGlobalScale() * 13.0F)); auto font = m_fontAtlas->AddFontDefault(&config); m_fontSizes.emplace_back(false, config.SizePixels); - m_config.MergeMode = true; + m_defaultConfig.MergeMode = true; return Font(font); } @@ -95,14 +76,14 @@ namespace hex::fonts { Font addFontFromMemory(const std::vector &fontData, float fontSize, bool scalable, ImVec2 offset, const ImVector &glyphRange = {}) { auto &storedFontData = m_fontData.emplace_back(fontData); - ImFontConfig config = m_config; + auto &config = m_fontConfigs.emplace_back(m_defaultConfig); config.FontDataOwnedByAtlas = false; config.GlyphOffset = { offset.x, offset.y }; auto font = m_fontAtlas->AddFontFromMemoryTTF(storedFontData.data(), int(storedFontData.size()), getAdjustedFontSize(fontSize), &config, !glyphRange.empty() ? glyphRange.Data : m_glyphRange.Data); m_fontSizes.emplace_back(scalable, fontSize); - m_config.MergeMode = true; + m_defaultConfig.MergeMode = true; return Font(font); } @@ -121,23 +102,23 @@ namespace hex::fonts { void setBold(bool enabled) { if (enabled) - m_config.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Bold; + m_defaultConfig.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Bold; else - m_config.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Bold; + m_defaultConfig.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Bold; } void setItalic(bool enabled) { if (enabled) - m_config.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Oblique; + m_defaultConfig.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Oblique; else - m_config.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Oblique; + m_defaultConfig.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Oblique; } void setAntiAliasing(bool enabled) { if (enabled) - m_config.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Monochrome | ImGuiFreeTypeBuilderFlags_MonoHinting; + m_defaultConfig.FontBuilderFlags &= ~ImGuiFreeTypeBuilderFlags_Monochrome | ImGuiFreeTypeBuilderFlags_MonoHinting; else - m_config.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Monochrome | ImGuiFreeTypeBuilderFlags_MonoHinting; + m_defaultConfig.FontBuilderFlags |= ImGuiFreeTypeBuilderFlags_Monochrome | ImGuiFreeTypeBuilderFlags_MonoHinting; } void enableUnicodeCharacters(bool enabled) { @@ -177,14 +158,12 @@ namespace hex::fonts { } [[nodiscard]] ImFontAtlas* getAtlas() { - auto result = m_fontAtlas; - - return result; + return m_fontAtlas; } float calculateFontDescend(const ImHexApi::Fonts::Font &font, float fontSize) const { auto atlas = std::make_unique(); - auto cfg = m_config; + auto cfg = m_defaultConfig; // Calculate the expected font size auto size = fontSize; @@ -214,11 +193,8 @@ namespace hex::fonts { } void reset() { - /*IM_DELETE(m_fontAtlas); - m_fontAtlas = IM_NEW(ImFontAtlas);*/ - m_fontData.clear(); - m_config.MergeMode = false; + m_defaultConfig.MergeMode = false; } void updateFontScaling(float newScaling) { @@ -242,9 +218,10 @@ namespace hex::fonts { } private: - ImFontAtlas* m_fontAtlas; + ImFontAtlas* m_fontAtlas = nullptr; std::vector> m_fontSizes; - ImFontConfig m_config; + ImFontConfig m_defaultConfig; + std::list m_fontConfigs; ImVector m_glyphRange; std::list> m_fontData; diff --git a/plugins/fonts/include/font_settings.hpp b/plugins/fonts/include/font_settings.hpp index f98130019d294..fd92810a3f841 100644 --- a/plugins/fonts/include/font_settings.hpp +++ b/plugins/fonts/include/font_settings.hpp @@ -19,7 +19,7 @@ namespace hex::fonts { private: std::string m_selectedFontName; - bool m_pixelPerfectFont = false; + bool m_pixelPerfectFont = true; }; class SliderPoints : public ContentRegistry::Settings::Widgets::SliderFloat { diff --git a/plugins/fonts/source/library_fonts.cpp b/plugins/fonts/source/library_fonts.cpp index 88ed6a7a00bbe..7739e2049f2d8 100644 --- a/plugins/fonts/source/library_fonts.cpp +++ b/plugins/fonts/source/library_fonts.cpp @@ -15,18 +15,18 @@ namespace hex::fonts { bool buildFontAtlas(FontAtlas *fontAtlas, std::fs::path fontPath, bool pixelPerfectFont, float fontSize, bool loadUnicodeCharacters, bool bold, bool italic, bool antialias); - static AutoReset> s_fontAtlases; + static AutoReset>> s_fontAtlases; - void loadFont(const ContentRegistry::Settings::Widgets::Widget &widget, const UnlocalizedString &name, ImFont **font) { + void loadFont(const ContentRegistry::Settings::Widgets::Widget &widget, const UnlocalizedString &name, ImFont **font, float scale) { const auto &settings = static_cast(widget); - FontAtlas atlas; + auto atlas = std::make_unique(); const bool atlasBuilt = buildFontAtlas( - &atlas, + atlas.get(), settings.getFontPath(), settings.isPixelPerfectFont(), - settings.getFontSize(), + settings.getFontSize() * scale, true, settings.isBold(), settings.isItalic(), @@ -35,10 +35,10 @@ namespace hex::fonts { if (!atlasBuilt) { buildFontAtlas( - &atlas, + atlas.get(), "", false, - settings.getFontSize(), + settings.getFontSize() * scale, false, settings.isBold(), settings.isItalic(), @@ -48,32 +48,9 @@ namespace hex::fonts { log::error("Failed to load font {}! Reverting back to default font!", name.get()); } - if (*font != nullptr) { - EventDPIChanged::unsubscribe(*font); - } - - *font = atlas.getAtlas()->Fonts[0]; + *font = atlas->getAtlas()->Fonts[0]; - s_fontAtlases->insert_or_assign(*font, std::move(atlas)); (*s_fontAtlases)[*font] = std::move(atlas); - - EventDPIChanged::subscribe(*font, [&atlas, font](float, float newScaling) { - atlas.updateFontScaling(newScaling); - - if (atlas.build()) { - auto &io = ImGui::GetIO(); - auto prevFont = io.Fonts; - - { - io.Fonts = atlas.getAtlas(); - ImGui_ImplOpenGL3_DestroyFontsTexture(); - ImGui_ImplOpenGL3_CreateFontsTexture(); - io.Fonts = prevFont; - } - - *font = atlas.getAtlas()->Fonts[0]; - } - }); } bool setupFonts() { @@ -87,10 +64,13 @@ namespace hex::fonts { return; } - loadFont(widget, name, &font); + loadFont(widget, name, &font, ImHexApi::System::getGlobalScale()); }); - loadFont(widget.getWidget(), name, &font); + loadFont(widget.getWidget(), name, &font, ImHexApi::System::getGlobalScale()); + EventDPIChanged::subscribe(font, [&widget, name, &font](float, float newScaling) { + loadFont(widget.getWidget(), name, &font, newScaling); + }); } return true;