522 lines
12 KiB
C
522 lines
12 KiB
C
/*
|
|
* IEnumIDList
|
|
*
|
|
* Copyright 1998 Juergen Schmied <juergen.schmied@metronet.de>
|
|
*
|
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "wine/debug.h"
|
|
#include "winreg.h"
|
|
#include "undocshell.h"
|
|
#include "shlwapi.h"
|
|
#include "winerror.h"
|
|
#include "objbase.h"
|
|
|
|
#include "pidl.h"
|
|
#include "shlguid.h"
|
|
#include "shell32_main.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shell);
|
|
|
|
typedef struct tagENUMLIST
|
|
{
|
|
struct tagENUMLIST *pNext;
|
|
LPITEMIDLIST pidl;
|
|
|
|
} ENUMLIST, *LPENUMLIST;
|
|
|
|
typedef struct
|
|
{
|
|
ICOM_VFIELD(IEnumIDList);
|
|
DWORD ref;
|
|
LPENUMLIST mpFirst;
|
|
LPENUMLIST mpLast;
|
|
LPENUMLIST mpCurrent;
|
|
|
|
} IEnumIDListImpl;
|
|
|
|
static struct ICOM_VTABLE(IEnumIDList) eidlvt;
|
|
|
|
/**************************************************************************
|
|
* AddToEnumList()
|
|
*/
|
|
static BOOL AddToEnumList(
|
|
IEnumIDList * iface,
|
|
LPITEMIDLIST pidl)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
LPENUMLIST pNew;
|
|
|
|
TRACE("(%p)->(pidl=%p)\n",This,pidl);
|
|
pNew = (LPENUMLIST)SHAlloc(sizeof(ENUMLIST));
|
|
if(pNew)
|
|
{
|
|
/*set the next pointer */
|
|
pNew->pNext = NULL;
|
|
pNew->pidl = pidl;
|
|
|
|
/*is This the first item in the list? */
|
|
if(!This->mpFirst)
|
|
{
|
|
This->mpFirst = pNew;
|
|
This->mpCurrent = pNew;
|
|
}
|
|
|
|
if(This->mpLast)
|
|
{
|
|
/*add the new item to the end of the list */
|
|
This->mpLast->pNext = pNew;
|
|
}
|
|
|
|
/*update the last item pointer */
|
|
This->mpLast = pNew;
|
|
TRACE("-- (%p)->(first=%p, last=%p)\n",This,This->mpFirst,This->mpLast);
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* CreateFolderEnumList()
|
|
*/
|
|
static BOOL CreateFolderEnumList(
|
|
IEnumIDList * iface,
|
|
LPCSTR lpszPath,
|
|
DWORD dwFlags)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
LPITEMIDLIST pidl=NULL;
|
|
WIN32_FIND_DATAA stffile;
|
|
HANDLE hFile;
|
|
CHAR szPath[MAX_PATH];
|
|
|
|
TRACE("(%p)->(path=%s flags=0x%08lx) \n",This,debugstr_a(lpszPath),dwFlags);
|
|
|
|
if(!lpszPath || !lpszPath[0]) return FALSE;
|
|
|
|
strcpy(szPath, lpszPath);
|
|
PathAddBackslashA(szPath);
|
|
strcat(szPath,"*.*");
|
|
|
|
/*enumerate the folders*/
|
|
if(dwFlags & SHCONTF_FOLDERS)
|
|
{
|
|
TRACE("-- (%p)-> enumerate SHCONTF_FOLDERS of %s\n",This,debugstr_a(szPath));
|
|
hFile = FindFirstFileA(szPath,&stffile);
|
|
if ( hFile != INVALID_HANDLE_VALUE )
|
|
{
|
|
do
|
|
{
|
|
if ( !(dwFlags & SHCONTF_INCLUDEHIDDEN) && (stffile.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) ) continue;
|
|
if ( (stffile.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) && strcmp (stffile.cFileName, ".") && strcmp (stffile.cFileName, ".."))
|
|
{
|
|
pidl = _ILCreateFolder (&stffile);
|
|
if(pidl && AddToEnumList((IEnumIDList*)This, pidl))
|
|
{
|
|
continue;
|
|
}
|
|
return FALSE;
|
|
}
|
|
} while( FindNextFileA(hFile,&stffile));
|
|
FindClose (hFile);
|
|
}
|
|
}
|
|
|
|
/*enumerate the non-folder items (values) */
|
|
if(dwFlags & SHCONTF_NONFOLDERS)
|
|
{
|
|
TRACE("-- (%p)-> enumerate SHCONTF_NONFOLDERS of %s\n",This,debugstr_a(szPath));
|
|
hFile = FindFirstFileA(szPath,&stffile);
|
|
if ( hFile != INVALID_HANDLE_VALUE )
|
|
{
|
|
do
|
|
{
|
|
if ( !(dwFlags & SHCONTF_INCLUDEHIDDEN) && (stffile.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN) ) continue;
|
|
if (! (stffile.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) )
|
|
{
|
|
pidl = _ILCreateValue(&stffile);
|
|
if(pidl && AddToEnumList((IEnumIDList*)This, pidl))
|
|
{
|
|
continue;
|
|
}
|
|
return FALSE;
|
|
}
|
|
} while( FindNextFileA(hFile,&stffile));
|
|
FindClose (hFile);
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* CreateDesktopEnumList()
|
|
*/
|
|
static BOOL CreateDesktopEnumList(
|
|
IEnumIDList * iface,
|
|
DWORD dwFlags)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
LPITEMIDLIST pidl=NULL;
|
|
HKEY hkey;
|
|
char szPath[MAX_PATH];
|
|
|
|
TRACE("(%p)->(flags=0x%08lx) \n",This,dwFlags);
|
|
|
|
/*enumerate the root folders */
|
|
if(dwFlags & SHCONTF_FOLDERS)
|
|
{
|
|
/*create the pidl for This item */
|
|
pidl = _ILCreateMyComputer();
|
|
if(pidl)
|
|
{
|
|
if(!AddToEnumList((IEnumIDList*)This, pidl))
|
|
return FALSE;
|
|
}
|
|
|
|
if (! RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\explorer\\desktop\\NameSpace", 0, KEY_READ, &hkey))
|
|
{
|
|
char iid[50];
|
|
int i=0;
|
|
|
|
while (1)
|
|
{
|
|
DWORD size = sizeof (iid);
|
|
|
|
if (ERROR_SUCCESS!=RegEnumKeyExA(hkey, i, iid, &size, 0, NULL, NULL, NULL))
|
|
break;
|
|
|
|
pidl = _ILCreateSpecial(iid);
|
|
|
|
if(pidl)
|
|
AddToEnumList((IEnumIDList*)This, pidl);
|
|
|
|
i++;
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
|
|
/*enumerate the elements in %windir%\desktop */
|
|
SHGetSpecialFolderPathA(0, szPath, CSIDL_DESKTOPDIRECTORY, FALSE);
|
|
CreateFolderEnumList( (IEnumIDList*)This, szPath, dwFlags);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* CreateMyCompEnumList()
|
|
*/
|
|
static BOOL CreateMyCompEnumList(
|
|
IEnumIDList * iface,
|
|
DWORD dwFlags)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
LPITEMIDLIST pidl=NULL;
|
|
DWORD dwDrivemap;
|
|
CHAR szDriveName[4];
|
|
HKEY hkey;
|
|
|
|
TRACE("(%p)->(flags=0x%08lx) \n",This,dwFlags);
|
|
|
|
/*enumerate the folders*/
|
|
if(dwFlags & SHCONTF_FOLDERS)
|
|
{
|
|
dwDrivemap = GetLogicalDrives();
|
|
strcpy (szDriveName,"A:\\");
|
|
while (szDriveName[0]<='Z')
|
|
{
|
|
if(dwDrivemap & 0x00000001L)
|
|
{
|
|
pidl = _ILCreateDrive(szDriveName);
|
|
if(pidl)
|
|
{
|
|
if(!AddToEnumList((IEnumIDList*)This, pidl))
|
|
return FALSE;
|
|
}
|
|
}
|
|
szDriveName[0]++;
|
|
dwDrivemap = dwDrivemap >> 1;
|
|
}
|
|
|
|
TRACE("-- (%p)-> enumerate (mycomputer shell extensions)\n",This);
|
|
if (! RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\explorer\\mycomputer\\NameSpace", 0, KEY_READ, &hkey))
|
|
{
|
|
char iid[50];
|
|
int i=0;
|
|
|
|
while (1)
|
|
{
|
|
DWORD size = sizeof (iid);
|
|
|
|
if (ERROR_SUCCESS!=RegEnumKeyExA(hkey, i, iid, &size, 0, NULL, NULL, NULL))
|
|
break;
|
|
|
|
pidl = _ILCreateSpecial(iid);
|
|
|
|
if(pidl)
|
|
AddToEnumList((IEnumIDList*)This, pidl);
|
|
|
|
i++;
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* DeleteList()
|
|
*/
|
|
static BOOL DeleteList(
|
|
IEnumIDList * iface)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
LPENUMLIST pDelete;
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
while(This->mpFirst)
|
|
{ pDelete = This->mpFirst;
|
|
This->mpFirst = pDelete->pNext;
|
|
SHFree(pDelete->pidl);
|
|
SHFree(pDelete);
|
|
}
|
|
This->mpFirst = This->mpLast = This->mpCurrent = NULL;
|
|
return TRUE;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* IEnumIDList_Folder_Constructor
|
|
*
|
|
*/
|
|
|
|
IEnumIDList * IEnumIDList_Constructor(
|
|
LPCSTR lpszPath,
|
|
DWORD dwFlags,
|
|
DWORD dwKind)
|
|
{
|
|
IEnumIDListImpl* lpeidl;
|
|
BOOL ret = FALSE;
|
|
|
|
TRACE("()->(%s flags=0x%08lx kind=0x%08lx)\n",debugstr_a(lpszPath),dwFlags, dwKind);
|
|
|
|
lpeidl = (IEnumIDListImpl*)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IEnumIDListImpl));
|
|
|
|
if (lpeidl)
|
|
{
|
|
lpeidl->ref = 1;
|
|
lpeidl->lpVtbl = &eidlvt;
|
|
|
|
switch (dwKind)
|
|
{
|
|
case EIDL_DESK:
|
|
ret = CreateDesktopEnumList((IEnumIDList*)lpeidl, dwFlags);
|
|
break;
|
|
|
|
case EIDL_MYCOMP:
|
|
ret = CreateMyCompEnumList((IEnumIDList*)lpeidl, dwFlags);
|
|
break;
|
|
|
|
case EIDL_FILE:
|
|
ret = CreateFolderEnumList((IEnumIDList*)lpeidl, lpszPath, dwFlags);
|
|
break;
|
|
}
|
|
|
|
if(!ret) {
|
|
HeapFree(GetProcessHeap(),0,lpeidl);
|
|
lpeidl = NULL;
|
|
}
|
|
}
|
|
|
|
TRACE("-- (%p)->()\n",lpeidl);
|
|
|
|
return (IEnumIDList*)lpeidl;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* EnumIDList_QueryInterface
|
|
*/
|
|
static HRESULT WINAPI IEnumIDList_fnQueryInterface(
|
|
IEnumIDList * iface,
|
|
REFIID riid,
|
|
LPVOID *ppvObj)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
TRACE("(%p)->(\n\tIID:\t%s,%p)\n",This,debugstr_guid(riid),ppvObj);
|
|
|
|
*ppvObj = NULL;
|
|
|
|
if(IsEqualIID(riid, &IID_IUnknown)) /*IUnknown*/
|
|
{ *ppvObj = This;
|
|
}
|
|
else if(IsEqualIID(riid, &IID_IEnumIDList)) /*IEnumIDList*/
|
|
{ *ppvObj = (IEnumIDList*)This;
|
|
}
|
|
|
|
if(*ppvObj)
|
|
{ IEnumIDList_AddRef((IEnumIDList*)*ppvObj);
|
|
TRACE("-- Interface: (%p)->(%p)\n",ppvObj,*ppvObj);
|
|
return S_OK;
|
|
}
|
|
|
|
TRACE("-- Interface: E_NOINTERFACE\n");
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* IEnumIDList_fnAddRef
|
|
*/
|
|
static ULONG WINAPI IEnumIDList_fnAddRef(
|
|
IEnumIDList * iface)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
TRACE("(%p)->(%lu)\n",This,This->ref);
|
|
return ++(This->ref);
|
|
}
|
|
/******************************************************************************
|
|
* IEnumIDList_fnRelease
|
|
*/
|
|
static ULONG WINAPI IEnumIDList_fnRelease(
|
|
IEnumIDList * iface)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
TRACE("(%p)->(%lu)\n",This,This->ref);
|
|
|
|
if (!--(This->ref)) {
|
|
TRACE(" destroying IEnumIDList(%p)\n",This);
|
|
DeleteList((IEnumIDList*)This);
|
|
HeapFree(GetProcessHeap(),0,This);
|
|
return 0;
|
|
}
|
|
return This->ref;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* IEnumIDList_fnNext
|
|
*/
|
|
|
|
static HRESULT WINAPI IEnumIDList_fnNext(
|
|
IEnumIDList * iface,
|
|
ULONG celt,
|
|
LPITEMIDLIST * rgelt,
|
|
ULONG *pceltFetched)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
ULONG i;
|
|
HRESULT hr = S_OK;
|
|
LPITEMIDLIST temp;
|
|
|
|
TRACE("(%p)->(%ld,%p, %p)\n",This,celt,rgelt,pceltFetched);
|
|
|
|
/* It is valid to leave pceltFetched NULL when celt is 1. Some of explorer's
|
|
* subsystems actually use it (and so may a third party browser)
|
|
*/
|
|
if(pceltFetched)
|
|
*pceltFetched = 0;
|
|
|
|
*rgelt=0;
|
|
|
|
if(celt > 1 && !pceltFetched)
|
|
{ return E_INVALIDARG;
|
|
}
|
|
|
|
for(i = 0; i < celt; i++)
|
|
{ if(!(This->mpCurrent))
|
|
{ hr = S_FALSE;
|
|
break;
|
|
}
|
|
temp = ILClone(This->mpCurrent->pidl);
|
|
rgelt[i] = temp;
|
|
This->mpCurrent = This->mpCurrent->pNext;
|
|
}
|
|
if(pceltFetched)
|
|
{ *pceltFetched = i;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* IEnumIDList_fnSkip
|
|
*/
|
|
static HRESULT WINAPI IEnumIDList_fnSkip(
|
|
IEnumIDList * iface,ULONG celt)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
DWORD dwIndex;
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("(%p)->(%lu)\n",This,celt);
|
|
|
|
for(dwIndex = 0; dwIndex < celt; dwIndex++)
|
|
{ if(!This->mpCurrent)
|
|
{ hr = S_FALSE;
|
|
break;
|
|
}
|
|
This->mpCurrent = This->mpCurrent->pNext;
|
|
}
|
|
return hr;
|
|
}
|
|
/**************************************************************************
|
|
* IEnumIDList_fnReset
|
|
*/
|
|
static HRESULT WINAPI IEnumIDList_fnReset(
|
|
IEnumIDList * iface)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
TRACE("(%p)\n",This);
|
|
This->mpCurrent = This->mpFirst;
|
|
return S_OK;
|
|
}
|
|
/**************************************************************************
|
|
* IEnumIDList_fnClone
|
|
*/
|
|
static HRESULT WINAPI IEnumIDList_fnClone(
|
|
IEnumIDList * iface,LPENUMIDLIST * ppenum)
|
|
{
|
|
ICOM_THIS(IEnumIDListImpl,iface);
|
|
|
|
TRACE("(%p)->() to (%p)->() E_NOTIMPL\n",This,ppenum);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/**************************************************************************
|
|
* IEnumIDList_fnVTable
|
|
*/
|
|
static ICOM_VTABLE (IEnumIDList) eidlvt =
|
|
{
|
|
ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE
|
|
IEnumIDList_fnQueryInterface,
|
|
IEnumIDList_fnAddRef,
|
|
IEnumIDList_fnRelease,
|
|
IEnumIDList_fnNext,
|
|
IEnumIDList_fnSkip,
|
|
IEnumIDList_fnReset,
|
|
IEnumIDList_fnClone,
|
|
};
|