#include "pch.h" #include "XamlBridge.h" bool XamlBridge::FilterMessage(const MSG* msg) { // When multiple child windows are present it is needed to pre dispatch messages to all // DesktopWindowXamlSource instances so keyboard accelerators and // keyboard focus work correctly. BOOL xamlSourceProcessedMessage = FALSE; { for (auto xamlSource : m_xamlSources) { auto xamlSourceNative2 = xamlSource.as(); const auto hr = xamlSourceNative2->PreTranslateMessage(msg, &xamlSourceProcessedMessage); winrt::check_hresult(hr); if (xamlSourceProcessedMessage) { break; } } } return !!xamlSourceProcessedMessage; } const auto static invalidReason = static_cast(-1); winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason GetReasonFromKey(WPARAM key) { auto reason = invalidReason; if (key == VK_TAB) { byte keyboardState[256] = {}; WINRT_VERIFY(::GetKeyboardState(keyboardState)); reason = (keyboardState[VK_SHIFT] & 0x80) ? winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Last : winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::First; } else if (key == VK_LEFT) { reason = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Left; } else if (key == VK_RIGHT) { reason = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Right; } else if (key == VK_UP) { reason = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Up; } else if (key == VK_DOWN) { reason = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Down; } return reason; } // Function to return the next xaml island in focus winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSource XamlBridge::GetNextFocusedIsland(MSG* msg) { if (msg->message == WM_KEYDOWN) { const auto key = msg->wParam; auto reason = GetReasonFromKey(key); if (reason != invalidReason) { const BOOL previous = (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::First || reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Down || reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Right) ? false : true; const auto currentFocusedWindow = ::GetFocus(); const auto nextElement = ::GetNextDlgTabItem(parentWindow, currentFocusedWindow, previous); for (auto xamlSource : m_xamlSources) { const auto nativeIsland = xamlSource.as(); HWND islandWnd = nullptr; winrt::check_hresult(nativeIsland->get_WindowHandle(&islandWnd)); if (nextElement == islandWnd) { return xamlSource; } } } } return nullptr; } // Function to return the xaml island currently in focus winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSource XamlBridge::GetFocusedIsland() { for (auto xamlSource : m_xamlSources) { if (xamlSource.HasFocus()) { return xamlSource; } } return nullptr; } // Function to handle focus navigation bool XamlBridge::NavigateFocus(MSG* msg) { if (const auto nextFocusedIsland = GetNextFocusedIsland(msg)) { const auto previousFocusedWindow = ::GetFocus(); RECT rect = {}; WINRT_VERIFY(::GetWindowRect(previousFocusedWindow, &rect)); const auto nativeIsland = nextFocusedIsland.as(); HWND islandWnd = nullptr; winrt::check_hresult(nativeIsland->get_WindowHandle(&islandWnd)); POINT pt = { rect.left, rect.top }; SIZE size = { rect.right - rect.left, rect.bottom - rect.top }; ::ScreenToClient(islandWnd, &pt); const auto hintRect = winrt::Windows::Foundation::Rect({ static_cast(pt.x), static_cast(pt.y), static_cast(size.cx), static_cast(size.cy) }); const auto reason = GetReasonFromKey(msg->wParam); const auto request = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationRequest(reason, hintRect); lastFocusRequestId = request.CorrelationId(); const auto result = nextFocusedIsland.NavigateFocus(request); return result.WasFocusMoved(); } else { const bool islandIsFocused = GetFocusedIsland() != nullptr; byte keyboardState[256] = {}; WINRT_VERIFY(::GetKeyboardState(keyboardState)); const bool isMenuModifier = keyboardState[VK_MENU] & 0x80; if (islandIsFocused && !isMenuModifier) { return false; } const bool isDialogMessage = !!IsDialogMessage(parentWindow, msg); return isDialogMessage; } } // Function to run the message loop for the xaml island window int XamlBridge::MessageLoop() { MSG msg = {}; HRESULT hr = S_OK; while (GetMessage(&msg, nullptr, 0, 0)) { const bool xamlSourceProcessedMessage = FilterMessage(&msg); if (!xamlSourceProcessedMessage) { if (!NavigateFocus(&msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } } } return (int)msg.wParam; } static const WPARAM invalidKey = (WPARAM)-1; WPARAM GetKeyFromReason(winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason reason) { auto key = invalidKey; if (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Last || reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::First) { key = VK_TAB; } else if (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Left) { key = VK_LEFT; } else if (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Right) { key = VK_RIGHT; } else if (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Up) { key = VK_UP; } else if (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Down) { key = VK_DOWN; } return key; } // Event triggered when focus is requested void XamlBridge::OnTakeFocusRequested(winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSource const& sender, winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSourceTakeFocusRequestedEventArgs const& args) { if (args.Request().CorrelationId() != lastFocusRequestId) { const auto reason = args.Request().Reason(); const BOOL previous = (reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::First || reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Down || reason == winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Right) ? false : true; const auto nativeXamlSource = sender.as(); HWND senderHwnd = nullptr; winrt::check_hresult(nativeXamlSource->get_WindowHandle(&senderHwnd)); MSG msg = {}; msg.hwnd = senderHwnd; msg.message = WM_KEYDOWN; msg.wParam = GetKeyFromReason(reason); if (!NavigateFocus(&msg)) { const auto nextElement = ::GetNextDlgTabItem(parentWindow, senderHwnd, previous); ::SetFocus(nextElement); } } else { const auto request = winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationRequest(winrt::Windows::UI::Xaml::Hosting::XamlSourceFocusNavigationReason::Restore); lastFocusRequestId = request.CorrelationId(); sender.NavigateFocus(request); } } // Function to initialise the xaml source object HWND XamlBridge::InitDesktopWindowsXamlSource(winrt::Windows::UI::Xaml::Hosting::DesktopWindowXamlSource desktopSource) { HRESULT hr = S_OK; auto interop = desktopSource.as(); // Parent the DesktopWindowXamlSource object to current window hr = interop->AttachToWindow(parentWindow); winrt::check_hresult(hr); // Get the new child window's hwnd HWND hWndXamlIsland = nullptr; hr = interop->get_WindowHandle(&hWndXamlIsland); winrt::check_hresult(hr); m_takeFocusEventRevokers.push_back(desktopSource.TakeFocusRequested(winrt::auto_revoke, { this, &XamlBridge::OnTakeFocusRequested })); m_xamlSources.push_back(desktopSource); return hWndXamlIsland; } // Function to close and delete all the xaml source objects void XamlBridge::ClearXamlIslands() { for (auto& takeFocusRevoker : m_takeFocusEventRevokers) { takeFocusRevoker.revoke(); } m_takeFocusEventRevokers.clear(); for (auto xamlSource : m_xamlSources) { xamlSource.Close(); } m_xamlSources.clear(); } // Function invoked when the window is destroyed void XamlBridge::OnDestroy(HWND) { PostQuitMessage(0); } // Function invoked when the window is activated void XamlBridge::OnActivate(HWND, UINT state, HWND hwndActDeact, BOOL fMinimized) { if (state == WA_INACTIVE) { m_hwndLastFocus = GetFocus(); } } // Function invoked when the window is set to focus void XamlBridge::OnSetFocus(HWND, HWND hwndOldFocus) { if (m_hwndLastFocus) { SetFocus(m_hwndLastFocus); } } // Message Handler function for Xaml Island windows LRESULT XamlBridge::MessageHandler(UINT const message, WPARAM const wParam, LPARAM const lParam) noexcept { switch (message) { HANDLE_MSG(parentWindow, WM_DESTROY, OnDestroy); HANDLE_MSG(parentWindow, WM_ACTIVATE, OnActivate); HANDLE_MSG(parentWindow, WM_SETFOCUS, OnSetFocus); } return DefWindowProc(parentWindow, message, wParam, lParam); }