This function always returns TRUE on Windows 7 and above, and now that we have stubs for all the event raising functions there should be no harm in doing the same. Signed-off-by: Connor McAdams <cmcadams@codeweavers.com>
527 lines
14 KiB
C
527 lines
14 KiB
C
/*
|
|
* Copyright 2017 Jacek Caban for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "combaseapi.h"
|
|
#include "initguid.h"
|
|
#include "uia_private.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(uiautomation);
|
|
|
|
HMODULE huia_module;
|
|
|
|
struct uia_object_wrapper
|
|
{
|
|
IUnknown IUnknown_iface;
|
|
LONG refcount;
|
|
|
|
IUnknown *marshaler;
|
|
IUnknown *marshal_object;
|
|
};
|
|
|
|
static struct uia_object_wrapper *impl_uia_object_wrapper_from_IUnknown(IUnknown *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct uia_object_wrapper, IUnknown_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI uia_object_wrapper_QueryInterface(IUnknown *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
struct uia_object_wrapper *wrapper = impl_uia_object_wrapper_from_IUnknown(iface);
|
|
return IUnknown_QueryInterface(wrapper->marshal_object, riid, ppv);
|
|
}
|
|
|
|
static ULONG WINAPI uia_object_wrapper_AddRef(IUnknown *iface)
|
|
{
|
|
struct uia_object_wrapper *wrapper = impl_uia_object_wrapper_from_IUnknown(iface);
|
|
ULONG refcount = InterlockedIncrement(&wrapper->refcount);
|
|
|
|
TRACE("%p, refcount %ld\n", iface, refcount);
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static ULONG WINAPI uia_object_wrapper_Release(IUnknown *iface)
|
|
{
|
|
struct uia_object_wrapper *wrapper = impl_uia_object_wrapper_from_IUnknown(iface);
|
|
ULONG refcount = InterlockedDecrement(&wrapper->refcount);
|
|
|
|
TRACE("%p, refcount %ld\n", iface, refcount);
|
|
if (!refcount)
|
|
{
|
|
IUnknown_Release(wrapper->marshaler);
|
|
free(wrapper);
|
|
}
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static const IUnknownVtbl uia_object_wrapper_vtbl = {
|
|
uia_object_wrapper_QueryInterface,
|
|
uia_object_wrapper_AddRef,
|
|
uia_object_wrapper_Release,
|
|
};
|
|
|
|
/*
|
|
* When passing the ReservedNotSupportedValue/ReservedMixedAttributeValue
|
|
* interface pointers across apartments within the same process, create a free
|
|
* threaded marshaler so that the pointer value is preserved.
|
|
*/
|
|
static HRESULT create_uia_object_wrapper(IUnknown *reserved, void **ppv)
|
|
{
|
|
struct uia_object_wrapper *wrapper;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p\n", reserved, ppv);
|
|
|
|
wrapper = calloc(1, sizeof(*wrapper));
|
|
if (!wrapper)
|
|
return E_OUTOFMEMORY;
|
|
|
|
wrapper->IUnknown_iface.lpVtbl = &uia_object_wrapper_vtbl;
|
|
wrapper->marshal_object = reserved;
|
|
wrapper->refcount = 1;
|
|
|
|
if (FAILED(hr = CoCreateFreeThreadedMarshaler(&wrapper->IUnknown_iface, &wrapper->marshaler)))
|
|
{
|
|
free(wrapper);
|
|
return hr;
|
|
}
|
|
|
|
hr = IUnknown_QueryInterface(wrapper->marshaler, &IID_IMarshal, ppv);
|
|
IUnknown_Release(&wrapper->IUnknown_iface);
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*
|
|
* UiaReservedNotSupportedValue/UiaReservedMixedAttributeValue object.
|
|
*/
|
|
static HRESULT WINAPI uia_reserved_obj_QueryInterface(IUnknown *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
if (IsEqualIID(riid, &IID_IUnknown))
|
|
*ppv = iface;
|
|
else if (IsEqualIID(riid, &IID_IMarshal))
|
|
return create_uia_object_wrapper(iface, ppv);
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI uia_reserved_obj_AddRef(IUnknown *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static ULONG WINAPI uia_reserved_obj_Release(IUnknown *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static const IUnknownVtbl uia_reserved_obj_vtbl = {
|
|
uia_reserved_obj_QueryInterface,
|
|
uia_reserved_obj_AddRef,
|
|
uia_reserved_obj_Release,
|
|
};
|
|
|
|
static IUnknown uia_reserved_ns_iface = {&uia_reserved_obj_vtbl};
|
|
static IUnknown uia_reserved_ma_iface = {&uia_reserved_obj_vtbl};
|
|
|
|
/*
|
|
* UiaHostProviderFromHwnd IRawElementProviderSimple interface.
|
|
*/
|
|
struct hwnd_host_provider {
|
|
IRawElementProviderSimple IRawElementProviderSimple_iface;
|
|
LONG refcount;
|
|
|
|
HWND hwnd;
|
|
};
|
|
|
|
static inline struct hwnd_host_provider *impl_from_hwnd_host_provider(IRawElementProviderSimple *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct hwnd_host_provider, IRawElementProviderSimple_iface);
|
|
}
|
|
|
|
HRESULT WINAPI hwnd_host_provider_QueryInterface(IRawElementProviderSimple *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
if (IsEqualIID(riid, &IID_IRawElementProviderSimple) || IsEqualIID(riid, &IID_IUnknown))
|
|
*ppv = iface;
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
IRawElementProviderSimple_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
ULONG WINAPI hwnd_host_provider_AddRef(IRawElementProviderSimple *iface)
|
|
{
|
|
struct hwnd_host_provider *host_prov = impl_from_hwnd_host_provider(iface);
|
|
ULONG refcount = InterlockedIncrement(&host_prov->refcount);
|
|
|
|
TRACE("%p, refcount %ld\n", iface, refcount);
|
|
|
|
return refcount;
|
|
}
|
|
|
|
ULONG WINAPI hwnd_host_provider_Release(IRawElementProviderSimple *iface)
|
|
{
|
|
struct hwnd_host_provider *host_prov = impl_from_hwnd_host_provider(iface);
|
|
ULONG refcount = InterlockedDecrement(&host_prov->refcount);
|
|
|
|
TRACE("%p, refcount %ld\n", iface, refcount);
|
|
|
|
if (!refcount)
|
|
free(host_prov);
|
|
|
|
return refcount;
|
|
}
|
|
|
|
HRESULT WINAPI hwnd_host_provider_get_ProviderOptions(IRawElementProviderSimple *iface,
|
|
enum ProviderOptions *ret_val)
|
|
{
|
|
TRACE("%p, %p\n", iface, ret_val);
|
|
*ret_val = ProviderOptions_ServerSideProvider;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI hwnd_host_provider_GetPatternProvider(IRawElementProviderSimple *iface,
|
|
PATTERNID pattern_id, IUnknown **ret_val)
|
|
{
|
|
TRACE("%p, %d, %p\n", iface, pattern_id, ret_val);
|
|
*ret_val = NULL;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI hwnd_host_provider_GetPropertyValue(IRawElementProviderSimple *iface,
|
|
PROPERTYID prop_id, VARIANT *ret_val)
|
|
{
|
|
struct hwnd_host_provider *host_prov = impl_from_hwnd_host_provider(iface);
|
|
|
|
TRACE("%p, %d, %p\n", iface, prop_id, ret_val);
|
|
|
|
VariantInit(ret_val);
|
|
switch (prop_id)
|
|
{
|
|
case UIA_NativeWindowHandlePropertyId:
|
|
V_VT(ret_val) = VT_I4;
|
|
V_I4(ret_val) = HandleToUlong(host_prov->hwnd);
|
|
break;
|
|
|
|
case UIA_ProviderDescriptionPropertyId:
|
|
V_VT(ret_val) = VT_BSTR;
|
|
V_BSTR(ret_val) = SysAllocString(L"Wine: HWND Provider Proxy");
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI hwnd_host_provider_get_HostRawElementProvider(IRawElementProviderSimple *iface,
|
|
IRawElementProviderSimple **ret_val)
|
|
{
|
|
TRACE("%p, %p\n", iface, ret_val);
|
|
*ret_val = NULL;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IRawElementProviderSimpleVtbl hwnd_host_provider_vtbl = {
|
|
hwnd_host_provider_QueryInterface,
|
|
hwnd_host_provider_AddRef,
|
|
hwnd_host_provider_Release,
|
|
hwnd_host_provider_get_ProviderOptions,
|
|
hwnd_host_provider_GetPatternProvider,
|
|
hwnd_host_provider_GetPropertyValue,
|
|
hwnd_host_provider_get_HostRawElementProvider,
|
|
};
|
|
|
|
/***********************************************************************
|
|
* UiaClientsAreListening (uiautomationcore.@)
|
|
*/
|
|
BOOL WINAPI UiaClientsAreListening(void)
|
|
{
|
|
TRACE("()\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaGetReservedMixedAttributeValue (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaGetReservedMixedAttributeValue(IUnknown **value)
|
|
{
|
|
TRACE("(%p)\n", value);
|
|
|
|
if (!value)
|
|
return E_INVALIDARG;
|
|
|
|
*value = &uia_reserved_ma_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaGetReservedNotSupportedValue (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaGetReservedNotSupportedValue(IUnknown **value)
|
|
{
|
|
TRACE("(%p)\n", value);
|
|
|
|
if (!value)
|
|
return E_INVALIDARG;
|
|
|
|
*value = &uia_reserved_ns_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseAutomationPropertyChangedEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseAutomationPropertyChangedEvent(IRawElementProviderSimple *provider, PROPERTYID id, VARIANT old, VARIANT new)
|
|
{
|
|
FIXME("(%p, %d, %s, %s): stub\n", provider, id, debugstr_variant(&old), debugstr_variant(&new));
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseStructureChangedEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseStructureChangedEvent(IRawElementProviderSimple *provider, enum StructureChangeType struct_change_type,
|
|
int *runtime_id, int runtime_id_len)
|
|
{
|
|
FIXME("(%p, %d, %p, %d): stub\n", provider, struct_change_type, runtime_id, runtime_id_len);
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseAsyncContentLoadedEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseAsyncContentLoadedEvent(IRawElementProviderSimple *provider,
|
|
enum AsyncContentLoadedState async_content_loaded_state, double percent_complete)
|
|
{
|
|
FIXME("(%p, %d, %f): stub\n", provider, async_content_loaded_state, percent_complete);
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseTextEditTextChangedEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseTextEditTextChangedEvent(IRawElementProviderSimple *provider,
|
|
enum TextEditChangeType text_edit_change_type, SAFEARRAY *changed_data)
|
|
{
|
|
FIXME("(%p, %d, %p): stub\n", provider, text_edit_change_type, changed_data);
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseNotificationEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseNotificationEvent(IRawElementProviderSimple *provider, enum NotificationKind notification_kind,
|
|
enum NotificationProcessing notification_processing, BSTR display_str, BSTR activity_id)
|
|
{
|
|
FIXME("(%p, %d, %d, %s, %s): stub\n", provider, notification_kind, notification_processing,
|
|
debugstr_w(display_str), debugstr_w(activity_id));
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseChangesEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseChangesEvent(IRawElementProviderSimple *provider, int event_id_count,
|
|
struct UiaChangeInfo *uia_changes)
|
|
{
|
|
FIXME("(%p, %d, %p): stub\n", provider, event_id_count, uia_changes);
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI UiaHostProviderFromHwnd(HWND hwnd, IRawElementProviderSimple **provider)
|
|
{
|
|
struct hwnd_host_provider *host_prov;
|
|
|
|
TRACE("(%p, %p)\n", hwnd, provider);
|
|
|
|
if (provider)
|
|
*provider = NULL;
|
|
|
|
if (!IsWindow(hwnd) || !provider)
|
|
return E_INVALIDARG;
|
|
|
|
host_prov = calloc(1, sizeof(*host_prov));
|
|
if (!host_prov)
|
|
return E_OUTOFMEMORY;
|
|
|
|
host_prov->IRawElementProviderSimple_iface.lpVtbl = &hwnd_host_provider_vtbl;
|
|
host_prov->refcount = 1;
|
|
host_prov->hwnd = hwnd;
|
|
*provider = &host_prov->IRawElementProviderSimple_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllMain (uiautomationcore.@)
|
|
*/
|
|
BOOL WINAPI DllMain(HINSTANCE hinst, DWORD reason, void *reserved)
|
|
{
|
|
TRACE("(%p, %ld, %p)\n", hinst, reason, reserved);
|
|
|
|
switch (reason)
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
DisableThreadLibraryCalls(hinst);
|
|
huia_module = hinst;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* UIAutomation ClassFactory */
|
|
struct uia_cf {
|
|
IClassFactory IClassFactory_iface;
|
|
LONG ref;
|
|
|
|
const GUID *clsid;
|
|
};
|
|
|
|
static struct uia_cf *impl_from_IClassFactory(IClassFactory *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct uia_cf, IClassFactory_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI uia_cf_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
if (IsEqualIID(riid, &IID_IClassFactory) || IsEqualIID(riid, &IID_IUnknown))
|
|
*ppv = iface;
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
IClassFactory_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI uia_cf_AddRef(IClassFactory *iface)
|
|
{
|
|
struct uia_cf *cf = impl_from_IClassFactory(iface);
|
|
ULONG ref = InterlockedIncrement(&cf->ref);
|
|
|
|
TRACE("%p, refcount %ld\n", cf, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI uia_cf_Release(IClassFactory *iface)
|
|
{
|
|
struct uia_cf *cf = impl_from_IClassFactory(iface);
|
|
ULONG ref = InterlockedDecrement(&cf->ref);
|
|
|
|
TRACE("%p, refcount %ld\n", cf, ref);
|
|
|
|
if (!ref)
|
|
free(cf);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI uia_cf_CreateInstance(IClassFactory *iface, IUnknown *pouter, REFIID riid, void **ppv)
|
|
{
|
|
struct uia_cf *cf = impl_from_IClassFactory(iface);
|
|
IUnknown *obj = NULL;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %s, %p\n", iface, pouter, debugstr_guid(riid), ppv);
|
|
|
|
*ppv = NULL;
|
|
if (pouter)
|
|
return CLASS_E_NOAGGREGATION;
|
|
|
|
if (IsEqualGUID(cf->clsid, &CLSID_CUIAutomation))
|
|
hr = create_uia_iface(&obj, FALSE);
|
|
else if (IsEqualGUID(cf->clsid, &CLSID_CUIAutomation8))
|
|
hr = create_uia_iface(&obj, TRUE);
|
|
else
|
|
return E_NOINTERFACE;
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = IUnknown_QueryInterface(obj, riid, ppv);
|
|
IUnknown_Release(obj);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI uia_cf_LockServer(IClassFactory *iface, BOOL do_lock)
|
|
{
|
|
FIXME("%p, %d: stub\n", iface, do_lock);
|
|
return S_OK;
|
|
}
|
|
|
|
static const IClassFactoryVtbl uia_cf_Vtbl =
|
|
{
|
|
uia_cf_QueryInterface,
|
|
uia_cf_AddRef,
|
|
uia_cf_Release,
|
|
uia_cf_CreateInstance,
|
|
uia_cf_LockServer
|
|
};
|
|
|
|
static inline HRESULT create_uia_cf(REFCLSID clsid, REFIID riid, void **ppv)
|
|
{
|
|
struct uia_cf *cf = calloc(1, sizeof(*cf));
|
|
HRESULT hr;
|
|
|
|
*ppv = NULL;
|
|
if (!cf)
|
|
return E_OUTOFMEMORY;
|
|
|
|
cf->IClassFactory_iface.lpVtbl = &uia_cf_Vtbl;
|
|
cf->clsid = clsid;
|
|
cf->ref = 1;
|
|
|
|
hr = IClassFactory_QueryInterface(&cf->IClassFactory_iface, riid, ppv);
|
|
IClassFactory_Release(&cf->IClassFactory_iface);
|
|
|
|
return hr;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllGetClassObject (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI DllGetClassObject(REFCLSID clsid, REFIID riid, void **ppv)
|
|
{
|
|
TRACE("(%s, %s, %p)\n", debugstr_guid(clsid), debugstr_guid(riid), ppv);
|
|
|
|
if (IsEqualGUID(clsid, &CLSID_CUIAutomation) || IsEqualGUID(clsid, &CLSID_CUIAutomation8))
|
|
return create_uia_cf(clsid, riid, ppv);
|
|
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|