/* * IShellDispatch implementation * * Copyright 2010 Alexander Morozov for Etersoft * * 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 "config.h" #include "wine/port.h" #include #define COBJMACROS #include "windef.h" #include "winbase.h" #include "winreg.h" #include "winsvc.h" #include "shlwapi.h" #include "shlobj.h" #include "shldisp.h" #include "debughlp.h" #include "shell32_main.h" #include "pidl.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(shell); static ITypeLib *typelib; static const IID * const tid_ids[] = { &IID_NULL, &IID_IShellDispatch6, &IID_IShellFolderViewDual3, &IID_Folder3, &IID_FolderItem2, &IID_FolderItems3, &IID_FolderItemVerb, &IID_FolderItemVerbs }; static ITypeInfo *typeinfos[LAST_tid]; typedef struct { IShellDispatch6 IShellDispatch6_iface; LONG ref; } ShellDispatch; typedef struct { Folder3 Folder3_iface; LONG ref; IDispatch *application; IShellFolder2 *folder; PIDLIST_ABSOLUTE pidl; BSTR path; } FolderImpl; typedef struct { FolderItems3 FolderItems3_iface; LONG ref; FolderImpl *folder; BSTR *item_names; LONG item_count; } FolderItemsImpl; typedef struct { FolderItem2 FolderItem2_iface; LONG ref; FolderImpl *folder; WCHAR *path; /* if NULL, folder path is used */ DWORD attributes; } FolderItemImpl; typedef struct { FolderItemVerbs FolderItemVerbs_iface; LONG ref; IContextMenu *contextmenu; HMENU hMenu; LONG count; } FolderItemVerbsImpl; typedef struct { FolderItemVerb FolderItemVerb_iface; LONG ref; IContextMenu *contextmenu; BSTR name; } FolderItemVerbImpl; static inline ShellDispatch *impl_from_IShellDispatch6(IShellDispatch6 *iface) { return CONTAINING_RECORD(iface, ShellDispatch, IShellDispatch6_iface); } static inline FolderImpl *impl_from_Folder(Folder3 *iface) { return CONTAINING_RECORD(iface, FolderImpl, Folder3_iface); } static inline FolderItemsImpl *impl_from_FolderItems(FolderItems3 *iface) { return CONTAINING_RECORD(iface, FolderItemsImpl, FolderItems3_iface); } static inline FolderItemImpl *impl_from_FolderItem(FolderItem2 *iface) { return CONTAINING_RECORD(iface, FolderItemImpl, FolderItem2_iface); } static inline FolderItemVerbsImpl *impl_from_FolderItemVerbs(FolderItemVerbs *iface) { return CONTAINING_RECORD(iface, FolderItemVerbsImpl, FolderItemVerbs_iface); } static inline FolderItemVerbImpl *impl_from_FolderItemVerb(FolderItemVerb *iface) { return CONTAINING_RECORD(iface, FolderItemVerbImpl, FolderItemVerb_iface); } static HRESULT load_typelib(void) { ITypeLib *tl; HRESULT hr; hr = LoadRegTypeLib(&LIBID_Shell32, 1, 0, LOCALE_SYSTEM_DEFAULT, &tl); if (FAILED(hr)) { ERR("LoadRegTypeLib failed: %08x\n", hr); return hr; } if (InterlockedCompareExchangePointer((void**)&typelib, tl, NULL)) ITypeLib_Release(tl); return hr; } void release_typelib(void) { unsigned i; if (!typelib) return; for (i = 0; i < ARRAY_SIZE(typeinfos); i++) if (typeinfos[i]) ITypeInfo_Release(typeinfos[i]); ITypeLib_Release(typelib); } HRESULT get_typeinfo(enum tid_t tid, ITypeInfo **typeinfo) { HRESULT hr; if (!typelib) hr = load_typelib(); if (!typelib) return hr; if (!typeinfos[tid]) { ITypeInfo *ti; hr = ITypeLib_GetTypeInfoOfGuid(typelib, tid_ids[tid], &ti); if (FAILED(hr)) { ERR("GetTypeInfoOfGuid(%s) failed: %08x\n", debugstr_guid(tid_ids[tid]), hr); return hr; } if (InterlockedCompareExchangePointer((void**)(typeinfos+tid), ti, NULL)) ITypeInfo_Release(ti); } *typeinfo = typeinfos[tid]; return S_OK; } /* FolderItemVerb */ static HRESULT WINAPI FolderItemVerbImpl_QueryInterface(FolderItemVerb *iface, REFIID riid, void **ppv) { FolderItemVerbImpl *This = impl_from_FolderItemVerb(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_FolderItemVerb, riid)) *ppv = &This->FolderItemVerb_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FolderItemVerbImpl_AddRef(FolderItemVerb *iface) { FolderItemVerbImpl *This = impl_from_FolderItemVerb(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI FolderItemVerbImpl_Release(FolderItemVerb *iface) { FolderItemVerbImpl *This = impl_from_FolderItemVerb(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) { IContextMenu_Release(This->contextmenu); SysFreeString(This->name); heap_free(This); } return ref; } static HRESULT WINAPI FolderItemVerbImpl_GetTypeInfoCount(FolderItemVerb *iface, UINT *pctinfo) { TRACE("(%p,%p)\n", iface, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI FolderItemVerbImpl_GetTypeInfo(FolderItemVerb *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, iTInfo, lcid, ppTInfo); hr = get_typeinfo(FolderItemVerb_tid, ppTInfo); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppTInfo); return hr; } static HRESULT WINAPI FolderItemVerbImpl_GetIDsOfNames(FolderItemVerb *iface, REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); hr = get_typeinfo(FolderItemVerb_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, rgszNames, cNames, rgDispId); return hr; } static HRESULT WINAPI FolderItemVerbImpl_Invoke(FolderItemVerb *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispIdMember, shdebugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(FolderItemVerb_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, iface, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); return hr; } static HRESULT WINAPI FolderItemVerbImpl_get_Application(FolderItemVerb *iface, IDispatch **disp) { TRACE("(%p, %p)\n", iface, disp); if (disp) *disp = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemVerbImpl_get_Parent(FolderItemVerb *iface, IDispatch **disp) { TRACE("(%p, %p)\n", iface, disp); if (disp) *disp = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemVerbImpl_get_Name(FolderItemVerb *iface, BSTR *name) { FolderItemVerbImpl *This = impl_from_FolderItemVerb(iface); TRACE("(%p, %p)\n", iface, name); *name = SysAllocString(This->name); return *name ? S_OK : E_OUTOFMEMORY; } static HRESULT WINAPI FolderItemVerbImpl_DoIt(FolderItemVerb *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static FolderItemVerbVtbl folderitemverbvtbl = { FolderItemVerbImpl_QueryInterface, FolderItemVerbImpl_AddRef, FolderItemVerbImpl_Release, FolderItemVerbImpl_GetTypeInfoCount, FolderItemVerbImpl_GetTypeInfo, FolderItemVerbImpl_GetIDsOfNames, FolderItemVerbImpl_Invoke, FolderItemVerbImpl_get_Application, FolderItemVerbImpl_get_Parent, FolderItemVerbImpl_get_Name, FolderItemVerbImpl_DoIt }; static HRESULT FolderItemVerb_Constructor(IContextMenu *contextmenu, BSTR name, FolderItemVerb **verb) { FolderItemVerbImpl *This; TRACE("%p, %s\n", contextmenu, debugstr_w(name)); This = heap_alloc(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->FolderItemVerb_iface.lpVtbl = &folderitemverbvtbl; This->ref = 1; This->contextmenu = contextmenu; IContextMenu_AddRef(contextmenu); This->name = name; *verb = &This->FolderItemVerb_iface; return S_OK; } /* FolderItemVerbs */ static HRESULT WINAPI FolderItemVerbsImpl_QueryInterface(FolderItemVerbs *iface, REFIID riid, void **ppv) { FolderItemVerbsImpl *This = impl_from_FolderItemVerbs(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_FolderItemVerbs, riid)) *ppv = &This->FolderItemVerbs_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FolderItemVerbsImpl_AddRef(FolderItemVerbs *iface) { FolderItemVerbsImpl *This = impl_from_FolderItemVerbs(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI FolderItemVerbsImpl_Release(FolderItemVerbs *iface) { FolderItemVerbsImpl *This = impl_from_FolderItemVerbs(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) { IContextMenu_Release(This->contextmenu); DestroyMenu(This->hMenu); heap_free(This); } return ref; } static HRESULT WINAPI FolderItemVerbsImpl_GetTypeInfoCount(FolderItemVerbs *iface, UINT *pctinfo) { TRACE("(%p,%p)\n", iface, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI FolderItemVerbsImpl_GetTypeInfo(FolderItemVerbs *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, iTInfo, lcid, ppTInfo); hr = get_typeinfo(FolderItemVerbs_tid, ppTInfo); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppTInfo); return hr; } static HRESULT WINAPI FolderItemVerbsImpl_GetIDsOfNames(FolderItemVerbs *iface, REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); hr = get_typeinfo(FolderItemVerbs_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, rgszNames, cNames, rgDispId); return hr; } static HRESULT WINAPI FolderItemVerbsImpl_Invoke(FolderItemVerbs *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispIdMember, shdebugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(FolderItemVerbs_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, iface, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); return hr; } static HRESULT WINAPI FolderItemVerbsImpl_get_Count(FolderItemVerbs *iface, LONG *count) { FolderItemVerbsImpl *This = impl_from_FolderItemVerbs(iface); TRACE("(%p, %p)\n", iface, count); if (!count) return E_INVALIDARG; *count = This->count; return S_OK; } static HRESULT WINAPI FolderItemVerbsImpl_get_Application(FolderItemVerbs *iface, IDispatch **disp) { TRACE("(%p, %p)\n", iface, disp); if (disp) *disp = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemVerbsImpl_get_Parent(FolderItemVerbs *iface, IDispatch **disp) { TRACE("(%p, %p)\n", iface, disp); if (disp) *disp = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemVerbsImpl_Item(FolderItemVerbs *iface, VARIANT index, FolderItemVerb **verb) { FolderItemVerbsImpl *This = impl_from_FolderItemVerbs(iface); MENUITEMINFOW info; HRESULT hr; VARIANT v; BSTR name; TRACE("(%p, %s, %p)\n", iface, debugstr_variant(&index), verb); if (!verb) return E_INVALIDARG; *verb = NULL; VariantInit(&v); VariantCopyInd(&v, &index); hr = VariantChangeType(&v, &v, 0, VT_I4); if (FAILED(hr)) { FIXME("failed to coerce to VT_I4, %s\n", debugstr_variant(&v)); return hr; } if (V_I4(&v) > This->count) return S_OK; if (V_I4(&v) == This->count) name = SysAllocStringLen(NULL, 0); else { /* get item name */ memset(&info, 0, sizeof(info)); info.cbSize = sizeof(info); info.fMask = MIIM_STRING; if (!GetMenuItemInfoW(This->hMenu, V_I4(&v), TRUE, &info)) return E_FAIL; name = SysAllocStringLen(NULL, info.cch); if (name) { info.dwTypeData = name; info.cch++; GetMenuItemInfoW(This->hMenu, V_I4(&v), TRUE, &info); } } if (!name) return E_OUTOFMEMORY; return FolderItemVerb_Constructor(This->contextmenu, name, verb); } static HRESULT WINAPI FolderItemVerbsImpl__NewEnum(FolderItemVerbs *iface, IUnknown **ret) { FIXME("(%p, %p)\n", iface, ret); return E_NOTIMPL; } static FolderItemVerbsVtbl folderitemverbsvtbl = { FolderItemVerbsImpl_QueryInterface, FolderItemVerbsImpl_AddRef, FolderItemVerbsImpl_Release, FolderItemVerbsImpl_GetTypeInfoCount, FolderItemVerbsImpl_GetTypeInfo, FolderItemVerbsImpl_GetIDsOfNames, FolderItemVerbsImpl_Invoke, FolderItemVerbsImpl_get_Count, FolderItemVerbsImpl_get_Application, FolderItemVerbsImpl_get_Parent, FolderItemVerbsImpl_Item, FolderItemVerbsImpl__NewEnum }; static HRESULT FolderItemVerbs_Constructor(BSTR path, FolderItemVerbs **verbs) { FolderItemVerbsImpl *This; IShellFolder *folder; LPCITEMIDLIST child; LPITEMIDLIST pidl; HRESULT hr; *verbs = NULL; This = heap_alloc(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->FolderItemVerbs_iface.lpVtbl = &folderitemverbsvtbl; This->ref = 1; /* build context menu for this path */ hr = SHParseDisplayName(path, NULL, &pidl, 0, NULL); if (FAILED(hr)) goto failed; hr = SHBindToParent(pidl, &IID_IShellFolder, (void**)&folder, &child); CoTaskMemFree(pidl); if (FAILED(hr)) goto failed; hr = IShellFolder_GetUIObjectOf(folder, NULL, 1, &child, &IID_IContextMenu, NULL, (void**)&This->contextmenu); IShellFolder_Release(folder); if (FAILED(hr)) goto failed; This->hMenu = CreatePopupMenu(); hr = IContextMenu_QueryContextMenu(This->contextmenu, This->hMenu, 0, FCIDM_SHVIEWFIRST, FCIDM_SHVIEWLAST, CMF_NORMAL); if (FAILED(hr)) { FolderItemVerbs_Release(&This->FolderItemVerbs_iface); return hr; } This->count = GetMenuItemCount(This->hMenu); *verbs = &This->FolderItemVerbs_iface; return S_OK; failed: heap_free(This); return hr; } static HRESULT WINAPI FolderItemImpl_QueryInterface(FolderItem2 *iface, REFIID riid, LPVOID *ppv) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_FolderItem, riid) || IsEqualIID(&IID_FolderItem2, riid)) *ppv = &This->FolderItem2_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FolderItemImpl_AddRef(FolderItem2 *iface) { FolderItemImpl *This = impl_from_FolderItem(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI FolderItemImpl_Release(FolderItem2 *iface) { FolderItemImpl *This = impl_from_FolderItem(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) { Folder3_Release(&This->folder->Folder3_iface); heap_free(This->path); heap_free(This); } return ref; } static HRESULT WINAPI FolderItemImpl_GetTypeInfoCount(FolderItem2 *iface, UINT *pctinfo) { TRACE("(%p,%p)\n", iface, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI FolderItemImpl_GetTypeInfo(FolderItem2 *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, iTInfo, lcid, ppTInfo); hr = get_typeinfo(FolderItem2_tid, ppTInfo); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppTInfo); return hr; } static HRESULT WINAPI FolderItemImpl_GetIDsOfNames(FolderItem2 *iface, REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); hr = get_typeinfo(FolderItem2_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, rgszNames, cNames, rgDispId); return hr; } static HRESULT WINAPI FolderItemImpl_Invoke(FolderItem2 *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) { FolderItemImpl *This = impl_from_FolderItem(iface); ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispIdMember, shdebugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(FolderItem2_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, This, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); return hr; } static HRESULT WINAPI FolderItemImpl_get_Application(FolderItem2 *iface, IDispatch **disp) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, disp); return Folder3_get_Application(&This->folder->Folder3_iface, disp); } static HRESULT WINAPI FolderItemImpl_get_Parent(FolderItem2 *iface, IDispatch **disp) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, disp); if (disp) { *disp = (IDispatch *)&This->folder->Folder3_iface; IDispatch_AddRef(*disp); } return S_OK; } static HRESULT WINAPI FolderItemImpl_get_Name(FolderItem2 *iface, BSTR *name) { FolderItemImpl *This = impl_from_FolderItem(iface); LPCITEMIDLIST last_part; IShellFolder2 *parent; HRESULT hr = S_OK; LPITEMIDLIST pidl; STRRET strret; TRACE("(%p,%p)\n", iface, name); *name = NULL; if (This->path) hr = SHParseDisplayName(This->path, NULL, &pidl, 0, NULL); else pidl = This->folder->pidl; if (FAILED(hr)) return S_FALSE; hr = SHBindToParent(pidl, &IID_IShellFolder2, (void **)&parent, &last_part); if (hr == S_OK) hr = IShellFolder2_GetDisplayNameOf(parent, last_part, SHGDN_INFOLDER, &strret); IShellFolder2_Release(parent); if (hr == S_OK) hr = StrRetToBSTR(&strret, last_part, name); if (This->path) ILFree(pidl); return hr; } static HRESULT WINAPI FolderItemImpl_put_Name(FolderItem2 *iface, BSTR bs) { FIXME("(%p,%s)\n", iface, debugstr_w(bs)); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_get_Path(FolderItem2 *iface, BSTR *path) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, path); *path = SysAllocString(This->path ? This->path : This->folder->path); return *path ? S_OK : E_OUTOFMEMORY; } static HRESULT WINAPI FolderItemImpl_get_GetLink(FolderItem2 *iface, IDispatch **ppid) { FIXME("(%p,%p)\n", iface, ppid); *ppid = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_get_GetFolder(FolderItem2 *iface, IDispatch **ppid) { FIXME("(%p,%p)\n", iface, ppid); *ppid = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_get_IsLink(FolderItem2 *iface, VARIANT_BOOL *b) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, b); *b = This->attributes & SFGAO_LINK ? VARIANT_TRUE : VARIANT_FALSE; return S_OK; } static HRESULT WINAPI FolderItemImpl_get_IsFolder(FolderItem2 *iface, VARIANT_BOOL *b) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, b); *b = This->attributes & SFGAO_FOLDER ? VARIANT_TRUE : VARIANT_FALSE; return S_OK; } static HRESULT WINAPI FolderItemImpl_get_IsFileSystem(FolderItem2 *iface, VARIANT_BOOL *b) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, b); *b = This->attributes & SFGAO_FILESYSTEM ? VARIANT_TRUE : VARIANT_FALSE; return S_OK; } static HRESULT WINAPI FolderItemImpl_get_IsBrowsable(FolderItem2 *iface, VARIANT_BOOL *b) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p,%p)\n", iface, b); *b = This->attributes & SFGAO_BROWSABLE ? VARIANT_TRUE : VARIANT_FALSE; return S_OK; } static HRESULT WINAPI FolderItemImpl_get_ModifyDate(FolderItem2 *iface, DATE *pdt) { FIXME("(%p,%p)\n", iface, pdt); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_put_ModifyDate(FolderItem2 *iface, DATE dt) { FIXME("(%p,%f)\n", iface, dt); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_get_Size(FolderItem2 *iface, LONG *pul) { FIXME("(%p,%p)\n", iface, pul); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_get_Type(FolderItem2 *iface, BSTR *pbs) { FIXME("(%p,%p)\n", iface, pbs); *pbs = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_Verbs(FolderItem2 *iface, FolderItemVerbs **verbs) { FolderItemImpl *This = impl_from_FolderItem(iface); TRACE("(%p, %p)\n", iface, verbs); if (!verbs) return E_INVALIDARG; return FolderItemVerbs_Constructor(This->path ? This->path : This->folder->path, verbs); } static HRESULT WINAPI FolderItemImpl_InvokeVerb(FolderItem2 *iface, VARIANT vVerb) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_InvokeVerbEx(FolderItem2 *iface, VARIANT verb, VARIANT args) { FIXME("(%p): stub\n", iface); return E_NOTIMPL; } static HRESULT WINAPI FolderItemImpl_ExtendedProperty(FolderItem2 *iface, BSTR propname, VARIANT *ret) { FIXME("(%p)->(%s %p): stub\n", iface, debugstr_w(propname), ret); return E_NOTIMPL; } static const FolderItem2Vtbl FolderItemImpl_Vtbl = { FolderItemImpl_QueryInterface, FolderItemImpl_AddRef, FolderItemImpl_Release, FolderItemImpl_GetTypeInfoCount, FolderItemImpl_GetTypeInfo, FolderItemImpl_GetIDsOfNames, FolderItemImpl_Invoke, FolderItemImpl_get_Application, FolderItemImpl_get_Parent, FolderItemImpl_get_Name, FolderItemImpl_put_Name, FolderItemImpl_get_Path, FolderItemImpl_get_GetLink, FolderItemImpl_get_GetFolder, FolderItemImpl_get_IsLink, FolderItemImpl_get_IsFolder, FolderItemImpl_get_IsFileSystem, FolderItemImpl_get_IsBrowsable, FolderItemImpl_get_ModifyDate, FolderItemImpl_put_ModifyDate, FolderItemImpl_get_Size, FolderItemImpl_get_Type, FolderItemImpl_Verbs, FolderItemImpl_InvokeVerb, FolderItemImpl_InvokeVerbEx, FolderItemImpl_ExtendedProperty }; static HRESULT FolderItem_Constructor(FolderImpl *folder, const WCHAR *path, FolderItem **item) { PIDLIST_ABSOLUTE pidl; FolderItemImpl *This; TRACE("%s\n", debugstr_w(path)); *item = NULL; This = heap_alloc_zero(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->FolderItem2_iface.lpVtbl = &FolderItemImpl_Vtbl; This->ref = 1; if (path) This->path = strdupW(path); This->folder = folder; Folder3_AddRef(&folder->Folder3_iface); if (SHParseDisplayName(This->path, NULL, &pidl, ~0u, &This->attributes) == S_OK) ILFree(pidl); *item = (FolderItem *)&This->FolderItem2_iface; return S_OK; } static HRESULT WINAPI FolderItemsImpl_QueryInterface(FolderItems3 *iface, REFIID riid, LPVOID *ppv) { FolderItemsImpl *This = impl_from_FolderItems(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_FolderItems, riid) || IsEqualIID(&IID_FolderItems2, riid) || IsEqualIID(&IID_FolderItems3, riid)) *ppv = &This->FolderItems3_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FolderItemsImpl_AddRef(FolderItems3 *iface) { FolderItemsImpl *This = impl_from_FolderItems(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI FolderItemsImpl_Release(FolderItems3 *iface) { FolderItemsImpl *This = impl_from_FolderItems(iface); ULONG ref = InterlockedDecrement(&This->ref); LONG i; TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) { Folder3_Release(&This->folder->Folder3_iface); for (i = 0; i < This->item_count; i++) SysFreeString(This->item_names[i]); heap_free(This->item_names); heap_free(This); } return ref; } static HRESULT WINAPI FolderItemsImpl_GetTypeInfoCount(FolderItems3 *iface, UINT *count) { TRACE("(%p,%p)\n", iface, count); *count = 1; return S_OK; } static HRESULT WINAPI FolderItemsImpl_GetTypeInfo(FolderItems3 *iface, UINT type, LCID lcid, ITypeInfo **ppti) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, type, lcid, ppti); hr = get_typeinfo(FolderItems3_tid, ppti); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppti); return hr; } static HRESULT WINAPI FolderItemsImpl_GetIDsOfNames(FolderItems3 *iface, REFIID riid, LPOLESTR *names, UINT count, LCID lcid, DISPID *dispid) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), names, count, lcid, dispid); hr = get_typeinfo(FolderItems3_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, names, count, dispid); return hr; } static HRESULT WINAPI FolderItemsImpl_Invoke(FolderItems3 *iface, DISPID dispid, REFIID riid, LCID lcid, WORD flags, DISPPARAMS *params, VARIANT *result, EXCEPINFO *ei, UINT *err) { FolderItemsImpl *This = impl_from_FolderItems(iface); ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispid, shdebugstr_guid(riid), lcid, flags, params, result, ei, err); hr = get_typeinfo(FolderItems3_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, This, dispid, flags, params, result, ei, err); return hr; } static BOOL shellfolder_exists(const WCHAR *path) { PIDLIST_ABSOLUTE pidl = NULL; HRESULT hr; hr = SHParseDisplayName(path, NULL, &pidl, 0, NULL); ILFree(pidl); return SUCCEEDED(hr); } static HRESULT WINAPI FolderItemsImpl_get_Count(FolderItems3 *iface, LONG *count) { FolderItemsImpl *This = impl_from_FolderItems(iface); TRACE("(%p,%p)\n", iface, count); *count = shellfolder_exists(This->folder->path) ? This->item_count : 0; return S_OK; } static HRESULT WINAPI FolderItemsImpl_get_Application(FolderItems3 *iface, IDispatch **disp) { FolderItemsImpl *This = impl_from_FolderItems(iface); TRACE("(%p,%p)\n", iface, disp); return Folder3_get_Application(&This->folder->Folder3_iface, disp); } static HRESULT WINAPI FolderItemsImpl_get_Parent(FolderItems3 *iface, IDispatch **ppid) { TRACE("(%p,%p)\n", iface, ppid); if (ppid) *ppid = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemsImpl_Item(FolderItems3 *iface, VARIANT var, FolderItem **item) { FolderItemsImpl *This = impl_from_FolderItems(iface); BSTR display_name = NULL; VARIANT index; HRESULT hr; TRACE("(%p,%s,%p)\n", iface, debugstr_variant(&var), item); *item = NULL; if (!shellfolder_exists(This->folder->path)) return S_FALSE; VariantInit(&index); if (FAILED(hr = VariantCopyInd(&index, &var))) return hr; switch (V_VT(&index)) { case VT_I2: VariantChangeType(&index, &index, 0, VT_I4); /* fall through */ case VT_I4: if (V_I4(&index) >= This->item_count || V_I4(&index) < 0) return S_FALSE; display_name = SysAllocString(This->item_names[V_I4(&index)]); break; case VT_BSTR: { LPITEMIDLIST pidl; STRRET strret; if (!V_BSTR(&index)) return S_FALSE; hr = IShellFolder2_ParseDisplayName(This->folder->folder, NULL, NULL, V_BSTR(&index), NULL, &pidl, NULL); VariantClear(&index); if (FAILED(hr)) return S_FALSE; if (IShellFolder2_GetDisplayNameOf(This->folder->folder, pidl, SHGDN_FORPARSING, &strret) == S_OK) StrRetToBSTR(&strret, pidl, &display_name); ILFree(pidl); break; } case VT_ERROR: break; default: FIXME("Index type %#x not handled.\n", V_VT(&index)); VariantClear(&index); /* fall through */ case VT_EMPTY: return E_NOTIMPL; } hr = FolderItem_Constructor(This->folder, display_name, item); SysFreeString(display_name); return hr; } static HRESULT WINAPI FolderItemsImpl__NewEnum(FolderItems3 *iface, IUnknown **ppunk) { FIXME("(%p,%p)\n", iface, ppunk); if (!ppunk) return E_INVALIDARG; *ppunk = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderItemsImpl_InvokeVerbEx(FolderItems3 *iface, VARIANT verb, VARIANT args) { FIXME("(%p,%s,%s)\n", iface, debugstr_variant(&verb), debugstr_variant(&args)); return E_NOTIMPL; } static HRESULT WINAPI FolderItemsImpl_Filter(FolderItems3 *iface, LONG flags, BSTR spec) { FIXME("(%p,%d,%s)\n", iface, flags, wine_dbgstr_w(spec)); return E_NOTIMPL; } static HRESULT WINAPI FolderItemsImpl_get_Verbs(FolderItems3 *iface, FolderItemVerbs **ppfic) { FIXME("(%p,%p)\n", iface, ppfic); if (!ppfic) return E_INVALIDARG; *ppfic = NULL; return E_NOTIMPL; } static const FolderItems3Vtbl FolderItemsImpl_Vtbl = { FolderItemsImpl_QueryInterface, FolderItemsImpl_AddRef, FolderItemsImpl_Release, FolderItemsImpl_GetTypeInfoCount, FolderItemsImpl_GetTypeInfo, FolderItemsImpl_GetIDsOfNames, FolderItemsImpl_Invoke, FolderItemsImpl_get_Count, FolderItemsImpl_get_Application, FolderItemsImpl_get_Parent, FolderItemsImpl_Item, FolderItemsImpl__NewEnum, FolderItemsImpl_InvokeVerbEx, FolderItemsImpl_Filter, FolderItemsImpl_get_Verbs }; static void idlist_sort(LPITEMIDLIST *idlist, unsigned int l, unsigned int r, IShellFolder2 *folder) { unsigned int m; if (l == r) return; if (r < l) { idlist_sort(idlist, r, l, folder); return; } m = (l + r) / 2; idlist_sort(idlist, l, m, folder); idlist_sort(idlist, m + 1, r, folder); /* join the two sides */ while (l <= m && m < r) { if ((short)IShellFolder2_CompareIDs(folder, 0, idlist[l], idlist[m + 1]) > 0) { LPITEMIDLIST t = idlist[m + 1]; memmove(&idlist[l + 1], &idlist[l], (m - l + 1) * sizeof(idlist[l])); idlist[l] = t; m++; } l++; } } static HRESULT FolderItems_Constructor(FolderImpl *folder, FolderItems **ret) { IEnumIDList *enumidlist; FolderItemsImpl *This; LPITEMIDLIST pidl; unsigned int i; HRESULT hr; TRACE("(%s,%p)\n", debugstr_w(folder->path), ret); *ret = NULL; This = heap_alloc_zero(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->FolderItems3_iface.lpVtbl = &FolderItemsImpl_Vtbl; This->ref = 1; This->folder = folder; Folder3_AddRef(&folder->Folder3_iface); enumidlist = NULL; if (FAILED(hr = IShellFolder2_EnumObjects(folder->folder, NULL, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS, &enumidlist))) { goto failed; } while (IEnumIDList_Next(enumidlist, 1, &pidl, NULL) == S_OK) { This->item_count++; ILFree(pidl); } if (This->item_count) { LPITEMIDLIST *pidls; ULONG fetched; pidls = heap_alloc(This->item_count * sizeof(*pidls)); This->item_names = heap_alloc_zero(This->item_count * sizeof(*This->item_names)); if (!pidls || !This->item_names) { heap_free(pidls); heap_free(This->item_names); hr = E_OUTOFMEMORY; goto failed; } IEnumIDList_Reset(enumidlist); if (IEnumIDList_Next(enumidlist, This->item_count, pidls, &fetched) == S_OK) idlist_sort(pidls, 0, This->item_count - 1, folder->folder); for (i = 0; i < This->item_count; i++) { STRRET strret; if (IShellFolder2_GetDisplayNameOf(folder->folder, pidls[i], SHGDN_FORPARSING, &strret) == S_OK) StrRetToBSTR(&strret, pidls[i], &This->item_names[i]); ILFree(pidls[i]); } heap_free(pidls); } IEnumIDList_Release(enumidlist); *ret = (FolderItems *)&This->FolderItems3_iface; return S_OK; failed: if (enumidlist) IEnumIDList_Release(enumidlist); return hr; } static HRESULT WINAPI FolderImpl_QueryInterface(Folder3 *iface, REFIID riid, LPVOID *ppv) { FolderImpl *This = impl_from_Folder(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_Folder, riid) || IsEqualIID(&IID_Folder2, riid) || IsEqualIID(&IID_Folder3, riid)) *ppv = &This->Folder3_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FolderImpl_AddRef(Folder3 *iface) { FolderImpl *This = impl_from_Folder(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI FolderImpl_Release(Folder3 *iface) { FolderImpl *This = impl_from_Folder(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) { ILFree(This->pidl); SysFreeString(This->path); IShellFolder2_Release(This->folder); IDispatch_Release(This->application); heap_free(This); } return ref; } static HRESULT WINAPI FolderImpl_GetTypeInfoCount(Folder3 *iface, UINT *pctinfo) { TRACE("(%p,%p)\n", iface, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI FolderImpl_GetTypeInfo(Folder3 *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, iTInfo, lcid, ppTInfo); hr = get_typeinfo(Folder3_tid, ppTInfo); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppTInfo); return hr; } static HRESULT WINAPI FolderImpl_GetIDsOfNames(Folder3 *iface, REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); hr = get_typeinfo(Folder3_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, rgszNames, cNames, rgDispId); return hr; } static HRESULT WINAPI FolderImpl_Invoke(Folder3 *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) { FolderImpl *This = impl_from_Folder(iface); ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispIdMember, shdebugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(Folder3_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, This, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); return hr; } static HRESULT WINAPI FolderImpl_get_Title(Folder3 *iface, BSTR *title) { FolderImpl *This = impl_from_Folder(iface); PCUITEMID_CHILD last_part; IShellFolder2 *parent; SHELLDETAILS sd; HRESULT hr; TRACE("(%p,%p)\n", iface, title); *title = NULL; if (FAILED(hr = SHBindToParent(This->pidl, &IID_IShellFolder2, (void **)&parent, &last_part))) return hr; hr = IShellFolder2_GetDetailsOf(parent, last_part, 0, &sd); IShellFolder2_Release(parent); if (FAILED(hr)) return hr; return StrRetToBSTR(&sd.str, last_part, title); } static HRESULT WINAPI FolderImpl_get_Application(Folder3 *iface, IDispatch **disp) { FolderImpl *This = impl_from_Folder(iface); TRACE("(%p,%p)\n", iface, disp); if (!disp) return E_INVALIDARG; *disp = This->application; IDispatch_AddRef(*disp); return S_OK; } static HRESULT WINAPI FolderImpl_get_Parent(Folder3 *iface, IDispatch **disp) { TRACE("(%p,%p)\n", iface, disp); if (disp) *disp = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_get_ParentFolder(Folder3 *iface, Folder **ppsf) { FIXME("(%p,%p)\n", iface, ppsf); *ppsf = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_Items(Folder3 *iface, FolderItems **ppid) { FolderImpl *This = impl_from_Folder(iface); TRACE("(%p,%p)\n", iface, ppid); return FolderItems_Constructor(This, ppid); } static HRESULT WINAPI FolderImpl_ParseName(Folder3 *iface, BSTR name, FolderItem **item) { FolderImpl *This = impl_from_Folder(iface); LPITEMIDLIST pidl; STRRET strret; HRESULT hr; BSTR path; TRACE("(%p,%s,%p)\n", iface, debugstr_w(name), item); *item = NULL; if (FAILED(IShellFolder2_ParseDisplayName(This->folder, NULL, NULL, name, NULL, &pidl, NULL))) return S_FALSE; if ((hr = IShellFolder2_GetDisplayNameOf(This->folder, pidl, SHGDN_FORPARSING, &strret)) == S_OK) hr = StrRetToBSTR(&strret, pidl, &path); ILFree(pidl); if (hr != S_OK) return S_FALSE; hr = FolderItem_Constructor(This, path, item); SysFreeString(path); return hr; } static HRESULT WINAPI FolderImpl_NewFolder(Folder3 *iface, BSTR name, VARIANT options) { FIXME("(%p,%s,%s)\n", iface, debugstr_w(name), debugstr_variant(&options)); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_MoveHere(Folder3 *iface, VARIANT item, VARIANT options) { FIXME("(%p,%s,%s)\n", iface, debugstr_variant(&item), debugstr_variant(&options)); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_CopyHere(Folder3 *iface, VARIANT item, VARIANT options) { FIXME("(%p,%s,%s)\n", iface, debugstr_variant(&item), debugstr_variant(&options)); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_GetDetailsOf(Folder3 *iface, VARIANT item, int column, BSTR *str) { FIXME("(%p,%s,%d,%p)\n", iface, debugstr_variant(&item), column, str); *str = NULL; return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_get_Self(Folder3 *iface, FolderItem **item) { FolderImpl *This = impl_from_Folder(iface); TRACE("(%p,%p)\n", iface, item); return FolderItem_Constructor(This, NULL, item); } static HRESULT WINAPI FolderImpl_get_OfflineStatus(Folder3 *iface, LONG *pul) { FIXME("(%p,%p)\n", iface, pul); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_Synchronize(Folder3 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_get_HaveToShowWebViewBarricade(Folder3 *iface, VARIANT_BOOL *pbHaveToShowWebViewBarricade) { FIXME("(%p,%p)\n", iface, pbHaveToShowWebViewBarricade); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_DismissedWebViewBarricade(Folder3 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_get_ShowWebViewBarricade(Folder3 *iface, VARIANT_BOOL *pbShowWebViewBarricade) { FIXME("(%p,%p)\n", iface, pbShowWebViewBarricade); return E_NOTIMPL; } static HRESULT WINAPI FolderImpl_put_ShowWebViewBarricade(Folder3 *iface, VARIANT_BOOL bShowWebViewBarricade) { FIXME("(%p,%d)\n", iface, bShowWebViewBarricade); return E_NOTIMPL; } static const Folder3Vtbl FolderImpl_Vtbl = { FolderImpl_QueryInterface, FolderImpl_AddRef, FolderImpl_Release, FolderImpl_GetTypeInfoCount, FolderImpl_GetTypeInfo, FolderImpl_GetIDsOfNames, FolderImpl_Invoke, FolderImpl_get_Title, FolderImpl_get_Application, FolderImpl_get_Parent, FolderImpl_get_ParentFolder, FolderImpl_Items, FolderImpl_ParseName, FolderImpl_NewFolder, FolderImpl_MoveHere, FolderImpl_CopyHere, FolderImpl_GetDetailsOf, FolderImpl_get_Self, FolderImpl_get_OfflineStatus, FolderImpl_Synchronize, FolderImpl_get_HaveToShowWebViewBarricade, FolderImpl_DismissedWebViewBarricade, FolderImpl_get_ShowWebViewBarricade, FolderImpl_put_ShowWebViewBarricade }; static HRESULT Folder_Constructor(IShellFolder2 *folder, LPITEMIDLIST pidl, Folder **ret) { PCUITEMID_CHILD last_part; IShellFolder2 *parent; FolderImpl *This; STRRET strret; HRESULT hr; *ret = NULL; This = heap_alloc(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->Folder3_iface.lpVtbl = &FolderImpl_Vtbl; This->ref = 1; This->folder = folder; This->pidl = ILClone(pidl); hr = SHBindToParent(pidl, &IID_IShellFolder2, (void **)&parent, &last_part); IShellFolder2_GetDisplayNameOf(parent, last_part, SHGDN_FORPARSING, &strret); StrRetToBSTR(&strret, last_part, &This->path); IShellFolder2_Release(parent); IShellDispatch_Constructor(NULL, &IID_IDispatch, (void **)&This->application); *ret = (Folder *)&This->Folder3_iface; return hr; } static HRESULT WINAPI ShellDispatch_QueryInterface(IShellDispatch6 *iface, REFIID riid, LPVOID *ppv) { ShellDispatch *This = impl_from_IShellDispatch6(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IDispatch, riid) || IsEqualIID(&IID_IShellDispatch, riid) || IsEqualIID(&IID_IShellDispatch2, riid) || IsEqualIID(&IID_IShellDispatch3, riid) || IsEqualIID(&IID_IShellDispatch4, riid) || IsEqualIID(&IID_IShellDispatch5, riid) || IsEqualIID(&IID_IShellDispatch6, riid)) *ppv = &This->IShellDispatch6_iface; else { WARN("not implemented for %s\n", debugstr_guid(riid)); *ppv = NULL; return E_NOINTERFACE; } IShellDispatch6_AddRef(iface); return S_OK; } static ULONG WINAPI ShellDispatch_AddRef(IShellDispatch6 *iface) { ShellDispatch *This = impl_from_IShellDispatch6(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); return ref; } static ULONG WINAPI ShellDispatch_Release(IShellDispatch6 *iface) { ShellDispatch *This = impl_from_IShellDispatch6(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p), new refcount=%i\n", iface, ref); if (!ref) heap_free(This); return ref; } static HRESULT WINAPI ShellDispatch_GetTypeInfoCount(IShellDispatch6 *iface, UINT *pctinfo) { TRACE("(%p,%p)\n", iface, pctinfo); *pctinfo = 1; return S_OK; } static HRESULT WINAPI ShellDispatch_GetTypeInfo(IShellDispatch6 *iface, UINT iTInfo, LCID lcid, ITypeInfo **ppTInfo) { HRESULT hr; TRACE("(%p,%u,%d,%p)\n", iface, iTInfo, lcid, ppTInfo); hr = get_typeinfo(IShellDispatch6_tid, ppTInfo); if (SUCCEEDED(hr)) ITypeInfo_AddRef(*ppTInfo); return hr; } static HRESULT WINAPI ShellDispatch_GetIDsOfNames(IShellDispatch6 *iface, REFIID riid, LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId) { ITypeInfo *ti; HRESULT hr; TRACE("(%p,%s,%p,%u,%d,%p)\n", iface, shdebugstr_guid(riid), rgszNames, cNames, lcid, rgDispId); hr = get_typeinfo(IShellDispatch6_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_GetIDsOfNames(ti, rgszNames, cNames, rgDispId); return hr; } static HRESULT WINAPI ShellDispatch_Invoke(IShellDispatch6 *iface, DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr) { ShellDispatch *This = impl_from_IShellDispatch6(iface); ITypeInfo *ti; HRESULT hr; TRACE("(%p,%d,%s,%d,%u,%p,%p,%p,%p)\n", iface, dispIdMember, shdebugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); hr = get_typeinfo(IShellDispatch6_tid, &ti); if (SUCCEEDED(hr)) hr = ITypeInfo_Invoke(ti, &This->IShellDispatch6_iface, dispIdMember, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr); return hr; } static HRESULT WINAPI ShellDispatch_get_Application(IShellDispatch6 *iface, IDispatch **disp) { TRACE("(%p,%p)\n", iface, disp); if (!disp) return E_INVALIDARG; *disp = (IDispatch *)iface; IDispatch_AddRef(*disp); return S_OK; } static HRESULT WINAPI ShellDispatch_get_Parent(IShellDispatch6 *iface, IDispatch **disp) { TRACE("(%p,%p)\n", iface, disp); if (disp) { *disp = (IDispatch *)iface; IDispatch_AddRef(*disp); } return S_OK; } static HRESULT create_folder_for_pidl(LPITEMIDLIST pidl, Folder **ret) { IShellFolder2 *folder; IShellFolder *desktop; HRESULT hr; *ret = NULL; if (FAILED(hr = SHGetDesktopFolder(&desktop))) return hr; if (_ILIsDesktop(pidl)) hr = IShellFolder_QueryInterface(desktop, &IID_IShellFolder2, (void **)&folder); else hr = IShellFolder_BindToObject(desktop, pidl, NULL, &IID_IShellFolder2, (void **)&folder); IShellFolder_Release(desktop); if (FAILED(hr)) return S_FALSE; return Folder_Constructor(folder, pidl, ret); } static HRESULT WINAPI ShellDispatch_NameSpace(IShellDispatch6 *iface, VARIANT dir, Folder **ret) { LPITEMIDLIST pidl; HRESULT hr; TRACE("(%p,%s,%p)\n", iface, debugstr_variant(&dir), ret); *ret = NULL; switch (V_VT(&dir)) { case VT_I2: if (FAILED(hr = VariantChangeType(&dir, &dir, 0, VT_I4))) return hr; /* fallthrough */ case VT_I4: if (FAILED(hr = SHGetFolderLocation(NULL, V_I4(&dir), NULL, 0, &pidl))) return S_FALSE; break; case VT_BSTR: if (FAILED(hr = SHParseDisplayName(V_BSTR(&dir), NULL, &pidl, 0, NULL))) return S_FALSE; break; default: WARN("Ignoring directory value %s\n", debugstr_variant(&dir)); return S_FALSE; } hr = create_folder_for_pidl(pidl, ret); ILFree(pidl); return hr; } static BOOL is_optional_argument(const VARIANT *arg) { return V_VT(arg) == VT_ERROR && V_ERROR(arg) == DISP_E_PARAMNOTFOUND; } static HRESULT WINAPI ShellDispatch_BrowseForFolder(IShellDispatch6 *iface, LONG hwnd, BSTR title, LONG options, VARIANT rootfolder, Folder **folder) { PIDLIST_ABSOLUTE selection; BROWSEINFOW bi = { 0 }; HRESULT hr; TRACE("(%p,%x,%s,%x,%s,%p)\n", iface, hwnd, debugstr_w(title), options, debugstr_variant(&rootfolder), folder); *folder = NULL; if (!is_optional_argument(&rootfolder)) FIXME("root folder is ignored\n"); bi.hwndOwner = LongToHandle(hwnd); bi.lpszTitle = title; bi.ulFlags = options; selection = SHBrowseForFolderW(&bi); if (selection) { hr = create_folder_for_pidl(selection, folder); ILFree(selection); } else hr = S_FALSE; return hr; } static HRESULT WINAPI ShellDispatch_Windows(IShellDispatch6 *iface, IDispatch **ppid) { FIXME("(%p,%p)\n", iface, ppid); *ppid = NULL; return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_Open(IShellDispatch6 *iface, VARIANT dir) { FIXME("(%p,%s)\n", iface, debugstr_variant(&dir)); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_Explore(IShellDispatch6 *iface, VARIANT dir) { FIXME("(%p,%s)\n", iface, debugstr_variant(&dir)); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_MinimizeAll(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_UndoMinimizeALL(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_FileRun(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_CascadeWindows(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_TileVertically(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_TileHorizontally(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ShutdownWindows(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_Suspend(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_EjectPC(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_SetTime(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_TrayProperties(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_Help(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_FindFiles(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_FindComputer(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_RefreshMenu(IShellDispatch6 *iface) { FIXME("(%p)\n", iface); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ControlPanelItem(IShellDispatch6 *iface, BSTR szDir) { FIXME("(%p,%s)\n", iface, debugstr_w(szDir)); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_IsRestricted(IShellDispatch6 *iface, BSTR group, BSTR restriction, LONG *value) { FIXME("(%s, %s, %p): stub\n", debugstr_w(group), debugstr_w(restriction), value); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ShellExecute(IShellDispatch6 *iface, BSTR file, VARIANT v_args, VARIANT v_dir, VARIANT v_op, VARIANT v_show) { VARIANT args_str, dir_str, op_str, show_int; WCHAR *args = NULL, *dir = NULL, *op = NULL; INT show = 0; HINSTANCE ret; TRACE("(%s, %s, %s, %s, %s)\n", debugstr_w(file), debugstr_variant(&v_args), debugstr_variant(&v_dir), debugstr_variant(&v_op), debugstr_variant(&v_show)); VariantInit(&args_str); VariantChangeType(&args_str, &v_args, 0, VT_BSTR); if (V_VT(&args_str) == VT_BSTR) args = V_BSTR(&args_str); VariantInit(&dir_str); VariantChangeType(&dir_str, &v_dir, 0, VT_BSTR); if (V_VT(&dir_str) == VT_BSTR) dir = V_BSTR(&dir_str); VariantInit(&op_str); VariantChangeType(&op_str, &v_op, 0, VT_BSTR); if (V_VT(&op_str) == VT_BSTR) op = V_BSTR(&op_str); VariantInit(&show_int); VariantChangeType(&show_int, &v_show, 0, VT_I4); if (V_VT(&show_int) == VT_I4) show = V_I4(&show_int); ret = ShellExecuteW(NULL, op, file, args, dir, show); VariantClear(&args_str); VariantClear(&dir_str); VariantClear(&op_str); VariantClear(&show_int); return (ULONG_PTR)ret > 32 ? S_OK : S_FALSE; } static HRESULT WINAPI ShellDispatch_FindPrinter(IShellDispatch6 *iface, BSTR name, BSTR location, BSTR model) { FIXME("(%s, %s, %s): stub\n", debugstr_w(name), debugstr_w(location), debugstr_w(model)); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_GetSystemInformation(IShellDispatch6 *iface, BSTR name, VARIANT *ret) { FIXME("(%s, %p): stub\n", debugstr_w(name), ret); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ServiceStart(IShellDispatch6 *iface, BSTR service, VARIANT persistent, VARIANT *ret) { FIXME("(%s, %s, %p): stub\n", debugstr_w(service), debugstr_variant(&persistent), ret); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ServiceStop(IShellDispatch6 *iface, BSTR service, VARIANT persistent, VARIANT *ret) { FIXME("(%s, %s, %p): stub\n", debugstr_w(service), debugstr_variant(&persistent), ret); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_IsServiceRunning(IShellDispatch6 *iface, BSTR name, VARIANT *running) { SERVICE_STATUS_PROCESS status; SC_HANDLE scm, service; DWORD dummy; TRACE("(%s, %p)\n", debugstr_w(name), running); V_VT(running) = VT_BOOL; V_BOOL(running) = VARIANT_FALSE; scm = OpenSCManagerW(NULL, NULL, SC_MANAGER_CONNECT); if (!scm) { ERR("failed to connect to service manager\n"); return S_OK; } service = OpenServiceW(scm, name, SERVICE_QUERY_STATUS); if (!service) { ERR("Failed to open service %s (%u)\n", debugstr_w(name), GetLastError()); CloseServiceHandle(scm); return S_OK; } if (!QueryServiceStatusEx(service, SC_STATUS_PROCESS_INFO, (BYTE *)&status, sizeof(SERVICE_STATUS_PROCESS), &dummy)) { TRACE("failed to query service status (%u)\n", GetLastError()); CloseServiceHandle(service); CloseServiceHandle(scm); return S_OK; } if (status.dwCurrentState == SERVICE_RUNNING) V_BOOL(running) = VARIANT_TRUE; CloseServiceHandle(service); CloseServiceHandle(scm); return S_OK; } static HRESULT WINAPI ShellDispatch_CanStartStopService(IShellDispatch6 *iface, BSTR service, VARIANT *ret) { FIXME("(%s, %p): stub\n", debugstr_w(service), ret); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ShowBrowserBar(IShellDispatch6 *iface, BSTR clsid, VARIANT show, VARIANT *ret) { FIXME("(%s, %s, %p): stub\n", debugstr_w(clsid), debugstr_variant(&show), ret); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_AddToRecent(IShellDispatch6 *iface, VARIANT file, BSTR category) { FIXME("(%s, %s): stub\n", debugstr_variant(&file), debugstr_w(category)); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_WindowsSecurity(IShellDispatch6 *iface) { FIXME("stub\n"); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ToggleDesktop(IShellDispatch6 *iface) { FIXME("stub\n"); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_ExplorerPolicy(IShellDispatch6 *iface, BSTR policy, VARIANT *value) { FIXME("(%s, %p): stub\n", debugstr_w(policy), value); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_GetSetting(IShellDispatch6 *iface, LONG setting, VARIANT_BOOL *result) { FIXME("(%d %p): stub\n", setting, result); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_WindowSwitcher(IShellDispatch6 *iface) { FIXME("stub\n"); return E_NOTIMPL; } static HRESULT WINAPI ShellDispatch_SearchCommand(IShellDispatch6 *iface) { FIXME("stub\n"); return E_NOTIMPL; } static const IShellDispatch6Vtbl ShellDispatchVtbl = { ShellDispatch_QueryInterface, ShellDispatch_AddRef, ShellDispatch_Release, ShellDispatch_GetTypeInfoCount, ShellDispatch_GetTypeInfo, ShellDispatch_GetIDsOfNames, ShellDispatch_Invoke, ShellDispatch_get_Application, ShellDispatch_get_Parent, ShellDispatch_NameSpace, ShellDispatch_BrowseForFolder, ShellDispatch_Windows, ShellDispatch_Open, ShellDispatch_Explore, ShellDispatch_MinimizeAll, ShellDispatch_UndoMinimizeALL, ShellDispatch_FileRun, ShellDispatch_CascadeWindows, ShellDispatch_TileVertically, ShellDispatch_TileHorizontally, ShellDispatch_ShutdownWindows, ShellDispatch_Suspend, ShellDispatch_EjectPC, ShellDispatch_SetTime, ShellDispatch_TrayProperties, ShellDispatch_Help, ShellDispatch_FindFiles, ShellDispatch_FindComputer, ShellDispatch_RefreshMenu, ShellDispatch_ControlPanelItem, ShellDispatch_IsRestricted, ShellDispatch_ShellExecute, ShellDispatch_FindPrinter, ShellDispatch_GetSystemInformation, ShellDispatch_ServiceStart, ShellDispatch_ServiceStop, ShellDispatch_IsServiceRunning, ShellDispatch_CanStartStopService, ShellDispatch_ShowBrowserBar, ShellDispatch_AddToRecent, ShellDispatch_WindowsSecurity, ShellDispatch_ToggleDesktop, ShellDispatch_ExplorerPolicy, ShellDispatch_GetSetting, ShellDispatch_WindowSwitcher, ShellDispatch_SearchCommand }; HRESULT WINAPI IShellDispatch_Constructor(IUnknown *outer, REFIID riid, void **ppv) { ShellDispatch *This; HRESULT ret; TRACE("(%p, %s)\n", outer, debugstr_guid(riid)); *ppv = NULL; if (outer) return CLASS_E_NOAGGREGATION; This = heap_alloc(sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->IShellDispatch6_iface.lpVtbl = &ShellDispatchVtbl; This->ref = 1; ret = IShellDispatch6_QueryInterface(&This->IShellDispatch6_iface, riid, ppv); IShellDispatch6_Release(&This->IShellDispatch6_iface); return ret; }