Commit 53376c55 authored by Connor McAdams's avatar Connor McAdams Committed by Alexandre Julliard

uiautomationcore: Add support for calling COM event handlers.

parent 9ab915a7
...@@ -13132,8 +13132,8 @@ static DWORD WINAPI uia_com_event_handler_test_thread(LPVOID param) ...@@ -13132,8 +13132,8 @@ static DWORD WINAPI uia_com_event_handler_test_thread(LPVOID param)
set_com_event_data(&exp_node_desc); set_com_event_data(&exp_node_desc);
hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId); hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId);
ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr);
todo_wine CHECK_CALLED(uia_com_event_callback); CHECK_CALLED(uia_com_event_callback);
todo_wine ok(ComEventData.last_call_tid == GetCurrentThreadId(), "Event handler called on unexpected thread %ld\n", ok(ComEventData.last_call_tid == GetCurrentThreadId(), "Event handler called on unexpected thread %ld\n",
ComEventData.last_call_tid); ComEventData.last_call_tid);
CoUninitialize(); CoUninitialize();
...@@ -13146,7 +13146,7 @@ static DWORD WINAPI uia_com_event_handler_test_thread(LPVOID param) ...@@ -13146,7 +13146,7 @@ static DWORD WINAPI uia_com_event_handler_test_thread(LPVOID param)
set_com_event_data(&exp_node_desc); set_com_event_data(&exp_node_desc);
hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId); hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId);
ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr);
todo_wine CHECK_CALLED(uia_com_event_callback); CHECK_CALLED(uia_com_event_callback);
ok(ComEventData.last_call_tid != GetCurrentThreadId(), "Event handler called on unexpected thread %ld\n", ok(ComEventData.last_call_tid != GetCurrentThreadId(), "Event handler called on unexpected thread %ld\n",
ComEventData.last_call_tid); ComEventData.last_call_tid);
CoUninitialize(); CoUninitialize();
...@@ -13233,7 +13233,7 @@ static void test_IUIAutomationEventHandler(IUIAutomation *uia_iface, IUIAutomati ...@@ -13233,7 +13233,7 @@ static void test_IUIAutomationEventHandler(IUIAutomation *uia_iface, IUIAutomati
set_com_event_data(&exp_node_desc); set_com_event_data(&exp_node_desc);
hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId); hr = UiaRaiseAutomationEvent(&Provider2.IRawElementProviderSimple_iface, UIA_LiveRegionChangedEventId);
ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr);
todo_wine CHECK_CALLED(uia_com_event_callback); CHECK_CALLED(uia_com_event_callback);
/* /*
* If no cache request is provided by the user in * If no cache request is provided by the user in
......
...@@ -950,26 +950,22 @@ static int uia_com_event_handler_id_compare(const void *key, const struct rb_ent ...@@ -950,26 +950,22 @@ static int uia_com_event_handler_id_compare(const void *key, const struct rb_ent
} }
struct uia_com_event { struct uia_com_event {
IUnknown *handler_iface; DWORD git_cookie;
HUIAEVENT event;
BOOL from_cui8;
struct list event_handler_map_list_entry; struct list event_handler_map_list_entry;
struct uia_event_handler_map_entry *handler_map; struct uia_event_handler_map_entry *handler_map;
}; };
static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY *runtime_id, int event_id) static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY *runtime_id, int event_id,
struct uia_com_event *event)
{ {
struct uia_event_handler_identifier event_ident = { handler_iface, runtime_id, event_id }; struct uia_event_handler_identifier event_ident = { handler_iface, runtime_id, event_id };
struct uia_event_handler_map_entry *event_map; struct uia_event_handler_map_entry *event_map;
struct uia_com_event *event;
struct rb_entry *rb_entry; struct rb_entry *rb_entry;
HRESULT hr = S_OK; HRESULT hr = S_OK;
if (!(event = heap_alloc_zero(sizeof(*event))))
return E_OUTOFMEMORY;
event->handler_iface = handler_iface;
IUnknown_AddRef(handler_iface);
EnterCriticalSection(&com_event_handlers_cs); EnterCriticalSection(&com_event_handlers_cs);
if (!com_event_handlers.handler_count) if (!com_event_handlers.handler_count)
...@@ -993,7 +989,7 @@ static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY ...@@ -993,7 +989,7 @@ static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY
} }
event_map->event_id = event_id; event_map->event_id = event_id;
event_map->handler_iface = event->handler_iface; event_map->handler_iface = handler_iface;
IUnknown_AddRef(event_map->handler_iface); IUnknown_AddRef(event_map->handler_iface);
list_init(&event_map->handlers_list); list_init(&event_map->handlers_list);
...@@ -1006,25 +1002,28 @@ static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY ...@@ -1006,25 +1002,28 @@ static HRESULT uia_event_handlers_add_handler(IUnknown *handler_iface, SAFEARRAY
exit: exit:
LeaveCriticalSection(&com_event_handlers_cs); LeaveCriticalSection(&com_event_handlers_cs);
if (FAILED(hr))
{
IUnknown_Release(event->handler_iface);
heap_free(event);
}
return hr; return hr;
} }
static void uia_event_handler_destroy(struct uia_com_event *event)
{
list_remove(&event->event_handler_map_list_entry);
if (event->event)
UiaRemoveEvent(event->event);
if (event->git_cookie)
unregister_interface_in_git(event->git_cookie);
heap_free(event);
}
static void uia_event_handler_map_entry_destroy(struct uia_event_handler_map_entry *entry) static void uia_event_handler_map_entry_destroy(struct uia_event_handler_map_entry *entry)
{ {
struct uia_com_event *event, *event2; struct uia_com_event *event, *event2;
LIST_FOR_EACH_ENTRY_SAFE(event, event2, &entry->handlers_list, struct uia_com_event, event_handler_map_list_entry) LIST_FOR_EACH_ENTRY_SAFE(event, event2, &entry->handlers_list, struct uia_com_event, event_handler_map_list_entry)
{ {
list_remove(&event->event_handler_map_list_entry); uia_event_handler_destroy(event);
IUnknown_Release(event->handler_iface);
com_event_handlers.handler_count--; com_event_handlers.handler_count--;
heap_free(event);
} }
rb_remove(&com_event_handlers.handler_map, &entry->entry); rb_remove(&com_event_handlers.handler_map, &entry->entry);
...@@ -1046,6 +1045,39 @@ static void uia_event_handlers_remove_handlers(IUnknown *handler_iface, SAFEARRA ...@@ -1046,6 +1045,39 @@ static void uia_event_handlers_remove_handlers(IUnknown *handler_iface, SAFEARRA
LeaveCriticalSection(&com_event_handlers_cs); LeaveCriticalSection(&com_event_handlers_cs);
} }
static HRESULT create_uia_element_from_cache_req(IUIAutomationElement **iface, BOOL from_cui8,
struct UiaCacheRequest *cache_req, LONG start_idx, SAFEARRAY *req_data, BSTR tree_struct);
static HRESULT uia_com_event_callback(struct uia_event *event, struct uia_event_args *args,
SAFEARRAY *cache_req, BSTR tree_struct)
{
struct uia_com_event *com_event = (struct uia_com_event *)event->u.clientside.callback_data;
IUIAutomationEventHandler *handler;
IUIAutomationElement *elem;
BSTR tree_struct2;
HRESULT hr;
/* Nothing matches the cache request view condition, do nothing. */
if (!cache_req)
return S_OK;
/* create_uia_element_from_cache_req frees the passed in BSTR. */
tree_struct2 = SysAllocString(tree_struct);
hr = create_uia_element_from_cache_req(&elem, com_event->from_cui8, &event->u.clientside.cache_req, 0, cache_req,
tree_struct2);
if (FAILED(hr))
return hr;
hr = get_interface_in_git(&IID_IUIAutomationEventHandler, com_event->git_cookie, (IUnknown **)&handler);
if (SUCCEEDED(hr))
{
hr = IUIAutomationEventHandler_HandleAutomationEvent(handler, elem, event->event_id);
IUIAutomationEventHandler_Release(handler);
}
IUIAutomationElement_Release(elem);
return hr;
}
/* /*
* IUIAutomationElementArray interface. * IUIAutomationElementArray interface.
*/ */
...@@ -1332,8 +1364,6 @@ static HRESULT set_find_params_struct(struct UiaFindParams *params, IUIAutomatio ...@@ -1332,8 +1364,6 @@ static HRESULT set_find_params_struct(struct UiaFindParams *params, IUIAutomatio
return S_OK; return S_OK;
} }
static HRESULT create_uia_element_from_cache_req(IUIAutomationElement **iface, BOOL from_cui8,
struct UiaCacheRequest *cache_req, LONG start_idx, SAFEARRAY *req_data, BSTR tree_struct);
static HRESULT WINAPI uia_element_FindFirstBuildCache(IUIAutomationElement9 *iface, enum TreeScope scope, static HRESULT WINAPI uia_element_FindFirstBuildCache(IUIAutomationElement9 *iface, enum TreeScope scope,
IUIAutomationCondition *condition, IUIAutomationCacheRequest *cache_req, IUIAutomationElement **found) IUIAutomationCondition *condition, IUIAutomationCacheRequest *cache_req, IUIAutomationElement **found)
{ {
...@@ -3213,9 +3243,11 @@ static HRESULT WINAPI uia_iface_AddAutomationEventHandler(IUIAutomation6 *iface, ...@@ -3213,9 +3243,11 @@ static HRESULT WINAPI uia_iface_AddAutomationEventHandler(IUIAutomation6 *iface,
IUIAutomationElement *elem, enum TreeScope scope, IUIAutomationCacheRequest *cache_req, IUIAutomationElement *elem, enum TreeScope scope, IUIAutomationCacheRequest *cache_req,
IUIAutomationEventHandler *handler) IUIAutomationEventHandler *handler)
{ {
struct UiaCacheRequest *cache_req_struct;
struct uia_com_event *com_event = NULL;
SAFEARRAY *runtime_id = NULL;
struct uia_element *element; struct uia_element *element;
IUnknown *handler_iface; IUnknown *handler_iface;
SAFEARRAY *runtime_id;
HRESULT hr; HRESULT hr;
TRACE("%p, %d, %p, %#x, %p, %p\n", iface, event_id, elem, scope, cache_req, handler); TRACE("%p, %d, %p, %#x, %p, %p\n", iface, event_id, elem, scope, cache_req, handler);
...@@ -3233,11 +3265,48 @@ static HRESULT WINAPI uia_iface_AddAutomationEventHandler(IUIAutomation6 *iface, ...@@ -3233,11 +3265,48 @@ static HRESULT WINAPI uia_iface_AddAutomationEventHandler(IUIAutomation6 *iface,
element = impl_from_IUIAutomationElement9((IUIAutomationElement9 *)elem); element = impl_from_IUIAutomationElement9((IUIAutomationElement9 *)elem);
hr = UiaGetRuntimeId(element->node, &runtime_id); hr = UiaGetRuntimeId(element->node, &runtime_id);
if (FAILED(hr)) if (FAILED(hr))
{
IUnknown_Release(handler_iface);
return hr;
}
if (!cache_req)
{
hr = create_uia_cache_request_iface(&cache_req);
if (FAILED(hr))
goto exit;
}
else
IUIAutomationCacheRequest_AddRef(cache_req);
hr = get_uia_cache_request_struct_from_iface(cache_req, &cache_req_struct);
if (FAILED(hr))
goto exit;
if (!(com_event = heap_alloc_zero(sizeof(*com_event))))
{
hr = E_OUTOFMEMORY;
goto exit;
}
com_event->from_cui8 = element->from_cui8;
list_init(&com_event->event_handler_map_list_entry);
hr = register_interface_in_git((IUnknown *)handler, &IID_IUIAutomationEventHandler, &com_event->git_cookie);
if (FAILED(hr))
goto exit; goto exit;
hr = uia_event_handlers_add_handler(handler_iface, runtime_id, event_id); hr = uia_add_clientside_event(element->node, event_id, scope, NULL, 0, cache_req_struct, runtime_id,
uia_com_event_callback, (void *)com_event, &com_event->event);
if (FAILED(hr))
goto exit;
hr = uia_event_handlers_add_handler(handler_iface, runtime_id, event_id, com_event);
exit: exit:
if (FAILED(hr) && com_event)
uia_event_handler_destroy(com_event);
if (cache_req)
IUIAutomationCacheRequest_Release(cache_req);
IUnknown_Release(handler_iface); IUnknown_Release(handler_iface);
SafeArrayDestroy(runtime_id); SafeArrayDestroy(runtime_id);
......
...@@ -1140,7 +1140,7 @@ static HRESULT uia_clientside_event_callback(struct uia_event *event, struct uia ...@@ -1140,7 +1140,7 @@ static HRESULT uia_clientside_event_callback(struct uia_event *event, struct uia
return S_OK; return S_OK;
} }
static HRESULT uia_add_clientside_event(HUIANODE huianode, EVENTID event_id, enum TreeScope scope, PROPERTYID *prop_ids, HRESULT uia_add_clientside_event(HUIANODE huianode, EVENTID event_id, enum TreeScope scope, PROPERTYID *prop_ids,
int prop_ids_count, struct UiaCacheRequest *cache_req, SAFEARRAY *rt_id, UiaWineEventCallback *cback, int prop_ids_count, struct UiaCacheRequest *cache_req, SAFEARRAY *rt_id, UiaWineEventCallback *cback,
void *cback_data, HUIAEVENT *huiaevent) void *cback_data, HUIAEVENT *huiaevent)
{ {
......
...@@ -218,6 +218,9 @@ HRESULT create_serverside_uia_event(struct uia_event **out_event, LONG process_i ...@@ -218,6 +218,9 @@ HRESULT create_serverside_uia_event(struct uia_event **out_event, LONG process_i
HRESULT uia_event_add_provider_event_adviser(IRawElementProviderAdviseEvents *advise_events, HRESULT uia_event_add_provider_event_adviser(IRawElementProviderAdviseEvents *advise_events,
struct uia_event *event) DECLSPEC_HIDDEN; struct uia_event *event) DECLSPEC_HIDDEN;
HRESULT uia_event_add_serverside_event_adviser(IWineUiaEvent *serverside_event, struct uia_event *event) DECLSPEC_HIDDEN; HRESULT uia_event_add_serverside_event_adviser(IWineUiaEvent *serverside_event, struct uia_event *event) DECLSPEC_HIDDEN;
HRESULT uia_add_clientside_event(HUIANODE huianode, EVENTID event_id, enum TreeScope scope, PROPERTYID *prop_ids,
int prop_ids_count, struct UiaCacheRequest *cache_req, SAFEARRAY *rt_id, UiaWineEventCallback *cback,
void *cback_data, HUIAEVENT *huiaevent) DECLSPEC_HIDDEN;
/* uia_ids.c */ /* uia_ids.c */
const struct uia_prop_info *uia_prop_info_from_id(PROPERTYID prop_id) DECLSPEC_HIDDEN; const struct uia_prop_info *uia_prop_info_from_id(PROPERTYID prop_id) DECLSPEC_HIDDEN;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment