mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 10:04:49 +00:00
432 lines
11 KiB
C++
432 lines
11 KiB
C++
/*
|
|
* IShellItem implementation
|
|
*
|
|
* Copyright 2008 Vincent Povirk for CodeWeavers
|
|
* Copyright 2009 Andrew Hill
|
|
* Copyright 2013 Katayama Hirofumi MZ
|
|
*
|
|
* 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shell);
|
|
|
|
EXTERN_C HRESULT WINAPI SHCreateShellItem(PCIDLIST_ABSOLUTE pidlParent,
|
|
IShellFolder *psfParent, PCUITEMID_CHILD pidl, IShellItem **ppsi);
|
|
|
|
CShellItem::CShellItem() :
|
|
m_pidl(NULL)
|
|
{
|
|
}
|
|
|
|
CShellItem::~CShellItem()
|
|
{
|
|
ILFree(m_pidl);
|
|
}
|
|
|
|
HRESULT CShellItem::get_parent_pidl(LPITEMIDLIST *parent_pidl)
|
|
{
|
|
*parent_pidl = ILClone(m_pidl);
|
|
if (*parent_pidl)
|
|
{
|
|
if (ILRemoveLastID(*parent_pidl))
|
|
return S_OK;
|
|
else
|
|
{
|
|
ILFree(*parent_pidl);
|
|
*parent_pidl = NULL;
|
|
return E_INVALIDARG;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
*parent_pidl = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
HRESULT CShellItem::get_parent_shellfolder(IShellFolder **ppsf)
|
|
{
|
|
HRESULT hr;
|
|
LPITEMIDLIST parent_pidl;
|
|
CComPtr<IShellFolder> desktop;
|
|
|
|
hr = get_parent_pidl(&parent_pidl);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SHGetDesktopFolder(&desktop);
|
|
if (SUCCEEDED(hr))
|
|
hr = desktop->BindToObject(parent_pidl, NULL, IID_PPV_ARG(IShellFolder, ppsf));
|
|
ILFree(parent_pidl);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CShellItem::get_shellfolder(IBindCtx *pbc, REFIID riid, void **ppvOut)
|
|
{
|
|
CComPtr<IShellFolder> psf;
|
|
CComPtr<IShellFolder> psfDesktop;
|
|
HRESULT ret;
|
|
|
|
ret = SHGetDesktopFolder(&psfDesktop);
|
|
if (FAILED_UNEXPECTEDLY(ret))
|
|
return ret;
|
|
|
|
if (_ILIsDesktop(m_pidl))
|
|
psf = psfDesktop;
|
|
else
|
|
{
|
|
ret = psfDesktop->BindToObject(m_pidl, pbc, IID_PPV_ARG(IShellFolder, &psf));
|
|
if (FAILED_UNEXPECTEDLY(ret))
|
|
return ret;
|
|
}
|
|
|
|
return psf->QueryInterface(riid, ppvOut);
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::BindToHandler(IBindCtx *pbc, REFGUID rbhid, REFIID riid, void **ppvOut)
|
|
{
|
|
HRESULT ret;
|
|
TRACE("(%p, %p,%s,%p,%p)\n", this, pbc, shdebugstr_guid(&rbhid), riid, ppvOut);
|
|
|
|
*ppvOut = NULL;
|
|
if (IsEqualGUID(rbhid, BHID_SFObject))
|
|
{
|
|
return get_shellfolder(pbc, riid, ppvOut);
|
|
}
|
|
else if (IsEqualGUID(rbhid, BHID_SFUIObject))
|
|
{
|
|
CComPtr<IShellFolder> psf_parent;
|
|
if (_ILIsDesktop(m_pidl))
|
|
ret = SHGetDesktopFolder(&psf_parent);
|
|
else
|
|
ret = get_parent_shellfolder(&psf_parent);
|
|
if (FAILED_UNEXPECTEDLY(ret))
|
|
return ret;
|
|
|
|
LPCITEMIDLIST pidl = ILFindLastID(m_pidl);
|
|
return psf_parent->GetUIObjectOf(NULL, 1, &pidl, riid, NULL, ppvOut);
|
|
}
|
|
else if (IsEqualGUID(rbhid, BHID_DataObject))
|
|
{
|
|
return BindToHandler(pbc, BHID_SFUIObject, IID_IDataObject, ppvOut);
|
|
}
|
|
else if (IsEqualGUID(rbhid, BHID_SFViewObject))
|
|
{
|
|
CComPtr<IShellFolder> psf;
|
|
ret = get_shellfolder(NULL, IID_PPV_ARG(IShellFolder, &psf));
|
|
if (FAILED_UNEXPECTEDLY(ret))
|
|
return ret;
|
|
|
|
return psf->CreateViewObject(NULL, riid, ppvOut);
|
|
}
|
|
|
|
FIXME("Unsupported BHID %s.\n", debugstr_guid(&rbhid));
|
|
|
|
return MK_E_NOOBJECT;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::GetParent(IShellItem **ppsi)
|
|
{
|
|
HRESULT hr;
|
|
LPITEMIDLIST parent_pidl;
|
|
|
|
TRACE("(%p,%p)\n", this, ppsi);
|
|
|
|
hr = get_parent_pidl(&parent_pidl);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SHCreateShellItem(NULL, NULL, parent_pidl, ppsi);
|
|
ILFree(parent_pidl);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::GetDisplayName(SIGDN sigdnName, LPWSTR *ppszName)
|
|
{
|
|
return SHGetNameFromIDList(m_pidl, sigdnName, ppszName);
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::GetAttributes(SFGAOF sfgaoMask, SFGAOF *psfgaoAttribs)
|
|
{
|
|
CComPtr<IShellFolder> parent_folder;
|
|
LPCITEMIDLIST child_pidl;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p,%x,%p)\n", this, sfgaoMask, psfgaoAttribs);
|
|
|
|
if (_ILIsDesktop(m_pidl))
|
|
hr = SHGetDesktopFolder(&parent_folder);
|
|
else
|
|
hr = get_parent_shellfolder(&parent_folder);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
child_pidl = ILFindLastID(m_pidl);
|
|
*psfgaoAttribs = sfgaoMask;
|
|
hr = parent_folder->GetAttributesOf(1, &child_pidl, psfgaoAttribs);
|
|
*psfgaoAttribs &= sfgaoMask;
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
return (sfgaoMask == *psfgaoAttribs) ? S_OK : S_FALSE;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::Compare(IShellItem *oth, SICHINTF hint, int *piOrder)
|
|
{
|
|
HRESULT hr;
|
|
CComPtr<IPersistIDList> pIDList;
|
|
CComPtr<IShellFolder> psfDesktop;
|
|
LPITEMIDLIST pidl;
|
|
|
|
TRACE("(%p,%p,%x,%p)\n", this, oth, hint, piOrder);
|
|
|
|
if (piOrder == NULL || oth == NULL)
|
|
return E_POINTER;
|
|
|
|
hr = oth->QueryInterface(IID_PPV_ARG(IPersistIDList, &pIDList));
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pIDList->GetIDList(&pidl);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SHGetDesktopFolder(&psfDesktop);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psfDesktop->CompareIDs(hint, m_pidl, pidl);
|
|
*piOrder = (int)(short)SCODE_CODE(hr);
|
|
}
|
|
ILFree(pidl);
|
|
}
|
|
}
|
|
|
|
if(FAILED(hr))
|
|
return hr;
|
|
|
|
if(*piOrder)
|
|
return S_FALSE;
|
|
else
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::GetClassID(CLSID *pClassID)
|
|
{
|
|
TRACE("(%p,%p)\n", this, pClassID);
|
|
|
|
*pClassID = CLSID_ShellItem;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::SetIDList(PCIDLIST_ABSOLUTE pidlx)
|
|
{
|
|
LPITEMIDLIST new_pidl;
|
|
|
|
TRACE("(%p,%p)\n", this, pidlx);
|
|
|
|
new_pidl = ILClone(pidlx);
|
|
if (new_pidl)
|
|
{
|
|
ILFree(m_pidl);
|
|
m_pidl = new_pidl;
|
|
return S_OK;
|
|
}
|
|
else
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
HRESULT WINAPI CShellItem::GetIDList(PIDLIST_ABSOLUTE *ppidl)
|
|
{
|
|
TRACE("(%p,%p)\n", this, ppidl);
|
|
|
|
*ppidl = ILClone(m_pidl);
|
|
if (*ppidl)
|
|
return S_OK;
|
|
else
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
HRESULT WINAPI SHCreateShellItem(PCIDLIST_ABSOLUTE pidlParent,
|
|
IShellFolder *psfParent, PCUITEMID_CHILD pidl, IShellItem **ppsi)
|
|
{
|
|
HRESULT hr;
|
|
CComPtr<IShellItem> newShellItem;
|
|
LPITEMIDLIST new_pidl;
|
|
CComPtr<IPersistIDList> newPersistIDList;
|
|
|
|
TRACE("(%p,%p,%p,%p)\n", pidlParent, psfParent, pidl, ppsi);
|
|
|
|
*ppsi = NULL;
|
|
|
|
if (!pidl)
|
|
return E_INVALIDARG;
|
|
|
|
if (pidlParent || psfParent)
|
|
{
|
|
LPITEMIDLIST temp_parent = NULL;
|
|
if (!pidlParent)
|
|
{
|
|
CComPtr<IPersistFolder2> ppf2Parent;
|
|
|
|
if (FAILED(psfParent->QueryInterface(IID_PPV_ARG(IPersistFolder2, &ppf2Parent))))
|
|
{
|
|
FIXME("couldn't get IPersistFolder2 interface of parent\n");
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
if (FAILED(ppf2Parent->GetCurFolder(&temp_parent)))
|
|
{
|
|
FIXME("couldn't get parent PIDL\n");
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
pidlParent = temp_parent;
|
|
}
|
|
|
|
new_pidl = ILCombine(pidlParent, pidl);
|
|
ILFree(temp_parent);
|
|
|
|
if (!new_pidl)
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
new_pidl = ILClone(pidl);
|
|
if (!new_pidl)
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = CShellItem::_CreatorClass::CreateInstance(NULL, IID_PPV_ARG(IShellItem, &newShellItem));
|
|
if (FAILED(hr))
|
|
{
|
|
ILFree(new_pidl);
|
|
return hr;
|
|
}
|
|
hr = newShellItem->QueryInterface(IID_PPV_ARG(IPersistIDList, &newPersistIDList));
|
|
if (FAILED(hr))
|
|
{
|
|
ILFree(new_pidl);
|
|
return hr;
|
|
}
|
|
hr = newPersistIDList->SetIDList(new_pidl);
|
|
if (FAILED(hr))
|
|
{
|
|
ILFree(new_pidl);
|
|
return hr;
|
|
}
|
|
ILFree(new_pidl);
|
|
|
|
*ppsi = newShellItem.Detach();
|
|
|
|
return hr;
|
|
}
|
|
|
|
class CShellItemArray :
|
|
public CComCoClass<CShellItemArray, &CLSID_NULL>,
|
|
public CComObjectRootEx<CComMultiThreadModelNoCS>,
|
|
public IShellItemArray
|
|
{
|
|
CIDA *m_pCIDA;
|
|
STGMEDIUM m_Medium;
|
|
|
|
public:
|
|
CShellItemArray() : m_pCIDA(NULL)
|
|
{
|
|
m_Medium.tymed = TYMED_NULL;
|
|
}
|
|
|
|
virtual ~CShellItemArray()
|
|
{
|
|
CDataObjectHIDA::DestroyCIDA(m_pCIDA, m_Medium);
|
|
}
|
|
|
|
HRESULT Initialize(IDataObject *pdo)
|
|
{
|
|
return CDataObjectHIDA::CreateCIDA(pdo, &m_pCIDA, m_Medium);
|
|
}
|
|
|
|
inline UINT GetCount() const { return m_pCIDA->cidl; }
|
|
|
|
// IShellItemArray
|
|
STDMETHODIMP BindToHandler(IBindCtx *pbc, REFGUID rbhid, REFIID riid, void **ppv) override
|
|
{
|
|
UNIMPLEMENTED;
|
|
*ppv = NULL;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP GetPropertyStore(GETPROPERTYSTOREFLAGS flags, REFIID riid, void **ppv) override
|
|
{
|
|
UNIMPLEMENTED;
|
|
*ppv = NULL;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP GetPropertyDescriptionList(REFPROPERTYKEY keyType, REFIID riid, void **ppv) override
|
|
{
|
|
UNIMPLEMENTED;
|
|
*ppv = NULL;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP GetAttributes(SIATTRIBFLAGS dwAttribFlags, SFGAOF sfgaoMask, SFGAOF *psfgaoAttribs) override
|
|
{
|
|
UNIMPLEMENTED;
|
|
*psfgaoAttribs = 0;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP GetCount(DWORD*pCount) override
|
|
{
|
|
*pCount = m_pCIDA ? GetCount() : 0;
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP GetItemAt(DWORD nIndex, IShellItem **ppItem) override
|
|
{
|
|
if (!ppItem)
|
|
return E_INVALIDARG;
|
|
*ppItem = NULL;
|
|
if (!m_pCIDA)
|
|
return E_UNEXPECTED;
|
|
if (nIndex >= GetCount())
|
|
return E_FAIL;
|
|
return SHCreateShellItem(HIDA_GetPIDLFolder(m_pCIDA), NULL,
|
|
HIDA_GetPIDLItem(m_pCIDA, nIndex), ppItem);
|
|
}
|
|
|
|
STDMETHODIMP EnumItems(IEnumShellItems **ppESI) override
|
|
{
|
|
UNIMPLEMENTED;
|
|
*ppESI = NULL;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
DECLARE_NO_REGISTRY()
|
|
DECLARE_NOT_AGGREGATABLE(CShellItemArray)
|
|
|
|
BEGIN_COM_MAP(CShellItemArray)
|
|
COM_INTERFACE_ENTRY_IID(IID_IShellItemArray, IShellItemArray)
|
|
END_COM_MAP()
|
|
};
|
|
|
|
EXTERN_C HRESULT WINAPI
|
|
SHCreateShellItemArrayFromDataObject(_In_ IDataObject *pdo, _In_ REFIID riid, _Out_ void **ppv)
|
|
{
|
|
return ShellObjectCreatorInit<CShellItemArray>(pdo, riid, ppv);
|
|
}
|