/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim:expandtab:shiftwidth=2:tabstop=2: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "nsWinUtils.h" #include "Compatibility.h" #include "DocAccessible.h" #include "nsAccessibilityService.h" #include "nsCoreUtils.h" #include "mozilla/a11y/DocAccessibleParent.h" #include "mozilla/Preferences.h" #include "nsArrayUtils.h" #include "nsICSSDeclaration.h" #include "mozilla/dom/Document.h" #include "mozilla/dom/Element.h" #include "nsXULAppAPI.h" #include "ProxyWrappers.h" using namespace mozilla; using namespace mozilla::a11y; using mozilla::dom::Element; // Window property used by ipc related code in identifying accessible // tab windows. const wchar_t* kPropNameTabContent = L"AccessibleTabWindow"; /** * WindowProc to process WM_GETOBJECT messages, used in windows emulation mode. */ static LRESULT CALLBACK WindowProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam); bool nsWinUtils::sWindowEmulationStarted = false; already_AddRefed nsWinUtils::GetComputedStyleDeclaration( nsIContent* aContent) { nsIContent* elm = nsCoreUtils::GetDOMElementFor(aContent); if (!elm) return nullptr; // Returns number of items in style declaration nsCOMPtr window = elm->OwnerDoc()->GetInnerWindow(); if (!window) return nullptr; ErrorResult dummy; nsCOMPtr domElement(do_QueryInterface(elm)); nsCOMPtr cssDecl = window->GetComputedStyle(*domElement, u""_ns, dummy); dummy.SuppressException(); return cssDecl.forget(); } bool nsWinUtils::MaybeStartWindowEmulation() { // Register window class that'll be used for document accessibles associated // with tabs. if (IPCAccessibilityActive()) return false; if (Compatibility::IsJAWS() || Compatibility::IsWE() || Compatibility::IsDolphin() || Compatibility::IsVisperoShared()) { RegisterNativeWindow(kClassNameTabContent); sWindowEmulationStarted = true; return true; } return false; } void nsWinUtils::ShutdownWindowEmulation() { // Unregister window call that's used for document accessibles associated // with tabs. if (IsWindowEmulationStarted()) { ::UnregisterClassW(kClassNameTabContent, GetModuleHandle(nullptr)); sWindowEmulationStarted = false; } } void nsWinUtils::RegisterNativeWindow(LPCWSTR aWindowClass) { WNDCLASSW wc; wc.style = CS_GLOBALCLASS; wc.lpfnWndProc = WindowProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = GetModuleHandle(nullptr); wc.hIcon = nullptr; wc.hCursor = nullptr; wc.hbrBackground = nullptr; wc.lpszMenuName = nullptr; wc.lpszClassName = aWindowClass; ::RegisterClassW(&wc); } HWND nsWinUtils::CreateNativeWindow(LPCWSTR aWindowClass, HWND aParentWnd, int aX, int aY, int aWidth, int aHeight, bool aIsActive, NativeWindowCreateProc* aOnCreateProc) { return ::CreateWindowExW( WS_EX_TRANSPARENT, aWindowClass, L"NetscapeDispatchWnd", WS_CHILD | (aIsActive ? WS_VISIBLE : 0), aX, aY, aWidth, aHeight, aParentWnd, nullptr, GetModuleHandle(nullptr), aOnCreateProc); } void nsWinUtils::ShowNativeWindow(HWND aWnd) { ::ShowWindow(aWnd, SW_SHOW); } void nsWinUtils::HideNativeWindow(HWND aWnd) { ::SetWindowPos( aWnd, nullptr, 0, 0, 0, 0, SWP_HIDEWINDOW | SWP_NOSIZE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE); } LRESULT CALLBACK WindowProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { // Note, this window's message handling should not invoke any call that // may result in a cross-process ipc call. Doing so may violate RPC // message semantics. switch (msg) { case WM_CREATE: { // Mark this window so that ipc related code can identify it. ::SetPropW(hWnd, kPropNameTabContent, reinterpret_cast(1)); auto createStruct = reinterpret_cast(lParam); auto createProc = reinterpret_cast( createStruct->lpCreateParams); if (createProc && *createProc) { (*createProc)(hWnd); } return 0; } case WM_GETOBJECT: { // Do explicit casting to make it working on 64bit systems (see bug 649236 // for details). int32_t objId = static_cast(lParam); if (objId == OBJID_CLIENT) { IAccessible* msaaAccessible = nullptr; DocAccessible* document = reinterpret_cast(::GetPropW(hWnd, kPropNameDocAcc)); if (document) { document->GetNativeInterface( (void**)&msaaAccessible); // does an addref } else { DocAccessibleParent* docParent = static_cast( ::GetPropW(hWnd, kPropNameDocAccParent)); if (docParent) { auto wrapper = WrapperFor(docParent); wrapper->GetNativeInterface( (void**)&msaaAccessible); // does an addref } } if (msaaAccessible) { LRESULT result = ::LresultFromObject(IID_IAccessible, wParam, msaaAccessible); // does an addref msaaAccessible->Release(); // release extra addref return result; } } return 0; } case WM_NCHITTEST: { LRESULT lRet = ::DefWindowProc(hWnd, msg, wParam, lParam); if (HTCLIENT == lRet) lRet = HTTRANSPARENT; return lRet; } } return ::DefWindowProcW(hWnd, msg, wParam, lParam); }