ReactOS 0.4.17-dev-243-g1369312
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 <userenv.h>
23#include <atlstr.h>
24
25#include <shlwapi_undoc.h>
27
28#include <strsafe.h>
29
30#ifndef FAILED_UNEXPECTEDLY
31#define FAILED_UNEXPECTEDLY FAILED /* FIXME: Make shellutils.h usable without ATL */
32#endif
33
35
37
38static inline WORD
40{
43}
44
45static BOOL
48 _In_ PCSTR lpString,
49 _In_ PCSTR lpSrc,
50 _Out_ PSTR pszDest,
52{
53 CHAR szBuff[MAX_PATH];
54 INT cchExpanded;
55
56 if (hUserToken)
57 {
58 if (ExpandEnvironmentStringsForUserA(hUserToken, lpSrc, szBuff, _countof(szBuff)))
59 cchExpanded = lstrlenA(szBuff) + 1;
60 else
61 cchExpanded = 0;
62 }
63 else
64 {
65 cchExpanded = ExpandEnvironmentStringsA(lpSrc, szBuff, _countof(szBuff));
66 }
67
68 if (!cchExpanded || cchExpanded > cchDest)
69 return FALSE;
70
71 INT cchEnvPath = cchExpanded - 1;
73 szBuff, cchEnvPath, lpString, cchEnvPath) != CSTR_EQUAL)
74 {
75 return FALSE;
76 }
77
78 INT cchSuffix = lstrlenA(lpString) - cchEnvPath;
79 if (lstrlenA(lpSrc) + cchSuffix >= cchDest)
80 return FALSE;
81
82 StringCchCopyA(pszDest, cchDest, lpSrc);
83 StringCchCatA(pszDest, cchDest, &lpString[cchEnvPath]);
84 return TRUE;
85}
86
87static BOOL
90 _In_ PCWSTR lpString,
91 _In_ PCWSTR lpSrc,
92 _Out_ PWSTR pszDest,
94{
95 WCHAR szBuff[MAX_PATH];
96 INT cchExpanded;
97
98 if (hUserToken)
99 {
100 if (ExpandEnvironmentStringsForUserW(hUserToken, lpSrc, szBuff, _countof(szBuff)))
101 cchExpanded = lstrlenW(szBuff) + 1;
102 else
103 cchExpanded = 0;
104 }
105 else
106 {
107 cchExpanded = ExpandEnvironmentStringsW(lpSrc, szBuff, _countof(szBuff));
108 }
109
110 if (!cchExpanded || cchExpanded > cchDest)
111 return FALSE;
112
113 INT cchEnvPath = cchExpanded - 1;
115 szBuff, cchEnvPath, lpString, cchEnvPath) != CSTR_EQUAL)
116 {
117 return FALSE;
118 }
119
120 INT cchSuffix = lstrlenW(lpString) - cchEnvPath;
121 if (lstrlenW(lpSrc) + cchSuffix >= cchDest)
122 return FALSE;
123
124 StringCchCopyW(pszDest, cchDest, lpSrc);
125 StringCchCatW(pszDest, cchDest, &lpString[cchEnvPath]);
126 return TRUE;
127}
128
129/*************************************************************************
130 * PathUnExpandEnvStringsForUserA [SHLWAPI.465]
131 *
132 * See PathUnExpandEnvStringsForUserW.
133 */
138 _In_ PCSTR pszPath,
139 _Out_writes_(cchBuff) PSTR pszBuff,
140 _In_ INT cchBuff)
141{
142 static const PCSTR c_varsA[] =
143 {
144 "%APPDATA%",
145 "%USERPROFILE%",
146 "%ALLUSERSPROFILE%",
147 "%ProgramFiles%",
148 "%SystemRoot%",
149 "%SystemDrive%",
150 };
151
152 if (!pszPath)
153 {
154 if (pszBuff && cchBuff)
155 *pszBuff = ANSI_NULL;
156
157 return FALSE;
158 }
159
160 if (!pszBuff)
161 return FALSE;
162
163 for (size_t iVar = 0; iVar < _countof(c_varsA); ++iVar)
164 {
165 if (UnExpandEnvironmentStringForUserA(hUserToken, pszPath, c_varsA[iVar],
166 pszBuff, cchBuff))
167 {
168 return TRUE;
169 }
170 }
171
172 return FALSE;
173}
174
175/*************************************************************************
176 * PathUnExpandEnvStringsForUserW [SHLWAPI.466]
177 *
178 * https://undoc.airesoft.co.uk/shlwapi.dll/PathUnExpandEnvStringsForUserW.php
179 */
184 _In_ PCWSTR pwszPath,
185 _Out_writes_(cchBuff) PWSTR pszBuff,
186 _In_ INT cchBuff)
187{
188 static const PCWSTR c_varsW[] =
189 {
190 L"%APPDATA%",
191 L"%USERPROFILE%",
192 L"%ALLUSERSPROFILE%",
193 L"%ProgramFiles%",
194 L"%SystemRoot%",
195 L"%SystemDrive%",
196 };
197
198 if (!pwszPath)
199 {
200 if (pszBuff && cchBuff)
201 *pszBuff = UNICODE_NULL;
202
203 return FALSE;
204 }
205
206 if (!pszBuff)
207 return FALSE;
208
209 for (size_t iVar = 0; iVar < _countof(c_varsW); ++iVar)
210 {
211 if (UnExpandEnvironmentStringForUserW(hUserToken, pwszPath, c_varsW[iVar],
212 pszBuff, cchBuff))
213 {
214 return TRUE;
215 }
216 }
217
218 return FALSE;
219}
220
221static BOOL CharLowerNoDBCSAWorker(PSTR lpString, INT cchMax, BOOL bUppercase)
222{
223 CHAR szBuff[MAX_PATH];
224 INT cch;
225 if (!lpString)
226 return FALSE;
227 cch = cchMax ? cchMax : lstrlenA(lpString);
228 if (FAILED(StringCchCopyA(szBuff, _countof(szBuff), lpString)))
229 return FALSE;
231 szBuff, cch, lpString, cch);
232}
233
234static BOOL CharLowerNoDBCSWWorker(PWSTR lpString, INT cchMax, BOOL bUppercase)
235{
236 WCHAR szDest[MAX_PATH];
237 INT cch;
238 if (!lpString)
239 return FALSE;
240 cch = cchMax ? cchMax : lstrlenW(lpString);
241 if (FAILED(StringCchCopyW(szDest, _countof(szDest), lpString)))
242 return FALSE;
244 szDest, cch, lpString, cch);
245}
246
247/*************************************************************************
248 * CharLowerNoDBCSA [SHLWAPI.453]
249 */
251{
252 return CharLowerNoDBCSAWorker(lpString, 0, FALSE) ? lpString : NULL;
253}
254
255/*************************************************************************
256 * CharLowerNoDBCSW [SHLWAPI.454]
257 */
259{
260 return CharLowerNoDBCSWWorker(lpString, 0, FALSE) ? lpString : NULL;
261}
262
263/*************************************************************************
264 * CharUpperNoDBCSA [SHLWAPI.451]
265 */
267{
268 return CharLowerNoDBCSAWorker(lpString, 0, TRUE) ? lpString : NULL;
269}
270
271/*************************************************************************
272 * CharUpperNoDBCSW [SHLWAPI.452]
273 */
275{
276 return CharLowerNoDBCSWWorker(lpString, 0, TRUE) ? lpString : NULL;
277}
278
279static HRESULT
283 _In_ IContextMenu* pCM,
284 _In_ UINT fCMIC,
285 _In_ UINT fCMF,
286 _In_opt_ LPCSTR pszVerb,
287 _In_opt_ LPCWSTR pwszDir,
288 _In_ bool ForceQCM)
289{
290 CMINVOKECOMMANDINFOEX info = { sizeof(info), fCMIC, hWnd, pszVerb };
291 INT iDefItem = 0;
292 HMENU hMenu = NULL;
293 HCURSOR hOldCursor;
294 HRESULT hr = S_OK;
295 WCHAR wideverb[MAX_PATH];
296
297 if (!pCM)
298 return E_INVALIDARG;
299
300 hOldCursor = SetCursor(LoadCursorW(NULL, (LPCWSTR)IDC_WAIT));
301 info.nShow = SW_NORMAL;
302 if (pUnk)
304
305 if (IS_INTRESOURCE(pszVerb))
306 {
307 hMenu = CreatePopupMenu();
308 if (hMenu)
309 {
310 hr = pCM->QueryContextMenu(hMenu, 0, 1, MAXSHORT, fCMF | CMF_DEFAULTONLY);
311 iDefItem = GetMenuDefaultItem(hMenu, 0, 0);
312 if (iDefItem != -1)
313 info.lpVerb = MAKEINTRESOURCEA(iDefItem - 1);
314 }
315 info.lpVerbW = MAKEINTRESOURCEW(info.lpVerb);
316 }
317 else
318 {
320 {
321 info.fMask |= CMF_OPTIMIZEFORINVOKE;
322 }
323 if (pszVerb && SHAnsiToUnicode(pszVerb, wideverb, _countof(wideverb)))
324 {
325 info.fMask |= CMIC_MASK_UNICODE;
326 info.lpVerbW = wideverb;
327 }
328 if (ForceQCM)
329 {
330 hMenu = CreatePopupMenu();
331 hr = pCM->QueryContextMenu(hMenu, 0, 1, MAXSHORT, fCMF);
332 }
333 }
334
335 SetCursor(hOldCursor);
336
337 if (!FAILED_UNEXPECTEDLY(hr) && (iDefItem != -1 || info.lpVerb))
338 {
339 if (!hWnd)
340 info.fMask |= CMIC_MASK_FLAG_NO_UI;
341
343 if (pwszDir)
344 {
345 info.fMask |= CMIC_MASK_UNICODE;
346 info.lpDirectoryW = pwszDir;
347 if (SHUnicodeToAnsi(pwszDir, dir, _countof(dir)))
348 info.lpDirectory = dir;
349 }
350
351 hr = pCM->InvokeCommand((LPCMINVOKECOMMANDINFO)&info);
352 if (FAILED_UNEXPECTEDLY(hr)) { /* Diagnostic message */ }
353 }
354
355 if (pUnk)
357 if (hMenu)
358 DestroyMenu(hMenu);
359
360 return hr;
361}
362
363/*************************************************************************
364 * SHInvokeCommandOnContextMenuEx [SHLWAPI.639]
365 */
371 _In_ IContextMenu* pCM,
372 _In_ UINT fCMIC,
373 _In_ UINT fCMF,
374 _In_opt_ LPCSTR pszVerb,
375 _In_opt_ LPCWSTR pwszDir)
376{
377 return SHInvokeCommandOnContextMenuInternal(hWnd, pUnk, pCM, fCMIC, fCMF, pszVerb, pwszDir, true);
378}
379
380/*************************************************************************
381 * SHInvokeCommandOnContextMenu [SHLWAPI.540]
382 */
388 _In_ IContextMenu* pCM,
389 _In_ UINT fCMIC,
390 _In_opt_ LPCSTR pszVerb)
391{
392 return SHInvokeCommandOnContextMenuEx(hWnd, pUnk, pCM, fCMIC, CMF_EXTENDEDVERBS, pszVerb, NULL);
393}
394
395static inline BOOL
397{
398 for (const signed char *pch = (const signed char *)psz; *pch; ++pch)
399 {
400 if (*pch < 0)
401 return FALSE;
402 }
403 return TRUE;
404}
405
406/*************************************************************************
407 * SHInvokeCommandsOnContextMenu [SHLWAPI.541]
408 */
413 _In_opt_ IUnknown *punkSite,
414 _In_ IContextMenu *pCM,
415 _In_ DWORD fMask,
416 _In_reads_opt_(cVerbs) PCSTR *pVerbs,
417 _In_ UINT cVerbs)
418{
419 HRESULT hr;
420 CMINVOKECOMMANDINFOEX ici;
421 WCHAR szVerbW[MAX_PATH];
422 HMENU hMenu = NULL;
423 UINT iVerb, idDefault = (UINT)-1;
424 PCSTR pszVerbA = NULL;
425
426 if (!pCM)
427 return E_INVALIDARG;
428
429 hMenu = CreatePopupMenu();
430 if (!hMenu)
431 return E_OUTOFMEMORY;
432
433 if (punkSite)
434 IUnknown_SetSite(pCM, punkSite);
435
436 hr = pCM->QueryContextMenu(hMenu, 0, 1, MAXSHORT, (cVerbs ? 0 : CMF_DEFAULTONLY));
437 if (FAILED(hr))
438 goto Cleanup;
439
440 if (!cVerbs)
441 {
442 idDefault = GetMenuDefaultItem(hMenu, FALSE, 0);
443 if (idDefault != (UINT)-1)
444 pszVerbA = MAKEINTRESOURCEA(idDefault - 1);
445 }
446
447 ZeroMemory(&ici, sizeof(ici));
448 ici.cbSize = sizeof(ici);
449 ici.hwnd = hwnd;
450 ici.nShow = SW_SHOWNORMAL;
451
452 iVerb = 0;
453 do
454 {
455 if (cVerbs)
456 pszVerbA = pVerbs[iVerb];
457
458 if (!pszVerbA && idDefault == (UINT)-1)
459 {
460 hr = E_FAIL;
461 break;
462 }
463
464 ici.fMask = fMask;
465 ici.lpVerb = pszVerbA;
466 ici.lpVerbW = NULL;
467
468 if (idDefault == (UINT)-1 && !IS_INTRESOURCE(pszVerbA) && IsTextAsciiOnly(pszVerbA))
469 {
470 size_t ich;
471 for (ich = 0; pszVerbA[ich] && ich + 1 < _countof(szVerbW); ++ich)
472 {
473 szVerbW[ich] = (BYTE)pszVerbA[ich];
474 }
475 szVerbW[ich] = UNICODE_NULL;
476
477 ici.lpVerbW = szVerbW;
478 ici.fMask |= CMIC_MASK_UNICODE;
479 }
480
481 hr = pCM->InvokeCommand((LPCMINVOKECOMMANDINFO)&ici);
482
484 break;
485
486 ++iVerb;
487 } while (iVerb < cVerbs);
488
489Cleanup:
490 if (punkSite)
492 DestroyMenu(hMenu);
493 return hr;
494}
495
496/*************************************************************************
497 * SHInvokeCommandWithFlagsAndSite [SHLWAPI.571]
498 */
504 _In_ IShellFolder* pShellFolder,
505 _In_ LPCITEMIDLIST pidl,
506 _In_ UINT fCMIC,
507 _In_opt_ LPCSTR pszVerb)
508{
510 if (pShellFolder)
511 {
512 IContextMenu *pCM;
513 hr = pShellFolder->GetUIObjectOf(hWnd, 1, &pidl, IID_IContextMenu, NULL, (void**)&pCM);
514 if (SUCCEEDED(hr))
515 {
516 fCMIC |= CMIC_MASK_FLAG_LOG_USAGE;
517 hr = SHInvokeCommandOnContextMenuEx(hWnd, pUnk, pCM, fCMIC, 0, pszVerb, NULL);
518 pCM->Release();
519 }
520 }
521 return hr;
522}
523
524
525/*************************************************************************
526 * IContextMenu_Invoke [SHLWAPI.207]
527 *
528 * Used by Win:SHELL32!CISFBand::_TrySimpleInvoke.
529 */
533 _In_ IContextMenu *pContextMenu,
534 _In_ HWND hwnd,
535 _In_ LPCSTR lpVerb,
537{
538 TRACE("(%p, %p, %s, %u)\n", pContextMenu, hwnd, debugstr_a(lpVerb), uFlags);
540 uFlags, lpVerb, NULL, false);
541 return !FAILED_UNEXPECTEDLY(hr);
542}
543
544/*************************************************************************
545 * ShellExecuteCommand [INTERNAL]
546 */
547static HRESULT
549{
550 WCHAR szCmd[MAX_PATH * 2];
552 if (len <= 0) // Could not resolve the command, just use the input
553 {
554 HRESULT hr = StringCchCopyW(szCmd, _countof(szCmd), Command);
555 if (FAILED(hr))
556 return hr;
557 }
558 PWSTR pszArgs = PathGetArgsW(szCmd);
559 PathRemoveArgsW(szCmd);
560 PathUnquoteSpacesW(szCmd);
561
562 SHELLEXECUTEINFOW sei = { sizeof(sei), Flags, hWnd, NULL, szCmd, pszArgs };
563 sei.nShow = SW_SHOW;
566}
567
568/*************************************************************************
569 * RunRegCommand [SHLWAPI.469]
570 */
573{
574 WCHAR szCmd[MAX_PATH * 2];
575 DWORD cb = sizeof(szCmd);
577 if (error)
580}
581
582/*************************************************************************
583 * RunIndirectRegCommand [SHLWAPI.468]
584 */
587{
588 WCHAR szKey[MAX_PATH];
589 HRESULT hr;
590 if (pszSubKey)
591 hr = StringCchPrintfW(szKey, _countof(szKey), L"%s\\shell\\%s\\command", pszSubKey, pszVerb);
592 else
593 hr = StringCchPrintfW(szKey, _countof(szKey), L"shell\\%s\\command", pszVerb);
594 return SUCCEEDED(hr) ? RunRegCommand(hWnd, hKey, szKey) : hr;
595}
596
597/*************************************************************************
598 * SHRunIndirectRegClientCommand [SHLWAPI.467]
599 */
602{
603 WCHAR szKey[MAX_PATH], szClient[MAX_PATH];
604 HRESULT hr = StringCchPrintfW(szKey, _countof(szKey), L"Software\\Clients\\%s", pszClientType);
605 if (FAILED(hr))
606 return hr;
607
608 // Find the default client
609 DWORD error, cb;
610 cb = sizeof(szClient);
612 if (error)
613 {
614 cb = sizeof(szClient);
617 if (error)
619 }
620
621 hr = StringCchPrintfW(szKey, _countof(szKey), L"Software\\Clients\\%s\\%s", pszClientType, szClient);
622 if (SUCCEEDED(hr))
624 return hr;
625}
626
627/*************************************************************************
628 * PathFileExistsDefExtAndAttributesW [SHLWAPI.511]
629 *
630 * @param pszPath The path string.
631 * @param dwWhich The WHICH_... flags.
632 * @param pdwFileAttributes A pointer to the file attributes. Optional.
633 * @return TRUE if successful.
634 */
637 _Inout_ LPWSTR pszPath,
638 _In_ DWORD dwWhich,
639 _Out_opt_ LPDWORD pdwFileAttributes)
640{
641 TRACE("(%s, 0x%lX, %p)\n", debugstr_w(pszPath), dwWhich, pdwFileAttributes);
642
643 if (pdwFileAttributes)
644 *pdwFileAttributes = INVALID_FILE_ATTRIBUTES;
645
646 if (!pszPath)
647 return FALSE;
648
649 if (!dwWhich || (*PathFindExtensionW(pszPath) && (dwWhich & WHICH_OPTIONAL)))
650 return PathFileExistsAndAttributesW(pszPath, pdwFileAttributes);
651
652 if (!PathFileExistsDefExtW(pszPath, dwWhich))
653 {
654 if (pdwFileAttributes)
655 *pdwFileAttributes = INVALID_FILE_ATTRIBUTES;
656 return FALSE;
657 }
658
659 if (pdwFileAttributes)
660 *pdwFileAttributes = GetFileAttributesW(pszPath);
661
662 return TRUE;
663}
664
665static inline BOOL
667{
668 return (hr == E_FAIL || hr == E_INVALIDARG || hr == E_NOTIMPL);
669}
670
671// Used for IShellFolder_GetDisplayNameOf
673{
674 SHGDNF uRemove;
675 SHGDNF uAdd;
677};
678static const RETRY_DATA g_RetryData[] =
679{
680 { SHGDN_FOREDITING, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
681 { SHGDN_FORADDRESSBAR, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
682 { SHGDN_NORMAL, SHGDN_FORPARSING, SFGDNO_RETRYALWAYS },
683 { SHGDN_FORPARSING, SHGDN_NORMAL, SFGDNO_RETRYWITHFORPARSING },
684 { SHGDN_INFOLDER, SHGDN_NORMAL, SFGDNO_RETRYALWAYS },
685};
686
687/*************************************************************************
688 * IShellFolder_GetDisplayNameOf [SHLWAPI.316]
689 *
690 * @note Don't confuse with <shobjidl.h> inline function of the same name.
691 * If the original call fails with the given uFlags, this function will
692 * retry with other flags to attempt retrieving any meaningful description.
693 */
696 _In_ IShellFolder *psf,
697 _In_ LPCITEMIDLIST pidl,
698 _In_ SHGDNF uFlags,
700 _In_ DWORD dwRetryFlags) // dwRetryFlags is an additional parameter
701{
702 HRESULT hr;
703
704 TRACE("(%p)->(%p, 0x%lX, %p, 0x%lX)\n", psf, pidl, uFlags, lpName, dwRetryFlags);
705
706 hr = psf->GetDisplayNameOf(pidl, uFlags, lpName);
708 return hr;
709
710 dwRetryFlags |= SFGDNO_RETRYALWAYS;
711
712 if ((uFlags & SHGDN_FORPARSING) == 0)
713 dwRetryFlags |= SFGDNO_RETRYWITHFORPARSING;
714
715 // Retry with other flags to get successful results
716 for (SIZE_T iEntry = 0; iEntry < _countof(g_RetryData); ++iEntry)
717 {
718 const RETRY_DATA *pData = &g_RetryData[iEntry];
719 if (!(dwRetryFlags & pData->dwRetryFlags))
720 continue;
721
722 SHGDNF uNewFlags = ((uFlags & ~pData->uRemove) | pData->uAdd);
723 if (uNewFlags == uFlags)
724 continue;
725
726 hr = psf->GetDisplayNameOf(pidl, uNewFlags, lpName);
728 break;
729
730 uFlags = uNewFlags; // Update flags every time
731 }
732
733 return hr;
734}
735
736/*************************************************************************
737 * IShellFolder_ParseDisplayName [SHLWAPI.317]
738 *
739 * @note Don't confuse with <shobjidl.h> inline function of the same name.
740 * This function is safer than IShellFolder::ParseDisplayName.
741 */
744 _In_ IShellFolder *psf,
745 _In_opt_ HWND hwndOwner,
746 _In_opt_ LPBC pbcReserved,
747 _In_ LPOLESTR lpszDisplayName,
748 _Out_opt_ ULONG *pchEaten,
750 _Out_opt_ ULONG *pdwAttributes)
751{
752 ULONG dummy1, dummy2;
753
754 TRACE("(%p)->(%p, %p, %s, %p, %p, %p)\n", psf, hwndOwner, pbcReserved,
755 debugstr_w(lpszDisplayName), pchEaten, ppidl, pdwAttributes);
756
757 if (!pdwAttributes)
758 {
759 dummy1 = 0;
760 pdwAttributes = &dummy1;
761 }
762
763 if (!pchEaten)
764 {
765 dummy2 = 0;
766 pchEaten = &dummy2;
767 }
768
769 if (ppidl)
770 *ppidl = NULL;
771
772 return psf->ParseDisplayName(hwndOwner, pbcReserved, lpszDisplayName, pchEaten,
773 ppidl, pdwAttributes);
774}
775
776/*************************************************************************
777 * IShellFolder_CompareIDs [SHLWAPI.551]
778 *
779 * @note Don't confuse with <shobjidl.h> inline function of the same name.
780 * This function tries IShellFolder2 if possible.
781 */
784 _In_ IShellFolder *psf,
788{
789 TRACE("(%p, %p, %p, %p)\n", psf, lParam, pidl1, pidl2);
790
791 if (lParam & ~(SIZE_T)SHCIDS_COLUMNMASK)
792 {
793 /* Try as IShellFolder2 if possible */
794 HRESULT hr = psf->QueryInterface(IID_IShellFolder2, (void **)&psf);
795 if (FAILED(hr))
796 lParam &= SHCIDS_COLUMNMASK;
797 else
798 psf->Release();
799 }
800
801 return psf->CompareIDs(lParam, pidl1, pidl2);
802}
803
804/*************************************************************************
805 * SHDialogProc [INTERNAL]
806 *
807 * Used in SHDialogBox below
808 */
809
810typedef struct tagSHDIALOG
811{
815
816static INT_PTR CALLBACK
818{
821 HWND hwndItem;
822 LRESULT ret;
823
824 if (uMsg == WM_INITDIALOG)
825 {
828 lParam = (LPARAM)pData->pThis;
829 }
830 else
831 {
833 }
834
835 if (pData && pData->fn)
836 {
837 result = pData->fn(pData->pThis, hWnd, uMsg, wParam, lParam);
838 if (result)
839 return result;
840 }
841
842 switch (uMsg)
843 {
844 case WM_INITDIALOG:
845 return TRUE;
846
847 case WM_COMMAND:
848 if (LOWORD(wParam) == IDHELP)
849 return FALSE;
850
851 hwndItem = GetDlgItem(hWnd, LOWORD(wParam));
852 if (!hwndItem)
853 return FALSE;
854
855 ret = SendMessageA(hwndItem, WM_GETDLGCODE, 0, 0);
857 return FALSE;
858
860 return TRUE;
861
862 default:
863 return FALSE;
864 }
865}
866
867/*************************************************************************
868 * SHDialogBox [SHLWAPI.277]
869 */
873 _In_ PCSTR lpTemplateName,
876 _In_opt_ PVOID pThis)
877{
878 SHDIALOG data = { fn, pThis };
879 return DialogBoxParamA(hInstance, lpTemplateName, hWndParent, SHDialogProc, (LPARAM)&data);
880}
#define PathUnExpandEnvStringsForUserW
UINT cchMax
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
HANDLE hUserToken
Definition: install.c:39
#define EXTERN_C
Definition: basetyps.h:12
HINSTANCE hInstance
Definition: charmap.c:19
WPARAM wParam
Definition: combotst.c:138
LPARAM lParam
Definition: combotst.c:139
#define ERROR_MORE_DATA
Definition: dderror.h:13
#define E_OUTOFMEMORY
Definition: ddrawi.h:100
#define E_INVALIDARG
Definition: ddrawi.h:101
#define E_NOTIMPL
Definition: ddrawi.h:99
#define E_FAIL
Definition: ddrawi.h:102
HRESULT hr
Definition: delayimp.cpp:582
#define ERROR_SUCCESS
Definition: deptool.c:10
static LSTATUS(WINAPI *pRegDeleteTreeW)(HKEY
#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
#define CALLBACK
Definition: compat.h:35
#define lstrlenW
Definition: compat.h:750
static const WCHAR version[]
Definition: asmname.c:66
DWORD WINAPI ExpandEnvironmentStringsA(IN LPCSTR lpSrc, IN LPSTR lpDst, IN DWORD nSize)
Definition: environ.c:372
DWORD WINAPI ExpandEnvironmentStringsW(IN LPCWSTR lpSrc, IN LPWSTR lpDst, IN DWORD nSize)
Definition: environ.c:492
DWORD WINAPI GetFileAttributesW(LPCWSTR lpFileName)
Definition: fileinfo.c:636
INT WINAPI LCMapStringA(LCID lcid, DWORD flags, LPCSTR src, INT srclen, LPSTR dst, INT dstlen)
Definition: locale.c:3834
INT WINAPI CompareStringA(LCID lcid, DWORD flags, LPCSTR str1, INT len1, LPCSTR str2, INT len2)
Definition: locale.c:4015
INT WINAPI CompareStringW(LCID lcid, DWORD flags, LPCWSTR str1, INT len1, LPCWSTR str2, INT len2)
Definition: locale.c:3946
INT WINAPI LCMapStringW(LCID lcid, DWORD flags, LPCWSTR src, INT srclen, LPWSTR dst, INT dstlen)
Definition: locale.c:3808
#define IS_INTRESOURCE(x)
Definition: loader.c:613
void WINAPI PathUnquoteSpacesW(WCHAR *path)
Definition: path.c:2006
LPWSTR WINAPI PathFindExtensionW(const WCHAR *path)
Definition: path.c:1274
WCHAR *WINAPI PathGetArgsW(const WCHAR *path)
Definition: path.c:1740
DWORD WINAPI GetVersion(void)
Definition: version.c:1458
static MonoProfilerRuntimeShutdownBeginCallback cb
Definition: metahost.c:118
HRESULT WINAPI IUnknown_SetSite(IUnknown *obj, IUnknown *site)
Definition: ordinal.c:1411
void WINAPI PathRemoveArgsW(LPWSTR lpszPath)
Definition: path.c:782
BOOL WINAPI PathFileExistsAndAttributesW(LPCWSTR lpszPath, DWORD *dwAttr)
Definition: path.c:1879
BOOL WINAPI PathFileExistsDefExtW(LPWSTR lpszPath, DWORD dwWhich)
Definition: path.c:1126
DWORD WINAPI SHAnsiToUnicode(LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen)
Definition: string.c:2803
INT WINAPI SHUnicodeToAnsi(LPCWSTR lpSrcStr, LPSTR lpDstStr, INT iLen)
Definition: string.c:2927
EXTERN_C HRESULT WINAPI RunRegCommand(_In_opt_ HWND hWnd, _In_ HKEY hKey, _In_opt_ PCWSTR pszSubKey)
Definition: utils.cpp:572
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:385
EXTERN_C LSTATUS WINAPI RegGetValueW(HKEY, LPCWSTR, LPCWSTR, DWORD, LPDWORD, PVOID, LPDWORD)
Definition: reg.c:1931
static BOOL IsTextAsciiOnly(PCSTR psz)
Definition: utils.cpp:396
EXTERN_C BOOL WINAPI PathUnExpandEnvStringsForUserA(_In_ HANDLE hUserToken, _In_ PCSTR pszPath, _Out_writes_(cchBuff) PSTR pszBuff, _In_ INT cchBuff)
Definition: utils.cpp:136
struct tagSHDIALOG * PSHDIALOG
EXTERN_C HRESULT WINAPI RunIndirectRegCommand(_In_opt_ HWND hWnd, _In_ HKEY hKey, _In_opt_ PCWSTR pszSubKey, _In_ PCWSTR pszVerb)
Definition: utils.cpp:586
static HRESULT ShellExecuteCommand(_In_opt_ HWND hWnd, _In_ PCWSTR Command, _In_opt_ UINT Flags)
Definition: utils.cpp:548
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:501
EXTERN_C PWSTR WINAPI CharLowerNoDBCSW(_Inout_ PWSTR lpString)
Definition: utils.cpp:258
static BOOL UnExpandEnvironmentStringForUserA(_In_ HANDLE hUserToken, _In_ PCSTR lpString, _In_ PCSTR lpSrc, _Out_ PSTR pszDest, _In_ INT cchDest)
Definition: utils.cpp:46
EXTERN_C INT_PTR WINAPI SHDialogBox(_In_opt_ HINSTANCE hInstance, _In_ PCSTR lpTemplateName, _In_opt_ HWND hWndParent, _In_opt_ SHDIALOGPROC fn, _In_opt_ PVOID pThis)
Definition: utils.cpp:871
EXTERN_C PSTR WINAPI CharUpperNoDBCSA(_Inout_ PSTR lpString)
Definition: utils.cpp:266
static BOOL UnExpandEnvironmentStringForUserW(_In_ HANDLE hUserToken, _In_ PCWSTR lpString, _In_ PCWSTR lpSrc, _Out_ PWSTR pszDest, _In_ INT cchDest)
Definition: utils.cpp:88
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:368
EXTERN_C BOOL WINAPI IContextMenu_Invoke(_In_ IContextMenu *pContextMenu, _In_ HWND hwnd, _In_ LPCSTR lpVerb, _In_ UINT uFlags)
Definition: utils.cpp:532
EXTERN_C PSTR WINAPI CharLowerNoDBCSA(_Inout_ PSTR lpString)
Definition: utils.cpp:250
static const RETRY_DATA g_RetryData[]
Definition: utils.cpp:678
EXTERN_C HRESULT WINAPI SHInvokeCommandsOnContextMenu(_In_opt_ HWND hwnd, _In_opt_ IUnknown *punkSite, _In_ IContextMenu *pCM, _In_ DWORD fMask, _In_reads_opt_(cVerbs) PCSTR *pVerbs, _In_ UINT cVerbs)
Definition: utils.cpp:411
struct tagSHDIALOG SHDIALOG
static BOOL CharLowerNoDBCSWWorker(PWSTR lpString, INT cchMax, BOOL bUppercase)
Definition: utils.cpp:234
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:280
#define FAILED_UNEXPECTEDLY
Definition: utils.cpp:31
static BOOL CharLowerNoDBCSAWorker(PSTR lpString, INT cchMax, BOOL bUppercase)
Definition: utils.cpp:221
EXTERN_C HRESULT WINAPI SHRunIndirectRegClientCommand(_In_opt_ HWND hWnd, _In_ PCWSTR pszClientType)
Definition: utils.cpp:601
#define IShellFolder_ParseDisplayName
Definition: utils.cpp:14
#define IShellFolder_CompareIDs
Definition: utils.cpp:15
static WORD GetVersionMajorMinor()
Definition: utils.cpp:39
EXTERN_C PWSTR WINAPI CharUpperNoDBCSW(_Inout_ PWSTR lpString)
Definition: utils.cpp:274
BOOL WINAPI PathFileExistsDefExtAndAttributesW(_Inout_ LPWSTR pszPath, _In_ DWORD dwWhich, _Out_opt_ LPDWORD pdwFileAttributes)
Definition: utils.cpp:636
static INT_PTR CALLBACK SHDialogProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
Definition: utils.cpp:817
#define IShellFolder_GetDisplayNameOf
Definition: utils.cpp:13
static BOOL SHLWAPI_IsBogusHRESULT(HRESULT hr)
Definition: utils.cpp:666
BOOL WINAPI ExpandEnvironmentStringsForUserW(IN HANDLE hToken, IN LPCWSTR lpSrc, OUT LPWSTR lpDest, IN DWORD dwSize)
Definition: environment.c:743
BOOL WINAPI ExpandEnvironmentStringsForUserA(IN HANDLE hToken, IN LPCSTR lpSrc, OUT LPSTR lpDest, IN DWORD dwSize)
Definition: environment.c:794
static const WCHAR Cleanup[]
Definition: register.c:80
static void *static void *static LPDIRECTPLAY IUnknown * pUnk
Definition: dplayx.c:30
return ret
Definition: mutex.c:146
#define L(x)
Definition: resources.c:13
unsigned short WORD
Definition: ntddk_ex.h:93
unsigned int BOOL
Definition: ntddk_ex.h:94
unsigned long DWORD
Definition: ntddk_ex.h:95
FxAutoRegKey hKey
GLint GLenum GLsizei GLsizei GLsizei GLint GLsizei const GLvoid * data
Definition: gl.h:1950
GLuint64EXT * result
Definition: glext.h:11304
GLenum GLsizei len
Definition: glext.h:6722
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
int WINAPI lstrlenA(LPCSTR lpString)
Definition: lstring.c:145
#define ZeroMemory
Definition: minwinbase.h:31
LONG_PTR LPARAM
Definition: minwindef.h:175
LONG_PTR LRESULT
Definition: minwindef.h:176
UINT_PTR WPARAM
Definition: minwindef.h:174
#define error(str)
Definition: mkdosfs.c:1605
#define pch(ap)
Definition: match.c:418
static BSTR *static LPOLESTR
Definition: varformat.c:44
static const CLSID *static CLSID *static const GUID VARIANT VARIANT *static IServiceProvider DWORD *static HMENU
Definition: ordinal.c:60
unsigned int UINT
Definition: ndis.h:50
#define _Out_opt_
Definition: no_sal2.h:214
#define _Inout_
Definition: no_sal2.h:162
#define _Out_writes_(s)
Definition: no_sal2.h:176
#define _Out_
Definition: no_sal2.h:160
#define _In_reads_opt_(s)
Definition: no_sal2.h:222
#define _In_
Definition: no_sal2.h:158
#define _In_opt_
Definition: no_sal2.h:212
#define LOCALE_SYSTEM_DEFAULT
#define UNICODE_NULL
#define ANSI_NULL
interface IBindCtx * LPBC
Definition: objfwd.h:18
#define LOWORD(l)
Definition: pedump.c:82
short WCHAR
Definition: pedump.c:58
char CHAR
Definition: pedump.c:57
_In_ INT cchDest
Definition: shlwapi.h:1151
_In_opt_ LPCSTR pszSubKey
Definition: shlwapi.h:783
#define IDHELP
Definition: resource_2.h:8
#define _WIN32_WINNT_WIN7
Definition: sdkddkver.h:28
#define SEE_MASK_FLAG_LOG_USAGE
Definition: shellapi.h:55
_In_ UINT _In_ UINT cch
Definition: shellapi.h:432
BOOL WINAPI DECLSPEC_HOTPATCH ShellExecuteExW(LPSHELLEXECUTEINFOW sei)
Definition: shlexec.cpp:2723
#define PPCF_FORCEQUALIFY
Definition: shlobj.h:2429
LONG WINAPI PathProcessCommand(_In_ LPCWSTR, _Out_writes_(cchDest) LPWSTR, int cchDest, DWORD)
#define PPCF_ADDARGUMENTS
Definition: shlobj.h:2426
INT_PTR(CALLBACK * SHDIALOGPROC)(PVOID pThis, HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
#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
STRSAFEAPI StringCchPrintfW(STRSAFE_LPWSTR pszDest, size_t cchDest, STRSAFE_LPCWSTR pszFormat,...)
Definition: strsafe.h:530
STRSAFEAPI StringCchCatA(STRSAFE_LPSTR pszDest, size_t cchDest, STRSAFE_LPCSTR pszSrc)
Definition: strsafe.h:320
STRSAFEAPI StringCchCatW(STRSAFE_LPWSTR pszDest, size_t cchDest, STRSAFE_LPCWSTR pszSrc)
Definition: strsafe.h:325
STRSAFEAPI StringCchCopyW(STRSAFE_LPWSTR pszDest, size_t cchDest, STRSAFE_LPCWSTR pszSrc)
Definition: strsafe.h:149
STRSAFEAPI StringCchCopyA(STRSAFE_LPSTR pszDest, size_t cchDest, STRSAFE_LPCSTR pszSrc)
Definition: strsafe.h:145
Definition: shell.h:41
SHGDNF uAdd
Definition: utils.cpp:675
SHGDNF uRemove
Definition: utils.cpp:674
DWORD dwRetryFlags
Definition: utils.cpp:676
SHDIALOGPROC fn
Definition: utils.cpp:812
PVOID pThis
Definition: utils.cpp:813
TW_UINT32 TW_UINT16 TW_UINT16 TW_MEMREF pData
Definition: twain.h:1830
uint16_t * PWSTR
Definition: typedefs.h:56
const char * LPCSTR
Definition: typedefs.h:52
int32_t INT_PTR
Definition: typedefs.h:64
char * PSTR
Definition: typedefs.h:51
const uint16_t * PCWSTR
Definition: typedefs.h:57
const uint16_t * LPCWSTR
Definition: typedefs.h:57
#define MAKEWORD(a, b)
Definition: typedefs.h:248
uint16_t * LPWSTR
Definition: typedefs.h:56
ULONG_PTR SIZE_T
Definition: typedefs.h:80
uint32_t * LPDWORD
Definition: typedefs.h:59
int32_t INT
Definition: typedefs.h:58
const char * PCSTR
Definition: typedefs.h:52
uint32_t ULONG
Definition: typedefs.h:59
#define MAXSHORT
Definition: umtypes.h:114
#define INVALID_FILE_ATTRIBUTES
Definition: vfdcmd.c:23
static GLenum _GLUfuncptr fn
Definition: wgl_font.c:159
DWORD WINAPI GetLastError(void)
Definition: except.c:1042
_In_ LPCSTR lpName
Definition: winbase.h:2543
_In_ LONG _In_ HWND hwnd
Definition: winddi.h:4023
HICON HCURSOR
Definition: windef.h:99
#define WINAPI
Definition: msvc.h:6
#define ERROR_BUFFER_OVERFLOW
Definition: winerror.h:307
static HRESULT HRESULT_FROM_WIN32(unsigned int x)
Definition: winerror.h:210
#define ERROR_CANCELLED
Definition: winerror.h:1055
#define NORM_IGNORECASE
Definition: winnls.h:187
#define LCMAP_UPPERCASE
Definition: winnls.h:198
#define CSTR_EQUAL
Definition: winnls.h:500
#define LCMAP_LOWERCASE
Definition: winnls.h:197
#define HKEY_LOCAL_MACHINE
Definition: winreg.h:12
#define HKEY_CURRENT_USER
Definition: winreg.h:11
#define RRF_RT_REG_SZ
Definition: winreg.h:58
#define SW_SHOWNORMAL
Definition: winuser.h:781
#define SetWindowLongPtrA
Definition: winuser.h:5511
HMENU WINAPI CreatePopupMenu(void)
Definition: menu.c:838
UINT WINAPI GetMenuDefaultItem(_In_ HMENU hMenu, _In_ UINT fByPos, _In_ UINT gmdiFlags)
#define DWLP_USER
Definition: winuser.h:883
#define WM_COMMAND
Definition: winuser.h:1768
#define DLGC_UNDEFPUSHBUTTON
Definition: winuser.h:2658
LRESULT WINAPI SendMessageA(_In_ HWND, _In_ UINT, _In_ WPARAM, _In_ LPARAM)
HCURSOR WINAPI SetCursor(_In_opt_ HCURSOR)
#define GetWindowLongPtrA
Definition: winuser.h:4982
#define WM_INITDIALOG
Definition: winuser.h:1767
HCURSOR WINAPI LoadCursorW(_In_opt_ HINSTANCE, _In_ LPCWSTR)
Definition: cursoricon.c:2474
#define DLGC_DEFPUSHBUTTON
Definition: winuser.h:2657
HWND WINAPI GetDlgItem(_In_opt_ HWND, _In_ int)
#define MAKEINTRESOURCEA(i)
Definition: winuser.h:581
BOOL WINAPI DestroyMenu(_In_ HMENU)
#define IDC_WAIT
Definition: winuser.h:697
#define SW_SHOW
Definition: winuser.h:786
#define MAKEINTRESOURCEW(i)
Definition: winuser.h:582
#define SW_NORMAL
Definition: winuser.h:780
#define WM_GETDLGCODE
Definition: winuser.h:1717
INT_PTR WINAPI DialogBoxParamA(_In_opt_ HINSTANCE, _In_ LPCSTR, _In_opt_ HWND, _In_opt_ DLGPROC, _In_ LPARAM)
BOOL WINAPI EndDialog(_In_ HWND, _In_ INT_PTR)
_Must_inspect_result_ _In_ ULONG Flags
Definition: wsk.h:170
unsigned char BYTE
Definition: xxhash.c:193