mirror of
https://github.com/reactos/reactos.git
synced 2025-01-12 01:00:06 +00:00
c424146e2c
svn path=/branches/cmake-bringup/; revision=48236
546 lines
15 KiB
C
546 lines
15 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS Registry namespace extension
|
|
* FILE: dll/win32/shell32/extracticon.c
|
|
* PURPOSE: Icon extraction
|
|
*
|
|
* PROGRAMMERS: Hervé Poussineau (hpoussin@reactos.org)
|
|
*/
|
|
|
|
#include <precomp.h>
|
|
|
|
#define CONST_VTABLE
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shell);
|
|
|
|
struct IconLocation
|
|
{
|
|
LPWSTR file;
|
|
UINT index;
|
|
};
|
|
|
|
struct IconExtraction
|
|
{
|
|
LONG ref;
|
|
IDefaultExtractIconInit defaultExtractIconInitImpl;
|
|
IExtractIconW extractIconWImpl;
|
|
IExtractIconA extractIconAImpl;
|
|
IPersistFile persistFileImpl;
|
|
|
|
UINT flags;
|
|
struct IconLocation defaultIcon;
|
|
struct IconLocation normalIcon;
|
|
struct IconLocation openIcon;
|
|
struct IconLocation shortcutIcon;
|
|
};
|
|
|
|
static VOID
|
|
DuplicateString(
|
|
LPCWSTR Source,
|
|
LPWSTR *Destination)
|
|
{
|
|
SIZE_T cb;
|
|
|
|
if (*Destination)
|
|
CoTaskMemFree(*Destination);
|
|
|
|
cb = (wcslen(Source) + 1) * sizeof(WCHAR);
|
|
*Destination = CoTaskMemAlloc(cb);
|
|
if (!*Destination)
|
|
return;
|
|
CopyMemory(*Destination, Source, cb);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_QueryInterface(
|
|
IDefaultExtractIconInit *This,
|
|
REFIID riid,
|
|
void **ppvObject)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, %s, %p)\n", This, debugstr_guid(riid), ppvObject);
|
|
|
|
if (!ppvObject)
|
|
return E_POINTER;
|
|
|
|
if (IsEqualIID(riid, &IID_IUnknown))
|
|
*ppvObject = &s->defaultExtractIconInitImpl;
|
|
else if (IsEqualIID(riid, &IID_IDefaultExtractIconInit))
|
|
*ppvObject = &s->defaultExtractIconInitImpl;
|
|
else if (IsEqualIID(riid, &IID_IExtractIconW))
|
|
*ppvObject = &s->extractIconWImpl;
|
|
else if (IsEqualIID(riid, &IID_IExtractIconA))
|
|
*ppvObject = &s->extractIconAImpl;
|
|
else if (IsEqualIID(riid, &IID_IPersist))
|
|
*ppvObject = &s->persistFileImpl;
|
|
else if (IsEqualIID(riid, &IID_IPersistFile))
|
|
*ppvObject = &s->persistFileImpl;
|
|
else
|
|
{
|
|
*ppvObject = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef(This);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_AddRef(
|
|
IDefaultExtractIconInit *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
ULONG refCount = InterlockedIncrement(&s->ref);
|
|
TRACE("(%p)\n", This);
|
|
return refCount;
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_Release(
|
|
IDefaultExtractIconInit *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
ULONG refCount;
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
refCount = InterlockedDecrement(&s->ref);
|
|
if (refCount == 0)
|
|
{
|
|
if (s->defaultIcon.file) CoTaskMemFree(s->defaultIcon.file);
|
|
if (s->normalIcon.file) CoTaskMemFree(s->normalIcon.file);
|
|
if (s->openIcon.file) CoTaskMemFree(s->openIcon.file);
|
|
if (s->shortcutIcon.file) CoTaskMemFree(s->shortcutIcon.file);
|
|
CoTaskMemFree(s);
|
|
}
|
|
|
|
return refCount;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetDefaultIcon(
|
|
IDefaultExtractIconInit *This,
|
|
LPCWSTR pszFile,
|
|
int iIcon)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, %s, %d)\n", This, debugstr_w(pszFile), iIcon);
|
|
|
|
DuplicateString(pszFile, &s->defaultIcon.file);
|
|
if (!s->defaultIcon.file)
|
|
return E_OUTOFMEMORY;
|
|
s->defaultIcon.index = iIcon;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetFlags(
|
|
IDefaultExtractIconInit *This,
|
|
UINT uFlags)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, 0x%x)\n", This, uFlags);
|
|
|
|
s->flags = uFlags;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetKey(
|
|
IDefaultExtractIconInit *This,
|
|
HKEY hkey)
|
|
{
|
|
FIXME("(%p, %p)\n", This, hkey);
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetNormalIcon(
|
|
IDefaultExtractIconInit *This,
|
|
LPCWSTR pszFile,
|
|
int iIcon)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, %s, %d)\n", This, debugstr_w(pszFile), iIcon);
|
|
|
|
DuplicateString(pszFile, &s->normalIcon.file);
|
|
if (!s->normalIcon.file)
|
|
return E_OUTOFMEMORY;
|
|
s->normalIcon.index = iIcon;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetOpenIcon(
|
|
IDefaultExtractIconInit *This,
|
|
LPCWSTR pszFile,
|
|
int iIcon)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, %s, %d)\n", This, debugstr_w(pszFile), iIcon);
|
|
|
|
DuplicateString(pszFile, &s->openIcon.file);
|
|
if (!s->openIcon.file)
|
|
return E_OUTOFMEMORY;
|
|
s->openIcon.index = iIcon;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_DefaultExtractIconInit_SetShortcutIcon(
|
|
IDefaultExtractIconInit *This,
|
|
LPCWSTR pszFile,
|
|
int iIcon)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, defaultExtractIconInitImpl);
|
|
|
|
TRACE("(%p, %s, %d)\n", This, debugstr_w(pszFile), iIcon);
|
|
|
|
DuplicateString(pszFile, &s->shortcutIcon.file);
|
|
if (!s->shortcutIcon.file)
|
|
return E_OUTOFMEMORY;
|
|
s->shortcutIcon.index = iIcon;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconW_QueryInterface(
|
|
IExtractIconW *This,
|
|
REFIID riid,
|
|
void **ppvObject)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconWImpl);
|
|
return IconExtraction_DefaultExtractIconInit_QueryInterface(&s->defaultExtractIconInitImpl, riid, ppvObject);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconW_AddRef(
|
|
IExtractIconW *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconWImpl);
|
|
return IconExtraction_DefaultExtractIconInit_AddRef(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconW_Release(
|
|
IExtractIconW *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconWImpl);
|
|
return IconExtraction_DefaultExtractIconInit_Release(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconW_GetIconLocation(
|
|
IExtractIconW *This,
|
|
UINT uFlags,
|
|
LPWSTR szIconFile,
|
|
UINT cchMax,
|
|
int *piIndex,
|
|
UINT *pwFlags)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconWImpl);
|
|
const struct IconLocation *icon = NULL;
|
|
SIZE_T cb;
|
|
|
|
TRACE("(%p, 0x%x, %s, 0x%x, %p, %p)\n", This, uFlags, debugstr_w(szIconFile), cchMax, piIndex, pwFlags);
|
|
|
|
if (!piIndex || !pwFlags)
|
|
return E_POINTER;
|
|
|
|
if (uFlags & GIL_DEFAULTICON)
|
|
icon = s->defaultIcon.file ? &s->defaultIcon : &s->normalIcon;
|
|
else if (uFlags & GIL_FORSHORTCUT)
|
|
icon = s->shortcutIcon.file ? &s->shortcutIcon : &s->normalIcon;
|
|
else if (uFlags & GIL_OPENICON)
|
|
icon = s->openIcon.file ? &s->openIcon : &s->normalIcon;
|
|
else
|
|
icon = &s->normalIcon;
|
|
|
|
if (!icon->file)
|
|
return E_FAIL;
|
|
|
|
cb = wcslen(icon->file) + 1;
|
|
if (cchMax < (UINT)cb)
|
|
return E_FAIL;
|
|
CopyMemory(szIconFile, icon->file, cb * sizeof(WCHAR));
|
|
*piIndex = icon->index;
|
|
*pwFlags = s->flags;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconW_Extract(
|
|
IExtractIconW *This,
|
|
LPCWSTR pszFile,
|
|
UINT nIconIndex,
|
|
HICON *phiconLarge,
|
|
HICON *phiconSmall,
|
|
UINT nIconSize)
|
|
{
|
|
TRACE("(%p, %s, %u, %p, %p, %u)\n", This, debugstr_w(pszFile),
|
|
nIconIndex, phiconLarge, phiconSmall, nIconSize);
|
|
|
|
/* Nothing to do, ExtractIconW::GetIconLocation should be enough */
|
|
return S_FALSE;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconA_QueryInterface(
|
|
IExtractIconA *This,
|
|
REFIID riid,
|
|
void **ppvObject)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconAImpl);
|
|
return IconExtraction_DefaultExtractIconInit_QueryInterface(&s->defaultExtractIconInitImpl, riid, ppvObject);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconA_AddRef(
|
|
IExtractIconA *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconAImpl);
|
|
return IconExtraction_DefaultExtractIconInit_AddRef(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconA_Release(
|
|
IExtractIconA *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconAImpl);
|
|
return IconExtraction_DefaultExtractIconInit_Release(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconA_GetIconLocation(
|
|
IExtractIconA *This,
|
|
UINT uFlags,
|
|
LPSTR szIconFile,
|
|
UINT cchMax,
|
|
int *piIndex,
|
|
UINT *pwFlags)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconAImpl);
|
|
LPWSTR szIconFileW = NULL;
|
|
HRESULT hr;
|
|
|
|
if (cchMax > 0)
|
|
{
|
|
szIconFileW = CoTaskMemAlloc(cchMax * sizeof(WCHAR));
|
|
if (!szIconFileW)
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = IconExtraction_ExtractIconW_GetIconLocation(
|
|
&s->extractIconWImpl, uFlags, szIconFileW, cchMax, piIndex, pwFlags);
|
|
if (SUCCEEDED(hr) && cchMax > 0)
|
|
if (0 == WideCharToMultiByte(CP_ACP, 0, szIconFileW, cchMax, szIconFile, cchMax, NULL, NULL))
|
|
hr = E_FAIL;
|
|
|
|
if (szIconFileW)
|
|
CoTaskMemFree(szIconFileW);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_ExtractIconA_Extract(
|
|
IExtractIconA *This,
|
|
LPCSTR pszFile,
|
|
UINT nIconIndex,
|
|
HICON *phiconLarge,
|
|
HICON *phiconSmall,
|
|
UINT nIconSize)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, extractIconAImpl);
|
|
LPWSTR pszFileW = NULL;
|
|
int nLength;
|
|
HRESULT hr;
|
|
|
|
if (pszFile)
|
|
{
|
|
nLength = MultiByteToWideChar(CP_ACP, 0, pszFile, -1, NULL, 0);
|
|
if (nLength == 0)
|
|
return E_FAIL;
|
|
pszFileW = CoTaskMemAlloc(nLength * sizeof(WCHAR));
|
|
if (!pszFileW)
|
|
return E_OUTOFMEMORY;
|
|
if (!MultiByteToWideChar(CP_ACP, 0, pszFile, nLength, pszFileW, nLength))
|
|
{
|
|
CoTaskMemFree(pszFileW);
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
hr = IconExtraction_ExtractIconW_Extract(
|
|
&s->extractIconWImpl, pszFileW, nIconIndex, phiconLarge, phiconSmall, nIconSize);
|
|
|
|
if (pszFileW)
|
|
CoTaskMemFree(pszFileW);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_QueryInterface(
|
|
IPersistFile *This,
|
|
REFIID riid,
|
|
void **ppvObject)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, persistFileImpl);
|
|
return IconExtraction_DefaultExtractIconInit_QueryInterface(&s->defaultExtractIconInitImpl, riid, ppvObject);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_AddRef(
|
|
IPersistFile *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, persistFileImpl);
|
|
return IconExtraction_DefaultExtractIconInit_AddRef(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_Release(
|
|
IPersistFile *This)
|
|
{
|
|
struct IconExtraction *s = CONTAINING_RECORD(This, struct IconExtraction, persistFileImpl);
|
|
return IconExtraction_DefaultExtractIconInit_Release(&s->defaultExtractIconInitImpl);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_GetClassID(
|
|
IPersistFile *This,
|
|
CLSID *pClassID)
|
|
{
|
|
TRACE("(%p, %p)\n", This, pClassID);
|
|
|
|
if (!pClassID)
|
|
return E_POINTER;
|
|
|
|
*pClassID = GUID_NULL;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_IsDirty(
|
|
IPersistFile *This)
|
|
{
|
|
FIXME("(%p)\n", This);
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_Load(
|
|
IPersistFile *This,
|
|
LPCOLESTR pszFileName,
|
|
DWORD dwMode)
|
|
{
|
|
FIXME("(%p, %s, %u)\n", This, debugstr_w(pszFileName), dwMode);
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_Save(
|
|
IPersistFile *This,
|
|
LPCOLESTR pszFileName,
|
|
BOOL fRemember)
|
|
{
|
|
FIXME("(%p, %s, %d)\n", This, debugstr_w(pszFileName), fRemember);
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_SaveCompleted(
|
|
IPersistFile *This,
|
|
LPCOLESTR pszFileName)
|
|
{
|
|
FIXME("(%p, %s)\n", This, debugstr_w(pszFileName));
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
IconExtraction_PersistFile_GetCurFile(
|
|
IPersistFile *This,
|
|
LPOLESTR *ppszFileName)
|
|
{
|
|
FIXME("(%p, %p)\n", This, ppszFileName);
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IDefaultExtractIconInitVtbl IconExtractionDefaultExtractIconInitVtbl =
|
|
{
|
|
IconExtraction_DefaultExtractIconInit_QueryInterface,
|
|
IconExtraction_DefaultExtractIconInit_AddRef,
|
|
IconExtraction_DefaultExtractIconInit_Release,
|
|
IconExtraction_DefaultExtractIconInit_SetDefaultIcon,
|
|
IconExtraction_DefaultExtractIconInit_SetFlags,
|
|
IconExtraction_DefaultExtractIconInit_SetKey,
|
|
IconExtraction_DefaultExtractIconInit_SetNormalIcon,
|
|
IconExtraction_DefaultExtractIconInit_SetOpenIcon,
|
|
IconExtraction_DefaultExtractIconInit_SetShortcutIcon,
|
|
};
|
|
|
|
static const IExtractIconWVtbl IconExtractionExtractIconWVtbl =
|
|
{
|
|
IconExtraction_ExtractIconW_QueryInterface,
|
|
IconExtraction_ExtractIconW_AddRef,
|
|
IconExtraction_ExtractIconW_Release,
|
|
IconExtraction_ExtractIconW_GetIconLocation,
|
|
IconExtraction_ExtractIconW_Extract,
|
|
};
|
|
|
|
static const IExtractIconAVtbl IconExtractionExtractIconAVtbl =
|
|
{
|
|
IconExtraction_ExtractIconA_QueryInterface,
|
|
IconExtraction_ExtractIconA_AddRef,
|
|
IconExtraction_ExtractIconA_Release,
|
|
IconExtraction_ExtractIconA_GetIconLocation,
|
|
IconExtraction_ExtractIconA_Extract,
|
|
};
|
|
|
|
static const IPersistFileVtbl IconExtractionPersistFileVtbl =
|
|
{
|
|
IconExtraction_PersistFile_QueryInterface,
|
|
IconExtraction_PersistFile_AddRef,
|
|
IconExtraction_PersistFile_Release,
|
|
IconExtraction_PersistFile_GetClassID,
|
|
IconExtraction_PersistFile_IsDirty,
|
|
IconExtraction_PersistFile_Load,
|
|
IconExtraction_PersistFile_Save,
|
|
IconExtraction_PersistFile_SaveCompleted,
|
|
IconExtraction_PersistFile_GetCurFile,
|
|
};
|
|
|
|
HRESULT WINAPI
|
|
SHCreateDefaultExtractIcon(
|
|
REFIID riid,
|
|
void **ppv)
|
|
{
|
|
struct IconExtraction *s;
|
|
|
|
if (!ppv)
|
|
return E_POINTER;
|
|
|
|
*ppv = NULL;
|
|
|
|
s = CoTaskMemAlloc(sizeof(struct IconExtraction));
|
|
if (!s)
|
|
return E_OUTOFMEMORY;
|
|
memset(s, 0, sizeof(struct IconExtraction));
|
|
s->defaultExtractIconInitImpl.lpVtbl = (IDefaultExtractIconInitVtbl*)&IconExtractionDefaultExtractIconInitVtbl;
|
|
s->extractIconAImpl.lpVtbl = (IExtractIconAVtbl*)&IconExtractionExtractIconAVtbl;
|
|
s->extractIconWImpl.lpVtbl = (IExtractIconWVtbl*)&IconExtractionExtractIconWVtbl;
|
|
s->persistFileImpl.lpVtbl = (IPersistFileVtbl*)&IconExtractionPersistFileVtbl;
|
|
s->ref = 1;
|
|
*ppv = &s->defaultExtractIconInitImpl;
|
|
|
|
return S_OK;
|
|
}
|