231 lines
6.2 KiB
C
231 lines
6.2 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 "uiautomation.h"
|
|
|
|
#include "wine/debug.h"
|
|
#include "wine/heap.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(uiautomation);
|
|
|
|
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 %d\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 %d\n", iface, refcount);
|
|
if (!refcount)
|
|
{
|
|
IUnknown_Release(wrapper->marshaler);
|
|
heap_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 = heap_alloc(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)))
|
|
{
|
|
heap_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};
|
|
|
|
/***********************************************************************
|
|
* UiaClientsAreListening (uiautomationcore.@)
|
|
*/
|
|
BOOL WINAPI UiaClientsAreListening(void)
|
|
{
|
|
FIXME("()\n");
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* 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;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaLookupId (uiautomationcore.@)
|
|
*/
|
|
int WINAPI UiaLookupId(enum AutomationIdentifierType type, const GUID *guid)
|
|
{
|
|
FIXME("(%d, %s) stub!\n", type, debugstr_guid(guid));
|
|
return 1;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaReturnRawElementProvider (uiautomationcore.@)
|
|
*/
|
|
LRESULT WINAPI UiaReturnRawElementProvider(HWND hwnd, WPARAM wParam,
|
|
LPARAM lParam, IRawElementProviderSimple *elprov)
|
|
{
|
|
FIXME("(%p, %lx, %lx, %p) stub!\n", hwnd, wParam, lParam, elprov);
|
|
return 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* UiaRaiseAutomationEvent (uiautomationcore.@)
|
|
*/
|
|
HRESULT WINAPI UiaRaiseAutomationEvent(IRawElementProviderSimple *provider, EVENTID id)
|
|
{
|
|
FIXME("(%p, %d): stub\n", provider, id);
|
|
return S_OK;
|
|
}
|
|
|
|
void WINAPI UiaRegisterProviderCallback(UiaProviderCallback *callback)
|
|
{
|
|
FIXME("(%p): stub\n", callback);
|
|
}
|
|
|
|
HRESULT WINAPI UiaHostProviderFromHwnd(HWND hwnd, IRawElementProviderSimple **provider)
|
|
{
|
|
FIXME("(%p, %p): stub\n", hwnd, provider);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
HRESULT WINAPI UiaDisconnectProvider(IRawElementProviderSimple *provider)
|
|
{
|
|
FIXME("(%p): stub\n", provider);
|
|
return E_NOTIMPL;
|
|
}
|