1 /*
2  * PROJECT:         ReactOS api tests
3  * LICENSE:         LGPLv2.1+ - See COPYING.LIB in the top level directory
4  * PURPOSE:         Test for Drag & Drop
5  * PROGRAMMER:      Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
6  */
7 
8 #include "shelltest.h"
9 #include <shlwapi.h>
10 
11 #define NDEBUG
12 #include <debug.h>
13 #include <stdio.h>
14 
15 #define TESTFILENAME L"DragDropTest.txt"
16 #define DROPPED_ON_FILE L"DragDroppedOn.lnk"
17 
18 static CComPtr<IShellFolder> s_pDesktop;
19 
20 static WCHAR s_szSrcTestFile[MAX_PATH];
21 static WCHAR s_szDestFolder[MAX_PATH];
22 static WCHAR s_szDestTestFile[MAX_PATH];
23 static WCHAR s_szDestLinkSpec[MAX_PATH];
24 static WCHAR s_szDroppedToItem[MAX_PATH];
25 
26 enum OP
27 {
28     OP_NONE,
29     OP_COPY,
30     OP_MOVE,
31     OP_LINK,
32     OP_NONE_OR_COPY,
33     OP_NONE_OR_MOVE,
34     OP_NONE_OR_LINK
35 };
36 
37 #define D_NONE DROPEFFECT_NONE
38 #define D_COPY DROPEFFECT_COPY
39 #define D_MOVE DROPEFFECT_MOVE
40 #define D_LINK DROPEFFECT_LINK
41 #define D_NONE_OR_COPY 0xAABBCCDD
42 #define D_NONE_OR_MOVE 0x11223344
43 #define D_NONE_OR_LINK 0x55667788
44 
45 struct TEST_ENTRY
46 {
47     int line;
48     OP op;
49     HRESULT hr1;
50     HRESULT hr2;
51     DWORD dwKeyState;
52     DWORD dwEffects1;
53     DWORD dwEffects2;
54     DWORD dwEffects3;
55 };
56 
57 static const TEST_ENTRY s_TestEntries[] =
58 {
59     // MK_LBUTTON
60     { __LINE__, OP_NONE, S_OK, S_OK, MK_LBUTTON, D_NONE, D_NONE, D_NONE },
61     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON, D_COPY, D_COPY, D_COPY },
62     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON, D_COPY | D_MOVE, D_MOVE, D_NONE },
63     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON, D_COPY | D_MOVE | D_LINK, D_MOVE, D_NONE },
64     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON, D_COPY | D_LINK, D_COPY, D_COPY },
65     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON, D_MOVE, D_MOVE, D_NONE },
66     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON, D_MOVE | D_LINK, D_MOVE, D_NONE },
67     { __LINE__, OP_LINK, S_OK, S_OK, MK_LBUTTON, D_LINK, D_LINK, D_LINK },
68 
69     // MK_LBUTTON | MK_SHIFT
70     { __LINE__, OP_NONE, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_NONE, D_NONE, D_NONE },
71     { __LINE__, OP_NONE_OR_COPY, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_COPY, D_NONE_OR_COPY, D_NONE_OR_COPY },
72     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_COPY | D_MOVE, D_MOVE, D_NONE },
73     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_COPY | D_MOVE | D_LINK, D_MOVE, D_NONE },
74     { __LINE__, OP_NONE_OR_COPY, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_COPY | D_LINK, D_NONE_OR_COPY, D_NONE_OR_COPY },
75     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_MOVE, D_MOVE, D_NONE },
76     { __LINE__, OP_MOVE, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_MOVE | D_LINK, D_MOVE, D_NONE },
77     { __LINE__, OP_NONE_OR_LINK, S_OK, S_OK, MK_LBUTTON | MK_SHIFT, D_LINK, D_NONE_OR_LINK, D_NONE_OR_LINK },
78 
79     // MK_LBUTTON | MK_SHIFT | MK_CONTROL
80 #define MK_LBUTTON_SHIFT_CTRL (MK_LBUTTON | MK_SHIFT | MK_CONTROL)
81     { __LINE__, OP_NONE, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_NONE, D_NONE, D_NONE },
82     { __LINE__, OP_NONE_OR_COPY, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_COPY, D_NONE_OR_COPY, D_NONE_OR_COPY },
83     { __LINE__, OP_NONE_OR_COPY, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_COPY | D_MOVE, D_NONE_OR_COPY, D_NONE_OR_COPY },
84     { __LINE__, OP_LINK, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_COPY | D_MOVE | D_LINK, D_LINK, D_LINK },
85     { __LINE__, OP_LINK, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_COPY | D_LINK, D_LINK, D_LINK },
86     { __LINE__, OP_NONE_OR_MOVE, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_MOVE, D_NONE_OR_MOVE, D_NONE },
87     { __LINE__, OP_LINK, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_MOVE | D_LINK, D_LINK, D_LINK },
88     { __LINE__, OP_LINK, S_OK, S_OK, MK_LBUTTON_SHIFT_CTRL, D_LINK, D_LINK, D_LINK },
89 #undef MK_LBUTTON_SHIFT_CTRL
90 
91     // MK_LBUTTON | MK_CONTROL
92     { __LINE__, OP_NONE, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_NONE, D_NONE, D_NONE },
93     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_COPY, D_COPY, D_COPY },
94     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_COPY | D_MOVE, D_COPY, D_COPY },
95     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_COPY | D_MOVE | D_LINK, D_COPY, D_COPY },
96     { __LINE__, OP_COPY, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_COPY | D_LINK, D_COPY, D_COPY },
97     { __LINE__, OP_NONE_OR_MOVE, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_MOVE, D_NONE_OR_MOVE, D_NONE },
98     { __LINE__, OP_NONE_OR_MOVE, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_MOVE | D_LINK, D_NONE_OR_MOVE, D_NONE },
99     { __LINE__, OP_NONE_OR_LINK, S_OK, S_OK, MK_LBUTTON | MK_CONTROL, D_LINK, D_NONE_OR_LINK, D_NONE_OR_LINK },
100 };
101 
102 static void DoCreateTestFile(LPCWSTR pszFileName)
103 {
104     FILE *fp = _wfopen(pszFileName, L"wb");
105     ok(fp != NULL, "fp is NULL for '%S'\n", pszFileName);
106     fclose(fp);
107 }
108 
109 HRESULT DoCreateShortcut(
110     LPCWSTR pszLnkFileName,
111     LPCWSTR pszTargetPathName)
112 {
113     CComPtr<IPersistFile> ppf;
114     CComPtr<IShellLinkW> psl;
115     HRESULT hr;
116 
117     hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
118                           IID_IShellLinkW, (LPVOID *)&psl);
119     if (SUCCEEDED(hr))
120     {
121         psl->SetPath(pszTargetPathName);
122 
123         hr = psl->QueryInterface(IID_IPersistFile, (LPVOID *)&ppf);
124         if (SUCCEEDED(hr))
125         {
126             hr = ppf->Save(pszLnkFileName, TRUE);
127         }
128     }
129 
130     return hr;
131 }
132 
133 static HRESULT
134 GetUIObjectOfAbsPidl(PIDLIST_ABSOLUTE pidl, REFIID riid, LPVOID *ppvOut)
135 {
136     *ppvOut = NULL;
137 
138     LPCITEMIDLIST pidlLast;
139     CComPtr<IShellFolder> psf;
140     HRESULT hr = SHBindToParent(pidl, IID_IShellFolder, (LPVOID *)&psf,
141                                 &pidlLast);
142     if (FAILED(hr))
143         return hr;
144 
145     hr = psf->GetUIObjectOf(NULL, 1, &pidlLast, riid, NULL, ppvOut);
146     return hr;
147 }
148 
149 static HRESULT
150 GetUIObjectOfPath(LPCWSTR pszPath, REFIID riid, LPVOID *ppvOut)
151 {
152     *ppvOut = NULL;
153 
154     PIDLIST_ABSOLUTE pidl = ILCreateFromPathW(pszPath);
155     if (!pidl)
156         return E_FAIL;
157 
158     HRESULT hr = GetUIObjectOfAbsPidl(pidl, riid, ppvOut);
159 
160     CoTaskMemFree(pidl);
161 
162     return hr;
163 }
164 
165 BOOL DoSpecExistsW(LPCWSTR pszSpec)
166 {
167     WIN32_FIND_DATAW find;
168     HANDLE hFind = FindFirstFileW(pszSpec, &find);
169     if (hFind != INVALID_HANDLE_VALUE)
170     {
171         FindClose(hFind);
172         return TRUE;
173     }
174     return FALSE;
175 }
176 
177 void DoDeleteSpecW(LPCWSTR pszSpec)
178 {
179     WCHAR szPath[MAX_PATH], szFile[MAX_PATH];
180     lstrcpyW(szPath, pszSpec);
181     PathRemoveFileSpecW(szPath);
182 
183     WIN32_FIND_DATAW find;
184     HANDLE hFind = FindFirstFileW(pszSpec, &find);
185     if (hFind != INVALID_HANDLE_VALUE)
186     {
187         do
188         {
189             lstrcpyW(szFile, szPath);
190             PathAppendW(szFile, find.cFileName);
191             DeleteFileW(szFile);
192         } while (FindNextFileW(hFind, &find));
193 
194         FindClose(hFind);
195     }
196 }
197 
198 static void DoTestEntry(const TEST_ENTRY *pEntry)
199 {
200     int line = pEntry->line;
201     HRESULT hr;
202     PIDLIST_ABSOLUTE pidlDesktop = NULL;
203     CComPtr<IDropTarget> pDropTarget;
204     CComPtr<IDataObject> pDataObject;
205 
206     // get the desktop PIDL
207     SHGetSpecialFolderLocation(NULL, CSIDL_DESKTOP, &pidlDesktop);
208     ok(!!pidlDesktop, "pidlDesktop is NULL\n");
209 
210     // build paths
211     //
212     SHGetPathFromIDListW(pidlDesktop, s_szDroppedToItem);
213     PathAppendW(s_szDroppedToItem, DROPPED_ON_FILE);
214 
215     GetModuleFileNameW(NULL, s_szSrcTestFile, _countof(s_szSrcTestFile));
216     PathRemoveFileSpecW(s_szSrcTestFile);
217     PathAppendW(s_szSrcTestFile, TESTFILENAME);
218 
219     lstrcpyW(s_szDestTestFile, s_szDestFolder);
220     PathAppendW(s_szDestTestFile, TESTFILENAME);
221 
222     lstrcpyW(s_szDestLinkSpec, s_szDestFolder);
223     PathAppendW(s_szDestLinkSpec, L"*DragDropTest*.lnk");
224 
225     //trace("s_szSrcTestFile: '%S'\n", s_szSrcTestFile);
226     //trace("s_szDestTestFile: '%S'\n", s_szDestTestFile);
227     //trace("s_szDestLinkSpec: '%S'\n", s_szDestLinkSpec);
228     //trace("s_szDroppedToItem: '%S'\n", s_szDroppedToItem);
229 
230     // create or delete files
231     //
232     DoCreateTestFile(s_szSrcTestFile);
233     DeleteFileW(s_szDestTestFile);
234     DoDeleteSpecW(s_szDestLinkSpec);
235     DeleteFileW(s_szDroppedToItem);
236     DoCreateShortcut(s_szDroppedToItem, s_szDestFolder);
237 
238     // check file existence
239     //
240     ok(PathIsDirectoryW(s_szDestFolder), "s_szDestFolder is not directory\n");
241     ok(PathFileExistsW(s_szSrcTestFile), "s_szSrcTestFile doesn't exist\n");
242     ok(!DoSpecExistsW(s_szDestLinkSpec), "s_szDestLinkSpec doesn't exist\n");
243     ok(!PathFileExistsW(s_szDestTestFile), "s_szDestTestFile exists\n");
244 
245     // get an IDataObject
246     pDataObject = NULL;
247     hr = GetUIObjectOfPath(s_szSrcTestFile, IID_IDataObject, (LPVOID *)&pDataObject);
248     ok_long(hr, S_OK);
249 
250     // get an IDropTarget
251     CComPtr<IEnumIDList> pEnumIDList;
252     PIDLIST_ABSOLUTE pidl = NULL;
253     hr = s_pDesktop->EnumObjects(NULL, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS,
254                                  &pEnumIDList);
255     ok_long(hr, S_OK);
256     while (pEnumIDList->Next(1, &pidl, NULL) == S_OK)
257     {
258         WCHAR szText[MAX_PATH];
259         SHGetPathFromIDListW(pidl, szText);
260         if (wcsstr(szText, DROPPED_ON_FILE) != NULL)
261         {
262             break;
263         }
264         CoTaskMemFree(pidl);
265         pidl = NULL;
266     }
267     ok(pidl != NULL, "Line %d: pidl is NULL\n", line);
268     pDropTarget = NULL;
269     PITEMID_CHILD pidlLast = ILFindLastID(pidl);
270     hr = s_pDesktop->GetUIObjectOf(NULL, 1, &pidlLast, IID_IDropTarget,
271                                    NULL, (LPVOID *)&pDropTarget);
272     CoTaskMemFree(pidl);
273     ok_long(hr, S_OK);
274 
275     if (!pDropTarget)
276     {
277         skip("Line %d: pDropTarget was NULL\n", line);
278 
279         // clean up
280         DeleteFileW(s_szSrcTestFile);
281         DeleteFileW(s_szDestTestFile);
282         DoDeleteSpecW(s_szDestLinkSpec);
283         ILFree(pidlDesktop);
284 
285         return;
286     }
287 
288     // DragEnter
289     POINTL ptl = { 0, 0 };
290     DWORD dwKeyState = pEntry->dwKeyState;
291     DWORD dwEffects = pEntry->dwEffects1;
292     hr = pDropTarget->DragEnter(pDataObject, dwKeyState, ptl, &dwEffects);
293 
294     ok(hr == pEntry->hr1, "Line %d: hr1 was %08lX\n", line, hr);
295 
296     switch (pEntry->dwEffects2)
297     {
298     case D_NONE_OR_COPY:
299         ok((dwEffects == D_NONE || dwEffects == D_COPY),
300            "Line %d: dwEffects2 was %08lX\n", line, dwEffects);
301         break;
302     case D_NONE_OR_MOVE:
303         ok((dwEffects == D_NONE || dwEffects == D_MOVE),
304            "Line %d: dwEffects2 was %08lX\n", line, dwEffects);
305         break;
306     case D_NONE_OR_LINK:
307         ok((dwEffects == D_NONE || dwEffects == D_LINK),
308            "Line %d: dwEffects2 was %08lX\n", line, dwEffects);
309         break;
310     default:
311         ok(dwEffects == pEntry->dwEffects2,
312            "Line %d: dwEffects2 was %08lX\n", line, dwEffects);
313         break;
314     }
315 
316     // Drop
317     hr = pDropTarget->Drop(pDataObject, dwKeyState, ptl, &dwEffects);
318     ok(hr == pEntry->hr2, "Line %d: hr2 was %08lX\n", line, hr);
319 
320     switch (pEntry->dwEffects3)
321     {
322     case D_NONE_OR_COPY:
323         ok((dwEffects == D_NONE || dwEffects == D_COPY),
324            "Line %d: dwEffects3 was %08lX\n", line, dwEffects);
325         break;
326     case D_NONE_OR_MOVE:
327         ok((dwEffects == D_NONE || dwEffects == D_MOVE),
328            "Line %d: dwEffects3 was %08lX\n", line, dwEffects);
329         break;
330     case D_NONE_OR_LINK:
331         ok((dwEffects == D_NONE || dwEffects == D_LINK),
332            "Line %d: dwEffects3 was %08lX\n", line, dwEffects);
333         break;
334     default:
335         ok(dwEffects == pEntry->dwEffects3,
336            "Line %d: dwEffects3 was %08lX\n", line, dwEffects);
337         break;
338     }
339 
340     // check file existence by pEntry->op
341     switch (pEntry->op)
342     {
343     case OP_NONE:
344         ok(PathFileExistsW(s_szSrcTestFile), "Line %d: src not exists\n", line);
345         ok(!PathFileExistsW(s_szDestTestFile), "Line %d: dest exists\n", line);
346         ok(!DoSpecExistsW(s_szDestLinkSpec), "Line %d: link exists\n", line);
347         break;
348     case OP_COPY:
349         ok(PathFileExistsW(s_szSrcTestFile), "Line %d: src not exists\n", line);
350         ok(PathFileExistsW(s_szDestTestFile), "Line %d: dest not exists\n", line);
351         ok(!DoSpecExistsW(s_szDestLinkSpec), "Line %d: link exists\n", line);
352         break;
353     case OP_MOVE:
354         ok(!PathFileExistsW(s_szSrcTestFile), "Line %d: src exists\n", line);
355         ok(PathFileExistsW(s_szDestTestFile), "Line %d: dest not exists\n", line);
356         ok(!DoSpecExistsW(s_szDestLinkSpec), "Line %d: link exists\n", line);
357         break;
358     case OP_LINK:
359         ok(PathFileExistsW(s_szSrcTestFile), "Line %d: src not exists\n", line);
360         ok(!PathFileExistsW(s_szDestTestFile), "Line %d: dest not exists\n", line);
361         ok(DoSpecExistsW(s_szDestLinkSpec), "Line %d: link not exists\n", line);
362         break;
363     case OP_NONE_OR_COPY:
364         ok(PathFileExistsW(s_szSrcTestFile), "Line %d: src not exists\n", line);
365         ok(!DoSpecExistsW(s_szDestLinkSpec), "Line %d: link exists\n", line);
366         break;
367     case OP_NONE_OR_MOVE:
368         ok(PathFileExistsW(s_szSrcTestFile) != PathFileExistsW(s_szDestTestFile),
369            "Line %d: It must be either None or Move\n", line);
370         break;
371     case OP_NONE_OR_LINK:
372         ok(PathFileExistsW(s_szSrcTestFile), "Line %d: src not exists\n", line);
373         ok(!PathFileExistsW(s_szDestTestFile), "Line %d: dest not exists\n", line);
374         break;
375     }
376 
377     // clean up
378     DeleteFileW(s_szSrcTestFile);
379     DeleteFileW(s_szDestTestFile);
380     DoDeleteSpecW(s_szDestLinkSpec);
381     ILFree(pidlDesktop);
382 }
383 
384 START_TEST(DragDrop)
385 {
386     HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
387     ok_int(SUCCEEDED(hr), TRUE);
388 
389     SHGetDesktopFolder(&s_pDesktop);
390     ok(!!s_pDesktop, "s_pDesktop is NULL\n");
391 
392     BOOL ret = SHGetSpecialFolderPathW(NULL, s_szDestFolder, CSIDL_DESKTOP, FALSE);
393     ok_int(ret, TRUE);
394 
395     for (size_t i = 0; i < _countof(s_TestEntries); ++i)
396     {
397         DoTestEntry(&s_TestEntries[i]);
398     }
399 
400     DeleteFileW(s_szSrcTestFile);
401     DeleteFileW(s_szDestTestFile);
402     DoDeleteSpecW(s_szDestLinkSpec);
403     DeleteFileW(s_szDroppedToItem);
404 
405     CoUninitialize();
406 }
407