ReactOS 0.4.16-dev-1007-g2e85425
utils.cpp
Go to the documentation of this file.
1/*
2 * PROJECT: ReactOS Shell
3 * LICENSE: LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
4 * PURPOSE: Implement shell light-weight utility functions
5 * COPYRIGHT: Copyright 2023-2024 Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
6 */
7
8#define _ATL_NO_EXCEPTIONS
9
10/*
11 * HACK! These functions are conflicting with <shobjidl.h> inline functions...
12 */
13#define IShellFolder_GetDisplayNameOf _disabled_IShellFolder_GetDisplayNameOf_
14#define IShellFolder_ParseDisplayName _disabled_IShellFolder_ParseDisplayName_
15#define IShellFolder_CompareIDs _disabled_IShellFolder_CompareIDs_
16
17#include "precomp.h"
18#include <shellapi.h>
19#include <shlwapi.h>
20#include <shlobj_undoc.h>
21#include <shlguid_undoc.h>
22#include <atlstr.h>
23
24#include <shlwapi_undoc.h>
26
27#include <strsafe.h>
28
29#ifndef FAILED_UNEXPECTEDLY
30#define FAILED_UNEXPECTEDLY FAILED /* FIXME: Make shellutils.h usable without ATL */
31#endif
32
34
35static inline WORD
37{
40}
41
42static HRESULT
46 _In_ IContextMenu* pCM,
47 _In_ UINT fCMIC,
48 _In_ UINT fCMF,
49 _In_opt_ LPCSTR pszVerb,
50 _In_opt_ LPCWSTR pwszDir,
51 _In_ bool ForceQCM)
52{
53 CMINVOKECOMMANDINFOEX info = { sizeof(info), fCMIC, hWnd, pszVerb };
54 INT iDefItem = 0;
55 HMENU hMenu = NULL;
56 HCURSOR hOldCursor;
57 HRESULT hr = S_OK;
58 WCHAR wideverb[MAX_PATH];
59
60 if (!pCM)
61 return E_INVALIDARG;
62
63 hOldCursor = SetCursor(LoadCursorW(NULL, (LPCWSTR)IDC_WAIT));
64 info.nShow = SW_NORMAL;
65 if (pUnk)
67
68 if (IS_INTRESOURCE(pszVerb))
69 {
70 hMenu = CreatePopupMenu();
71 if (hMenu)
72 {
73 hr = pCM->QueryContextMenu(hMenu, 0, 1, MAXSHORT, fCMF | CMF_DEFAULTONLY);
74 iDefItem = GetMenuDefaultItem(hMenu, 0, 0);
75 if (iDefItem != -1)
76 info.lpVerb = MAKEINTRESOURCEA(iDefItem - 1);
77 }
78 info.lpVerbW = MAKEINTRESOURCEW(info.lpVerb);
79 }
80 else
81 {
83 {
84 info.fMask |= CMF_OPTIMIZEFORINVOKE;
85 }
86 if (pszVerb && SHAnsiToUnicode(pszVerb, wideverb, _countof(wideverb)))
87 {
88 info.fMask |= CMIC_MASK_UNICODE;
89 info.lpVerbW = wideverb;
90 }
91 if (ForceQCM)
92 {
93 hMenu = CreatePopupMenu();
94 hr = pCM->QueryContextMenu(hMenu, 0, 1, MAXSHORT, fCMF);
95 }
96 }
97
98 SetCursor(hOldCursor);
99
100 if (!FAILED_UNEXPECTEDLY(hr) && (iDefItem != -1 || info.lpVerb))
101 {
102 if (!hWnd)
103 info.fMask |= CMIC_MASK_FLAG_NO_UI;
104
106 if (pwszDir)
107 {
108 info.fMask |= CMIC_MASK_UNICODE;
109 info.lpDirectoryW = pwszDir;
110 if (SHUnicodeToAnsi(pwszDir, dir, _countof(dir)))
111 info.lpDirectory = dir;
112 }
113
114 hr = pCM->InvokeCommand((LPCMINVOKECOMMANDINFO)&info);
115 if (FAILED_UNEXPECTEDLY(hr)) { /* Diagnostic message */ }
116 }
117
118 if (pUnk)
120 if (hMenu)
121 DestroyMenu(hMenu);
122
123 return hr;
124}
125
126/*************************************************************************
127 * SHInvokeCommandOnContextMenuEx [SHLWAPI.639]
128 */
134 _In_ IContextMenu* pCM,
135 _In_ UINT fCMIC,
136 _In_ UINT fCMF,
137 _In_opt_ LPCSTR pszVerb,
138 _In_opt_ LPCWSTR pwszDir)
139{
140 return SHInvokeCommandOnContextMenuInternal(hWnd, pUnk, pCM, fCMIC, fCMF, pszVerb, pwszDir, true);
141}
142
143/*************************************************************************
144 * SHInvokeCommandOnContextMenu [SHLWAPI.540]
145 */
151 _In_ IContextMenu* pCM,
152 _In_ UINT fCMIC,
153 _In_opt_ LPCSTR pszVerb)
154{
155 return SHInvokeCommandOnContextMenuEx(hWnd, pUnk, pCM, fCMIC, CMF_EXTENDEDVERBS, pszVerb, NULL);
156}
157
158/*************************************************************************
159 * SHInvokeCommandWithFlagsAndSite [SHLWAPI.571]
160 */
166 _In_ IShellFolder* pShellFolder,
167 _In_ LPCITEMIDLIST pidl,
168 _In_ UINT fCMIC,
169 _In_opt_ LPCSTR pszVerb)
170{
172 if (pShellFolder)
173 {
174 IContextMenu *pCM;
175 hr = pShellFolder->GetUIObjectOf(hWnd, 1, &pidl, IID_IContextMenu, NULL, (void**)&pCM);
176 if (SUCCEEDED(hr))
177 {
178 fCMIC |= CMIC_MASK_FLAG_LOG_USAGE;
179 hr = SHInvokeCommandOnContextMenuEx(hWnd, pUnk, pCM, fCMIC, 0, pszVerb, NULL);
180 pCM->Release();
181 }
182 }
183 return hr;
184}
185
186
187/*************************************************************************
188 * IContextMenu_Invoke [SHLWAPI.207]
189 *
190 * Used by Win:SHELL32!CISFBand::_TrySimpleInvoke.
191 */
195 _In_ IContextMenu *pContextMenu,
196 _In_ HWND hwnd,
197 _In_ LPCSTR lpVerb,
199{
200 TRACE("(%p, %p, %s, %u)\n", pContextMenu, hwnd, debugstr_a(lpVerb), uFlags);
202 uFlags, lpVerb, NULL, false);
203 return !FAILED_UNEXPECTEDLY(hr);
204}
205
206/*************************************************************************
207 * PathFileExistsDefExtAndAttributesW [SHLWAPI.511]
208 *
209 * @param pszPath The path string.
210 * @param dwWhich The WHICH_... flags.
211 * @param pdwFileAttributes A pointer to the file attributes. Optional.
212 * @return TRUE if successful.
213 */
216 _Inout_ LPWSTR pszPath,
217 _In_ DWORD dwWhich,
218 _Out_opt_ LPDWORD pdwFileAttributes)
219{
220 TRACE("(%s, 0x%lX, %p)\n", debugstr_w(pszPath), dwWhich, pdwFileAttributes);
221
222 if (pdwFileAttributes)
223 *pdwFileAttributes = INVALID_FILE_ATTRIBUTES;
224
225 if (!pszPath)
226 return FALSE;
227
228 if (!dwWhich || (*PathFindExtensionW(pszPath) && (dwWhich & WHICH_OPTIONAL)))
229 return PathFileExistsAndAttributesW(pszPath, pdwFileAttributes);
230
231 if (!PathFileExistsDefExtW(pszPath, dwWhich))
232 {
233 if (pdwFileAttributes)
234 *pdwFileAttributes = INVALID_FILE_ATTRIBUTES;
235 return FALSE;
236 }
237
238 if (pdwFileAttributes)
239 *pdwFileAttributes = GetFileAttributesW(pszPath);
240
241 return TRUE;
242}
243
244static inline BOOL
246{
247 return (hr == E_FAIL || hr == E_INVALIDARG || hr == E_NOTIMPL);
248}
249
250// Used for IShellFolder_GetDisplayNameOf
252{
253 SHGDNF uRemove;
254 SHGDNF uAdd;
256};
257static const RETRY_DATA g_RetryData[] =
258{
259 { SHGDN_FOREDITING, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
260 { SHGDN_FORADDRESSBAR, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
261 { SHGDN_NORMAL, SHGDN_FORPARSING, SFGDNO_RETRYALWAYS },
262 { SHGDN_FORPARSING, SHGDN_NORMAL, SFGDNO_RETRYWITHFORPARSING },
263 { SHGDN_INFOLDER, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
264};
265
266/*************************************************************************
267 * IShellFolder_GetDisplayNameOf [SHLWAPI.316]
268 *
269 * @note Don't confuse with <shobjidl.h> inline function of the same name.
270 * If the original call fails with the given uFlags, this function will
271 * retry with other flags to attempt retrieving any meaningful description.
272 */
275 _In_ IShellFolder *psf,
276 _In_ LPCITEMIDLIST pidl,
277 _In_ SHGDNF uFlags,
279 _In_ DWORD dwRetryFlags) // dwRetryFlags is an additional parameter
280{
281 HRESULT hr;
282
283 TRACE("(%p)->(%p, 0x%lX, %p, 0x%lX)\n", psf, pidl, uFlags, lpName, dwRetryFlags);
284
285 hr = psf->GetDisplayNameOf(pidl, uFlags, lpName);
287 return hr;
288
289 dwRetryFlags |= SFGDNO_RETRYALWAYS;
290
291 if ((uFlags & SHGDN_FORPARSING) == 0)
292 dwRetryFlags |= SFGDNO_RETRYWITHFORPARSING;
293
294 // Retry with other flags to get successful results
295 for (SIZE_T iEntry = 0; iEntry < _countof(g_RetryData); ++iEntry)
296 {
297 const RETRY_DATA *pData = &g_RetryData[iEntry];
298 if (!(dwRetryFlags & pData->dwRetryFlags))
299 continue;
300
301 SHGDNF uNewFlags = ((uFlags & ~pData->uRemove) | pData->uAdd);
302 if (uNewFlags == uFlags)
303 continue;
304
305 hr = psf->GetDisplayNameOf(pidl, uNewFlags, lpName);
307 break;
308
309 uFlags = uNewFlags; // Update flags every time
310 }
311
312 return hr;
313}
314
315/*************************************************************************
316 * IShellFolder_ParseDisplayName [SHLWAPI.317]
317 *
318 * @note Don't confuse with <shobjidl.h> inline function of the same name.
319 * This function is safer than IShellFolder::ParseDisplayName.
320 */
323 _In_ IShellFolder *psf,
324 _In_opt_ HWND hwndOwner,
325 _In_opt_ LPBC pbcReserved,
326 _In_ LPOLESTR lpszDisplayName,
327 _Out_opt_ ULONG *pchEaten,
329 _Out_opt_ ULONG *pdwAttributes)
330{
331 ULONG dummy1, dummy2;
332
333 TRACE("(%p)->(%p, %p, %s, %p, %p, %p)\n", psf, hwndOwner, pbcReserved,
334 debugstr_w(lpszDisplayName), pchEaten, ppidl, pdwAttributes);
335
336 if (!pdwAttributes)
337 {
338 dummy1 = 0;
339 pdwAttributes = &dummy1;
340 }
341
342 if (!pchEaten)
343 {
344 dummy2 = 0;
345 pchEaten = &dummy2;
346 }
347
348 if (ppidl)
349 *ppidl = NULL;
350
351 return psf->ParseDisplayName(hwndOwner, pbcReserved, lpszDisplayName, pchEaten,
352 ppidl, pdwAttributes);
353}
354
355/*************************************************************************
356 * IShellFolder_CompareIDs [SHLWAPI.551]
357 *
358 * @note Don't confuse with <shobjidl.h> inline function of the same name.
359 * This function tries IShellFolder2 if possible.
360 */
363 _In_ IShellFolder *psf,
367{
368 TRACE("(%p, %p, %p, %p)\n", psf, lParam, pidl1, pidl2);
369
370 if (lParam & ~(SIZE_T)SHCIDS_COLUMNMASK)
371 {
372 /* Try as IShellFolder2 if possible */
373 HRESULT hr = psf->QueryInterface(IID_IShellFolder2, (void **)&psf);
374 if (FAILED(hr))
375 lParam &= SHCIDS_COLUMNMASK;
376 else
377 psf->Release();
378 }
379
380 return psf->CompareIDs(lParam, pidl1, pidl2);
381}
unsigned int dir
Definition: maze.c:112
HWND hWnd
Definition: settings.c:17
#define WINE_DEFAULT_DEBUG_CHANNEL(t)
Definition: precomp.h:23
void shell(int argc, const char *argv[])
Definition: cmds.c:1231
#define EXTERN_C
Definition: basetyps.h:12
LPARAM lParam
Definition: combotst.c:139
#define E_INVALIDARG
Definition: ddrawi.h:101
#define E_NOTIMPL
Definition: ddrawi.h:99
#define E_FAIL
Definition: ddrawi.h:102
#define NULL
Definition: types.h:112
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
UINT uFlags
Definition: api.c:59
#define MAX_PATH
Definition: compat.h:34
static const WCHAR version[]
Definition: asmname.c:66
DWORD WINAPI GetFileAttributesW(LPCWSTR lpFileName)
Definition: fileinfo.c:652
HRESULT WINAPI IUnknown_SetSite(IUnknown *obj, IUnknown *site)
Definition: ordinal.c:1411
LPWSTR WINAPI PathFindExtensionW(LPCWSTR lpszPath)
Definition: path.c:447
BOOL WINAPI PathFileExistsAndAttributesW(LPCWSTR lpszPath, DWORD *dwAttr)
Definition: path.c:1838
BOOL WINAPI PathFileExistsDefExtW(LPWSTR lpszPath, DWORD dwWhich)
Definition: path.c:1123
DWORD WINAPI SHAnsiToUnicode(LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen)
Definition: string.c:2673
INT WINAPI SHUnicodeToAnsi(LPCWSTR lpSrcStr, LPSTR lpDstStr, INT iLen)
Definition: string.c:2797
EXTERN_C HRESULT WINAPI SHInvokeCommandOnContextMenu(_In_opt_ HWND hWnd, _In_opt_ IUnknown *pUnk, _In_ IContextMenu *pCM, _In_ UINT fCMIC, _In_opt_ LPCSTR pszVerb)
Definition: utils.cpp:148
EXTERN_C HRESULT WINAPI SHInvokeCommandWithFlagsAndSite(_In_opt_ HWND hWnd, _In_opt_ IUnknown *pUnk, _In_ IShellFolder *pShellFolder, _In_ LPCITEMIDLIST pidl, _In_ UINT fCMIC, _In_opt_ LPCSTR pszVerb)
Definition: utils.cpp:163
EXTERN_C HRESULT WINAPI SHInvokeCommandOnContextMenuEx(_In_opt_ HWND hWnd, _In_opt_ IUnknown *pUnk, _In_ IContextMenu *pCM, _In_ UINT fCMIC, _In_ UINT fCMF, _In_opt_ LPCSTR pszVerb, _In_opt_ LPCWSTR pwszDir)
Definition: utils.cpp:131
EXTERN_C BOOL WINAPI IContextMenu_Invoke(_In_ IContextMenu *pContextMenu, _In_ HWND hwnd, _In_ LPCSTR lpVerb, _In_ UINT uFlags)
Definition: utils.cpp:194
static const RETRY_DATA g_RetryData[]
Definition: utils.cpp:257
static HRESULT SHInvokeCommandOnContextMenuInternal(_In_opt_ HWND hWnd, _In_opt_ IUnknown *pUnk, _In_ IContextMenu *pCM, _In_ UINT fCMIC, _In_ UINT fCMF, _In_opt_ LPCSTR pszVerb, _In_opt_ LPCWSTR pwszDir, _In_ bool ForceQCM)
Definition: utils.cpp:43
#define FAILED_UNEXPECTEDLY
Definition: utils.cpp:30
#define IShellFolder_ParseDisplayName
Definition: utils.cpp:14
#define IShellFolder_CompareIDs
Definition: utils.cpp:15
static WORD GetVersionMajorMinor()
Definition: utils.cpp:36
BOOL WINAPI PathFileExistsDefExtAndAttributesW(_Inout_ LPWSTR pszPath, _In_ DWORD dwWhich, _Out_opt_ LPDWORD pdwFileAttributes)
Definition: utils.cpp:215
#define IShellFolder_GetDisplayNameOf
Definition: utils.cpp:13
static BOOL SHLWAPI_IsBogusHRESULT(HRESULT hr)
Definition: utils.cpp:245
static void *static void *static LPDIRECTPLAY IUnknown * pUnk
Definition: dplayx.c:30
unsigned int BOOL
Definition: ntddk_ex.h:94
unsigned long DWORD
Definition: ntddk_ex.h:95
unsigned short WORD
Definition: ntddk_ex.h:93
HRESULT QueryInterface([in] REFIID riid, [out, iid_is(riid)] void **ppvObject)
ULONG Release()
nsrefcnt Release()
#define S_OK
Definition: intsafe.h:52
#define SUCCEEDED(hr)
Definition: intsafe.h:50
#define FAILED(hr)
Definition: intsafe.h:51
#define SFGDNO_RETRYWITHFORPARSING
#define SFGDNO_RETRYALWAYS
#define LOBYTE(W)
Definition: jmemdos.c:487
#define HIBYTE(W)
Definition: jmemdos.c:486
#define debugstr_a
Definition: kernel32.h:31
#define debugstr_w
Definition: kernel32.h:32
static LPOLESTR
Definition: stg_prop.c:27
static const CLSID *static CLSID *static const GUID VARIANT VARIANT *static IServiceProvider DWORD *static HMENU
Definition: ordinal.c:63
unsigned int UINT
Definition: ndis.h:50
#define _Out_opt_
Definition: no_sal2.h:214
#define _Inout_
Definition: no_sal2.h:162
#define _Out_
Definition: no_sal2.h:160
#define _In_
Definition: no_sal2.h:158
#define _In_opt_
Definition: no_sal2.h:212
interface IBindCtx * LPBC
Definition: objfwd.h:18
DWORD WINAPI GetVersion()
Definition: redirtest.c:5
#define _WIN32_WINNT_WIN7
Definition: sdkddkver.h:28
HRESULT hr
Definition: shlfolder.c:183
#define WHICH_OPTIONAL
const ITEMIDLIST_RELATIVE UNALIGNED * PCUIDLIST_RELATIVE
Definition: shtypes.idl:57
const ITEMIDLIST UNALIGNED * LPCITEMIDLIST
Definition: shtypes.idl:42
#define _countof(array)
Definition: sndvol32.h:70
#define TRACE(s)
Definition: solgame.cpp:4
SHGDNF uAdd
Definition: utils.cpp:254
SHGDNF uRemove
Definition: utils.cpp:253
DWORD dwRetryFlags
Definition: utils.cpp:255
TW_UINT32 TW_UINT16 TW_UINT16 TW_MEMREF pData
Definition: twain.h:1830
#define MAKEWORD(a, b)
Definition: typedefs.h:248
ULONG_PTR SIZE_T
Definition: typedefs.h:80
uint32_t * LPDWORD
Definition: typedefs.h:59
int32_t INT
Definition: typedefs.h:58
uint32_t ULONG
Definition: typedefs.h:59
#define MAXSHORT
Definition: umtypes.h:114
#define INVALID_FILE_ATTRIBUTES
Definition: vfdcmd.c:23
_In_ LPCSTR lpName
Definition: winbase.h:2820
_In_ LONG _In_ HWND hwnd
Definition: winddi.h:4023
LONG_PTR LPARAM
Definition: windef.h:208
HICON HCURSOR
Definition: windef.h:299
#define WINAPI
Definition: msvc.h:6
HMENU WINAPI CreatePopupMenu(void)
Definition: menu.c:838
UINT WINAPI GetMenuDefaultItem(_In_ HMENU hMenu, _In_ UINT fByPos, _In_ UINT gmdiFlags)
#define IS_INTRESOURCE(i)
Definition: winuser.h:580
HCURSOR WINAPI SetCursor(_In_opt_ HCURSOR)
HCURSOR WINAPI LoadCursorW(_In_opt_ HINSTANCE, _In_ LPCWSTR)
Definition: cursoricon.c:2442
#define MAKEINTRESOURCEA(i)
Definition: winuser.h:581
BOOL WINAPI DestroyMenu(_In_ HMENU)
#define IDC_WAIT
Definition: winuser.h:697
#define MAKEINTRESOURCEW(i)
Definition: winuser.h:582
#define SW_NORMAL
Definition: winuser.h:780
const char * LPCSTR
Definition: xmlstorage.h:183
__wchar_t WCHAR
Definition: xmlstorage.h:180
WCHAR * LPWSTR
Definition: xmlstorage.h:184
const WCHAR * LPCWSTR
Definition: xmlstorage.h:185
char CHAR
Definition: xmlstorage.h:175