1 /*
2  * Trash virtual folder support. The trashing engine is implemented in trash.c
3  *
4  * Copyright (C) 2006 Mikolaj Zalewski
5  * Copyright (C) 2009 Andrew Hill
6  * Copyright (C) 2018 Russell Johnson
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public
10  * License as published by the Free Software Foundation; either
11  * version 2.1 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with this library; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21  */
22 
23 #include <precomp.h>
24 
25 #include <mmsystem.h>
26 #include <ntquery.h>
27 
28 WINE_DEFAULT_DEBUG_CHANNEL(CRecycleBin);
29 
30 typedef struct
31 {
32     int column_name_id;
33     const GUID *fmtId;
34     DWORD pid;
35     int pcsFlags;
36     int fmt;
37     int cxChars;
38 } columninfo;
39 
40 static const columninfo RecycleBinColumns[] =
41 {
42     {IDS_SHV_COLUMN_NAME,        &FMTID_Storage,   PID_STG_NAME,       SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT,  LVCFMT_LEFT,  30},
43     {IDS_SHV_COLUMN_DELFROM, &FMTID_Displaced, PID_DISPLACED_FROM, SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT,  LVCFMT_LEFT,  30},
44     {IDS_SHV_COLUMN_DELDATE, &FMTID_Displaced, PID_DISPLACED_DATE, SHCOLSTATE_TYPE_DATE | SHCOLSTATE_ONBYDEFAULT, LVCFMT_LEFT,  20},
45     {IDS_SHV_COLUMN_SIZE,        &FMTID_Storage,   PID_STG_SIZE,       SHCOLSTATE_TYPE_INT | SHCOLSTATE_ONBYDEFAULT,  LVCFMT_RIGHT, 20},
46     {IDS_SHV_COLUMN_TYPE,        &FMTID_Storage,   PID_STG_STORAGETYPE, SHCOLSTATE_TYPE_INT | SHCOLSTATE_ONBYDEFAULT,  LVCFMT_LEFT,  20},
47     {IDS_SHV_COLUMN_MODIFIED,        &FMTID_Storage,   PID_STG_WRITETIME,  SHCOLSTATE_TYPE_DATE | SHCOLSTATE_ONBYDEFAULT, LVCFMT_LEFT,  20},
48     /*    {"creation time",  &FMTID_Storage,   PID_STG_CREATETIME, SHCOLSTATE_TYPE_DATE,                        LVCFMT_LEFT,  20}, */
49     /*    {"attribs",        &FMTID_Storage,   PID_STG_ATTRIBUTES, SHCOLSTATE_TYPE_STR,                         LVCFMT_LEFT,  20},       */
50 };
51 
52 #define COLUMN_NAME    0
53 #define COLUMN_DELFROM 1
54 #define COLUMN_DATEDEL 2
55 #define COLUMN_SIZE    3
56 #define COLUMN_TYPE    4
57 #define COLUMN_MTIME   5
58 
59 #define COLUMNS_COUNT  6
60 
61 /*
62  * Recycle Bin folder
63  */
64 
65 HRESULT CRecyclerExtractIcon_CreateInstance(LPCITEMIDLIST pidl, REFIID riid, LPVOID * ppvOut)
66 {
67     CComPtr<IDefaultExtractIconInit> initIcon;
68     HRESULT hr = SHCreateDefaultExtractIcon(IID_PPV_ARG(IDefaultExtractIconInit, &initIcon));
69     if (FAILED_UNEXPECTEDLY(hr))
70         return hr;
71 
72     /* FIXME: This is completely unimplemented */
73     initIcon->SetNormalIcon(swShell32Name, 0);
74 
75     return initIcon->QueryInterface(riid, ppvOut);
76 }
77 
78 class CRecycleBinEnum :
79     public CEnumIDListBase
80 {
81     private:
82     public:
83         CRecycleBinEnum();
84         ~CRecycleBinEnum();
85         HRESULT WINAPI Initialize(DWORD dwFlags);
86         static BOOL WINAPI CBEnumRecycleBin(IN PVOID Context, IN HANDLE hDeletedFile);
87         BOOL WINAPI CBEnumRecycleBin(IN HANDLE hDeletedFile);
88 
89         BEGIN_COM_MAP(CRecycleBinEnum)
90         COM_INTERFACE_ENTRY_IID(IID_IEnumIDList, IEnumIDList)
91         END_COM_MAP()
92 };
93 
94 class CRecycleBinItemContextMenu :
95     public CComObjectRootEx<CComMultiThreadModelNoCS>,
96     public IContextMenu2
97 {
98     private:
99         LPITEMIDLIST                        apidl;
100     public:
101         CRecycleBinItemContextMenu();
102         ~CRecycleBinItemContextMenu();
103         HRESULT WINAPI Initialize(LPCITEMIDLIST pidl);
104 
105         // IContextMenu
106         virtual HRESULT WINAPI QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags);
107         virtual HRESULT WINAPI InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi);
108         virtual HRESULT WINAPI GetCommandString(UINT_PTR idCommand, UINT uFlags, UINT *lpReserved, LPSTR lpszName, UINT uMaxNameLen);
109 
110         // IContextMenu2
111         virtual HRESULT WINAPI HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam);
112 
113         BEGIN_COM_MAP(CRecycleBinItemContextMenu)
114         COM_INTERFACE_ENTRY_IID(IID_IContextMenu, IContextMenu)
115         COM_INTERFACE_ENTRY_IID(IID_IContextMenu2, IContextMenu2)
116         END_COM_MAP()
117 };
118 
119 typedef struct
120 {
121     PIDLRecycleStruct *pFileDetails;
122     HANDLE hDeletedFile;
123     BOOL bFound;
124 } SEARCH_CONTEXT, *PSEARCH_CONTEXT;
125 
126 BOOL WINAPI CBSearchRecycleBin(IN PVOID Context, IN HANDLE hDeletedFile)
127 {
128     PSEARCH_CONTEXT pContext = (PSEARCH_CONTEXT)Context;
129 
130     PDELETED_FILE_DETAILS_W pFileDetails;
131     DWORD dwSize;
132     BOOL ret;
133 
134     if (!GetDeletedFileDetailsW(hDeletedFile,
135                                 0,
136                                 NULL,
137                                 &dwSize) &&
138             GetLastError() != ERROR_INSUFFICIENT_BUFFER)
139     {
140         ERR("GetDeletedFileDetailsW failed\n");
141         return FALSE;
142     }
143 
144     pFileDetails = (DELETED_FILE_DETAILS_W *)SHAlloc(dwSize);
145     if (!pFileDetails)
146     {
147         ERR("No memory\n");
148         return FALSE;
149     }
150 
151     if (!GetDeletedFileDetailsW(hDeletedFile,
152                                 dwSize,
153                                 pFileDetails,
154                                 NULL))
155     {
156         ERR("GetDeletedFileDetailsW failed\n");
157         SHFree(pFileDetails);
158         return FALSE;
159     }
160 
161     ret = memcmp(pFileDetails, pContext->pFileDetails, dwSize);
162     if (!ret)
163     {
164         pContext->hDeletedFile = hDeletedFile;
165         pContext->bFound = TRUE;
166     }
167     else
168         CloseRecycleBinHandle(hDeletedFile);
169 
170     SHFree(pFileDetails);
171     return ret;
172 }
173 
174 static PIDLRecycleStruct * _ILGetRecycleStruct(LPCITEMIDLIST pidl)
175 {
176     LPPIDLDATA pdata = _ILGetDataPointer(pidl);
177 
178     if (pdata && pdata->type == 0x00)
179         return (PIDLRecycleStruct*) & (pdata->u.crecycle);
180 
181     return NULL;
182 }
183 
184 CRecycleBinEnum::CRecycleBinEnum()
185 {
186 }
187 
188 CRecycleBinEnum::~CRecycleBinEnum()
189 {
190 }
191 
192 HRESULT WINAPI CRecycleBinEnum::Initialize(DWORD dwFlags)
193 {
194     static LPCWSTR szDrive = L"C:\\";
195 
196     if (dwFlags & SHCONTF_NONFOLDERS)
197     {
198         TRACE("Starting Enumeration\n");
199 
200         if (!EnumerateRecycleBinW(szDrive /* FIXME */ , CBEnumRecycleBin, (PVOID)this))
201         {
202             WARN("Error: EnumerateCRecycleBinW failed\n");
203             return E_FAIL;
204         }
205     }
206     else
207     {
208         // do nothing
209     }
210     return S_OK;
211 }
212 
213 static LPITEMIDLIST _ILCreateRecycleItem(PDELETED_FILE_DETAILS_W pFileDetails)
214 {
215     PIDLDATA tmp;
216     LPITEMIDLIST pidl;
217     PIDLRecycleStruct * p;
218     int size0 = (char*)&tmp.u.crecycle.szName - (char*)&tmp.u.crecycle;
219     int size = size0;
220 
221     tmp.type = 0x00;
222     size += (wcslen(pFileDetails->FileName) + 1) * sizeof(WCHAR);
223 
224     pidl = (LPITEMIDLIST)SHAlloc(size + 4);
225     if (!pidl)
226         return pidl;
227 
228     pidl->mkid.cb = size + 2;
229     memcpy(pidl->mkid.abID, &tmp, 2 + size0);
230 
231     p = &((PIDLDATA*)pidl->mkid.abID)->u.crecycle;
232     RtlCopyMemory(p, pFileDetails, sizeof(DELETED_FILE_DETAILS_W));
233     wcscpy(p->szName, pFileDetails->FileName);
234     *(WORD*)((char*)pidl + (size + 2)) = 0;
235     return pidl;
236 }
237 
238 BOOL WINAPI CRecycleBinEnum::CBEnumRecycleBin(IN PVOID Context, IN HANDLE hDeletedFile)
239 {
240     return static_cast<CRecycleBinEnum *>(Context)->CBEnumRecycleBin(hDeletedFile);
241 }
242 
243 BOOL WINAPI CRecycleBinEnum::CBEnumRecycleBin(IN HANDLE hDeletedFile)
244 {
245     PDELETED_FILE_DETAILS_W pFileDetails;
246     DWORD dwSize;
247     LPITEMIDLIST pidl = NULL;
248     BOOL ret;
249 
250     if (!GetDeletedFileDetailsW(hDeletedFile,
251                                 0,
252                                 NULL,
253                                 &dwSize) &&
254             GetLastError() != ERROR_INSUFFICIENT_BUFFER)
255     {
256         ERR("GetDeletedFileDetailsW failed\n");
257         return FALSE;
258     }
259 
260     pFileDetails = (DELETED_FILE_DETAILS_W *)SHAlloc(dwSize);
261     if (!pFileDetails)
262     {
263         ERR("No memory\n");
264         return FALSE;
265     }
266 
267     if (!GetDeletedFileDetailsW(hDeletedFile,
268                                 dwSize,
269                                 pFileDetails,
270                                 NULL))
271     {
272         ERR("GetDeletedFileDetailsW failed\n");
273         SHFree(pFileDetails);
274         return FALSE;
275     }
276 
277     pidl = _ILCreateRecycleItem(pFileDetails);
278     if (!pidl)
279     {
280         SHFree(pFileDetails);
281         return FALSE;
282     }
283 
284     ret = AddToEnumList(pidl);
285 
286     if (!ret)
287         SHFree(pidl);
288     SHFree(pFileDetails);
289     TRACE("Returning %d\n", ret);
290     CloseRecycleBinHandle(hDeletedFile);
291     return ret;
292 }
293 
294 /**************************************************************************
295 * IContextMenu2 Bitbucket Item Implementation
296 */
297 
298 CRecycleBinItemContextMenu::CRecycleBinItemContextMenu()
299 {
300     apidl = NULL;
301 }
302 
303 CRecycleBinItemContextMenu::~CRecycleBinItemContextMenu()
304 {
305     ILFree(apidl);
306 }
307 
308 HRESULT WINAPI CRecycleBinItemContextMenu::Initialize(LPCITEMIDLIST pidl)
309 {
310     apidl = ILClone(pidl);
311     if (apidl == NULL)
312         return E_OUTOFMEMORY;
313     return S_OK;
314 }
315 
316 HRESULT WINAPI CRecycleBinItemContextMenu::QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
317 {
318     WCHAR szBuffer[30] = {0};
319     ULONG Count = 1;
320 
321     TRACE("(%p)->(hmenu=%p indexmenu=%x cmdfirst=%x cmdlast=%x flags=%x )\n", this, hMenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
322 
323     if (LoadStringW(shell32_hInstance, IDS_RESTORE, szBuffer, _countof(szBuffer)))
324     {
325         szBuffer[_countof(szBuffer)-1] = L'\0';
326         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count, MFT_STRING, szBuffer, MFS_ENABLED);
327         Count++;
328     }
329 
330     if (LoadStringW(shell32_hInstance, IDS_CUT, szBuffer, _countof(szBuffer)))
331     {
332         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count++, MFT_SEPARATOR, NULL, MFS_ENABLED);
333         szBuffer[_countof(szBuffer)-1] = L'\0';
334         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count++, MFT_STRING, szBuffer, MFS_ENABLED);
335     }
336 
337     if (LoadStringW(shell32_hInstance, IDS_DELETE, szBuffer, _countof(szBuffer)))
338     {
339         szBuffer[_countof(szBuffer)-1] = L'\0';
340         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count++, MFT_SEPARATOR, NULL, MFS_ENABLED);
341         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count++, MFT_STRING, szBuffer, MFS_ENABLED);
342     }
343 
344     if (LoadStringW(shell32_hInstance, IDS_PROPERTIES, szBuffer, _countof(szBuffer)))
345     {
346         szBuffer[_countof(szBuffer)-1] = L'\0';
347         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count++, MFT_SEPARATOR, NULL, MFS_ENABLED);
348         _InsertMenuItemW(hMenu, indexMenu++, TRUE, idCmdFirst + Count, MFT_STRING, szBuffer, MFS_DEFAULT);
349     }
350 
351     return MAKE_HRESULT(SEVERITY_SUCCESS, 0, Count);
352 }
353 
354 HRESULT WINAPI CRecycleBinItemContextMenu::InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi)
355 {
356     SEARCH_CONTEXT Context;
357     static LPCWSTR szDrive = L"C:\\";
358 
359     TRACE("(%p)->(invcom=%p verb=%p wnd=%p)\n", this, lpcmi, lpcmi->lpVerb, lpcmi->hwnd);
360 
361     if (lpcmi->lpVerb == MAKEINTRESOURCEA(1) || lpcmi->lpVerb == MAKEINTRESOURCEA(5))
362     {
363         Context.pFileDetails = _ILGetRecycleStruct(apidl);
364         Context.bFound = FALSE;
365 
366         EnumerateRecycleBinW(szDrive, CBSearchRecycleBin, (PVOID)&Context);
367         if (!Context.bFound)
368             return E_FAIL;
369 
370         if (lpcmi->lpVerb == MAKEINTRESOURCEA(1))
371         {
372             /* restore file */
373             if (RestoreFile(Context.hDeletedFile))
374                 return S_OK;
375             else
376                 return E_FAIL;
377         }
378         else
379         {
380             DeleteFileHandleToRecycleBin(Context.hDeletedFile);
381             return E_NOTIMPL;
382         }
383     }
384     else if (lpcmi->lpVerb == MAKEINTRESOURCEA(3))
385     {
386         FIXME("implement cut\n");
387         return E_NOTIMPL;
388     }
389     else if (lpcmi->lpVerb == MAKEINTRESOURCEA(7))
390     {
391         FIXME("implement properties\n");
392         return E_NOTIMPL;
393     }
394 
395     return S_OK;
396 }
397 
398 HRESULT WINAPI CRecycleBinItemContextMenu::GetCommandString(UINT_PTR idCommand, UINT uFlags, UINT *lpReserved, LPSTR lpszName, UINT uMaxNameLen)
399 {
400     TRACE("(%p)->(idcom=%lx flags=%x %p name=%p len=%x)\n", this, idCommand, uFlags, lpReserved, lpszName, uMaxNameLen);
401 
402     return E_FAIL;
403 }
404 
405 HRESULT WINAPI CRecycleBinItemContextMenu::HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam)
406 {
407     TRACE("CRecycleBin_IContextMenu2Item_HandleMenuMsg (%p)->(msg=%x wp=%lx lp=%lx)\n", this, uMsg, wParam, lParam);
408 
409     return E_NOTIMPL;
410 }
411 
412 CRecycleBin::CRecycleBin()
413 {
414     pidl = NULL;
415 }
416 
417 CRecycleBin::~CRecycleBin()
418 {
419     SHFree(pidl);
420 }
421 
422 /*************************************************************************
423  * RecycleBin IPersistFolder2 interface
424  */
425 
426 HRESULT WINAPI CRecycleBin::GetClassID(CLSID *pClassID)
427 {
428     TRACE("(%p, %p)\n", this, pClassID);
429     if (pClassID == NULL)
430         return E_INVALIDARG;
431     memcpy(pClassID, &CLSID_RecycleBin, sizeof(CLSID));
432     return S_OK;
433 }
434 
435 HRESULT WINAPI CRecycleBin::Initialize(PCIDLIST_ABSOLUTE pidl)
436 {
437     TRACE("(%p, %p)\n", this, pidl);
438 
439     SHFree((LPVOID)this->pidl);
440     this->pidl = ILClone(pidl);
441     if (this->pidl == NULL)
442         return E_OUTOFMEMORY;
443     return S_OK;
444 }
445 
446 HRESULT WINAPI CRecycleBin::GetCurFolder(PIDLIST_ABSOLUTE *ppidl)
447 {
448     TRACE("\n");
449     *ppidl = ILClone(pidl);
450     return S_OK;
451 }
452 
453 /*************************************************************************
454  * RecycleBin IShellFolder2 interface
455  */
456 
457 HRESULT WINAPI CRecycleBin::ParseDisplayName(HWND hwnd, LPBC pbc,
458         LPOLESTR pszDisplayName, ULONG *pchEaten, PIDLIST_RELATIVE *ppidl,
459         ULONG *pdwAttributes)
460 {
461     FIXME("stub\n");
462     return E_NOTIMPL;
463 }
464 
465 
466 PDELETED_FILE_DETAILS_W
467 UnpackDetailsFromPidl(LPCITEMIDLIST pidl)
468 {
469     return (PDELETED_FILE_DETAILS_W)&pidl->mkid.abID;
470 }
471 
472 HRESULT WINAPI CRecycleBin::EnumObjects(HWND hwndOwner, DWORD dwFlags, LPENUMIDLIST *ppEnumIDList)
473 {
474     return ShellObjectCreatorInit<CRecycleBinEnum>(dwFlags, IID_PPV_ARG(IEnumIDList, ppEnumIDList));
475 }
476 
477 HRESULT WINAPI CRecycleBin::BindToObject(PCUIDLIST_RELATIVE pidl, LPBC pbc, REFIID riid, void **ppv)
478 {
479     FIXME("(%p, %p, %p, %s, %p) - stub\n", this, pidl, pbc, debugstr_guid(&riid), ppv);
480     return E_NOTIMPL;
481 }
482 
483 HRESULT WINAPI CRecycleBin::BindToStorage(PCUIDLIST_RELATIVE pidl, LPBC pbc, REFIID riid, void **ppv)
484 {
485     FIXME("(%p, %p, %p, %s, %p) - stub\n", this, pidl, pbc, debugstr_guid(&riid), ppv);
486     return E_NOTIMPL;
487 }
488 
489 HRESULT WINAPI CRecycleBin::CompareIDs(LPARAM lParam, PCUIDLIST_RELATIVE pidl1, PCUIDLIST_RELATIVE pidl2)
490 {
491     PIDLRecycleStruct* pData1 = _ILGetRecycleStruct(pidl1);
492     PIDLRecycleStruct* pData2 = _ILGetRecycleStruct(pidl2);
493     LPWSTR pName1, pName2;
494 
495     if(!pData1 || !pData2 || LOWORD(lParam) >= COLUMNS_COUNT)
496         return E_INVALIDARG;
497 
498     SHORT result;
499     LONGLONG diff;
500     switch (LOWORD(lParam))
501     {
502         case 0: /* Name */
503             pName1 = PathFindFileNameW(pData1->szName);
504             pName2 = PathFindFileNameW(pData2->szName);
505             result = wcsicmp(pName1, pName2);
506             break;
507         case 1: /* Orig. Location */
508             result = wcsicmp(pData1->szName, pData2->szName);
509             break;
510         case 2: /* Date Deleted */
511             result = CompareFileTime(&pData1->DeletionTime, &pData2->DeletionTime);
512             break;
513         case 3: /* Size */
514             diff = pData1->FileSize.QuadPart - pData2->FileSize.QuadPart;
515             return MAKE_COMPARE_HRESULT(diff);
516         case 4: /* Type */
517             pName1 = PathFindExtensionW(pData1->szName);
518             pName2 = PathFindExtensionW(pData2->szName);
519             result = wcsicmp(pName1, pName2);
520             break;
521         case 5: /* Modified */
522             result = CompareFileTime(&pData1->LastModification, &pData2->LastModification);
523             break;
524     }
525     return MAKE_COMPARE_HRESULT(result);
526 }
527 
528 HRESULT WINAPI CRecycleBin::CreateViewObject(HWND hwndOwner, REFIID riid, void **ppv)
529 {
530     CComPtr<IShellView> pShellView;
531     HRESULT hr = E_NOINTERFACE;
532 
533     TRACE("(%p, %p, %s, %p)\n", this, hwndOwner, debugstr_guid(&riid), ppv);
534 
535     if (!ppv)
536         return hr;
537 
538     *ppv = NULL;
539 
540     if (IsEqualIID (riid, IID_IDropTarget))
541     {
542         hr = CRecyclerDropTarget_CreateInstance(riid, ppv);
543     }
544     else if (IsEqualIID (riid, IID_IContextMenu) || IsEqualIID (riid, IID_IContextMenu2))
545     {
546         hr = this->QueryInterface(riid, ppv);
547     }
548     else if (IsEqualIID (riid, IID_IShellView))
549     {
550         SFV_CREATE sfvparams = {sizeof(SFV_CREATE), this};
551         hr = SHCreateShellFolderView(&sfvparams, (IShellView**)ppv);
552     }
553     else
554         return hr;
555 
556     TRACE ("-- (%p)->(interface=%p)\n", this, ppv);
557     return hr;
558 
559 }
560 
561 HRESULT WINAPI CRecycleBin::GetAttributesOf(UINT cidl, PCUITEMID_CHILD_ARRAY apidl,
562         SFGAOF *rgfInOut)
563 {
564     TRACE("(%p, %d, {%p, ...}, {%x})\n", this, cidl, apidl ? apidl[0] : NULL, (unsigned int)*rgfInOut);
565     *rgfInOut &= SFGAO_FOLDER|SFGAO_DROPTARGET|SFGAO_HASPROPSHEET|SFGAO_CANLINK;
566     return S_OK;
567 }
568 
569 HRESULT WINAPI CRecycleBin::GetUIObjectOf(HWND hwndOwner, UINT cidl, PCUITEMID_CHILD_ARRAY apidl,
570         REFIID riid, UINT *prgfInOut, void **ppv)
571 {
572     LPVOID pObj = NULL;
573     HRESULT hr = E_INVALIDARG;
574 
575     TRACE ("(%p)->(%p,%u,apidl=%p, %p %p)\n", this,
576            hwndOwner, cidl, apidl, prgfInOut, ppv);
577 
578     if (!ppv)
579         return hr;
580 
581     *ppv = NULL;
582 
583     if ((IsEqualIID (riid, IID_IContextMenu) || IsEqualIID(riid, IID_IContextMenu2)) && (cidl >= 1))
584     {
585         hr = ShellObjectCreatorInit<CRecycleBinItemContextMenu>(apidl[0], riid, &pObj);
586     }
587     else if((IsEqualIID(riid, IID_IExtractIconA) || IsEqualIID(riid, IID_IExtractIconW)) && (cidl == 1))
588     {
589         hr = CRecyclerExtractIcon_CreateInstance(apidl[0], riid, &pObj);
590     }
591     else
592         hr = E_NOINTERFACE;
593 
594     if (SUCCEEDED(hr) && !pObj)
595         hr = E_OUTOFMEMORY;
596 
597     *ppv = pObj;
598     TRACE ("(%p)->hr=0x%08x\n", this, hr);
599     return hr;
600 }
601 
602 HRESULT WINAPI CRecycleBin::GetDisplayNameOf(PCUITEMID_CHILD pidl, SHGDNF uFlags, STRRET *pName)
603 {
604     PIDLRecycleStruct *pFileDetails;
605     LPWSTR pFileName;
606 
607     TRACE("(%p, %p, %x, %p)\n", this, pidl, (unsigned int)uFlags, pName);
608 
609     pFileDetails = _ILGetRecycleStruct(pidl);
610     if (!pFileDetails)
611     {
612         pName->cStr[0] = 0;
613         pName->uType = STRRET_CSTR;
614         return E_INVALIDARG;
615     }
616 
617     pFileName = wcsrchr(pFileDetails->szName, L'\\');
618     if (!pFileName)
619     {
620         pName->cStr[0] = 0;
621         pName->uType = STRRET_CSTR;
622         return E_UNEXPECTED;
623     }
624 
625     pName->pOleStr = StrDupW(pFileName + 1);
626     if (pName->pOleStr == NULL)
627         return E_OUTOFMEMORY;
628 
629     pName->uType = STRRET_WSTR;
630     return S_OK;
631 }
632 
633 HRESULT WINAPI CRecycleBin::SetNameOf(HWND hwnd, PCUITEMID_CHILD pidl, LPCOLESTR pszName,
634                                       SHGDNF uFlags, PITEMID_CHILD *ppidlOut)
635 {
636     TRACE("\n");
637     return E_FAIL; /* not supported */
638 }
639 
640 HRESULT WINAPI CRecycleBin::GetDefaultSearchGUID(GUID *pguid)
641 {
642     FIXME("stub\n");
643     return E_NOTIMPL;
644 }
645 
646 HRESULT WINAPI CRecycleBin::EnumSearches(IEnumExtraSearch **ppEnum)
647 {
648     FIXME("stub\n");
649     *ppEnum = NULL;
650     return E_NOTIMPL;
651 }
652 
653 HRESULT WINAPI CRecycleBin::GetDefaultColumn(DWORD dwReserved, ULONG *pSort, ULONG *pDisplay)
654 {
655     TRACE("(%p, %x, %p, %p)\n", this, (unsigned int)dwReserved, pSort, pDisplay);
656     if (pSort)
657         *pSort = 0;
658     if (pDisplay)
659         *pDisplay = 0;
660     return S_OK;
661 }
662 
663 HRESULT WINAPI CRecycleBin::GetDefaultColumnState(UINT iColumn, SHCOLSTATEF *pcsFlags)
664 {
665     TRACE("(%p, %d, %p)\n", this, iColumn, pcsFlags);
666     if (iColumn >= COLUMNS_COUNT)
667         return E_INVALIDARG;
668     *pcsFlags = RecycleBinColumns[iColumn].pcsFlags;
669     return S_OK;
670 }
671 
672 HRESULT WINAPI CRecycleBin::GetDetailsEx(PCUITEMID_CHILD pidl, const SHCOLUMNID *pscid, VARIANT *pv)
673 {
674     FIXME("stub\n");
675     return E_NOTIMPL;
676 }
677 
678 static HRESULT FormatDateTime(LPWSTR buffer, int size, FILETIME * ft)
679 {
680     FILETIME lft;
681     SYSTEMTIME time;
682     int ret;
683 
684     FileTimeToLocalFileTime(ft, &lft);
685     FileTimeToSystemTime(&lft, &time);
686 
687     ret = GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &time, NULL, buffer, size);
688     if (ret > 0 && ret < size)
689     {
690         /* Append space + time without seconds */
691         buffer[ret-1] = ' ';
692         GetTimeFormatW(LOCALE_USER_DEFAULT, TIME_NOSECONDS, &time, NULL, &buffer[ret], size - ret);
693     }
694 
695     return (ret != 0 ? E_FAIL : S_OK);
696 }
697 
698 HRESULT WINAPI CRecycleBin::GetDetailsOf(PCUITEMID_CHILD pidl, UINT iColumn, LPSHELLDETAILS pDetails)
699 {
700     PIDLRecycleStruct * pFileDetails;
701     WCHAR buffer[MAX_PATH];
702     WCHAR szTypeName[100];
703     LPWSTR pszBackslash;
704     UINT Length;
705 
706     TRACE("(%p, %p, %d, %p)\n", this, pidl, iColumn, pDetails);
707     if (iColumn >= COLUMNS_COUNT)
708         return E_FAIL;
709     pDetails->fmt = RecycleBinColumns[iColumn].fmt;
710     pDetails->cxChar = RecycleBinColumns[iColumn].cxChars;
711     if (pidl == NULL)
712         return SHSetStrRet(&pDetails->str, RecycleBinColumns[iColumn].column_name_id);
713 
714     if (iColumn == COLUMN_NAME)
715         return GetDisplayNameOf(pidl, SHGDN_NORMAL, &pDetails->str);
716 
717     pFileDetails = _ILGetRecycleStruct(pidl);
718     switch (iColumn)
719     {
720         case COLUMN_DATEDEL:
721             FormatDateTime(buffer, MAX_PATH, &pFileDetails->DeletionTime);
722             break;
723         case COLUMN_DELFROM:
724             pszBackslash = wcsrchr(pFileDetails->szName, L'\\');
725             Length = (pszBackslash - pFileDetails->szName);
726             memcpy((LPVOID)buffer, pFileDetails->szName, Length * sizeof(WCHAR));
727             buffer[Length] = L'\0';
728             break;
729         case COLUMN_SIZE:
730             StrFormatKBSizeW(pFileDetails->FileSize.QuadPart, buffer, MAX_PATH);
731             break;
732         case COLUMN_MTIME:
733             FormatDateTime(buffer, MAX_PATH, &pFileDetails->LastModification);
734             break;
735         case COLUMN_TYPE:
736             // FIXME: We should in fact use a UNICODE version of _ILGetFileType
737             szTypeName[0] = L'\0';
738             wcscpy(buffer, PathFindExtensionW(pFileDetails->szName));
739             if (!( HCR_MapTypeToValueW(buffer, buffer, _countof(buffer), TRUE) &&
740                     HCR_MapTypeToValueW(buffer, szTypeName, _countof(szTypeName), FALSE )))
741             {
742                 /* load localized file string */
743                 szTypeName[0] = '\0';
744                 if(LoadStringW(shell32_hInstance, IDS_ANY_FILE, szTypeName, _countof(szTypeName)))
745                 {
746                     szTypeName[63] = '\0';
747                     StringCchPrintfW(buffer, _countof(buffer), szTypeName, PathFindExtensionW(pFileDetails->szName));
748                 }
749             }
750             return SHSetStrRet(&pDetails->str, szTypeName);
751         default:
752             return E_FAIL;
753     }
754 
755     return SHSetStrRet(&pDetails->str, buffer);
756 }
757 
758 HRESULT WINAPI CRecycleBin::MapColumnToSCID(UINT iColumn, SHCOLUMNID *pscid)
759 {
760     TRACE("(%p, %d, %p)\n", this, iColumn, pscid);
761     if (iColumn >= COLUMNS_COUNT)
762         return E_INVALIDARG;
763     pscid->fmtid = *RecycleBinColumns[iColumn].fmtId;
764     pscid->pid = RecycleBinColumns[iColumn].pid;
765     return S_OK;
766 }
767 
768 BOOL CRecycleBin::RecycleBinIsEmpty()
769 {
770     CComPtr<IEnumIDList> spEnumFiles;
771     HRESULT hr = EnumObjects(NULL, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS, &spEnumFiles);
772     if (FAILED(hr))
773         return TRUE;
774     CComHeapPtr<ITEMIDLIST> spPidl;
775     ULONG itemcount;
776     return spEnumFiles->Next(1, &spPidl, &itemcount) != S_OK;
777     }
778 
779 /*************************************************************************
780  * RecycleBin IContextMenu interface
781  */
782 
783 HRESULT WINAPI CRecycleBin::QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
784 {
785     WCHAR szBuffer[100];
786     MENUITEMINFOW mii;
787     int id = 1;
788 
789     TRACE("QueryContextMenu %p %p %u %u %u %u\n", this, hMenu, indexMenu, idCmdFirst, idCmdLast, uFlags );
790 
791     if (!hMenu)
792         return E_INVALIDARG;
793 
794     ZeroMemory(&mii, sizeof(mii));
795     mii.cbSize = sizeof(mii);
796     mii.fMask = MIIM_TYPE | MIIM_ID | MIIM_STATE;
797     mii.fState = RecycleBinIsEmpty() ? MFS_DISABLED : MFS_ENABLED;
798     szBuffer[0] = L'\0';
799     LoadStringW(shell32_hInstance, IDS_EMPTY_BITBUCKET, szBuffer, _countof(szBuffer));
800     mii.dwTypeData = szBuffer;
801     mii.cch = wcslen(mii.dwTypeData);
802     mii.wID = idCmdFirst + id++;
803     mii.fType = MFT_STRING;
804     iIdEmpty = 1;
805 
806     if (!InsertMenuItemW(hMenu, indexMenu, TRUE, &mii))
807         return E_FAIL;
808 
809     return MAKE_HRESULT(SEVERITY_SUCCESS, 0, id);
810 }
811 
812 HRESULT WINAPI CRecycleBin::InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi)
813 {
814     HRESULT hr;
815     LPSHELLBROWSER lpSB;
816     IShellView * lpSV = NULL;
817 
818     TRACE("%p %p verb %p\n", this, lpcmi, lpcmi->lpVerb);
819 
820     if (LOWORD(lpcmi->lpVerb) == iIdEmpty)
821     {
822         // FIXME
823         // path & flags
824         hr = SHEmptyRecycleBinW(lpcmi->hwnd, L"C:\\", 0);
825         TRACE("result %x\n", hr);
826         if (hr != S_OK)
827             return hr;
828 
829         lpSB = (LPSHELLBROWSER)SendMessageA(lpcmi->hwnd, CWM_GETISHELLBROWSER, 0, 0);
830         if (lpSB && SUCCEEDED(lpSB->QueryActiveShellView(&lpSV)))
831             lpSV->Refresh();
832     }
833     return S_OK;
834 }
835 
836 HRESULT WINAPI CRecycleBin::GetCommandString(UINT_PTR idCommand, UINT uFlags, UINT *lpReserved, LPSTR lpszName, UINT uMaxNameLen)
837 {
838     FIXME("%p %lu %u %p %p %u\n", this, idCommand, uFlags, lpReserved, lpszName, uMaxNameLen);
839 
840     return E_NOTIMPL;
841 }
842 
843 /*************************************************************************
844  * RecycleBin IShellPropSheetExt interface
845  */
846 
847 HRESULT WINAPI CRecycleBin::AddPages(LPFNSVADDPROPSHEETPAGE pfnAddPage, LPARAM lParam)
848 {
849     FIXME("%p %p %lu\n", this, pfnAddPage, lParam);
850 
851     return E_NOTIMPL;
852 }
853 
854 HRESULT WINAPI CRecycleBin::ReplacePage(EXPPS uPageID, LPFNSVADDPROPSHEETPAGE pfnReplaceWith, LPARAM lParam)
855 {
856     FIXME("%p %lu %p %lu\n", this, uPageID, pfnReplaceWith, lParam);
857 
858     return E_NOTIMPL;
859 }
860 
861 /*************************************************************************
862  * RecycleBin IShellExtInit interface
863  */
864 
865 HRESULT WINAPI CRecycleBin::Initialize(PCIDLIST_ABSOLUTE pidlFolder, IDataObject *pdtobj, HKEY hkeyProgID)
866 {
867     TRACE("%p %p %p %p\n", this, pidlFolder, pdtobj, hkeyProgID );
868     return S_OK;
869 }
870 
871 /**
872  * Tests whether a file can be trashed
873  * @param wszPath Path to the file to be trash
874  * @returns TRUE if the file can be trashed, FALSE otherwise
875  */
876 BOOL
877 TRASH_CanTrashFile(LPCWSTR wszPath)
878 {
879     LONG ret;
880     DWORD dwNukeOnDelete, dwType, VolSerialNumber, MaxComponentLength;
881     DWORD FileSystemFlags, dwSize, dwDisposition;
882     HKEY hKey;
883     WCHAR szBuffer[10];
884     WCHAR szKey[150] = L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\BitBucket\\Volume\\";
885 
886     if (wszPath[1] != L':')
887     {
888         /* path is UNC */
889         return FALSE;
890     }
891 
892     // Copy and retrieve the root path from get given string
893     WCHAR wszRootPathName[MAX_PATH];
894     StringCbCopyW(wszRootPathName, sizeof(wszRootPathName), wszPath);
895     PathStripToRootW(wszRootPathName);
896 
897     // Test to see if the drive is fixed (non removable)
898     if (GetDriveTypeW(wszRootPathName) != DRIVE_FIXED)
899     {
900         /* no bitbucket on removable media */
901         return FALSE;
902     }
903 
904     if (!GetVolumeInformationW(wszRootPathName, NULL, 0, &VolSerialNumber, &MaxComponentLength, &FileSystemFlags, NULL, 0))
905     {
906         ERR("GetVolumeInformationW failed with %u wszRootPathName=%s\n", GetLastError(), debugstr_w(wszRootPathName));
907         return FALSE;
908     }
909 
910     swprintf(szBuffer, L"%04X-%04X", LOWORD(VolSerialNumber), HIWORD(VolSerialNumber));
911     wcscat(szKey, szBuffer);
912 
913     if (RegCreateKeyExW(HKEY_CURRENT_USER, szKey, 0, NULL, 0, KEY_WRITE, NULL, &hKey, &dwDisposition) != ERROR_SUCCESS)
914     {
915         ERR("RegCreateKeyExW failed\n");
916         return FALSE;
917     }
918 
919     if (dwDisposition  & REG_CREATED_NEW_KEY)
920     {
921         /* per default move to bitbucket */
922         dwNukeOnDelete = 0;
923         RegSetValueExW(hKey, L"NukeOnDelete", 0, REG_DWORD, (LPBYTE)&dwNukeOnDelete, sizeof(DWORD));
924         /* per default unlimited size */
925         dwSize = -1;
926         RegSetValueExW(hKey, L"MaxCapacity", 0, REG_DWORD, (LPBYTE)&dwSize, sizeof(DWORD));
927         RegCloseKey(hKey);
928         return TRUE;
929     }
930     else
931     {
932         dwSize = sizeof(dwNukeOnDelete);
933         ret = RegQueryValueExW(hKey, L"NukeOnDelete", NULL, &dwType, (LPBYTE)&dwNukeOnDelete, &dwSize);
934         if (ret != ERROR_SUCCESS)
935         {
936             if (ret ==  ERROR_FILE_NOT_FOUND)
937             {
938                 /* restore key and enable bitbucket */
939                 dwNukeOnDelete = 0;
940                 RegSetValueExW(hKey, L"NukeOnDelete", 0, REG_DWORD, (LPBYTE)&dwNukeOnDelete, sizeof(DWORD));
941             }
942             RegCloseKey(hKey);
943             return TRUE;
944         }
945         else if (dwNukeOnDelete)
946         {
947             /* do not delete to bitbucket */
948             RegCloseKey(hKey);
949             return FALSE;
950         }
951         /* FIXME
952          * check if bitbucket is full
953          */
954         RegCloseKey(hKey);
955         return TRUE;
956     }
957 }
958 
959 BOOL
960 TRASH_TrashFile(LPCWSTR wszPath)
961 {
962     TRACE("(%s)\n", debugstr_w(wszPath));
963     return DeleteFileToRecycleBin(wszPath);
964 }
965 
966 /*************************************************************************
967  * SHUpdateCRecycleBinIcon                                [SHELL32.@]
968  *
969  * Undocumented
970  */
971 EXTERN_C HRESULT WINAPI SHUpdateRecycleBinIcon(void)
972 {
973     FIXME("stub\n");
974 
975     return S_OK;
976 }
977 
978 /*************************************************************************
979  *              SHEmptyRecycleBinA (SHELL32.@)
980  */
981 HRESULT WINAPI SHEmptyRecycleBinA(HWND hwnd, LPCSTR pszRootPath, DWORD dwFlags)
982 {
983     LPWSTR szRootPathW = NULL;
984     int len;
985     HRESULT hr;
986 
987     TRACE("%p, %s, 0x%08x\n", hwnd, debugstr_a(pszRootPath), dwFlags);
988 
989     if (pszRootPath)
990     {
991         len = MultiByteToWideChar(CP_ACP, 0, pszRootPath, -1, NULL, 0);
992         if (len == 0)
993             return HRESULT_FROM_WIN32(GetLastError());
994         szRootPathW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
995         if (!szRootPathW)
996             return E_OUTOFMEMORY;
997         if (MultiByteToWideChar(CP_ACP, 0, pszRootPath, -1, szRootPathW, len) == 0)
998         {
999             HeapFree(GetProcessHeap(), 0, szRootPathW);
1000             return HRESULT_FROM_WIN32(GetLastError());
1001         }
1002     }
1003 
1004     hr = SHEmptyRecycleBinW(hwnd, szRootPathW, dwFlags);
1005     HeapFree(GetProcessHeap(), 0, szRootPathW);
1006 
1007     return hr;
1008 }
1009 
1010 HRESULT WINAPI SHEmptyRecycleBinW(HWND hwnd, LPCWSTR pszRootPath, DWORD dwFlags)
1011 {
1012     WCHAR szPath[MAX_PATH] = {0}, szBuffer[MAX_PATH];
1013     DWORD dwSize, dwType, count;
1014     LONG ret;
1015     IShellFolder *pDesktop, *pRecycleBin;
1016     PIDLIST_ABSOLUTE pidlRecycleBin;
1017     PITEMID_CHILD pidl;
1018     HRESULT hr = S_OK;
1019     LPENUMIDLIST penumFiles;
1020     STRRET StrRet;
1021 
1022     TRACE("%p, %s, 0x%08x\n", hwnd, debugstr_w(pszRootPath), dwFlags);
1023 
1024     if (!(dwFlags & SHERB_NOCONFIRMATION))
1025     {
1026         hr = SHGetDesktopFolder(&pDesktop);
1027         if (FAILED(hr))
1028             return hr;
1029         hr = SHGetFolderLocation(NULL, CSIDL_BITBUCKET, NULL, 0, &pidlRecycleBin);
1030         if (FAILED(hr))
1031         {
1032             pDesktop->Release();
1033             return hr;
1034         }
1035         hr = pDesktop->BindToObject(pidlRecycleBin, NULL, IID_PPV_ARG(IShellFolder, &pRecycleBin));
1036         CoTaskMemFree(pidlRecycleBin);
1037         pDesktop->Release();
1038         if (FAILED(hr))
1039             return hr;
1040         hr = pRecycleBin->EnumObjects(hwnd, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS | SHCONTF_INCLUDEHIDDEN, &penumFiles);
1041         if (FAILED(hr))
1042         {
1043             pRecycleBin->Release();
1044             return hr;
1045         }
1046 
1047         count = 0;
1048         if (hr != S_FALSE)
1049         {
1050             while (penumFiles->Next(1, &pidl, NULL) == S_OK)
1051             {
1052                 count++;
1053                 pRecycleBin->GetDisplayNameOf(pidl, SHGDN_NORMAL, &StrRet);
1054                 StrRetToBuf(&StrRet, pidl, szBuffer, _countof(szBuffer));
1055                 CoTaskMemFree(pidl);
1056             }
1057             penumFiles->Release();
1058         }
1059         pRecycleBin->Release();
1060 
1061         switch (count)
1062         {
1063             case 0:
1064                 /* no files, don't need confirmation */
1065                 break;
1066 
1067             case 1:
1068                 /* we have only one item inside the bin, so show a message box with its name */
1069                 if (ShellMessageBoxW(shell32_hInstance, hwnd, MAKEINTRESOURCEW(IDS_DELETEITEM_TEXT), MAKEINTRESOURCEW(IDS_EMPTY_BITBUCKET),
1070                                    MB_ICONEXCLAMATION | MB_YESNO | MB_DEFBUTTON2, szBuffer) == IDNO)
1071                 {
1072                     return S_OK;
1073                 }
1074                 break;
1075 
1076             default:
1077                 /* we have more than one item, so show a message box with the count of the items */
1078                 StringCbPrintfW(szBuffer, sizeof(szBuffer), L"%u", count);
1079                 if (ShellMessageBoxW(shell32_hInstance, hwnd, MAKEINTRESOURCEW(IDS_DELETEMULTIPLE_TEXT), MAKEINTRESOURCEW(IDS_EMPTY_BITBUCKET),
1080                                    MB_ICONEXCLAMATION | MB_YESNO | MB_DEFBUTTON2, szBuffer) == IDNO)
1081                 {
1082                     return S_OK;
1083                 }
1084                 break;
1085         }
1086     }
1087 
1088     if (dwFlags & SHERB_NOPROGRESSUI)
1089     {
1090         ret = EmptyRecycleBinW(pszRootPath);
1091     }
1092     else
1093     {
1094        /* FIXME
1095         * show a progress dialog
1096         */
1097         ret = EmptyRecycleBinW(pszRootPath);
1098     }
1099 
1100     if (!ret)
1101         return HRESULT_FROM_WIN32(GetLastError());
1102 
1103     if (!(dwFlags & SHERB_NOSOUND))
1104     {
1105         dwSize = sizeof(szPath);
1106         ret = RegGetValueW(HKEY_CURRENT_USER,
1107                            L"AppEvents\\Schemes\\Apps\\Explorer\\EmptyRecycleBin\\.Current",
1108                            NULL,
1109                            RRF_RT_REG_SZ,
1110                            &dwType,
1111                            (PVOID)szPath,
1112                            &dwSize);
1113         if (ret != ERROR_SUCCESS)
1114             return S_OK;
1115 
1116         if (dwType != REG_EXPAND_SZ) /* type dismatch */
1117             return S_OK;
1118 
1119         szPath[_countof(szPath)-1] = L'\0';
1120         PlaySoundW(szPath, NULL, SND_FILENAME);
1121     }
1122     return S_OK;
1123 }
1124 
1125 HRESULT WINAPI SHQueryRecycleBinA(LPCSTR pszRootPath, LPSHQUERYRBINFO pSHQueryRBInfo)
1126 {
1127     LPWSTR szRootPathW = NULL;
1128     int len;
1129     HRESULT hr;
1130 
1131     TRACE("%s, %p\n", debugstr_a(pszRootPath), pSHQueryRBInfo);
1132 
1133     if (pszRootPath)
1134     {
1135         len = MultiByteToWideChar(CP_ACP, 0, pszRootPath, -1, NULL, 0);
1136         if (len == 0)
1137             return HRESULT_FROM_WIN32(GetLastError());
1138         szRootPathW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
1139         if (!szRootPathW)
1140             return E_OUTOFMEMORY;
1141         if (MultiByteToWideChar(CP_ACP, 0, pszRootPath, -1, szRootPathW, len) == 0)
1142         {
1143             HeapFree(GetProcessHeap(), 0, szRootPathW);
1144             return HRESULT_FROM_WIN32(GetLastError());
1145         }
1146     }
1147 
1148     hr = SHQueryRecycleBinW(szRootPathW, pSHQueryRBInfo);
1149     HeapFree(GetProcessHeap(), 0, szRootPathW);
1150 
1151     return hr;
1152 }
1153 
1154 HRESULT WINAPI SHQueryRecycleBinW(LPCWSTR pszRootPath, LPSHQUERYRBINFO pSHQueryRBInfo)
1155 {
1156     FIXME("%s, %p - stub\n", debugstr_w(pszRootPath), pSHQueryRBInfo);
1157 
1158     if (!(pszRootPath) || (pszRootPath[0] == 0) ||
1159         !(pSHQueryRBInfo) || (pSHQueryRBInfo->cbSize < sizeof(SHQUERYRBINFO)))
1160     {
1161         return E_INVALIDARG;
1162     }
1163 
1164     pSHQueryRBInfo->i64Size = 0;
1165     pSHQueryRBInfo->i64NumItems = 0;
1166 
1167     return S_OK;
1168 }
1169