Moved all the documented shlwapi functions out of undocshell.h into
[wine/hacks.git] / dlls / shell32 / enumidlist.c
blob7edab9c9e560708ee72f0203d366d939243d5c8e
1 /*
2 * IEnumIDList
4 * Copyright 1998 Juergen Schmied <juergen.schmied@metronet.de>
5 */
7 #include <stdlib.h>
8 #include <string.h>
9 #include "debugtools.h"
10 #include "winreg.h"
11 #include "wine/undocshell.h"
12 #include "shlwapi.h"
13 #include "winerror.h"
14 #include "wine/obj_base.h"
15 #include "wine/obj_enumidlist.h"
17 #include "pidl.h"
18 #include "shlguid.h"
19 #include "shell32_main.h"
21 DEFAULT_DEBUG_CHANNEL(shell);
23 typedef struct tagENUMLIST
25 struct tagENUMLIST *pNext;
26 LPITEMIDLIST pidl;
28 } ENUMLIST, *LPENUMLIST;
30 typedef struct
32 ICOM_VFIELD(IEnumIDList);
33 DWORD ref;
34 LPENUMLIST mpFirst;
35 LPENUMLIST mpLast;
36 LPENUMLIST mpCurrent;
38 } IEnumIDListImpl;
40 static struct ICOM_VTABLE(IEnumIDList) eidlvt;
42 /**************************************************************************
43 * AddToEnumList()
45 static BOOL AddToEnumList(
46 IEnumIDList * iface,
47 LPITEMIDLIST pidl)
49 ICOM_THIS(IEnumIDListImpl,iface);
51 LPENUMLIST pNew;
53 TRACE("(%p)->(pidl=%p)\n",This,pidl);
54 pNew = (LPENUMLIST)SHAlloc(sizeof(ENUMLIST));
55 if(pNew)
57 /*set the next pointer */
58 pNew->pNext = NULL;
59 pNew->pidl = pidl;
61 /*is This the first item in the list? */
62 if(!This->mpFirst)
64 This->mpFirst = pNew;
65 This->mpCurrent = pNew;
68 if(This->mpLast)
70 /*add the new item to the end of the list */
71 This->mpLast->pNext = pNew;
74 /*update the last item pointer */
75 This->mpLast = pNew;
76 TRACE("-- (%p)->(first=%p, last=%p)\n",This,This->mpFirst,This->mpLast);
77 return TRUE;
79 return FALSE;
82 /**************************************************************************
83 * CreateFolderEnumList()
85 static BOOL CreateFolderEnumList(
86 IEnumIDList * iface,
87 LPCSTR lpszPath,
88 DWORD dwFlags)
90 ICOM_THIS(IEnumIDListImpl,iface);
92 LPITEMIDLIST pidl=NULL;
93 WIN32_FIND_DATAA stffile;
94 HANDLE hFile;
95 CHAR szPath[MAX_PATH];
97 TRACE("(%p)->(path=%s flags=0x%08lx) \n",This,debugstr_a(lpszPath),dwFlags);
99 if(!lpszPath || !lpszPath[0]) return FALSE;
101 strcpy(szPath, lpszPath);
102 PathAddBackslashA(szPath);
103 strcat(szPath,"*.*");
105 /*enumerate the folders*/
106 if(dwFlags & SHCONTF_FOLDERS)
108 TRACE("-- (%p)-> enumerate SHCONTF_FOLDERS of %s\n",This,debugstr_a(szPath));
109 hFile = FindFirstFileA(szPath,&stffile);
110 if ( hFile != INVALID_HANDLE_VALUE )
114 if ( (stffile.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) && strcmp (stffile.cFileName, ".") && strcmp (stffile.cFileName, ".."))
116 pidl = _ILCreateFolder (&stffile);
117 if(pidl && AddToEnumList((IEnumIDList*)This, pidl))
119 continue;
121 return FALSE;
123 } while( FindNextFileA(hFile,&stffile));
124 FindClose (hFile);
128 /*enumerate the non-folder items (values) */
129 if(dwFlags & SHCONTF_NONFOLDERS)
131 TRACE("-- (%p)-> enumerate SHCONTF_NONFOLDERS of %s\n",This,debugstr_a(szPath));
132 hFile = FindFirstFileA(szPath,&stffile);
133 if ( hFile != INVALID_HANDLE_VALUE )
137 if (! (stffile.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) )
139 pidl = _ILCreateValue(&stffile);
140 if(pidl && AddToEnumList((IEnumIDList*)This, pidl))
142 continue;
144 return FALSE;
146 } while( FindNextFileA(hFile,&stffile));
147 FindClose (hFile);
150 return TRUE;
153 /**************************************************************************
154 * CreateDesktopEnumList()
156 static BOOL CreateDesktopEnumList(
157 IEnumIDList * iface,
158 DWORD dwFlags)
160 ICOM_THIS(IEnumIDListImpl,iface);
162 LPITEMIDLIST pidl=NULL;
163 HKEY hkey;
164 char szPath[MAX_PATH];
166 TRACE("(%p)->(flags=0x%08lx) \n",This,dwFlags);
168 /*enumerate the root folders */
169 if(dwFlags & SHCONTF_FOLDERS)
171 /*create the pidl for This item */
172 pidl = _ILCreateMyComputer();
173 if(pidl)
175 if(!AddToEnumList((IEnumIDList*)This, pidl))
176 return FALSE;
179 if (! RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\explorer\\desktop\\NameSpace", 0, KEY_READ, &hkey))
181 char iid[50];
182 int i=0;
184 while (1)
186 DWORD size = sizeof (iid);
188 if (ERROR_SUCCESS!=RegEnumKeyExA(hkey, i, iid, &size, 0, NULL, NULL, NULL))
189 break;
191 pidl = _ILCreateSpecial(iid);
193 if(pidl)
194 AddToEnumList((IEnumIDList*)This, pidl);
196 i++;
198 RegCloseKey(hkey);
202 /*enumerate the elements in %windir%\desktop */
203 SHGetSpecialFolderPathA(0, szPath, CSIDL_DESKTOPDIRECTORY, FALSE);
204 CreateFolderEnumList( (IEnumIDList*)This, szPath, dwFlags);
206 return TRUE;
209 /**************************************************************************
210 * CreateMyCompEnumList()
212 static BOOL CreateMyCompEnumList(
213 IEnumIDList * iface,
214 DWORD dwFlags)
216 ICOM_THIS(IEnumIDListImpl,iface);
218 LPITEMIDLIST pidl=NULL;
219 DWORD dwDrivemap;
220 CHAR szDriveName[4];
221 HKEY hkey;
223 TRACE("(%p)->(flags=0x%08lx) \n",This,dwFlags);
225 /*enumerate the folders*/
226 if(dwFlags & SHCONTF_FOLDERS)
228 dwDrivemap = GetLogicalDrives();
229 strcpy (szDriveName,"A:\\");
230 while (szDriveName[0]<='Z')
232 if(dwDrivemap & 0x00000001L)
234 pidl = _ILCreateDrive(szDriveName);
235 if(pidl)
237 if(!AddToEnumList((IEnumIDList*)This, pidl))
238 return FALSE;
241 szDriveName[0]++;
242 dwDrivemap = dwDrivemap >> 1;
245 TRACE("-- (%p)-> enumerate (mycomputer shell extensions)\n",This);
246 if (! RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\explorer\\mycomputer\\NameSpace", 0, KEY_READ, &hkey))
248 char iid[50];
249 int i=0;
251 while (1)
253 DWORD size = sizeof (iid);
255 if (ERROR_SUCCESS!=RegEnumKeyExA(hkey, i, iid, &size, 0, NULL, NULL, NULL))
256 break;
258 pidl = _ILCreateSpecial(iid);
260 if(pidl)
261 AddToEnumList((IEnumIDList*)This, pidl);
263 i++;
265 RegCloseKey(hkey);
268 return TRUE;
271 /**************************************************************************
272 * DeleteList()
274 static BOOL DeleteList(
275 IEnumIDList * iface)
277 ICOM_THIS(IEnumIDListImpl,iface);
279 LPENUMLIST pDelete;
281 TRACE("(%p)->()\n",This);
283 while(This->mpFirst)
284 { pDelete = This->mpFirst;
285 This->mpFirst = pDelete->pNext;
286 SHFree(pDelete->pidl);
287 SHFree(pDelete);
289 This->mpFirst = This->mpLast = This->mpCurrent = NULL;
290 return TRUE;
293 /**************************************************************************
294 * IEnumIDList_Folder_Constructor
298 IEnumIDList * IEnumIDList_Constructor(
299 LPCSTR lpszPath,
300 DWORD dwFlags,
301 DWORD dwKind)
303 IEnumIDListImpl* lpeidl;
304 BOOL ret = FALSE;
306 lpeidl = (IEnumIDListImpl*)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IEnumIDListImpl));
308 TRACE("(%p)->(%s flags=0x%08lx kind=0x%08lx)\n",lpeidl,debugstr_a(lpszPath),dwFlags, dwKind);
310 if (lpeidl)
312 lpeidl->ref = 1;
313 ICOM_VTBL(lpeidl) = &eidlvt;
315 switch (dwKind)
317 case EIDL_DESK:
318 ret = CreateDesktopEnumList((IEnumIDList*)lpeidl, dwFlags);
319 break;
321 case EIDL_MYCOMP:
322 ret = CreateMyCompEnumList((IEnumIDList*)lpeidl, dwFlags);
323 break;
325 case EIDL_FILE:
326 ret = CreateFolderEnumList((IEnumIDList*)lpeidl, lpszPath, dwFlags);
327 break;
330 if(ret)
332 shell32_ObjCount++;
334 else
336 if (lpeidl)
338 HeapFree(GetProcessHeap(),0,lpeidl);
343 TRACE("-- (%p)->()\n",lpeidl);
345 return (IEnumIDList*)lpeidl;
348 /**************************************************************************
349 * EnumIDList_QueryInterface
351 static HRESULT WINAPI IEnumIDList_fnQueryInterface(
352 IEnumIDList * iface,
353 REFIID riid,
354 LPVOID *ppvObj)
356 ICOM_THIS(IEnumIDListImpl,iface);
358 TRACE("(%p)->(\n\tIID:\t%s,%p)\n",This,debugstr_guid(riid),ppvObj);
360 *ppvObj = NULL;
362 if(IsEqualIID(riid, &IID_IUnknown)) /*IUnknown*/
363 { *ppvObj = This;
365 else if(IsEqualIID(riid, &IID_IEnumIDList)) /*IEnumIDList*/
366 { *ppvObj = (IEnumIDList*)This;
369 if(*ppvObj)
370 { IEnumIDList_AddRef((IEnumIDList*)*ppvObj);
371 TRACE("-- Interface: (%p)->(%p)\n",ppvObj,*ppvObj);
372 return S_OK;
375 TRACE("-- Interface: E_NOINTERFACE\n");
376 return E_NOINTERFACE;
379 /******************************************************************************
380 * IEnumIDList_fnAddRef
382 static ULONG WINAPI IEnumIDList_fnAddRef(
383 IEnumIDList * iface)
385 ICOM_THIS(IEnumIDListImpl,iface);
387 TRACE("(%p)->(%lu)\n",This,This->ref);
389 shell32_ObjCount++;
390 return ++(This->ref);
392 /******************************************************************************
393 * IEnumIDList_fnRelease
395 static ULONG WINAPI IEnumIDList_fnRelease(
396 IEnumIDList * iface)
398 ICOM_THIS(IEnumIDListImpl,iface);
400 TRACE("(%p)->(%lu)\n",This,This->ref);
402 shell32_ObjCount--;
404 if (!--(This->ref))
405 { TRACE(" destroying IEnumIDList(%p)\n",This);
406 DeleteList((IEnumIDList*)This);
407 HeapFree(GetProcessHeap(),0,This);
408 return 0;
410 return This->ref;
413 /**************************************************************************
414 * IEnumIDList_fnNext
417 static HRESULT WINAPI IEnumIDList_fnNext(
418 IEnumIDList * iface,
419 ULONG celt,
420 LPITEMIDLIST * rgelt,
421 ULONG *pceltFetched)
423 ICOM_THIS(IEnumIDListImpl,iface);
425 ULONG i;
426 HRESULT hr = S_OK;
427 LPITEMIDLIST temp;
429 TRACE("(%p)->(%ld,%p, %p)\n",This,celt,rgelt,pceltFetched);
431 /* It is valid to leave pceltFetched NULL when celt is 1. Some of explorer's
432 * subsystems actually use it (and so may a third party browser)
434 if(pceltFetched)
435 *pceltFetched = 0;
437 *rgelt=0;
439 if(celt > 1 && !pceltFetched)
440 { return E_INVALIDARG;
443 for(i = 0; i < celt; i++)
444 { if(!(This->mpCurrent))
445 { hr = S_FALSE;
446 break;
448 temp = ILClone(This->mpCurrent->pidl);
449 rgelt[i] = temp;
450 This->mpCurrent = This->mpCurrent->pNext;
452 if(pceltFetched)
453 { *pceltFetched = i;
456 return hr;
459 /**************************************************************************
460 * IEnumIDList_fnSkip
462 static HRESULT WINAPI IEnumIDList_fnSkip(
463 IEnumIDList * iface,ULONG celt)
465 ICOM_THIS(IEnumIDListImpl,iface);
467 DWORD dwIndex;
468 HRESULT hr = S_OK;
470 TRACE("(%p)->(%lu)\n",This,celt);
472 for(dwIndex = 0; dwIndex < celt; dwIndex++)
473 { if(!This->mpCurrent)
474 { hr = S_FALSE;
475 break;
477 This->mpCurrent = This->mpCurrent->pNext;
479 return hr;
481 /**************************************************************************
482 * IEnumIDList_fnReset
484 static HRESULT WINAPI IEnumIDList_fnReset(
485 IEnumIDList * iface)
487 ICOM_THIS(IEnumIDListImpl,iface);
489 TRACE("(%p)\n",This);
490 This->mpCurrent = This->mpFirst;
491 return S_OK;
493 /**************************************************************************
494 * IEnumIDList_fnClone
496 static HRESULT WINAPI IEnumIDList_fnClone(
497 IEnumIDList * iface,LPENUMIDLIST * ppenum)
499 ICOM_THIS(IEnumIDListImpl,iface);
501 TRACE("(%p)->() to (%p)->() E_NOTIMPL\n",This,ppenum);
502 return E_NOTIMPL;
505 /**************************************************************************
506 * IEnumIDList_fnVTable
508 static ICOM_VTABLE (IEnumIDList) eidlvt =
510 ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE
511 IEnumIDList_fnQueryInterface,
512 IEnumIDList_fnAddRef,
513 IEnumIDList_fnRelease,
514 IEnumIDList_fnNext,
515 IEnumIDList_fnSkip,
516 IEnumIDList_fnReset,
517 IEnumIDList_fnClone,