| /* |
| * Drag and Drop Tests |
| * |
| * Copyright 2007 Robert Shearman |
| * |
| * 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 _WIN32_DCOM |
| #define COBJMACROS |
| #define CONST_VTABLE |
| |
| #include <stdarg.h> |
| #include <stdio.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "objbase.h" |
| |
| #include "wine/test.h" |
| |
| |
| #define METHOD_LIST \ |
| METHOD(DO_EnumFormatEtc), \ |
| METHOD(DO_QueryGetData), \ |
| METHOD(EnumFMT_Next), \ |
| METHOD(EnumFMT_Reset), \ |
| METHOD(EnumFMT_Skip), \ |
| METHOD(DS_QueryContinueDrag), \ |
| METHOD(DS_GiveFeedback), \ |
| METHOD(DT_DragEnter), \ |
| METHOD(DT_Drop), \ |
| METHOD(DT_DragLeave), \ |
| METHOD(DT_DragOver), \ |
| METHOD(DoDragDrop_effect_in), \ |
| METHOD(DoDragDrop_ret), \ |
| METHOD(DoDragDrop_effect_out), \ |
| METHOD(end_seq) |
| |
| #define METHOD(x) x |
| enum method |
| { |
| METHOD_LIST |
| }; |
| #undef METHOD |
| |
| #define METHOD(x) #x |
| static const char *method_names[] = |
| { |
| METHOD_LIST |
| }; |
| #undef METHOD |
| #undef METHOD_LIST |
| |
| struct method_call |
| { |
| enum method method; |
| DWORD expect_param; |
| |
| HRESULT set_ret; |
| DWORD set_param; |
| |
| int called_todo : 1; |
| }; |
| |
| const struct method_call *call_ptr; |
| |
| static HRESULT check_expect_(enum method func, DWORD expect_param, DWORD *set_param, const char *file, int line ) |
| { |
| HRESULT hr; |
| |
| do |
| { |
| todo_wine_if(call_ptr->called_todo) |
| ok_( file, line )( func == call_ptr->method, "unexpected call %s instead of %s\n", |
| method_names[func], method_names[call_ptr->method] ); |
| if (call_ptr->method == func) break; |
| } while ((++call_ptr)->method != end_seq); |
| |
| ok_( file, line )( expect_param == call_ptr->expect_param, "%s: unexpected param %08x expected %08x\n", |
| method_names[func], expect_param, call_ptr->expect_param ); |
| if (set_param) *set_param = call_ptr->set_param; |
| hr = call_ptr->set_ret; |
| if (call_ptr->method != end_seq) call_ptr++; |
| return hr; |
| } |
| |
| #define check_expect(func, expect_param, set_param) \ |
| check_expect_((func), (expect_param), (set_param), __FILE__, __LINE__) |
| |
| |
| struct method_call call_lists[][30] = |
| { |
| { /* First QueryContinueDrag rets DRAGDROP_S_DROP */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_DROP, 0, 0 }, |
| { DT_DragEnter, DROPEFFECT_COPY, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_Drop, DROPEFFECT_COPY, 0xbeefbeef, DROPEFFECT_COPY, 0 }, |
| |
| { DoDragDrop_ret, 0xbeefbeef, 0, 0, 0, }, |
| { DoDragDrop_effect_out, DROPEFFECT_COPY, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* As above, but initial effects == 0 */ |
| { DoDragDrop_effect_in, 0, 0, 0, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_DROP, 0, 0 }, |
| { DT_DragEnter, 0, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, 0, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_DragLeave, 0, 0, 0, 0 }, |
| |
| { DoDragDrop_ret, DRAGDROP_S_DROP, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* Multiple initial effects */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_DROP, 0, 0 }, |
| { DT_DragEnter, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_Drop, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0, 0, 0 }, |
| |
| { DoDragDrop_ret, DRAGDROP_S_DROP, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* First couple of QueryContinueDrag return S_OK followed by a drop */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragEnter, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_DROP, 0, 0 }, |
| { DT_Drop, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0, 0, 0 }, |
| |
| { DoDragDrop_ret, DRAGDROP_S_DROP, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* First QueryContinueDrag cancels */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_CANCEL, 0, 0 }, |
| |
| { DoDragDrop_ret, DRAGDROP_S_CANCEL, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* First couple of QueryContinueDrag return S_OK followed by a cancel */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragEnter, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, DRAGDROP_S_CANCEL, 0, 0 }, |
| { DT_DragLeave, 0, 0, 0, 0 }, |
| |
| { DoDragDrop_ret, DRAGDROP_S_CANCEL, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| { /* First couple of QueryContinueDrag return S_OK followed by a E_FAIL */ |
| { DoDragDrop_effect_in, 0, 0, DROPEFFECT_COPY | DROPEFFECT_MOVE, 0 }, |
| { DO_EnumFormatEtc, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { EnumFMT_Reset, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_OK, 0, 1 }, |
| { EnumFMT_Next, 0, S_FALSE, 0, 1 }, |
| { DO_QueryGetData, 0, S_OK, 0, 1 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragEnter, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, S_OK, 0, 0 }, |
| { DT_DragOver, DROPEFFECT_COPY | DROPEFFECT_MOVE, S_OK, DROPEFFECT_COPY, 0 }, |
| { DS_GiveFeedback, DROPEFFECT_COPY, DRAGDROP_S_USEDEFAULTCURSORS, 0, 0 }, |
| |
| { DS_QueryContinueDrag, 0, E_FAIL, 0, 0 }, |
| { DT_DragLeave, 0, 0, 0, 0 }, |
| |
| { DoDragDrop_ret, E_FAIL, 0, 0, 0 }, |
| { DoDragDrop_effect_out, 0, 0, 0, 0 }, |
| { end_seq, 0, 0, 0, 0 } |
| }, |
| }; |
| |
| static int droptarget_refs; |
| |
| /* helper macros to make tests a bit leaner */ |
| #define ok_ole_success(hr, func) ok(hr == S_OK, func " failed with error 0x%08x\n", hr) |
| |
| static HRESULT WINAPI DropTarget_QueryInterface(IDropTarget* iface, REFIID riid, |
| void** ppvObject) |
| { |
| ok(0, "DropTarget_QueryInterface() shouldn't be called\n"); |
| if (IsEqualIID(riid, &IID_IUnknown) || |
| IsEqualIID(riid, &IID_IDropTarget)) |
| { |
| IDropTarget_AddRef(iface); |
| *ppvObject = iface; |
| return S_OK; |
| } |
| *ppvObject = NULL; |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI DropTarget_AddRef(IDropTarget* iface) |
| { |
| droptarget_refs++; |
| return droptarget_refs; |
| } |
| |
| static ULONG WINAPI DropTarget_Release(IDropTarget* iface) |
| { |
| droptarget_refs--; |
| return droptarget_refs; |
| } |
| |
| static HRESULT WINAPI DropTarget_DragEnter(IDropTarget* iface, |
| IDataObject* pDataObj, |
| DWORD grfKeyState, POINTL pt, |
| DWORD* pdwEffect) |
| { |
| return check_expect(DT_DragEnter, *pdwEffect, pdwEffect); |
| } |
| |
| static HRESULT WINAPI DropTarget_DragOver(IDropTarget* iface, |
| DWORD grfKeyState, |
| POINTL pt, |
| DWORD* pdwEffect) |
| { |
| return check_expect(DT_DragOver, *pdwEffect, pdwEffect); |
| } |
| |
| static HRESULT WINAPI DropTarget_DragLeave(IDropTarget* iface) |
| { |
| return check_expect(DT_DragLeave, 0, NULL); |
| } |
| |
| static HRESULT WINAPI DropTarget_Drop(IDropTarget* iface, |
| IDataObject* pDataObj, DWORD grfKeyState, |
| POINTL pt, DWORD* pdwEffect) |
| { |
| return check_expect(DT_Drop, *pdwEffect, pdwEffect); |
| } |
| |
| static const IDropTargetVtbl DropTarget_VTbl = |
| { |
| DropTarget_QueryInterface, |
| DropTarget_AddRef, |
| DropTarget_Release, |
| DropTarget_DragEnter, |
| DropTarget_DragOver, |
| DropTarget_DragLeave, |
| DropTarget_Drop |
| }; |
| |
| static IDropTarget DropTarget = { &DropTarget_VTbl }; |
| |
| static HRESULT WINAPI DropSource_QueryInterface(IDropSource *iface, REFIID riid, void **ppObj) |
| { |
| if (IsEqualIID(riid, &IID_IUnknown) || |
| IsEqualIID(riid, &IID_IDropSource)) |
| { |
| *ppObj = iface; |
| IDropSource_AddRef(iface); |
| return S_OK; |
| } |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI DropSource_AddRef(IDropSource *iface) |
| { |
| return 2; |
| } |
| |
| static ULONG WINAPI DropSource_Release(IDropSource *iface) |
| { |
| return 1; |
| } |
| |
| static HRESULT WINAPI DropSource_QueryContinueDrag( |
| IDropSource *iface, |
| BOOL fEscapePressed, |
| DWORD grfKeyState) |
| { |
| return check_expect(DS_QueryContinueDrag, 0, NULL); |
| } |
| |
| static HRESULT WINAPI DropSource_GiveFeedback( |
| IDropSource *iface, |
| DWORD dwEffect) |
| { |
| return check_expect(DS_GiveFeedback, dwEffect, NULL); |
| } |
| |
| static const IDropSourceVtbl dropsource_vtbl = { |
| DropSource_QueryInterface, |
| DropSource_AddRef, |
| DropSource_Release, |
| DropSource_QueryContinueDrag, |
| DropSource_GiveFeedback |
| }; |
| |
| static IDropSource DropSource = { &dropsource_vtbl }; |
| |
| static HRESULT WINAPI EnumFORMATETC_QueryInterface(IEnumFORMATETC *iface, |
| REFIID riid, void **ppvObj) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static ULONG WINAPI EnumFORMATETC_AddRef(IEnumFORMATETC *iface) |
| { |
| return 2; |
| } |
| |
| static ULONG WINAPI EnumFORMATETC_Release(IEnumFORMATETC *iface) |
| { |
| return 1; |
| } |
| |
| static HRESULT WINAPI EnumFORMATETC_Next(IEnumFORMATETC *iface, |
| ULONG celt, FORMATETC *rgelt, ULONG *pceltFetched) |
| { |
| static FORMATETC format = { CF_TEXT, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }; |
| HRESULT hr = check_expect(EnumFMT_Next, 0, NULL); |
| |
| ok(celt == 1, "celt = %d\n", celt); |
| ok(rgelt != NULL, "rgelt == NULL\n"); |
| ok(pceltFetched == NULL, "pceltFetched != NULL\n"); |
| |
| *rgelt = format; |
| return hr; |
| } |
| |
| static HRESULT WINAPI EnumFORMATETC_Skip(IEnumFORMATETC *iface, ULONG celt) |
| { |
| return check_expect(EnumFMT_Skip, 0, NULL); |
| } |
| |
| static HRESULT WINAPI EnumFORMATETC_Reset(IEnumFORMATETC *iface) |
| { |
| return check_expect(EnumFMT_Reset, 0, NULL); |
| } |
| |
| static HRESULT WINAPI EnumFORMATETC_Clone(IEnumFORMATETC *iface, |
| IEnumFORMATETC **ppenum) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static const IEnumFORMATETCVtbl enumformatetc_vtbl = { |
| EnumFORMATETC_QueryInterface, |
| EnumFORMATETC_AddRef, |
| EnumFORMATETC_Release, |
| EnumFORMATETC_Next, |
| EnumFORMATETC_Skip, |
| EnumFORMATETC_Reset, |
| EnumFORMATETC_Clone |
| }; |
| |
| static IEnumFORMATETC EnumFORMATETC = { &enumformatetc_vtbl }; |
| |
| static HRESULT WINAPI DataObject_QueryInterface( |
| IDataObject *iface, |
| REFIID riid, |
| void **pObj) |
| { |
| if (IsEqualIID(riid, &IID_IUnknown) || |
| IsEqualIID(riid, &IID_IDataObject)) |
| { |
| *pObj = iface; |
| IDataObject_AddRef(iface); |
| return S_OK; |
| } |
| |
| trace("DataObject_QueryInterface: %s\n", wine_dbgstr_guid(riid)); |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI DataObject_AddRef(IDataObject *iface) |
| { |
| return 2; |
| } |
| |
| static ULONG WINAPI DataObject_Release(IDataObject *iface) |
| { |
| return 1; |
| } |
| |
| static HRESULT WINAPI DataObject_GetData( |
| IDataObject *iface, |
| FORMATETC *pformatetcIn, |
| STGMEDIUM *pmedium) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_GetDataHere( |
| IDataObject *iface, |
| FORMATETC *pformatetc, |
| STGMEDIUM *pmedium) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_QueryGetData( |
| IDataObject *iface, |
| FORMATETC *pformatetc) |
| { |
| return check_expect(DO_QueryGetData, 0, NULL); |
| } |
| |
| static HRESULT WINAPI DataObject_GetCanonicalFormatEtc( |
| IDataObject *iface, |
| FORMATETC *pformatectIn, |
| FORMATETC *pformatetcOut) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_SetData( |
| IDataObject *iface, |
| FORMATETC *pformatetc, |
| STGMEDIUM *pmedium, |
| BOOL fRelease) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_EnumFormatEtc( |
| IDataObject *iface, |
| DWORD dwDirection, |
| IEnumFORMATETC **ppenumFormatEtc) |
| { |
| HRESULT hr = check_expect(DO_EnumFormatEtc, 0, NULL); |
| *ppenumFormatEtc = &EnumFORMATETC; |
| return hr; |
| } |
| |
| static HRESULT WINAPI DataObject_DAdvise( |
| IDataObject *iface, |
| FORMATETC *pformatetc, |
| DWORD advf, |
| IAdviseSink *pAdvSink, |
| DWORD *pdwConnection) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_DUnadvise( |
| IDataObject *iface, |
| DWORD dwConnection) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI DataObject_EnumDAdvise( |
| IDataObject *iface, |
| IEnumSTATDATA **ppenumAdvise) |
| { |
| ok(0, "unexpected call\n"); |
| return E_NOTIMPL; |
| } |
| |
| static const IDataObjectVtbl dataobject_vtbl = { |
| DataObject_QueryInterface, |
| DataObject_AddRef, |
| DataObject_Release, |
| DataObject_GetData, |
| DataObject_GetDataHere, |
| DataObject_QueryGetData, |
| DataObject_GetCanonicalFormatEtc, |
| DataObject_SetData, |
| DataObject_EnumFormatEtc, |
| DataObject_DAdvise, |
| DataObject_DUnadvise, |
| DataObject_EnumDAdvise |
| }; |
| |
| static IDataObject DataObject = { &dataobject_vtbl }; |
| |
| static ATOM register_dummy_class(void) |
| { |
| WNDCLASSA wc = |
| { |
| 0, |
| DefWindowProcA, |
| 0, |
| 0, |
| GetModuleHandleA(NULL), |
| NULL, |
| LoadCursorA(NULL, (LPSTR)IDC_ARROW), |
| (HBRUSH)(COLOR_BTNFACE+1), |
| NULL, |
| "WineOleTestClass", |
| }; |
| |
| return RegisterClassA(&wc); |
| } |
| |
| static void test_Register_Revoke(void) |
| { |
| HANDLE prop; |
| HRESULT hr; |
| HWND hwnd; |
| |
| hwnd = CreateWindowA("WineOleTestClass", "Test", 0, |
| CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, |
| NULL, NULL, NULL); |
| |
| hr = RegisterDragDrop(hwnd, &DropTarget); |
| ok(hr == E_OUTOFMEMORY || |
| broken(hr == CO_E_NOTINITIALIZED), /* NT4 */ |
| "RegisterDragDrop without OLE initialized should have returned E_OUTOFMEMORY instead of 0x%08x\n", hr); |
| |
| OleInitialize(NULL); |
| |
| hr = RegisterDragDrop(hwnd, NULL); |
| ok(hr == E_INVALIDARG, "RegisterDragDrop with NULL IDropTarget * should return E_INVALIDARG instead of 0x%08x\n", hr); |
| |
| hr = RegisterDragDrop(NULL, &DropTarget); |
| ok(hr == DRAGDROP_E_INVALIDHWND, "RegisterDragDrop with NULL hwnd should return DRAGDROP_E_INVALIDHWND instead of 0x%08x\n", hr); |
| |
| hr = RegisterDragDrop((HWND)0xdeadbeef, &DropTarget); |
| ok(hr == DRAGDROP_E_INVALIDHWND, "RegisterDragDrop with garbage hwnd should return DRAGDROP_E_INVALIDHWND instead of 0x%08x\n", hr); |
| |
| ok(droptarget_refs == 0, "DropTarget refs should be zero not %d\n", droptarget_refs); |
| hr = RegisterDragDrop(hwnd, &DropTarget); |
| ok_ole_success(hr, "RegisterDragDrop"); |
| ok(droptarget_refs >= 1, "DropTarget refs should be at least one\n"); |
| |
| prop = GetPropA(hwnd, "OleDropTargetInterface"); |
| ok(prop == &DropTarget, "expected IDropTarget pointer %p, got %p\n", &DropTarget, prop); |
| |
| hr = RegisterDragDrop(hwnd, &DropTarget); |
| ok(hr == DRAGDROP_E_ALREADYREGISTERED, "RegisterDragDrop with already registered hwnd should return DRAGDROP_E_ALREADYREGISTERED instead of 0x%08x\n", hr); |
| |
| ok(droptarget_refs >= 1, "DropTarget refs should be at least one\n"); |
| OleUninitialize(); |
| |
| /* Win 8 releases the ref in OleUninitialize() */ |
| if (droptarget_refs >= 1) |
| { |
| hr = RevokeDragDrop(hwnd); |
| ok_ole_success(hr, "RevokeDragDrop"); |
| ok(droptarget_refs == 0 || |
| broken(droptarget_refs == 1), /* NT4 */ |
| "DropTarget refs should be zero not %d\n", droptarget_refs); |
| } |
| |
| hr = RevokeDragDrop(NULL); |
| ok(hr == DRAGDROP_E_INVALIDHWND, "RevokeDragDrop with NULL hwnd should return DRAGDROP_E_INVALIDHWND instead of 0x%08x\n", hr); |
| |
| DestroyWindow(hwnd); |
| |
| /* try to revoke with already destroyed window */ |
| OleInitialize(NULL); |
| |
| hwnd = CreateWindowA("WineOleTestClass", "Test", 0, |
| CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, |
| NULL, NULL, NULL); |
| |
| hr = RegisterDragDrop(hwnd, &DropTarget); |
| ok(hr == S_OK, "got 0x%08x\n", hr); |
| |
| DestroyWindow(hwnd); |
| |
| hr = RevokeDragDrop(hwnd); |
| ok(hr == DRAGDROP_E_INVALIDHWND, "got 0x%08x\n", hr); |
| |
| OleUninitialize(); |
| } |
| |
| static void test_DoDragDrop(void) |
| { |
| DWORD effect; |
| HRESULT hr; |
| HWND hwnd; |
| RECT rect; |
| int seq; |
| |
| hwnd = CreateWindowExA(WS_EX_TOPMOST, "WineOleTestClass", "Test", 0, |
| CW_USEDEFAULT, CW_USEDEFAULT, 100, 100, NULL, |
| NULL, NULL, NULL); |
| ok(IsWindow(hwnd), "failed to create window\n"); |
| |
| hr = OleInitialize(NULL); |
| ok(hr == S_OK, "got 0x%08x\n", hr); |
| |
| hr = RegisterDragDrop(hwnd, &DropTarget); |
| ok(hr == S_OK, "got 0x%08x\n", hr); |
| |
| /* incomplete arguments set */ |
| hr = DoDragDrop(NULL, NULL, 0, NULL); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(NULL, &DropSource, 0, NULL); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(&DataObject, NULL, 0, NULL); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(NULL, NULL, 0, &effect); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(&DataObject, &DropSource, 0, NULL); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(NULL, &DropSource, 0, &effect); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| hr = DoDragDrop(&DataObject, NULL, 0, &effect); |
| ok(hr == E_INVALIDARG, "got 0x%08x\n", hr); |
| |
| ShowWindow(hwnd, SW_SHOW); |
| GetWindowRect(hwnd, &rect); |
| ok(SetCursorPos(rect.left+50, rect.top+50), "SetCursorPos failed\n"); |
| |
| for (seq = 0; seq < sizeof(call_lists) / sizeof(call_lists[0]); seq++) |
| { |
| DWORD effect_in; |
| trace("%d\n", seq); |
| call_ptr = call_lists[seq]; |
| effect_in = call_ptr->set_param; |
| call_ptr++; |
| |
| hr = DoDragDrop(&DataObject, &DropSource, effect_in, &effect); |
| check_expect(DoDragDrop_ret, hr, NULL); |
| check_expect(DoDragDrop_effect_out, effect, NULL); |
| } |
| |
| OleUninitialize(); |
| |
| DestroyWindow(hwnd); |
| } |
| |
| START_TEST(dragdrop) |
| { |
| register_dummy_class(); |
| |
| test_Register_Revoke(); |
| test_DoDragDrop(); |
| } |