[SHELL32]

- Improve IShellItem implementation. Patch by Katayama Hirofumi MZ.
CORE-7063 #resolve

svn path=/trunk/; revision=58687
This commit is contained in:
Thomas Faber 2013-04-06 12:12:43 +00:00
parent 8d892dc056
commit 69889ed097
2 changed files with 104 additions and 42 deletions

View file

@ -3,6 +3,7 @@
* *
* Copyright 2008 Vincent Povirk for CodeWeavers * Copyright 2008 Vincent Povirk for CodeWeavers
* Copyright 2009 Andrew Hill * Copyright 2009 Andrew Hill
* Copyright 2013 Katayama Hirofumi MZ
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public * modify it under the terms of the GNU Lesser General Public
@ -28,17 +29,17 @@ EXTERN_C HRESULT WINAPI SHCreateShellItem(LPCITEMIDLIST pidlParent,
CShellItem::CShellItem() CShellItem::CShellItem()
{ {
pidl = NULL; m_pidl = NULL;
} }
CShellItem::~CShellItem() CShellItem::~CShellItem()
{ {
ILFree(pidl); ILFree(m_pidl);
} }
HRESULT CShellItem::get_parent_pidl(LPITEMIDLIST *parent_pidl) HRESULT CShellItem::get_parent_pidl(LPITEMIDLIST *parent_pidl)
{ {
*parent_pidl = ILClone(pidl); *parent_pidl = ILClone(m_pidl);
if (*parent_pidl) if (*parent_pidl)
{ {
if (ILRemoveLastID(*parent_pidl)) if (ILRemoveLastID(*parent_pidl))
@ -61,18 +62,18 @@ HRESULT CShellItem::get_parent_shellfolder(IShellFolder **ppsf)
{ {
LPITEMIDLIST parent_pidl; LPITEMIDLIST parent_pidl;
CComPtr<IShellFolder> desktop; CComPtr<IShellFolder> desktop;
HRESULT ret; HRESULT hr;
ret = get_parent_pidl(&parent_pidl); hr = get_parent_pidl(&parent_pidl);
if (SUCCEEDED(ret)) if (SUCCEEDED(hr))
{ {
ret = SHGetDesktopFolder(&desktop); hr = SHGetDesktopFolder(&desktop);
if (SUCCEEDED(ret)) if (SUCCEEDED(hr))
ret = desktop->BindToObject(parent_pidl, NULL, IID_IShellFolder, (void**)ppsf); hr = desktop->BindToObject(parent_pidl, NULL, IID_IShellFolder, (void**)ppsf);
ILFree(parent_pidl); ILFree(parent_pidl);
} }
return ret; return hr;
} }
HRESULT WINAPI CShellItem::BindToHandler(IBindCtx *pbc, REFGUID rbhid, REFIID riid, void **ppvOut) HRESULT WINAPI CShellItem::BindToHandler(IBindCtx *pbc, REFGUID rbhid, REFIID riid, void **ppvOut)
@ -87,53 +88,116 @@ HRESULT WINAPI CShellItem::BindToHandler(IBindCtx *pbc, REFGUID rbhid, REFIID ri
HRESULT WINAPI CShellItem::GetParent(IShellItem **ppsi) HRESULT WINAPI CShellItem::GetParent(IShellItem **ppsi)
{ {
LPITEMIDLIST parent_pidl; LPITEMIDLIST parent_pidl;
HRESULT ret; HRESULT hr;
TRACE("(%p,%p)\n", this, ppsi); TRACE("(%p,%p)\n", this, ppsi);
ret = get_parent_pidl(&parent_pidl); hr = get_parent_pidl(&parent_pidl);
if (SUCCEEDED(ret)) if (SUCCEEDED(hr))
{ {
ret = SHCreateShellItem(NULL, NULL, parent_pidl, ppsi); hr = SHCreateShellItem(NULL, NULL, parent_pidl, ppsi);
ILFree(parent_pidl); ILFree(parent_pidl);
} }
return ret; return hr;
} }
HRESULT WINAPI CShellItem::GetDisplayName(SIGDN sigdnName, LPWSTR *ppszName) HRESULT WINAPI CShellItem::GetDisplayName(SIGDN sigdnName, LPWSTR *ppszName)
{ {
FIXME("(%p,%x,%p)\n", this, sigdnName, ppszName); CComPtr<IShellFolder> parent_folder;
HRESULT hr;
STRRET name;
DWORD uFlags;
TRACE("(%p,%x,%p)\n", this, sigdnName, ppszName);
if (sigdnName & SIGDN_URL)
return E_NOTIMPL;
if (ppszName == NULL)
return E_POINTER;
*ppszName = NULL; *ppszName = NULL;
return E_NOTIMPL; hr = get_parent_shellfolder(&parent_folder);
if (SUCCEEDED(hr))
{
if (sigdnName == SIGDN_PARENTRELATIVEEDITING)
uFlags = SHGDN_FOREDITING | SHGDN_INFOLDER;
else if (sigdnName == SIGDN_DESKTOPABSOLUTEEDITING)
uFlags = SHGDN_FOREDITING;
else if (sigdnName == SIGDN_PARENTRELATIVEEDITING)
uFlags = SHGDN_FOREDITING | SHGDN_INFOLDER;
else if (sigdnName == SIGDN_DESKTOPABSOLUTEEDITING)
uFlags = SHGDN_FOREDITING;
else if (sigdnName == SIGDN_PARENTRELATIVEPARSING)
uFlags = SHGDN_FORPARSING | SHGDN_INFOLDER;
else if (sigdnName == SIGDN_DESKTOPABSOLUTEPARSING)
uFlags = SHGDN_FORPARSING;
else
uFlags = SHGDN_NORMAL;
hr = parent_folder->GetDisplayNameOf(m_pidl, uFlags, &name);
if (SUCCEEDED(hr))
{
StrRetToStrW(&name, m_pidl, ppszName);
return S_OK;
}
}
return hr;
} }
HRESULT WINAPI CShellItem::GetAttributes(SFGAOF sfgaoMask, SFGAOF *psfgaoAttribs) HRESULT WINAPI CShellItem::GetAttributes(SFGAOF sfgaoMask, SFGAOF *psfgaoAttribs)
{ {
CComPtr<IShellFolder> parent_folder; CComPtr<IShellFolder> parent_folder;
LPITEMIDLIST child_pidl; LPITEMIDLIST child_pidl;
HRESULT ret; HRESULT hr;
TRACE("(%p,%x,%p)\n", this, sfgaoMask, psfgaoAttribs); TRACE("(%p,%x,%p)\n", this, sfgaoMask, psfgaoAttribs);
ret = get_parent_shellfolder(&parent_folder); hr = get_parent_shellfolder(&parent_folder);
if (SUCCEEDED(ret)) if (SUCCEEDED(hr))
{ {
child_pidl = ILFindLastID(pidl); child_pidl = ILFindLastID(m_pidl);
*psfgaoAttribs = sfgaoMask; *psfgaoAttribs = sfgaoMask;
ret = parent_folder->GetAttributesOf(1, (LPCITEMIDLIST*)&child_pidl, psfgaoAttribs); hr = parent_folder->GetAttributesOf(1, (LPCITEMIDLIST*)&child_pidl, psfgaoAttribs);
} }
return ret; return hr;
} }
HRESULT WINAPI CShellItem::Compare(IShellItem *oth, SICHINTF hint, int *piOrder) HRESULT WINAPI CShellItem::Compare(IShellItem *oth, SICHINTF hint, int *piOrder)
{ {
FIXME("(%p,%p,%x,%p)\n", this, oth, hint, piOrder); CComPtr<IShellFolder> parent_folder;
CComPtr<IPersistIDList> pIDList;
HRESULT hr;
LPITEMIDLIST pidl;
return E_NOTIMPL; TRACE("(%p,%p,%x,%p)\n", this, oth, hint, piOrder);
if (piOrder == NULL || oth == NULL)
return E_POINTER;
hr = oth->QueryInterface(IID_IPersistIDList, (void **)&pIDList);
if (SUCCEEDED(hr))
{
hr = pIDList->GetIDList(&pidl);
if (SUCCEEDED(hr))
{
hr = get_parent_shellfolder(&parent_folder);
if (SUCCEEDED(hr))
{
hr = parent_folder->CompareIDs(hint, m_pidl, pidl);
*piOrder = (int)SCODE_CODE(hr);
}
ILFree(pidl);
}
}
if (SUCCEEDED(hr))
return S_OK;
return hr;
} }
HRESULT WINAPI CShellItem::GetClassID(CLSID *pClassID) HRESULT WINAPI CShellItem::GetClassID(CLSID *pClassID)
@ -144,7 +208,6 @@ HRESULT WINAPI CShellItem::GetClassID(CLSID *pClassID)
return S_OK; return S_OK;
} }
HRESULT WINAPI CShellItem::SetIDList(LPCITEMIDLIST pidlx) HRESULT WINAPI CShellItem::SetIDList(LPCITEMIDLIST pidlx)
{ {
LPITEMIDLIST new_pidl; LPITEMIDLIST new_pidl;
@ -152,11 +215,10 @@ HRESULT WINAPI CShellItem::SetIDList(LPCITEMIDLIST pidlx)
TRACE("(%p,%p)\n", this, pidlx); TRACE("(%p,%p)\n", this, pidlx);
new_pidl = ILClone(pidlx); new_pidl = ILClone(pidlx);
if (new_pidl) if (new_pidl)
{ {
ILFree(pidl); ILFree(m_pidl);
pidl = new_pidl; m_pidl = new_pidl;
return S_OK; return S_OK;
} }
else else
@ -167,7 +229,7 @@ HRESULT WINAPI CShellItem::GetIDList(LPITEMIDLIST *ppidl)
{ {
TRACE("(%p,%p)\n", this, ppidl); TRACE("(%p,%p)\n", this, ppidl);
*ppidl = ILClone(pidl); *ppidl = ILClone(m_pidl);
if (*ppidl) if (*ppidl)
return S_OK; return S_OK;
else else
@ -180,7 +242,7 @@ HRESULT WINAPI SHCreateShellItem(LPCITEMIDLIST pidlParent,
IShellItem *newShellItem; IShellItem *newShellItem;
LPITEMIDLIST new_pidl; LPITEMIDLIST new_pidl;
CComPtr<IPersistIDList> newPersistIDList; CComPtr<IPersistIDList> newPersistIDList;
HRESULT ret; HRESULT hr;
TRACE("(%p,%p,%p,%p)\n", pidlParent, psfParent, pidl, ppsi); TRACE("(%p,%p,%p,%p)\n", pidlParent, psfParent, pidl, ppsi);
@ -223,26 +285,26 @@ HRESULT WINAPI SHCreateShellItem(LPCITEMIDLIST pidlParent,
return E_OUTOFMEMORY; return E_OUTOFMEMORY;
} }
ret = CShellItem::_CreatorClass::CreateInstance(NULL, IID_IShellItem, (void**)&newShellItem); hr = CShellItem::_CreatorClass::CreateInstance(NULL, IID_IShellItem, (void**)&newShellItem);
if (FAILED(ret)) if (FAILED(hr))
{ {
*ppsi = NULL; *ppsi = NULL;
ILFree(new_pidl); ILFree(new_pidl);
return ret; return hr;
} }
ret = newShellItem->QueryInterface(IID_IPersistIDList, (void **)&newPersistIDList); hr = newShellItem->QueryInterface(IID_IPersistIDList, (void **)&newPersistIDList);
if (FAILED(ret)) if (FAILED(hr))
{ {
ILFree(new_pidl); ILFree(new_pidl);
return ret; return hr;
} }
ret = newPersistIDList->SetIDList(new_pidl); hr = newPersistIDList->SetIDList(new_pidl);
if (FAILED(ret)) if (FAILED(hr))
{ {
ILFree(new_pidl); ILFree(new_pidl);
return ret; return hr;
} }
ILFree(new_pidl); ILFree(new_pidl);
*ppsi = newShellItem; *ppsi = newShellItem;
return ret; return hr;
} }

View file

@ -29,7 +29,7 @@ class CShellItem :
public IPersistIDList public IPersistIDList
{ {
private: private:
LPITEMIDLIST pidl; LPITEMIDLIST m_pidl;
public: public:
CShellItem(); CShellItem();
~CShellItem(); ~CShellItem();