From c70a286d01728bd2fa01103191c9da23ad3f8be2 Mon Sep 17 00:00:00 2001 From: Sine Striker <trueful@163.com> Date: 周五, 23 2月 2024 22:03:56 +0800 Subject: [PATCH] Delay filtering the native events --- src/core/contexts/win32windowcontext.cpp | 158 +++++++++++++++++++++++++++++++++++++++------------- 1 files changed, 117 insertions(+), 41 deletions(-) diff --git a/src/core/contexts/win32windowcontext.cpp b/src/core/contexts/win32windowcontext.cpp index e0db522..841273a 100644 --- a/src/core/contexts/win32windowcontext.cpp +++ b/src/core/contexts/win32windowcontext.cpp @@ -359,6 +359,104 @@ return true; } + static inline constexpr bool isNonClientMessage(const UINT message) { + if (((message >= WM_NCCREATE) && (message <= WM_NCACTIVATE)) || + ((message >= WM_NCMOUSEMOVE) && (message <= WM_NCMBUTTONDBLCLK)) || + ((message >= WM_NCXBUTTONDOWN) && (message <= WM_NCXBUTTONDBLCLK)) +#if (WINVER >= _WIN32_WINNT_WIN8) + || ((message >= WM_NCPOINTERUPDATE) && (message <= WM_NCPOINTERUP)) +#endif + || ((message == WM_NCMOUSEHOVER) || (message == WM_NCMOUSELEAVE))) { + return true; + } else { + return false; + } + } + + static MSG createMessageBlock(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { + MSG msg; + msg.hwnd = hWnd; // re-create MSG structure + msg.message = message; // time and pt fields ignored + msg.wParam = wParam; + msg.lParam = lParam; + + const DWORD dwScreenPos = ::GetMessagePos(); + msg.pt.x = GET_X_LPARAM(dwScreenPos); + msg.pt.y = GET_Y_LPARAM(dwScreenPos); + if (!isNonClientMessage(message)) { + ::ScreenToClient(hWnd, &msg.pt); + } + return msg; + } + + static inline constexpr bool isInputMessage(UINT m) { + switch (m) { + case WM_IME_STARTCOMPOSITION: + case WM_IME_ENDCOMPOSITION: + case WM_IME_COMPOSITION: + case WM_INPUT: + case WM_TOUCH: + case WM_MOUSEHOVER: + case WM_MOUSELEAVE: + case WM_NCMOUSEHOVER: + case WM_NCMOUSELEAVE: + case WM_SIZING: + case WM_MOVING: + case WM_SYSCOMMAND: + case WM_COMMAND: + case WM_DWMNCRENDERINGCHANGED: + case WM_PAINT: + return true; + default: + break; + } + return (m >= WM_MOUSEFIRST && m <= WM_MOUSELAST) || + (m >= WM_NCMOUSEMOVE && m <= WM_NCXBUTTONDBLCLK) || + (m >= WM_KEYFIRST && m <= WM_KEYLAST); + } + + static inline QByteArray nativeEventType() { + return QByteArrayLiteral("windows_generic_MSG"); + } + + // Send to QAbstractEventDispatcher + bool filterNativeEvent(MSG *msg, LRESULT *result) { + auto dispatcher = QAbstractEventDispatcher::instance(); + QT_NATIVE_EVENT_RESULT_TYPE filterResult = *result; + if (dispatcher && dispatcher->filterNativeEvent(nativeEventType(), msg, &filterResult)) { + *result = LRESULT(filterResult); + return true; + } + return false; + } + + // Send to QWindowSystemInterface + bool filterNativeEvent(QWindow *window, MSG *msg, LRESULT *result) { + QT_NATIVE_EVENT_RESULT_TYPE filterResult = *result; + if (QWindowSystemInterface::handleNativeEvent(window, nativeEventType(), msg, + &filterResult)) { + *result = LRESULT(filterResult); + return true; + } + return false; + } + + static inline bool forwardFilteredEvents(QWindow *window, HWND hWnd, UINT message, + WPARAM wParam, LPARAM lParam, LRESULT *result) { + MSG msg = createMessageBlock(hWnd, message, wParam, lParam); + + // Run the native event filters. QTBUG-67095: Exclude input messages which are sent + // by QEventDispatcherWin32::processEvents() + if (!isInputMessage(msg.message) && filterNativeEvent(&msg, result)) + return true; + + auto platformWindow = window->handle(); + if (platformWindow && filterNativeEvent(platformWindow->window(), &msg, result)) + return true; + + return false; + } + // https://github.com/qt/qtbase/blob/e26a87f1ecc40bc8c6aa5b889fce67410a57a702/src/plugins/platforms/windows/qwindowscontext.cpp#L1556 // In QWindowsContext::windowsProc(), the messages will be passed to all global native event // filters, but because we have already filtered the messages in the hook WndProc function for @@ -489,6 +587,10 @@ // Try hooked procedure and save result LRESULT result; if (ctx->windowProc(hWnd, message, wParam, lParam, &result)) { + // Forward the event to user-defined native event filters, there may be some messages + // that need to be processed by the user. + std::ignore = + forwardFilteredEvents(ctx->window(), hWnd, message, wParam, lParam, &result); return result; } @@ -631,7 +733,7 @@ } case DrawWindows10BorderHook2: { - if (!m_windowHandle) + if (!m_windowId) return; // https://github.com/microsoft/terminal/blob/71a6f26e6ece656084e87de1a528c4a8072eeabd/src/cascadia/WindowsTerminal/NonClientIslandWindow.cpp#L1025 @@ -663,7 +765,7 @@ QVariant Win32WindowContext::windowAttribute(const QString &key) const { if (key == QStringLiteral("window-rect")) { - if (!m_windowHandle) + if (!m_windowId) return {}; RECT frame{}; @@ -686,7 +788,7 @@ } if (key == QStringLiteral("border-thickness")) { - return m_windowHandle + return m_windowId ? int(getWindowFrameBorderThickness(reinterpret_cast<HWND>(m_windowId))) : 0; } @@ -695,12 +797,16 @@ } void Win32WindowContext::winIdChanged(WId winId, WId oldWinId) { + // Reset the context data + mouseLeaveBlocked = false; + lastHitTestResult = WindowPart::Outside; + // If the original window id is valid, remove all resources related if (oldWinId) { removeManagedWindow(reinterpret_cast<HWND>(oldWinId)); } - if (!m_windowHandle || ! winId) { + if (!winId) { return; } @@ -735,15 +841,11 @@ LRESULT *result) { *result = FALSE; - if (message == WM_DESTROY) { - qDebug() << "WM_DESTROY"; - } - // We should skip these messages otherwise we will get crashes. // NOTE: WM_QUIT won't be posted to the WindowProc function. switch (message) { - case WM_CLOSE: case WM_DESTROY: + case WM_CLOSE: case WM_NCDESTROY: // Undocumented messages: case WM_UAHDESTROYWINDOW: @@ -755,20 +857,6 @@ if (!isValidWindow(hWnd, false, true)) { return false; - } - - switch (message) { - case WM_NCMOUSELEAVE: - qDebug() << "WM_NCMOUSELEAVE" << mouseLeaveBlocked; - break; - case WM_MOUSEHOVER: - qDebug() << "WM_MOUSEHOVER" << mouseLeaveBlocked; - break; - case WM_NCMOUSEHOVER: - qDebug() << "WM_NCMOUSEHOVER" << mouseLeaveBlocked; - break; - default: - break; } // Test snap layout @@ -788,13 +876,9 @@ // Forward to native event filter subscribers if (!m_nativeEventFilters.isEmpty()) { - MSG msg; - msg.hwnd = hWnd; - msg.message = message; - msg.wParam = wParam; - msg.lParam = lParam; + MSG msg = createMessageBlock(hWnd, message, wParam, lParam); QT_NATIVE_EVENT_RESULT_TYPE res = 0; - if (nativeDispatch(QByteArrayLiteral("windows_generic_MSG"), &msg, &res)) { + if (nativeDispatch(nativeEventType(), &msg, &res)) { *result = LRESULT(res); return true; } @@ -1156,8 +1240,6 @@ } } mouseLeaveBlocked = false; - - qDebug() << "WM_MOUSELEAVE"; break; } @@ -1166,7 +1248,6 @@ // we unset `mouseLeaveBlocked` mark and pretend as if Qt has received // WM_MOUSELEAVE. if (lastHitTestResult != WindowPart::ChromeButton && mouseLeaveBlocked) { - qDebug() << lastHitTestResult << "Track"; mouseLeaveBlocked = false; requestForMouseLeaveMessage(hWnd, false); } @@ -1243,19 +1324,14 @@ // the above problems would not arise. m_delegate->resetQtGrabbedControl(m_host); + + // If the mouse moves from chrome buttons to other non-client areas, a + // WM_MOUSELEAVE message should be sent. if (mouseLeaveBlocked) { emulateClientAreaMessage(hWnd, message, wParam, lParam, WM_NCMOUSELEAVE); } } - - // We need to make sure we get the right hit-test result when a WM_NCMOUSELEAVE - // comes, so we reset it when we receive a WM_NCMOUSEMOVE. - - // If the mouse is entering the client area, there must be a WM_NCHITTEST - // setting it to `Client` before the WM_NCMOUSELEAVE comes; if the mouse is - // leaving the window, current window part remains as `Outside`. - // lastHitTestResult = WindowPart::Outside; } if (currentWindowPart == WindowPart::ChromeButton) { @@ -1287,7 +1363,7 @@ // pressing area as HTCLIENT which maybe because of our former retransmission of // WM_NCLBUTTONDOWN, as a result, a WM_NCMOUSELEAVE will come immediately and a // lot of WM_MOUSEMOVE will come if we move the mouse, we should track the mouse - // in advance. + // in advance. (May be redundant?) if (mouseLeaveBlocked) { mouseLeaveBlocked = false; requestForMouseLeaveMessage(hWnd, false); -- Gitblit v1.9.1