860 lines
22 KiB
C++
860 lines
22 KiB
C++
|
#include "precomp.h"
|
||
|
#include "shconv.h"
|
||
|
#pragma hdrstop
|
||
|
|
||
|
|
||
|
class CEnumFolderItems;
|
||
|
|
||
|
class CFolderItems : public FolderItems3,
|
||
|
public IPersistFolder,
|
||
|
public CObjectSafety,
|
||
|
public CObjectWithSite,
|
||
|
protected CImpIDispatch
|
||
|
{
|
||
|
friend CEnumFolderItems;
|
||
|
|
||
|
public:
|
||
|
CFolderItems(CFolder *psdf, BOOL fSelected);
|
||
|
|
||
|
// IUnknown
|
||
|
STDMETHOD(QueryInterface)(REFIID riid, void ** ppv);
|
||
|
STDMETHOD_(ULONG, AddRef)(void);
|
||
|
STDMETHOD_(ULONG, Release)(void);
|
||
|
|
||
|
// IDispatch
|
||
|
STDMETHODIMP GetTypeInfoCount(UINT *pctinfo)
|
||
|
{ return CImpIDispatch::GetTypeInfoCount(pctinfo); }
|
||
|
STDMETHODIMP GetTypeInfo(UINT itinfo, LCID lcid, ITypeInfo **pptinfo)
|
||
|
{ return CImpIDispatch::GetTypeInfo(itinfo, lcid, pptinfo); }
|
||
|
STDMETHODIMP GetIDsOfNames(REFIID riid, OLECHAR ** rgszNames, UINT cNames, LCID lcid, DISPID *rgdispid)
|
||
|
{ return CImpIDispatch::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid); }
|
||
|
STDMETHODIMP Invoke(DISPID dispidMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS * pdispparams, VARIANT * pvarResult, EXCEPINFO * pexcepinfo, UINT * puArgErr)
|
||
|
{ return CImpIDispatch::Invoke(dispidMember, riid, lcid, wFlags, pdispparams, pvarResult, pexcepinfo, puArgErr); }
|
||
|
|
||
|
// FolderItems
|
||
|
STDMETHODIMP get_Application(IDispatch **ppid);
|
||
|
STDMETHODIMP get_Parent (IDispatch **ppid);
|
||
|
STDMETHODIMP get_Count(long *pCount);
|
||
|
STDMETHODIMP Item(VARIANT, FolderItem**);
|
||
|
STDMETHODIMP _NewEnum(IUnknown **);
|
||
|
|
||
|
// FolderItems2
|
||
|
STDMETHODIMP InvokeVerbEx(VARIANT vVerb, VARIANT vArgs);
|
||
|
|
||
|
// FolderItems3
|
||
|
STDMETHODIMP Filter(long grfFlags, BSTR bstrFilter);
|
||
|
STDMETHODIMP get_Verbs(FolderItemVerbs **ppfic);
|
||
|
|
||
|
// IPersist
|
||
|
STDMETHODIMP GetClassID(CLSID *pClassID);
|
||
|
|
||
|
// IPersistFolder
|
||
|
STDMETHODIMP Initialize(LPCITEMIDLIST pidl);
|
||
|
|
||
|
protected:
|
||
|
HDPA _GetHDPA();
|
||
|
UINT _GetHDPACount();
|
||
|
BOOL_PTR _CopyItem(UINT iItem, LPCITEMIDLIST pidl);
|
||
|
|
||
|
LONG _cRef;
|
||
|
CFolder *_psdf;
|
||
|
HDPA _hdpa;
|
||
|
BOOL _fSelected;
|
||
|
BOOL _fGotAllItems;
|
||
|
IEnumIDList *_penum;
|
||
|
UINT _cNumEnumed;
|
||
|
LONG _grfFlags;
|
||
|
LPTSTR _pszFileSpec;
|
||
|
|
||
|
HRESULT _SecurityCheck();
|
||
|
void _ResetIDListArray();
|
||
|
virtual ~CFolderItems(void);
|
||
|
BOOL _IncludeItem(IShellFolder *psf, LPCITEMIDLIST pidl);
|
||
|
virtual HRESULT _EnsureItem(UINT iItemNeeded, LPCITEMIDLIST *ppidl);
|
||
|
STDMETHODIMP _GetUIObjectOf(REFIID riid, void ** ppv);
|
||
|
};
|
||
|
|
||
|
// CFolderItemsF(rom)D(ifferent)F(olders)
|
||
|
class CFolderItemsFDF : public CFolderItems,
|
||
|
public IInsertItem
|
||
|
{
|
||
|
public:
|
||
|
// TODO: add callback pointer to constructor
|
||
|
CFolderItemsFDF(CFolder *psdf);
|
||
|
|
||
|
// IUnknown override
|
||
|
STDMETHOD(QueryInterface)(REFIID riid, void ** ppv);
|
||
|
STDMETHOD_(ULONG, AddRef)(void) {return CFolderItems::AddRef();};
|
||
|
STDMETHOD_(ULONG, Release)(void) {return CFolderItems::Release();};
|
||
|
|
||
|
// IInsertItem
|
||
|
STDMETHOD(InsertItem)(LPCITEMIDLIST pidl);
|
||
|
protected:
|
||
|
virtual HRESULT _EnsureItem(UINT iItemNeeded, LPCITEMIDLIST *ppidl);
|
||
|
};
|
||
|
|
||
|
//Enumerator of whatever is held in the collection
|
||
|
class CEnumFolderItems : public IEnumVARIANT
|
||
|
{
|
||
|
public:
|
||
|
CEnumFolderItems(CFolderItems *pfdritms);
|
||
|
|
||
|
// IUnknown
|
||
|
STDMETHODIMP QueryInterface(REFIID, void **);
|
||
|
STDMETHODIMP_(ULONG) AddRef(void);
|
||
|
STDMETHODIMP_(ULONG) Release(void);
|
||
|
|
||
|
// IEnumFORMATETC
|
||
|
STDMETHODIMP Next(ULONG, VARIANT *, ULONG *);
|
||
|
STDMETHODIMP Skip(ULONG);
|
||
|
STDMETHODIMP Reset(void);
|
||
|
STDMETHODIMP Clone(IEnumVARIANT **);
|
||
|
|
||
|
private:
|
||
|
~CEnumFolderItems();
|
||
|
|
||
|
LONG _cRef;
|
||
|
CFolderItems *_pfdritms;
|
||
|
UINT _iCur;
|
||
|
};
|
||
|
|
||
|
HRESULT CFolderItems_Create(CFolder *psdf, BOOL fSelected, FolderItems **ppitems)
|
||
|
{
|
||
|
*ppitems = NULL;
|
||
|
HRESULT hr = E_OUTOFMEMORY;
|
||
|
CFolderItems* psdfi = new CFolderItems(psdf, fSelected);
|
||
|
if (psdfi)
|
||
|
{
|
||
|
hr = psdfi->QueryInterface(IID_PPV_ARG(FolderItems, ppitems));
|
||
|
psdfi->Release();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
CFolderItems::CFolderItems(CFolder *psdf, BOOL fSelected) :
|
||
|
_cRef(1),
|
||
|
_psdf(psdf),
|
||
|
_fSelected(fSelected),
|
||
|
_grfFlags(SHCONTF_FOLDERS | SHCONTF_NONFOLDERS),
|
||
|
CImpIDispatch(SDSPATCH_TYPELIB, IID_FolderItems3)
|
||
|
{
|
||
|
DllAddRef();
|
||
|
|
||
|
if (_psdf)
|
||
|
_psdf->AddRef();
|
||
|
|
||
|
ASSERT(_hdpa == NULL);
|
||
|
ASSERT(_pszFileSpec == NULL);
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItems::GetClassID(CLSID *pClassID)
|
||
|
{
|
||
|
*pClassID = CLSID_FolderItemsFDF;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItems::Initialize(LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
ASSERT(_psdf == NULL);
|
||
|
return CFolder_Create2(NULL, pidl, NULL, &_psdf);
|
||
|
}
|
||
|
|
||
|
int FreePidlCallBack(void *pv, void *)
|
||
|
{
|
||
|
ILFree((LPITEMIDLIST)pv);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItems::_SecurityCheck()
|
||
|
{
|
||
|
if (!_dwSafetyOptions)
|
||
|
return S_OK;
|
||
|
|
||
|
return _psdf->_SecurityCheck();
|
||
|
}
|
||
|
|
||
|
void CFolderItems::_ResetIDListArray(void)
|
||
|
{
|
||
|
// destory the DPA, and lets reset the counters and pointers
|
||
|
|
||
|
if (_hdpa)
|
||
|
{
|
||
|
DPA_DestroyCallback(_hdpa, FreePidlCallBack, 0);
|
||
|
_hdpa = NULL;
|
||
|
}
|
||
|
|
||
|
_fGotAllItems = FALSE;
|
||
|
_cNumEnumed = 0;
|
||
|
|
||
|
ATOMICRELEASE(_penum); // may be NULL
|
||
|
}
|
||
|
|
||
|
CFolderItems::~CFolderItems(void)
|
||
|
{
|
||
|
_ResetIDListArray();
|
||
|
Str_SetPtr(&_pszFileSpec, NULL);
|
||
|
|
||
|
if (_psdf)
|
||
|
_psdf->Release();
|
||
|
|
||
|
DllRelease();
|
||
|
}
|
||
|
|
||
|
HDPA CFolderItems::_GetHDPA()
|
||
|
{
|
||
|
if (NULL == _hdpa)
|
||
|
_hdpa = ::DPA_Create(0);
|
||
|
return _hdpa;
|
||
|
}
|
||
|
|
||
|
UINT CFolderItems::_GetHDPACount()
|
||
|
{
|
||
|
UINT count = 0;
|
||
|
HDPA hdpa = _GetHDPA();
|
||
|
if (hdpa)
|
||
|
count = DPA_GetPtrCount(hdpa);
|
||
|
return count;
|
||
|
}
|
||
|
|
||
|
BOOL_PTR CFolderItems::_CopyItem(UINT iItem, LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
LPITEMIDLIST pidlT = ILClone(pidl);
|
||
|
if (pidlT)
|
||
|
{
|
||
|
if ( -1 == ::DPA_InsertPtr(_hdpa, iItem, pidlT) )
|
||
|
{
|
||
|
ILFree(pidlT);
|
||
|
pidlT = NULL; // failure
|
||
|
}
|
||
|
}
|
||
|
return (BOOL_PTR)pidlT;
|
||
|
}
|
||
|
|
||
|
|
||
|
// check the item name against the file spec and see if we should include it
|
||
|
|
||
|
BOOL CFolderItems::_IncludeItem(IShellFolder *psf, LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
BOOL fInclude = TRUE; // by default include it...
|
||
|
|
||
|
if ( _pszFileSpec )
|
||
|
{
|
||
|
// see if we can resolve the link on this object
|
||
|
|
||
|
LPITEMIDLIST pidlFromLink = NULL;
|
||
|
IShellLink *psl;
|
||
|
|
||
|
if ( SUCCEEDED(psf->GetUIObjectOf(NULL, 1, &pidl, IID_IShellLink, NULL, (void **)&psl)) )
|
||
|
{
|
||
|
psl->GetIDList(&pidlFromLink);
|
||
|
psl->Release();
|
||
|
pidl = pidlFromLink;
|
||
|
}
|
||
|
|
||
|
// then apply the file spec
|
||
|
|
||
|
TCHAR szName[MAX_PATH];
|
||
|
SHGetNameAndFlags(pidl, SHGDN_INFOLDER|SHGDN_FORPARSING, szName, ARRAYSIZE(szName), NULL);
|
||
|
fInclude = PathMatchSpec(szName, _pszFileSpec);
|
||
|
|
||
|
ILFree(pidlFromLink);
|
||
|
}
|
||
|
|
||
|
return fInclude;
|
||
|
}
|
||
|
|
||
|
|
||
|
// in:
|
||
|
// iItemNeeded zero based index
|
||
|
//
|
||
|
//
|
||
|
// returns:
|
||
|
// S_OK in range value
|
||
|
// S_FALSE out of range value
|
||
|
//
|
||
|
|
||
|
HRESULT CFolderItems::_EnsureItem(UINT iItemNeeded, LPCITEMIDLIST *ppidlItem)
|
||
|
{
|
||
|
HRESULT hr = S_FALSE; // assume out of range
|
||
|
|
||
|
if (_GetHDPA())
|
||
|
{
|
||
|
LPCITEMIDLIST pidl = (LPCITEMIDLIST)DPA_GetPtr(_hdpa, iItemNeeded);
|
||
|
if (pidl)
|
||
|
{
|
||
|
if (ppidlItem)
|
||
|
*ppidlItem = pidl;
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
else if (!_fGotAllItems)
|
||
|
{
|
||
|
IShellFolderView *psfv;
|
||
|
if (SUCCEEDED(_psdf->GetShellFolderView(&psfv)))
|
||
|
{
|
||
|
if (_fSelected)
|
||
|
{
|
||
|
// we can only request the entire selection, therefore
|
||
|
// do so and populate our array with those.
|
||
|
|
||
|
UINT cItems;
|
||
|
LPCITEMIDLIST *ppidl = NULL;
|
||
|
if (SUCCEEDED(psfv->GetSelectedObjects(&ppidl, &cItems)) && ppidl)
|
||
|
{
|
||
|
for (UINT i = 0; i < cItems; i++)
|
||
|
_CopyItem(i, ppidl[i]);
|
||
|
|
||
|
LocalFree(ppidl);
|
||
|
}
|
||
|
_fGotAllItems = TRUE;
|
||
|
hr = _EnsureItem(iItemNeeded, ppidlItem);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
UINT cItems;
|
||
|
if (SUCCEEDED(psfv->GetObjectCount(&cItems)))
|
||
|
{
|
||
|
// if there is no file spec then we can just request the item
|
||
|
// that we want, otherwise we must collect them all
|
||
|
// from the view.
|
||
|
|
||
|
if (iItemNeeded < cItems)
|
||
|
{
|
||
|
LPCITEMIDLIST pidl;
|
||
|
if (SUCCEEDED(GetObjectSafely(psfv, &pidl, iItemNeeded)))
|
||
|
{
|
||
|
if (_CopyItem(iItemNeeded, pidl))
|
||
|
{
|
||
|
hr = _EnsureItem(iItemNeeded, ppidlItem);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
psfv->Release();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// we don't have an enumerator, so lets request it
|
||
|
|
||
|
if (NULL == _penum)
|
||
|
_psdf->_psf->EnumObjects(NULL, _grfFlags, &_penum);
|
||
|
|
||
|
if (NULL == _penum)
|
||
|
{
|
||
|
_fGotAllItems = TRUE; // enum empty, we are done
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// get more while our count is less than the index
|
||
|
while (_cNumEnumed <= iItemNeeded)
|
||
|
{
|
||
|
LPITEMIDLIST pidl;
|
||
|
if (S_OK == _penum->Next(1, &pidl, NULL))
|
||
|
{
|
||
|
if ( _IncludeItem(_psdf->_psf, pidl) &&
|
||
|
(-1 != ::DPA_AppendPtr(_hdpa, pidl)) )
|
||
|
{
|
||
|
_cNumEnumed++;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ILFree(pidl);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ATOMICRELEASE(_penum);
|
||
|
_fGotAllItems = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
hr = _EnsureItem(iItemNeeded, ppidlItem);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
hr = E_OUTOFMEMORY;
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::QueryInterface(REFIID riid, void ** ppv)
|
||
|
{
|
||
|
static const QITAB qit[] = {
|
||
|
QITABENT(CFolderItems, FolderItems3),
|
||
|
QITABENTMULTI(CFolderItems, FolderItems, FolderItems3),
|
||
|
QITABENTMULTI(CFolderItems, IDispatch, FolderItems3),
|
||
|
QITABENT(CFolderItems, IObjectSafety),
|
||
|
QITABENT(CFolderItems, IPersistFolder),
|
||
|
{ 0 },
|
||
|
};
|
||
|
return QISearch(this, qit, riid, ppv);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CFolderItems::AddRef(void)
|
||
|
{
|
||
|
return InterlockedIncrement(&_cRef);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CFolderItems::Release(void)
|
||
|
{
|
||
|
if (InterlockedDecrement(&_cRef))
|
||
|
return _cRef;
|
||
|
|
||
|
delete this;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
// FolderItems implementation
|
||
|
|
||
|
STDMETHODIMP CFolderItems::get_Application(IDispatch **ppid)
|
||
|
{
|
||
|
// let the folder object do the work...
|
||
|
return _psdf->get_Application(ppid);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::get_Parent(IDispatch **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::get_Count(long *pCount)
|
||
|
{
|
||
|
HRESULT hr = _SecurityCheck();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = E_FAIL;
|
||
|
IShellFolderView *psfv = NULL;
|
||
|
|
||
|
// get the items from the view, we can do this if we don't have
|
||
|
// a spec.
|
||
|
|
||
|
if ( !_pszFileSpec )
|
||
|
{
|
||
|
hr = _psdf->GetShellFolderView(&psfv);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
UINT cCount;
|
||
|
hr = _fSelected ? psfv->GetSelectedCount(&cCount) : psfv->GetObjectCount(&cCount);
|
||
|
*pCount = cCount;
|
||
|
psfv->Release();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// either we failed to get to the view, or the file spec won't allow us
|
||
|
|
||
|
if ( _pszFileSpec || FAILED(hr) )
|
||
|
{
|
||
|
// Well it looks like we need to finish the iteration now to get this!
|
||
|
*pCount = SUCCEEDED(_EnsureItem(-1, NULL)) ? _GetHDPACount() : 0;
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
// Folder.Items.Item(1)
|
||
|
// Folder.Items.Item("file name")
|
||
|
// Folder.Items.Item() - same as Folder.Self
|
||
|
|
||
|
STDMETHODIMP CFolderItems::Item(VARIANT index, FolderItem **ppid)
|
||
|
{
|
||
|
HRESULT hr = _SecurityCheck();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = S_FALSE;
|
||
|
*ppid = NULL;
|
||
|
|
||
|
// This is sortof gross, but if we are passed a pointer to another variant, simply
|
||
|
// update our copy here...
|
||
|
if (index.vt == (VT_BYREF | VT_VARIANT) && index.pvarVal)
|
||
|
index = *index.pvarVal;
|
||
|
|
||
|
switch (index.vt)
|
||
|
{
|
||
|
case VT_ERROR:
|
||
|
{
|
||
|
// No Parameters, generate a folder item for the folder itself...
|
||
|
Folder * psdfParent;
|
||
|
hr = _psdf->get_ParentFolder(&psdfParent);
|
||
|
if (SUCCEEDED(hr) && psdfParent)
|
||
|
{
|
||
|
hr = CFolderItem_Create((CFolder*)psdfParent, ILFindLastID(_psdf->_pidl), ppid);
|
||
|
psdfParent->Release();
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case VT_I2:
|
||
|
index.lVal = (long)index.iVal;
|
||
|
// And fall through...
|
||
|
|
||
|
case VT_I4:
|
||
|
{
|
||
|
LPCITEMIDLIST pidl;
|
||
|
hr = _EnsureItem(index.lVal, &pidl); // Get the asked for item...
|
||
|
if (S_OK == hr)
|
||
|
hr = CFolderItem_Create(_psdf, pidl, ppid);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case VT_BSTR:
|
||
|
{
|
||
|
LPITEMIDLIST pidl;
|
||
|
hr = _psdf->_psf->ParseDisplayName(NULL, NULL, index.bstrVal, NULL, &pidl, NULL);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = CFolderItem_Create(_psdf, pidl, ppid);
|
||
|
ILFree(pidl);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
if (hr != S_OK) // Error values cause problems in Java script
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
hr = S_FALSE;
|
||
|
}
|
||
|
else if (ppid && _dwSafetyOptions)
|
||
|
{
|
||
|
hr = MakeSafeForScripting((IUnknown**)ppid);
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::InvokeVerbEx(VARIANT vVerb, VARIANT vArgs)
|
||
|
{
|
||
|
long cItems;
|
||
|
// Note: if not safe, we'll fail in get_Count with E_ACCESSDENIED
|
||
|
HRESULT hr = get_Count(&cItems);
|
||
|
if (SUCCEEDED(hr) && cItems)
|
||
|
{
|
||
|
LPCITEMIDLIST *ppidl = (LPCITEMIDLIST *)LocalAlloc(LPTR, SIZEOF(*ppidl) * cItems);
|
||
|
if (ppidl)
|
||
|
{
|
||
|
for (int i = 0; i < cItems; i++)
|
||
|
{
|
||
|
_EnsureItem(i, &ppidl[i]);
|
||
|
}
|
||
|
|
||
|
// BUGBUG: shouldn't worry about items from different folders here?
|
||
|
hr = _psdf->InvokeVerbHelper(vVerb, vArgs, ppidl, cItems, _dwSafetyOptions);
|
||
|
|
||
|
LocalFree(ppidl);
|
||
|
}
|
||
|
else
|
||
|
hr = E_OUTOFMEMORY;
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
//
|
||
|
// fIncludeFolders => includ folders in the enumeration (TRUE by default)
|
||
|
// bstrFilter = filespec to apply while enumerating
|
||
|
//
|
||
|
|
||
|
STDMETHODIMP CFolderItems::Filter(LONG grfFlags, BSTR bstrFileSpec)
|
||
|
{
|
||
|
HRESULT hr = _SecurityCheck();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
_grfFlags = grfFlags;
|
||
|
Str_SetPtr(&_pszFileSpec, bstrFileSpec);
|
||
|
_ResetIDListArray();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::_GetUIObjectOf(REFIID riid, void ** ppv)
|
||
|
{
|
||
|
*ppv = NULL;
|
||
|
HRESULT hr = E_FAIL;
|
||
|
|
||
|
long cItems;
|
||
|
if (SUCCEEDED(get_Count(&cItems)) && cItems)
|
||
|
{
|
||
|
LPCITEMIDLIST *ppidl = (LPCITEMIDLIST *)LocalAlloc(LPTR, SIZEOF(*ppidl) * cItems);
|
||
|
if (ppidl)
|
||
|
{
|
||
|
for (int i = 0; i < cItems; i++)
|
||
|
{
|
||
|
_EnsureItem(i, &ppidl[i]);
|
||
|
}
|
||
|
|
||
|
// BUGBUG: shouldn't worry about items from different folders here?
|
||
|
hr = _psdf->_psf->GetUIObjectOf(_psdf->_hwnd, cItems, ppidl, riid, NULL, ppv);
|
||
|
|
||
|
LocalFree(ppidl);
|
||
|
}
|
||
|
else
|
||
|
hr = E_OUTOFMEMORY;
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CFolderItems::get_Verbs(FolderItemVerbs **ppfic)
|
||
|
{
|
||
|
*ppfic = NULL;
|
||
|
HRESULT hr = _SecurityCheck();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = S_FALSE;
|
||
|
|
||
|
IContextMenu *pcm;
|
||
|
if (SUCCEEDED(_GetUIObjectOf(IID_PPV_ARG(IContextMenu, &pcm))))
|
||
|
{
|
||
|
hr = CFolderItemVerbs_Create(pcm, ppfic);
|
||
|
if (SUCCEEDED(hr) && _dwSafetyOptions)
|
||
|
{
|
||
|
hr = MakeSafeForScripting((IUnknown**)ppfic);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
// Set the folder's site to FolderItemVerbs
|
||
|
IUnknown_SetSite(*ppfic, _psdf->_punkSite);
|
||
|
}
|
||
|
}
|
||
|
pcm->Release();
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
// supports VB "For Each" statement
|
||
|
|
||
|
STDMETHODIMP CFolderItems::_NewEnum(IUnknown **ppunk)
|
||
|
{
|
||
|
*ppunk = NULL;
|
||
|
HRESULT hr = _SecurityCheck();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = E_OUTOFMEMORY;
|
||
|
CEnumFolderItems *pNew = new CEnumFolderItems(this);
|
||
|
if (pNew)
|
||
|
{
|
||
|
hr = pNew->QueryInterface(IID_PPV_ARG(IUnknown, ppunk));
|
||
|
pNew->Release();
|
||
|
|
||
|
if (SUCCEEDED(hr) && _dwSafetyOptions)
|
||
|
{
|
||
|
hr = MakeSafeForScripting(ppunk);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItemsFDF_Create(CFolder *psdf, FolderItems **ppitems)
|
||
|
{
|
||
|
*ppitems = NULL;
|
||
|
HRESULT hr = E_OUTOFMEMORY;
|
||
|
CFolderItemsFDF* psdfi = new CFolderItemsFDF(psdf);
|
||
|
if (psdfi)
|
||
|
{
|
||
|
hr = psdfi->QueryInterface(IID_PPV_ARG(FolderItems, ppitems));
|
||
|
psdfi->Release();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDAPI CFolderItemsFDF_CreateInstance(IUnknown *punk, REFIID riid, void **ppv)
|
||
|
{
|
||
|
HRESULT hr = E_OUTOFMEMORY;
|
||
|
*ppv = NULL;
|
||
|
|
||
|
CFolderItemsFDF *pfi = new CFolderItemsFDF(NULL);
|
||
|
if (pfi)
|
||
|
{
|
||
|
hr = pfi->QueryInterface(riid, ppv);
|
||
|
pfi->Release();
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
CFolderItemsFDF::CFolderItemsFDF(CFolder *psdf) : CFolderItems(psdf, FALSE)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItemsFDF::QueryInterface(REFIID riid, void ** ppv)
|
||
|
{
|
||
|
HRESULT hr = CFolderItems::QueryInterface(riid, ppv);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
static const QITAB qit[] = {
|
||
|
QITABENT(CFolderItemsFDF, IInsertItem),
|
||
|
{ 0 },
|
||
|
};
|
||
|
|
||
|
hr = QISearch(this, qit, riid, ppv);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItemsFDF::InsertItem(LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
HRESULT hr = S_FALSE;
|
||
|
|
||
|
if (_GetHDPA())
|
||
|
{
|
||
|
LPITEMIDLIST pidlTemp;
|
||
|
|
||
|
hr = SHILClone(pidl, &pidlTemp);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (DPA_AppendPtr(_hdpa, pidlTemp) == -1)
|
||
|
{
|
||
|
ILFree(pidlTemp);
|
||
|
hr = E_FAIL;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CFolderItemsFDF::_EnsureItem(UINT iItemNeeded, LPCITEMIDLIST *ppidlItem)
|
||
|
{
|
||
|
HRESULT hr = S_FALSE; // assume out of range
|
||
|
|
||
|
if (ppidlItem)
|
||
|
*ppidlItem = NULL;
|
||
|
|
||
|
if (_GetHDPA())
|
||
|
{
|
||
|
LPCITEMIDLIST pidl = (LPCITEMIDLIST)DPA_GetPtr(_hdpa, iItemNeeded);
|
||
|
if (pidl)
|
||
|
{
|
||
|
if (ppidlItem)
|
||
|
*ppidlItem = pidl;
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
// CEnumFolderItems implementation of IEnumVARIANT
|
||
|
|
||
|
CEnumFolderItems::CEnumFolderItems(CFolderItems *pfdritms) :
|
||
|
_cRef(1),
|
||
|
_pfdritms(pfdritms),
|
||
|
_iCur(0)
|
||
|
{
|
||
|
_pfdritms->AddRef();
|
||
|
DllAddRef();
|
||
|
}
|
||
|
|
||
|
|
||
|
CEnumFolderItems::~CEnumFolderItems(void)
|
||
|
{
|
||
|
_pfdritms->Release();
|
||
|
DllRelease();
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CEnumFolderItems::QueryInterface(REFIID riid, void ** ppv)
|
||
|
{
|
||
|
static const QITAB qit[] = {
|
||
|
QITABENT(CEnumFolderItems, IEnumVARIANT),
|
||
|
{ 0 },
|
||
|
};
|
||
|
return QISearch(this, qit, riid, ppv);
|
||
|
}
|
||
|
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CEnumFolderItems::AddRef(void)
|
||
|
{
|
||
|
return InterlockedIncrement(&_cRef);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CEnumFolderItems::Release(void)
|
||
|
{
|
||
|
if (InterlockedDecrement(&_cRef))
|
||
|
return _cRef;
|
||
|
|
||
|
delete this;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CEnumFolderItems::Next(ULONG cVar, VARIANT *pVar, ULONG *pulVar)
|
||
|
{
|
||
|
ULONG cReturn = 0;
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
if (!pulVar && (cVar != 1))
|
||
|
return E_POINTER;
|
||
|
|
||
|
while (cVar)
|
||
|
{
|
||
|
LPCITEMIDLIST pidl;
|
||
|
|
||
|
if (S_OK == _pfdritms->_EnsureItem(_iCur + cVar - 1, &pidl))
|
||
|
{
|
||
|
FolderItem *pid;
|
||
|
|
||
|
hr = CFolderItem_Create(_pfdritms->_psdf, pidl, &pid);
|
||
|
_iCur++;
|
||
|
|
||
|
if (_pfdritms->_dwSafetyOptions && SUCCEEDED(hr))
|
||
|
hr = MakeSafeForScripting((IUnknown**)&pid);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pVar->pdispVal = pid;
|
||
|
pVar->vt = VT_DISPATCH;
|
||
|
pVar++;
|
||
|
cReturn++;
|
||
|
cVar--;
|
||
|
}
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (pulVar)
|
||
|
*pulVar = cReturn;
|
||
|
hr = cReturn ? S_OK : S_FALSE;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CEnumFolderItems::Skip(ULONG cSkip)
|
||
|
{
|
||
|
if ((_iCur + cSkip) >= _pfdritms->_GetHDPACount())
|
||
|
return S_FALSE;
|
||
|
|
||
|
_iCur += cSkip;
|
||
|
return NOERROR;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CEnumFolderItems::Reset(void)
|
||
|
{
|
||
|
_iCur = 0;
|
||
|
return NOERROR;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CEnumFolderItems::Clone(IEnumVARIANT **ppenum)
|
||
|
{
|
||
|
*ppenum = NULL;
|
||
|
HRESULT hr = E_OUTOFMEMORY;
|
||
|
CEnumFolderItems *pNew = new CEnumFolderItems(_pfdritms);
|
||
|
if (pNew)
|
||
|
{
|
||
|
hr = pNew->QueryInterface(IID_PPV_ARG(IEnumVARIANT, ppenum));
|
||
|
pNew->Release();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|