From 165403da6794ba50e950fde3353b1205e51c9396 Mon Sep 17 00:00:00 2001 From: WerWolv Date: Wed, 1 Jan 2025 20:39:12 +0100 Subject: [PATCH] fix: ImHex freezing on AMD GPUs when resizing Fixes #1842 --- main/gui/source/window/win_window.cpp | 27 ++++++++++++++++++++++++--- 1 file changed, 24 insertions(+), 3 deletions(-) diff --git a/main/gui/source/window/win_window.cpp b/main/gui/source/window/win_window.cpp index 7a0b37470..5e1baac31 100644 --- a/main/gui/source/window/win_window.cpp +++ b/main/gui/source/window/win_window.cpp @@ -37,6 +37,7 @@ namespace hex { static LONG_PTR s_oldWndProc; static float s_titleBarHeight; static Microsoft::WRL::ComPtr s_taskbarList; + static bool s_useLayeredWindow = true; void nativeErrorMessage(const std::string &message) { log::fatal(message); @@ -644,6 +645,10 @@ namespace hex { win->fullFrame(); DwmFlush(); }); + + // AMD GPUs seem to have issues with Layered Window rendering. Until we figure out + // why that is or AMD fixes the issue on their side, disable it on these GPUs. + s_useLayeredWindow = ImHexApi::System::getGPUVendor() != "ATI Technologies Inc."; } void Window::beginNativeWindowFrame() { @@ -651,11 +656,27 @@ namespace hex { // Remove WS_POPUP style from the window to make various window management tools work auto hwnd = glfwGetWin32Window(m_window); - ::SetWindowLong(hwnd, GWL_STYLE, (GetWindowLong(hwnd, GWL_STYLE) | WS_OVERLAPPEDWINDOW) & ~WS_POPUP); - ::SetWindowLong(hwnd, GWL_EXSTYLE, GetWindowLong(hwnd, GWL_EXSTYLE) | WS_EX_COMPOSITED | WS_EX_LAYERED); + { + auto style = GetWindowLong(hwnd, GWL_STYLE); + style |= WS_OVERLAPPEDWINDOW; + style &= ~WS_POPUP; + + ::SetWindowLong(hwnd, GWL_STYLE, style); + } + + // Make window composited and layered when supported to eradicate any window flickering that happens while resizing + { + auto style = GetWindowLong(hwnd, GWL_EXSTYLE); + style |= WS_EX_COMPOSITED; + + if (s_useLayeredWindow) + style |= WS_EX_LAYERED; + + ::SetWindowLong(hwnd, GWL_EXSTYLE, style); + } if (!ImHexApi::System::impl::isWindowResizable()) { - if (glfwGetWindowAttrib(m_window, GLFW_MAXIMIZED)) { + if (glfwGetWindowAttrib(m_window, GLFW_MAXIMIZED) == GLFW_TRUE) { glfwRestoreWindow(m_window); } }