/* * 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 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 psf; CComPtr 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 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 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 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 pIDList; CComPtr 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 newShellItem; LPITEMIDLIST new_pidl; CComPtr 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 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, public CComObjectRootEx, 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(pdo, riid, ppv); }