2 * Copyright (C) 2008, 2009, 2010 Apple Inc. All Rights Reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "AccessibilityController.h"
29 #include "AccessibilityUIElement.h"
30 #include "DumpRenderTree.h"
31 #include "FrameLoadDelegate.h"
32 #include <JavaScriptCore/Assertions.h>
33 #include <JavaScriptCore/JSRetainPtr.h>
34 #include <JavaScriptCore/JSStringRef.h>
35 #include <WebCore/COMPtr.h>
36 #include <WebKit/WebKit.h>
42 AccessibilityController::AccessibilityController()
44 , m_scrollingStartEventHook(0)
45 , m_valueChangeEventHook(0)
47 , m_notificationsEventHook(0)
51 AccessibilityController::~AccessibilityController()
53 setLogFocusEvents(false);
54 setLogAccessibilityEvents(false);
55 setLogValueChangeEvents(false);
57 if (m_notificationsEventHook)
58 UnhookWinEvent(m_notificationsEventHook);
60 for (HashMap<PlatformUIElement, JSObjectRef>::iterator it = m_notificationListeners.begin(); it != m_notificationListeners.end(); ++it)
61 JSValueUnprotect(frame->globalContext(), it->second);
64 AccessibilityUIElement AccessibilityController::elementAtPoint(int x, int y)
70 AccessibilityUIElement AccessibilityController::focusedElement()
72 COMPtr<IAccessible> rootAccessible = rootElement().platformUIElement();
75 if (FAILED(rootAccessible->get_accFocus(&vFocus)))
78 if (V_VT(&vFocus) == VT_I4) {
79 ASSERT(V_I4(&vFocus) == CHILDID_SELF);
80 // The root accessible object is the focused object.
81 return rootAccessible;
84 ASSERT(V_VT(&vFocus) == VT_DISPATCH);
85 // We have an IDispatch; query for IAccessible.
86 return COMPtr<IAccessible>(Query, V_DISPATCH(&vFocus));
89 AccessibilityUIElement AccessibilityController::rootElement()
91 COMPtr<IWebView> view;
92 if (FAILED(frame->webView(&view)))
95 COMPtr<IWebViewPrivate> viewPrivate(Query, view);
100 if (FAILED(viewPrivate->viewWindow((OLE_HANDLE*)&webViewWindow)))
103 // Get the root accessible object by querying for the accessible object for the
105 COMPtr<IAccessible> rootAccessible;
106 if (FAILED(AccessibleObjectFromWindow(webViewWindow, static_cast<DWORD>(OBJID_CLIENT), __uuidof(IAccessible), reinterpret_cast<void**>(&rootAccessible))))
109 return rootAccessible;
112 static void CALLBACK logEventProc(HWINEVENTHOOK, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD, DWORD)
114 // Get the accessible object for this event.
115 COMPtr<IAccessible> parentObject;
118 VariantInit(&vChild);
120 HRESULT hr = AccessibleObjectFromEvent(hwnd, idObject, idChild, &parentObject, &vChild);
121 ASSERT(SUCCEEDED(hr));
123 // Get the name of the focused element, and log it to stdout.
125 hr = parentObject->get_accName(vChild, &nameBSTR);
126 ASSERT(SUCCEEDED(hr));
127 wstring name(nameBSTR, ::SysStringLen(nameBSTR));
128 SysFreeString(nameBSTR);
131 case EVENT_OBJECT_FOCUS:
132 printf("Received focus event for object '%S'.\n", name.c_str());
135 case EVENT_OBJECT_SELECTION:
136 printf("Received selection event for object '%S'.\n", name.c_str());
139 case EVENT_OBJECT_VALUECHANGE: {
141 hr = parentObject->get_accValue(vChild, &valueBSTR);
142 ASSERT(SUCCEEDED(hr));
143 wstring value(valueBSTR, ::SysStringLen(valueBSTR));
144 SysFreeString(valueBSTR);
146 printf("Received value change event for object '%S', value '%S'.\n", name.c_str(), value.c_str());
150 case EVENT_SYSTEM_SCROLLINGSTART:
151 printf("Received scrolling start event for object '%S'.\n", name.c_str());
155 printf("Received unknown event for object '%S'.\n", name.c_str());
159 VariantClear(&vChild);
162 void AccessibilityController::setLogFocusEvents(bool logFocusEvents)
164 if (!!m_focusEventHook == logFocusEvents)
167 if (!logFocusEvents) {
168 UnhookWinEvent(m_focusEventHook);
169 m_focusEventHook = 0;
173 // Ensure that accessibility is initialized for the WebView by querying for
174 // the root accessible object.
177 m_focusEventHook = SetWinEventHook(EVENT_OBJECT_FOCUS, EVENT_OBJECT_FOCUS, GetModuleHandle(0), logEventProc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
179 ASSERT(m_focusEventHook);
182 void AccessibilityController::setLogValueChangeEvents(bool logValueChangeEvents)
184 if (!!m_valueChangeEventHook == logValueChangeEvents)
187 if (!logValueChangeEvents) {
188 UnhookWinEvent(m_valueChangeEventHook);
189 m_valueChangeEventHook = 0;
193 // Ensure that accessibility is initialized for the WebView by querying for
194 // the root accessible object.
197 m_valueChangeEventHook = SetWinEventHook(EVENT_OBJECT_VALUECHANGE, EVENT_OBJECT_VALUECHANGE, GetModuleHandle(0), logEventProc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
199 ASSERT(m_valueChangeEventHook);
202 void AccessibilityController::setLogScrollingStartEvents(bool logScrollingStartEvents)
204 if (!!m_scrollingStartEventHook == logScrollingStartEvents)
207 if (!logScrollingStartEvents) {
208 UnhookWinEvent(m_scrollingStartEventHook);
209 m_scrollingStartEventHook = 0;
213 // Ensure that accessibility is initialized for the WebView by querying for
214 // the root accessible object.
217 m_scrollingStartEventHook = SetWinEventHook(EVENT_SYSTEM_SCROLLINGSTART, EVENT_SYSTEM_SCROLLINGSTART, GetModuleHandle(0), logEventProc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
219 ASSERT(m_scrollingStartEventHook);
222 void AccessibilityController::setLogAccessibilityEvents(bool logAccessibilityEvents)
224 if (!!m_allEventsHook == logAccessibilityEvents)
227 if (!logAccessibilityEvents) {
228 UnhookWinEvent(m_allEventsHook);
233 // Ensure that accessibility is initialized for the WebView by querying for
234 // the root accessible object.
237 m_allEventsHook = SetWinEventHook(EVENT_MIN, EVENT_MAX, GetModuleHandle(0), logEventProc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
239 ASSERT(m_allEventsHook);
242 static string stringEvent(DWORD event)
245 case EVENT_OBJECT_VALUECHANGE:
246 return "value change event";
248 return "unknown event";
252 static void CALLBACK notificationListenerProc(HWINEVENTHOOK, DWORD event, HWND hwnd, LONG idObject, LONG idChild, DWORD, DWORD)
254 // Get the accessible object for this event.
255 COMPtr<IAccessible> parentObject;
258 VariantInit(&vChild);
260 HRESULT hr = AccessibleObjectFromEvent(hwnd, idObject, idChild, &parentObject, &vChild);
261 if (FAILED(hr) || !parentObject)
264 COMPtr<IDispatch> childDispatch;
265 if (FAILED(parentObject->get_accChild(vChild, &childDispatch))) {
266 VariantClear(&vChild);
270 COMPtr<IAccessible> childAccessible(Query, childDispatch);
272 sharedFrameLoadDelegate->accessibilityController()->notificationReceived(childAccessible, stringEvent(event));
274 VariantClear(&vChild);
277 static COMPtr<IAccessibleComparable> comparableObject(const COMPtr<IServiceProvider>& serviceProvider)
279 COMPtr<IAccessibleComparable> comparable;
280 serviceProvider->QueryService(SID_AccessibleComparable, __uuidof(IAccessibleComparable), reinterpret_cast<void**>(&comparable));
284 void AccessibilityController::notificationReceived(PlatformUIElement element, const string& eventName)
286 for (HashMap<PlatformUIElement, JSObjectRef>::iterator it = m_notificationListeners.begin(); it != m_notificationListeners.end(); ++it) {
287 COMPtr<IServiceProvider> thisServiceProvider(Query, it->first);
288 if (!thisServiceProvider)
291 COMPtr<IAccessibleComparable> thisComparable = comparableObject(thisServiceProvider);
295 COMPtr<IServiceProvider> elementServiceProvider(Query, element);
296 if (!elementServiceProvider)
299 COMPtr<IAccessibleComparable> elementComparable = comparableObject(elementServiceProvider);
300 if (!elementComparable)
304 thisComparable->isSameObject(elementComparable.get(), &isSame);
308 JSRetainPtr<JSStringRef> jsNotification(Adopt, JSStringCreateWithUTF8CString(eventName.c_str()));
309 JSValueRef argument = JSValueMakeString(frame->globalContext(), jsNotification.get());
310 JSObjectCallAsFunction(frame->globalContext(), it->second, NULL, 1, &argument, NULL);
314 void AccessibilityController::addNotificationListener(PlatformUIElement element, JSObjectRef functionCallback)
316 if (!m_notificationsEventHook)
317 m_notificationsEventHook = SetWinEventHook(EVENT_MIN, EVENT_MAX, GetModuleHandle(0), notificationListenerProc, GetCurrentProcessId(), 0, WINEVENT_INCONTEXT);
319 JSValueProtect(frame->globalContext(), functionCallback);
320 m_notificationListeners.add(element, functionCallback);