2011-05-15 15:55:49 +00:00
|
|
|
/*
|
|
|
|
* Virtual Workplace folder
|
|
|
|
*
|
|
|
|
* Copyright 1997 Marcus Meissner
|
|
|
|
* Copyright 1998, 1999, 2002 Juergen Schmied
|
|
|
|
* Copyright 2009 Andrew Hill
|
2024-04-14 18:25:20 +00:00
|
|
|
* Copyright 2017-2024 Katayama Hirofumi MZ
|
2011-05-15 15:55:49 +00:00
|
|
|
*
|
|
|
|
* 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 <precomp.h>
|
2019-11-20 01:00:26 +00:00
|
|
|
#include <process.h>
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2024-02-20 01:08:56 +00:00
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shell);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
CDrivesFolder should create a CRegFolder to represent the virtual items that exist only in
|
|
|
|
the registry. The CRegFolder is aggregated by the CDrivesFolder.
|
|
|
|
The CDrivesFolderEnum class should enumerate only drives on the system. Since the CRegFolder
|
|
|
|
implementation of IShellFolder::EnumObjects enumerates the virtual items, the
|
|
|
|
CDrivesFolderEnum is only responsible for returning the physical items.
|
|
|
|
|
|
|
|
2. At least on my XP system, the drive pidls returned are of type PT_DRIVE1, not PT_DRIVE
|
|
|
|
3. The parsing name returned for my computer is incorrect. It should be "My Computer"
|
|
|
|
*/
|
|
|
|
|
2017-06-11 14:02:14 +00:00
|
|
|
static int iDriveIconIds[7] = { IDI_SHELL_DRIVE, /* DRIVE_UNKNOWN */
|
|
|
|
IDI_SHELL_CDROM, /* DRIVE_NO_ROOT_DIR*/
|
|
|
|
IDI_SHELL_3_14_FLOPPY, /* DRIVE_REMOVABLE*/
|
|
|
|
IDI_SHELL_DRIVE, /* DRIVE_FIXED*/
|
|
|
|
IDI_SHELL_NETDRIVE, /* DRIVE_REMOTE*/
|
|
|
|
IDI_SHELL_CDROM, /* DRIVE_CDROM*/
|
|
|
|
IDI_SHELL_RAMDISK /* DRIVE_RAMDISK*/
|
|
|
|
};
|
|
|
|
|
|
|
|
static int iDriveTypeIds[7] = { IDS_DRIVE_FIXED, /* DRIVE_UNKNOWN */
|
|
|
|
IDS_DRIVE_FIXED, /* DRIVE_NO_ROOT_DIR*/
|
|
|
|
IDS_DRIVE_FLOPPY, /* DRIVE_REMOVABLE*/
|
|
|
|
IDS_DRIVE_FIXED, /* DRIVE_FIXED*/
|
|
|
|
IDS_DRIVE_NETWORK, /* DRIVE_REMOTE*/
|
|
|
|
IDS_DRIVE_CDROM, /* DRIVE_CDROM*/
|
|
|
|
IDS_DRIVE_FIXED /* DRIVE_RAMDISK*/
|
|
|
|
};
|
|
|
|
|
2023-07-20 07:53:04 +00:00
|
|
|
BOOL _ILGetDriveType(LPCITEMIDLIST pidl)
|
|
|
|
{
|
|
|
|
char szDrive[8];
|
|
|
|
if (!_ILGetDrive(pidl, szDrive, _countof(szDrive)))
|
|
|
|
{
|
|
|
|
ERR("pidl %p is not a drive\n", pidl);
|
|
|
|
return DRIVE_UNKNOWN;
|
|
|
|
}
|
|
|
|
return ::GetDriveTypeA(szDrive);
|
|
|
|
}
|
|
|
|
|
2011-05-15 15:55:49 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* IShellFolder implementation
|
|
|
|
*/
|
|
|
|
|
2017-11-18 14:55:08 +00:00
|
|
|
#define RETRY_COUNT 3
|
|
|
|
#define RETRY_SLEEP 250
|
|
|
|
static BOOL TryToLockOrUnlockDrive(HANDLE hDrive, BOOL bLock)
|
|
|
|
{
|
|
|
|
DWORD dwError, dwBytesReturned;
|
|
|
|
DWORD dwCode = (bLock ? FSCTL_LOCK_VOLUME : FSCTL_UNLOCK_VOLUME);
|
|
|
|
for (DWORD i = 0; i < RETRY_COUNT; ++i)
|
|
|
|
{
|
|
|
|
if (DeviceIoControl(hDrive, dwCode, NULL, 0, NULL, 0, &dwBytesReturned, NULL))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
dwError = GetLastError();
|
|
|
|
if (dwError == ERROR_INVALID_FUNCTION)
|
|
|
|
break; /* don't sleep if function is not implemented */
|
|
|
|
|
|
|
|
Sleep(RETRY_SLEEP);
|
|
|
|
}
|
|
|
|
SetLastError(dwError);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: See also https://support.microsoft.com/en-us/help/165721/how-to-ejecting-removable-media-in-windows-nt-windows-2000-windows-xp
|
|
|
|
static BOOL DoEjectDrive(const WCHAR *physical, UINT nDriveType, INT *pnStringID)
|
|
|
|
{
|
|
|
|
/* GENERIC_WRITE isn't needed for umount */
|
|
|
|
DWORD dwAccessMode = GENERIC_READ;
|
|
|
|
DWORD dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
|
|
|
|
|
|
|
|
HANDLE hDrive = CreateFile(physical, dwAccessMode, dwShareMode, 0, OPEN_EXISTING, 0, NULL);
|
|
|
|
if (hDrive == INVALID_HANDLE_VALUE)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
BOOL bResult, bNeedUnlock = FALSE;
|
|
|
|
DWORD dwBytesReturned, dwError = NO_ERROR;
|
|
|
|
PREVENT_MEDIA_REMOVAL removal;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
bResult = TryToLockOrUnlockDrive(hDrive, TRUE);
|
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
dwError = GetLastError();
|
|
|
|
*pnStringID = IDS_CANTLOCKVOLUME; /* Unable to lock volume */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
bResult = DeviceIoControl(hDrive, FSCTL_DISMOUNT_VOLUME, NULL, 0, NULL, 0, &dwBytesReturned, NULL);
|
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
dwError = GetLastError();
|
|
|
|
*pnStringID = IDS_CANTDISMOUNTVOLUME; /* Unable to dismount volume */
|
|
|
|
bNeedUnlock = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
removal.PreventMediaRemoval = FALSE;
|
|
|
|
bResult = DeviceIoControl(hDrive, IOCTL_STORAGE_MEDIA_REMOVAL, &removal, sizeof(removal), NULL,
|
|
|
|
0, &dwBytesReturned, NULL);
|
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
*pnStringID = IDS_CANTEJECTMEDIA; /* Unable to eject media */
|
|
|
|
dwError = GetLastError();
|
|
|
|
bNeedUnlock = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
bResult = DeviceIoControl(hDrive, IOCTL_STORAGE_EJECT_MEDIA, NULL, 0, NULL, 0, &dwBytesReturned, NULL);
|
|
|
|
if (!bResult)
|
|
|
|
{
|
|
|
|
*pnStringID = IDS_CANTEJECTMEDIA; /* Unable to eject media */
|
|
|
|
dwError = GetLastError();
|
|
|
|
bNeedUnlock = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
if (bNeedUnlock)
|
|
|
|
{
|
|
|
|
TryToLockOrUnlockDrive(hDrive, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle(hDrive);
|
|
|
|
|
|
|
|
SetLastError(dwError);
|
|
|
|
return bResult;
|
|
|
|
}
|
|
|
|
|
2019-11-20 01:00:26 +00:00
|
|
|
// A callback function for finding the stub windows.
|
|
|
|
static BOOL CALLBACK
|
|
|
|
EnumStubProc(HWND hwnd, LPARAM lParam)
|
|
|
|
{
|
|
|
|
CSimpleArray<HWND> *pStubs = reinterpret_cast<CSimpleArray<HWND> *>(lParam);
|
|
|
|
|
|
|
|
WCHAR szClass[64];
|
|
|
|
GetClassNameW(hwnd, szClass, _countof(szClass));
|
|
|
|
|
|
|
|
if (lstrcmpiW(szClass, L"StubWindow32") == 0)
|
|
|
|
{
|
|
|
|
pStubs->Add(hwnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Another callback function to find the owned window of the stub window.
|
|
|
|
static BOOL CALLBACK
|
|
|
|
EnumStubProc2(HWND hwnd, LPARAM lParam)
|
|
|
|
{
|
|
|
|
HWND *phwnd = reinterpret_cast<HWND *>(lParam);
|
|
|
|
|
|
|
|
if (phwnd[0] == GetWindow(hwnd, GW_OWNER))
|
|
|
|
{
|
|
|
|
phwnd[1] = hwnd;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parameters for format_drive_thread function below.
|
|
|
|
struct THREAD_PARAMS
|
|
|
|
{
|
|
|
|
UINT nDriveNumber;
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned __stdcall format_drive_thread(void *args)
|
|
|
|
{
|
|
|
|
THREAD_PARAMS *params = (THREAD_PARAMS *)args;
|
|
|
|
UINT nDriveNumber = params->nDriveNumber;
|
|
|
|
LONG_PTR nProp = nDriveNumber | 0x7F00;
|
|
|
|
|
|
|
|
// Search the stub windows that already exist.
|
|
|
|
CSimpleArray<HWND> old_stubs;
|
|
|
|
EnumWindows(EnumStubProc, (LPARAM)&old_stubs);
|
|
|
|
|
|
|
|
for (INT n = 0; n < old_stubs.GetSize(); ++n)
|
|
|
|
{
|
|
|
|
HWND hwndStub = old_stubs[n];
|
|
|
|
|
|
|
|
// The target stub window has the prop.
|
|
|
|
if (GetPropW(hwndStub, L"DriveNumber") == (HANDLE)nProp)
|
|
|
|
{
|
|
|
|
// Found.
|
|
|
|
HWND ahwnd[2];
|
|
|
|
ahwnd[0] = hwndStub;
|
|
|
|
ahwnd[1] = NULL;
|
|
|
|
EnumWindows(EnumStubProc2, (LPARAM)ahwnd);
|
|
|
|
|
|
|
|
// Activate.
|
|
|
|
BringWindowToTop(ahwnd[1]);
|
|
|
|
|
|
|
|
delete params;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a stub window.
|
|
|
|
DWORD style = WS_DISABLED | WS_CLIPSIBLINGS | WS_CAPTION;
|
|
|
|
DWORD exstyle = WS_EX_WINDOWEDGE | WS_EX_APPWINDOW;
|
|
|
|
CStubWindow32 stub;
|
|
|
|
if (!stub.Create(NULL, NULL, NULL, style, exstyle))
|
|
|
|
{
|
|
|
|
ERR("StubWindow32 creation failed\n");
|
|
|
|
delete params;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add prop to the target stub window.
|
|
|
|
SetPropW(stub, L"DriveNumber", (HANDLE)nProp);
|
|
|
|
|
|
|
|
// Do format.
|
|
|
|
SHFormatDrive(stub, nDriveNumber, SHFMT_ID_DEFAULT, 0);
|
|
|
|
|
|
|
|
// Clean up.
|
|
|
|
RemovePropW(stub, L"DriveNumber");
|
|
|
|
stub.DestroyWindow();
|
|
|
|
delete params;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT DoFormatDrive(HWND hwnd, UINT nDriveNumber)
|
|
|
|
{
|
|
|
|
THREAD_PARAMS *params = new THREAD_PARAMS;
|
|
|
|
params->nDriveNumber = nDriveNumber;
|
|
|
|
|
|
|
|
// Create thread to avoid locked.
|
|
|
|
unsigned tid;
|
|
|
|
HANDLE hThread = (HANDLE)_beginthreadex(NULL, 0, format_drive_thread, params, 0, &tid);
|
|
|
|
if (hThread == NULL)
|
|
|
|
{
|
|
|
|
delete params;
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle(hThread);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2016-11-17 14:35:19 +00:00
|
|
|
HRESULT CALLBACK DrivesContextMenuCallback(IShellFolder *psf,
|
|
|
|
HWND hwnd,
|
|
|
|
IDataObject *pdtobj,
|
|
|
|
UINT uMsg,
|
|
|
|
WPARAM wParam,
|
|
|
|
LPARAM lParam)
|
|
|
|
{
|
|
|
|
if (uMsg != DFM_MERGECONTEXTMENU && uMsg != DFM_INVOKECOMMAND)
|
2024-05-09 17:52:05 +00:00
|
|
|
return SHELL32_DefaultContextMenuCallBack(psf, pdtobj, uMsg);
|
2016-11-17 14:35:19 +00:00
|
|
|
|
|
|
|
PIDLIST_ABSOLUTE pidlFolder;
|
|
|
|
PUITEMID_CHILD *apidl;
|
|
|
|
UINT cidl;
|
2017-11-18 14:55:08 +00:00
|
|
|
UINT nDriveType;
|
|
|
|
DWORD dwFlags;
|
2016-11-17 14:35:19 +00:00
|
|
|
HRESULT hr = SH_GetApidlFromDataObject(pdtobj, &pidlFolder, &apidl, &cidl);
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
char szDrive[8] = {0};
|
|
|
|
if (!_ILGetDrive(apidl[0], szDrive, sizeof(szDrive)))
|
|
|
|
{
|
|
|
|
ERR("pidl is not a drive\n");
|
|
|
|
SHFree(pidlFolder);
|
|
|
|
_ILFreeaPidl(apidl, cidl);
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
2017-11-18 14:55:08 +00:00
|
|
|
nDriveType = GetDriveTypeA(szDrive);
|
|
|
|
GetVolumeInformationA(szDrive, NULL, 0, NULL, NULL, &dwFlags, NULL, 0);
|
|
|
|
|
|
|
|
// custom command IDs
|
2022-11-08 14:52:04 +00:00
|
|
|
#if 0 // Disabled until our menu building system is fixed
|
|
|
|
#define CMDID_FORMAT 0
|
|
|
|
#define CMDID_EJECT 1
|
|
|
|
#define CMDID_DISCONNECT 2
|
|
|
|
#else
|
|
|
|
/* FIXME: These IDs should start from 0, however there is difference
|
|
|
|
* between ours and Windows' menu building systems, which should be fixed. */
|
2017-11-18 14:55:08 +00:00
|
|
|
#define CMDID_FORMAT 1
|
|
|
|
#define CMDID_EJECT 2
|
|
|
|
#define CMDID_DISCONNECT 3
|
2022-11-08 14:52:04 +00:00
|
|
|
#endif
|
2016-11-17 14:35:19 +00:00
|
|
|
|
|
|
|
if (uMsg == DFM_MERGECONTEXTMENU)
|
|
|
|
{
|
|
|
|
QCMINFO *pqcminfo = (QCMINFO *)lParam;
|
|
|
|
|
2017-11-18 14:55:08 +00:00
|
|
|
UINT idCmdFirst = pqcminfo->idCmdFirst;
|
2022-11-08 14:52:04 +00:00
|
|
|
UINT idCmd = 0;
|
2017-11-18 14:55:08 +00:00
|
|
|
if (!(dwFlags & FILE_READ_ONLY_VOLUME) && nDriveType != DRIVE_REMOTE)
|
2016-11-17 14:35:19 +00:00
|
|
|
{
|
2017-11-18 14:55:08 +00:00
|
|
|
/* add separator and Format */
|
2022-11-08 14:52:04 +00:00
|
|
|
idCmd = idCmdFirst + CMDID_FORMAT;
|
2017-11-18 14:55:08 +00:00
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, 0, MFT_SEPARATOR, NULL, 0);
|
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, idCmd, MFT_STRING, MAKEINTRESOURCEW(IDS_FORMATDRIVE), MFS_ENABLED);
|
|
|
|
}
|
|
|
|
if (nDriveType == DRIVE_REMOVABLE || nDriveType == DRIVE_CDROM)
|
|
|
|
{
|
|
|
|
/* add separator and Eject */
|
2022-11-08 14:52:04 +00:00
|
|
|
idCmd = idCmdFirst + CMDID_EJECT;
|
2017-11-18 14:55:08 +00:00
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, 0, MFT_SEPARATOR, NULL, 0);
|
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, idCmd, MFT_STRING, MAKEINTRESOURCEW(IDS_EJECT), MFS_ENABLED);
|
|
|
|
}
|
|
|
|
if (nDriveType == DRIVE_REMOTE)
|
|
|
|
{
|
|
|
|
/* add separator and Disconnect */
|
2022-11-08 14:52:04 +00:00
|
|
|
idCmd = idCmdFirst + CMDID_DISCONNECT;
|
2017-11-18 14:55:08 +00:00
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, 0, MFT_SEPARATOR, NULL, 0);
|
|
|
|
_InsertMenuItemW(pqcminfo->hmenu, pqcminfo->indexMenu++, TRUE, idCmd, MFT_STRING, MAKEINTRESOURCEW(IDS_DISCONNECT), MFS_ENABLED);
|
2016-11-17 14:35:19 +00:00
|
|
|
}
|
2017-11-18 14:55:08 +00:00
|
|
|
|
2022-11-08 14:52:04 +00:00
|
|
|
if (idCmd)
|
|
|
|
#if 0 // see FIXME above
|
|
|
|
pqcminfo->idCmdFirst = ++idCmd;
|
|
|
|
#else
|
|
|
|
pqcminfo->idCmdFirst = (idCmd + 2);
|
|
|
|
#endif
|
2024-05-09 17:52:05 +00:00
|
|
|
hr = S_OK;
|
2016-11-17 14:35:19 +00:00
|
|
|
}
|
|
|
|
else if (uMsg == DFM_INVOKECOMMAND)
|
|
|
|
{
|
2017-11-18 14:55:08 +00:00
|
|
|
WCHAR wszBuf[4] = L"A:\\";
|
|
|
|
wszBuf[0] = (WCHAR)szDrive[0];
|
|
|
|
|
|
|
|
INT nStringID = 0;
|
|
|
|
DWORD dwError = NO_ERROR;
|
|
|
|
|
2017-01-24 16:35:36 +00:00
|
|
|
if (wParam == DFM_CMD_PROPERTIES)
|
2016-11-17 14:35:19 +00:00
|
|
|
{
|
2022-09-22 17:27:33 +00:00
|
|
|
// pdtobj should be valid at this point!
|
|
|
|
ATLASSERT(pdtobj);
|
2022-11-20 18:10:11 +00:00
|
|
|
hr = SH_ShowDriveProperties(wszBuf, pdtobj) ? S_OK : E_UNEXPECTED;
|
2017-11-25 09:27:20 +00:00
|
|
|
if (FAILED(hr))
|
2017-11-18 14:55:08 +00:00
|
|
|
{
|
|
|
|
dwError = ERROR_CAN_NOT_COMPLETE;
|
|
|
|
nStringID = IDS_CANTSHOWPROPERTIES;
|
|
|
|
}
|
2016-11-17 14:35:19 +00:00
|
|
|
}
|
2017-01-24 16:35:36 +00:00
|
|
|
else
|
|
|
|
{
|
2017-11-18 14:55:08 +00:00
|
|
|
if (wParam == CMDID_FORMAT)
|
|
|
|
{
|
2019-11-20 01:00:26 +00:00
|
|
|
hr = DoFormatDrive(hwnd, szDrive[0] - 'A');
|
2017-11-18 14:55:08 +00:00
|
|
|
}
|
|
|
|
else if (wParam == CMDID_EJECT)
|
|
|
|
{
|
|
|
|
/* do eject */
|
|
|
|
WCHAR physical[10];
|
|
|
|
wsprintfW(physical, _T("\\\\.\\%c:"), szDrive[0]);
|
|
|
|
|
|
|
|
if (DoEjectDrive(physical, nDriveType, &nStringID))
|
|
|
|
{
|
|
|
|
SHChangeNotify(SHCNE_MEDIAREMOVED, SHCNF_PATHW | SHCNF_FLUSHNOWAIT, wszBuf, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dwError = GetLastError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (wParam == CMDID_DISCONNECT)
|
|
|
|
{
|
|
|
|
/* do disconnect */
|
2017-11-18 20:26:18 +00:00
|
|
|
wszBuf[2] = UNICODE_NULL;
|
2017-11-18 14:55:08 +00:00
|
|
|
dwError = WNetCancelConnection2W(wszBuf, 0, FALSE);
|
|
|
|
if (dwError == NO_ERROR)
|
|
|
|
{
|
|
|
|
SHChangeNotify(SHCNE_DRIVEREMOVED, SHCNF_PATHW | SHCNF_FLUSHNOWAIT, wszBuf, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nStringID = IDS_CANTDISCONNECT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nStringID != 0)
|
|
|
|
{
|
|
|
|
/* show error message */
|
|
|
|
WCHAR szFormat[128], szMessage[128];
|
|
|
|
LoadStringW(shell32_hInstance, nStringID, szFormat, _countof(szFormat));
|
|
|
|
wsprintfW(szMessage, szFormat, dwError);
|
|
|
|
MessageBoxW(hwnd, szMessage, NULL, MB_ICONERROR);
|
2017-01-24 16:35:36 +00:00
|
|
|
}
|
2016-11-17 14:35:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SHFree(pidlFolder);
|
|
|
|
_ILFreeaPidl(apidl, cidl);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2016-11-15 20:14:56 +00:00
|
|
|
HRESULT CDrivesContextMenu_CreateInstance(PCIDLIST_ABSOLUTE pidlFolder,
|
|
|
|
HWND hwnd,
|
|
|
|
UINT cidl,
|
|
|
|
PCUITEMID_CHILD_ARRAY apidl,
|
|
|
|
IShellFolder *psf,
|
|
|
|
IContextMenu **ppcm)
|
|
|
|
{
|
|
|
|
HKEY hKeys[2];
|
|
|
|
UINT cKeys = 0;
|
|
|
|
AddClassKeyToArray(L"Drive", hKeys, &cKeys);
|
|
|
|
AddClassKeyToArray(L"Folder", hKeys, &cKeys);
|
|
|
|
|
2016-11-17 14:35:19 +00:00
|
|
|
return CDefFolderMenu_Create2(pidlFolder, hwnd, cidl, apidl, psf, DrivesContextMenuCallback, cKeys, hKeys, ppcm);
|
2016-11-15 20:14:56 +00:00
|
|
|
}
|
|
|
|
|
2018-06-27 21:06:54 +00:00
|
|
|
static HRESULT
|
|
|
|
getIconLocationForDrive(IShellFolder *psf, PCITEMID_CHILD pidl, UINT uFlags,
|
|
|
|
LPWSTR szIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
|
|
|
|
{
|
|
|
|
WCHAR wszPath[MAX_PATH];
|
|
|
|
WCHAR wszAutoRunInfPath[MAX_PATH];
|
|
|
|
WCHAR wszValue[MAX_PATH], wszTemp[MAX_PATH];
|
|
|
|
|
|
|
|
// get path
|
|
|
|
if (!ILGetDisplayNameExW(psf, pidl, wszPath, 0))
|
|
|
|
return E_FAIL;
|
|
|
|
if (!PathIsDirectoryW(wszPath))
|
|
|
|
return E_FAIL;
|
|
|
|
|
|
|
|
// build the full path of autorun.inf
|
|
|
|
StringCchCopyW(wszAutoRunInfPath, _countof(wszAutoRunInfPath), wszPath);
|
2022-01-09 21:38:31 +00:00
|
|
|
PathAppendW(wszAutoRunInfPath, L"autorun.inf");
|
2018-06-27 21:06:54 +00:00
|
|
|
|
|
|
|
// autorun.inf --> wszValue
|
2022-01-09 21:38:31 +00:00
|
|
|
if (GetPrivateProfileStringW(L"autorun", L"icon", NULL, wszValue, _countof(wszValue),
|
2018-06-27 21:06:54 +00:00
|
|
|
wszAutoRunInfPath) && wszValue[0] != 0)
|
|
|
|
{
|
|
|
|
// wszValue --> wszTemp
|
|
|
|
ExpandEnvironmentStringsW(wszValue, wszTemp, _countof(wszTemp));
|
|
|
|
|
|
|
|
// parse the icon location
|
|
|
|
*piIndex = PathParseIconLocationW(wszTemp);
|
|
|
|
|
|
|
|
// wszPath + wszTemp --> wszPath
|
|
|
|
if (PathIsRelativeW(wszTemp))
|
|
|
|
PathAppendW(wszPath, wszTemp);
|
|
|
|
else
|
|
|
|
StringCchCopyW(wszPath, _countof(wszPath), wszTemp);
|
|
|
|
|
|
|
|
// wszPath --> szIconFile
|
|
|
|
GetFullPathNameW(wszPath, cchMax, szIconFile, NULL);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
2022-12-20 19:30:09 +00:00
|
|
|
static HRESULT
|
|
|
|
getLabelForDrive(LPWSTR wszPath, LPWSTR wszLabel)
|
|
|
|
{
|
|
|
|
WCHAR wszAutoRunInfPath[MAX_PATH];
|
|
|
|
WCHAR wszTemp[MAX_PATH];
|
|
|
|
|
|
|
|
if (!PathIsDirectoryW(wszPath))
|
|
|
|
return E_FAIL;
|
|
|
|
|
|
|
|
StringCchCopyW(wszAutoRunInfPath, _countof(wszAutoRunInfPath), wszPath);
|
|
|
|
PathAppendW(wszAutoRunInfPath, L"autorun.inf");
|
|
|
|
|
|
|
|
if (GetPrivateProfileStringW(L"autorun", L"label", NULL, wszTemp, _countof(wszTemp),
|
|
|
|
wszAutoRunInfPath) && wszTemp[0] != 0)
|
|
|
|
{
|
|
|
|
StringCchCopyW(wszLabel, _countof(wszTemp), wszTemp);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
2020-05-05 15:35:01 +00:00
|
|
|
BOOL IsDriveFloppyA(LPCSTR pszDriveRoot);
|
|
|
|
|
2016-07-29 06:21:11 +00:00
|
|
|
HRESULT CDrivesExtractIcon_CreateInstance(IShellFolder * psf, LPCITEMIDLIST pidl, REFIID riid, LPVOID * ppvOut)
|
|
|
|
{
|
|
|
|
CComPtr<IDefaultExtractIconInit> initIcon;
|
|
|
|
HRESULT hr = SHCreateDefaultExtractIcon(IID_PPV_ARG(IDefaultExtractIconInit, &initIcon));
|
2016-11-17 14:35:19 +00:00
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
2016-07-29 06:21:11 +00:00
|
|
|
|
|
|
|
CHAR* pszDrive = _ILGetDataPointer(pidl)->u.drive.szDriveName;
|
2017-06-11 14:02:14 +00:00
|
|
|
UINT DriveType = GetDriveTypeA(pszDrive);
|
|
|
|
if (DriveType > DRIVE_RAMDISK)
|
|
|
|
DriveType = DRIVE_FIXED;
|
2016-07-29 06:21:11 +00:00
|
|
|
|
2017-06-11 14:02:14 +00:00
|
|
|
WCHAR wTemp[MAX_PATH];
|
[SHELL32] Allow using custom desktop/folders/drives icons (#6421)
Implement proper reading the current user's icons from registry.
CORE-14758
- Load the icons specified by user in registry in the following keys:
"HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\CLSID\{XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX}\DefaultIcon"
(virtual namespace folders)
"HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Shell Icons"
(filesystem folders and drives)
- Implement two functions HCU/HLM_GetIconW for reading the icons
from mentioned HKCU and HKLM keys accordingly.
- Use HCR_RegGetIconW for falling back to default icons.
This function always loads only default icons, even when
the custom ones are specified by user.
- Refactor SIC_LoadOverlayIcon to use newly implemented HLM_GetIconW.
These changes apply to:
- virtual namespace folders and other desktop items (like My Computer,
My Documents, Network Places, Recycle Bin, Web Browser (aka Internet
Explorer), Control Panel and some of its items);
- normal filesystem directories;
- all types of drives (fixed disk drives, removable drives, CD-ROMs,
RamDisks and network drives). Handle invalid drives, setting blank
icon for them, since they cannot be recognized or mounted correctly.
Also, load the autorun icons first to avoid overriding them by the
icons defined in registry.
I've rechecked twice: excluding Start Menu icons, Desktop Workspace icon
and some FS folder icons that have their own desktop.ini configuration
files (we probably should write the custom icons we load to these configs,
as Windows does it, perhaps with WritePrivateProfileStringW), all other
icons can be changed now (only ones that can be changed on XP SP3 / 2003
SP2) via built-in system tools (like Desktop icons in desk.cpl) or any
3rd-party tools without modifying system resources.
Also all icons for the known file types can be changed,
same as before my changes.
Regarding Start Menu icons:
- changing them should be correctly implemented in explorer
instead of shell32, as the former is responsible for the Start Menu
and partially for the taskbar;
- in order to actually use all of them, we need to implement modern
Start Menu first.
Useful reference: http://www.winfaq.de/faq_html/Content/tip0000/onlinefaq.php?h=tip0162.htm
2024-02-28 21:20:47 +00:00
|
|
|
int icon_idx, reg_idx;
|
2018-06-27 21:06:54 +00:00
|
|
|
UINT flags = 0;
|
[SHELL32] Allow using custom desktop/folders/drives icons (#6421)
Implement proper reading the current user's icons from registry.
CORE-14758
- Load the icons specified by user in registry in the following keys:
"HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\CLSID\{XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX}\DefaultIcon"
(virtual namespace folders)
"HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Shell Icons"
(filesystem folders and drives)
- Implement two functions HCU/HLM_GetIconW for reading the icons
from mentioned HKCU and HKLM keys accordingly.
- Use HCR_RegGetIconW for falling back to default icons.
This function always loads only default icons, even when
the custom ones are specified by user.
- Refactor SIC_LoadOverlayIcon to use newly implemented HLM_GetIconW.
These changes apply to:
- virtual namespace folders and other desktop items (like My Computer,
My Documents, Network Places, Recycle Bin, Web Browser (aka Internet
Explorer), Control Panel and some of its items);
- normal filesystem directories;
- all types of drives (fixed disk drives, removable drives, CD-ROMs,
RamDisks and network drives). Handle invalid drives, setting blank
icon for them, since they cannot be recognized or mounted correctly.
Also, load the autorun icons first to avoid overriding them by the
icons defined in registry.
I've rechecked twice: excluding Start Menu icons, Desktop Workspace icon
and some FS folder icons that have their own desktop.ini configuration
files (we probably should write the custom icons we load to these configs,
as Windows does it, perhaps with WritePrivateProfileStringW), all other
icons can be changed now (only ones that can be changed on XP SP3 / 2003
SP2) via built-in system tools (like Desktop icons in desk.cpl) or any
3rd-party tools without modifying system resources.
Also all icons for the known file types can be changed,
same as before my changes.
Regarding Start Menu icons:
- changing them should be correctly implemented in explorer
instead of shell32, as the former is responsible for the Start Menu
and partially for the taskbar;
- in order to actually use all of them, we need to implement modern
Start Menu first.
Useful reference: http://www.winfaq.de/faq_html/Content/tip0000/onlinefaq.php?h=tip0162.htm
2024-02-28 21:20:47 +00:00
|
|
|
|
|
|
|
switch (DriveType)
|
|
|
|
{
|
|
|
|
case DRIVE_FIXED:
|
|
|
|
case DRIVE_UNKNOWN:
|
|
|
|
reg_idx = IDI_SHELL_DRIVE;
|
|
|
|
break;
|
|
|
|
case DRIVE_CDROM:
|
|
|
|
reg_idx = IDI_SHELL_CDROM;
|
|
|
|
break;
|
|
|
|
case DRIVE_REMOTE:
|
|
|
|
reg_idx = IDI_SHELL_NETDRIVE;
|
|
|
|
break;
|
|
|
|
case DRIVE_REMOVABLE:
|
|
|
|
if (!IsDriveFloppyA(pszDrive))
|
|
|
|
reg_idx = IDI_SHELL_REMOVEABLE;
|
|
|
|
else
|
|
|
|
reg_idx = IDI_SHELL_3_14_FLOPPY;
|
|
|
|
break;
|
|
|
|
case DRIVE_RAMDISK:
|
|
|
|
reg_idx = IDI_SHELL_RAMDISK;
|
|
|
|
break;
|
|
|
|
case DRIVE_NO_ROOT_DIR:
|
|
|
|
default:
|
|
|
|
reg_idx = IDI_SHELL_DOCUMENT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = getIconLocationForDrive(psf, pidl, 0, wTemp, _countof(wTemp),
|
|
|
|
&icon_idx, &flags);
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
initIcon->SetNormalIcon(wTemp, icon_idx);
|
|
|
|
}
|
|
|
|
else if (HLM_GetIconW(reg_idx - 1, wTemp, _countof(wTemp), &icon_idx))
|
2016-07-29 06:21:11 +00:00
|
|
|
{
|
2017-06-11 14:02:14 +00:00
|
|
|
initIcon->SetNormalIcon(wTemp, icon_idx);
|
2016-07-29 06:21:11 +00:00
|
|
|
}
|
[SHELL32] Allow using custom desktop/folders/drives icons (#6421)
Implement proper reading the current user's icons from registry.
CORE-14758
- Load the icons specified by user in registry in the following keys:
"HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\CLSID\{XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX}\DefaultIcon"
(virtual namespace folders)
"HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Shell Icons"
(filesystem folders and drives)
- Implement two functions HCU/HLM_GetIconW for reading the icons
from mentioned HKCU and HKLM keys accordingly.
- Use HCR_RegGetIconW for falling back to default icons.
This function always loads only default icons, even when
the custom ones are specified by user.
- Refactor SIC_LoadOverlayIcon to use newly implemented HLM_GetIconW.
These changes apply to:
- virtual namespace folders and other desktop items (like My Computer,
My Documents, Network Places, Recycle Bin, Web Browser (aka Internet
Explorer), Control Panel and some of its items);
- normal filesystem directories;
- all types of drives (fixed disk drives, removable drives, CD-ROMs,
RamDisks and network drives). Handle invalid drives, setting blank
icon for them, since they cannot be recognized or mounted correctly.
Also, load the autorun icons first to avoid overriding them by the
icons defined in registry.
I've rechecked twice: excluding Start Menu icons, Desktop Workspace icon
and some FS folder icons that have their own desktop.ini configuration
files (we probably should write the custom icons we load to these configs,
as Windows does it, perhaps with WritePrivateProfileStringW), all other
icons can be changed now (only ones that can be changed on XP SP3 / 2003
SP2) via built-in system tools (like Desktop icons in desk.cpl) or any
3rd-party tools without modifying system resources.
Also all icons for the known file types can be changed,
same as before my changes.
Regarding Start Menu icons:
- changing them should be correctly implemented in explorer
instead of shell32, as the former is responsible for the Start Menu
and partially for the taskbar;
- in order to actually use all of them, we need to implement modern
Start Menu first.
Useful reference: http://www.winfaq.de/faq_html/Content/tip0000/onlinefaq.php?h=tip0162.htm
2024-02-28 21:20:47 +00:00
|
|
|
else if ((DriveType == DRIVE_FIXED || DriveType == DRIVE_UNKNOWN) &&
|
|
|
|
(HCR_GetIconW(L"Drive", wTemp, NULL, _countof(wTemp), &icon_idx)))
|
2018-06-27 21:06:54 +00:00
|
|
|
{
|
|
|
|
initIcon->SetNormalIcon(wTemp, icon_idx);
|
|
|
|
}
|
2016-07-29 06:21:11 +00:00
|
|
|
else
|
|
|
|
{
|
2020-05-05 15:35:01 +00:00
|
|
|
if (DriveType == DRIVE_REMOVABLE && !IsDriveFloppyA(pszDrive))
|
|
|
|
{
|
|
|
|
icon_idx = IDI_SHELL_REMOVEABLE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
icon_idx = iDriveIconIds[DriveType];
|
|
|
|
}
|
2017-06-11 14:02:14 +00:00
|
|
|
initIcon->SetNormalIcon(swShell32Name, -icon_idx);
|
2016-07-29 06:21:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return initIcon->QueryInterface(riid, ppvOut);
|
|
|
|
}
|
2016-08-10 18:56:48 +00:00
|
|
|
|
2011-05-15 15:55:49 +00:00
|
|
|
class CDrivesFolderEnum :
|
2014-11-04 12:49:36 +00:00
|
|
|
public CEnumIDListBase
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2011-12-17 22:53:44 +00:00
|
|
|
public:
|
2017-07-16 15:13:02 +00:00
|
|
|
HRESULT WINAPI Initialize(HWND hwndOwner, DWORD dwFlags, IEnumIDList* pRegEnumerator)
|
|
|
|
{
|
|
|
|
/* enumerate the folders */
|
|
|
|
if (dwFlags & SHCONTF_FOLDERS)
|
|
|
|
{
|
|
|
|
WCHAR wszDriveName[] = {'A', ':', '\\', '\0'};
|
|
|
|
DWORD dwDrivemap = GetLogicalDrives();
|
|
|
|
|
|
|
|
while (wszDriveName[0] <= 'Z')
|
|
|
|
{
|
|
|
|
if(dwDrivemap & 0x00000001L)
|
|
|
|
AddToEnumList(_ILCreateDrive(wszDriveName));
|
|
|
|
wszDriveName[0]++;
|
|
|
|
dwDrivemap = dwDrivemap >> 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enumerate the items of the reg folder */
|
|
|
|
AppendItemsFromEnumerator(pRegEnumerator);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
2011-12-17 22:53:44 +00:00
|
|
|
|
|
|
|
BEGIN_COM_MAP(CDrivesFolderEnum)
|
|
|
|
COM_INTERFACE_ENTRY_IID(IID_IEnumIDList, IEnumIDList)
|
|
|
|
END_COM_MAP()
|
2011-05-15 15:55:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* IShellFolder [MyComputer] implementation
|
|
|
|
*/
|
|
|
|
|
|
|
|
static const shvheader MyComputerSFHeader[] = {
|
2017-06-12 10:01:28 +00:00
|
|
|
{IDS_SHV_COLUMN_NAME, SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT, LVCFMT_LEFT, 15},
|
|
|
|
{IDS_SHV_COLUMN_TYPE, SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT, LVCFMT_LEFT, 10},
|
|
|
|
{IDS_SHV_COLUMN_DISK_CAPACITY, SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT, LVCFMT_RIGHT, 10},
|
|
|
|
{IDS_SHV_COLUMN_DISK_AVAILABLE, SHCOLSTATE_TYPE_STR | SHCOLSTATE_ONBYDEFAULT, LVCFMT_RIGHT, 10},
|
2022-05-05 15:05:18 +00:00
|
|
|
{IDS_SHV_COLUMN_COMMENTS, SHCOLSTATE_TYPE_STR, LVCFMT_LEFT, 10},
|
2011-05-15 15:55:49 +00:00
|
|
|
};
|
|
|
|
|
2015-09-02 09:31:16 +00:00
|
|
|
static const DWORD dwComputerAttributes =
|
|
|
|
SFGAO_CANRENAME | SFGAO_CANDELETE | SFGAO_HASPROPSHEET | SFGAO_DROPTARGET |
|
2018-02-19 01:23:15 +00:00
|
|
|
SFGAO_FILESYSANCESTOR | SFGAO_FOLDER | SFGAO_HASSUBFOLDER;
|
2015-09-02 09:31:16 +00:00
|
|
|
static const DWORD dwControlPanelAttributes =
|
|
|
|
SFGAO_HASSUBFOLDER | SFGAO_FOLDER | SFGAO_CANLINK;
|
|
|
|
static const DWORD dwDriveAttributes =
|
|
|
|
SFGAO_HASSUBFOLDER | SFGAO_FILESYSTEM | SFGAO_FOLDER | SFGAO_FILESYSANCESTOR |
|
|
|
|
SFGAO_DROPTARGET | SFGAO_HASPROPSHEET | SFGAO_CANRENAME | SFGAO_CANLINK;
|
|
|
|
|
2011-05-15 15:55:49 +00:00
|
|
|
CDrivesFolder::CDrivesFolder()
|
|
|
|
{
|
|
|
|
pidlRoot = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
CDrivesFolder::~CDrivesFolder()
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("-- destroying IShellFolder(%p)\n", this);
|
2011-09-08 22:43:43 +00:00
|
|
|
SHFree(pidlRoot);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::FinalConstruct()
|
|
|
|
{
|
|
|
|
pidlRoot = _ILCreateMyComputer(); /* my qualified pidl */
|
2011-09-08 22:43:43 +00:00
|
|
|
if (pidlRoot == NULL)
|
|
|
|
return E_OUTOFMEMORY;
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2021-09-13 01:33:14 +00:00
|
|
|
HRESULT hr = CRegFolder_CreateInstance(&CLSID_MyComputer,
|
|
|
|
pidlRoot,
|
|
|
|
L"::{20D04FE0-3AEA-1069-A2D8-08002B30309D}",
|
2017-07-16 15:13:02 +00:00
|
|
|
L"MyComputer",
|
2016-08-10 18:56:48 +00:00
|
|
|
IID_PPV_ARG(IShellFolder2, &m_regFolder));
|
|
|
|
|
|
|
|
return hr;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::ParseDisplayName
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2011-12-18 21:54:04 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::ParseDisplayName(HWND hwndOwner, LPBC pbc, LPOLESTR lpszDisplayName,
|
2014-08-15 10:38:11 +00:00
|
|
|
DWORD * pchEaten, PIDLIST_RELATIVE * ppidl, DWORD * pdwAttributes)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
|
|
|
|
TRACE("(%p)->(HWND=%p,%p,%p=%s,%p,pidl=%p,%p)\n", this,
|
|
|
|
hwndOwner, pbc, lpszDisplayName, debugstr_w (lpszDisplayName),
|
|
|
|
pchEaten, ppidl, pdwAttributes);
|
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
if (!ppidl)
|
|
|
|
return hr;
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
*ppidl = NULL;
|
2016-04-30 12:52:17 +00:00
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
if (!lpszDisplayName)
|
|
|
|
return hr;
|
2016-04-30 12:52:17 +00:00
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
/* handle CLSID paths */
|
|
|
|
if (lpszDisplayName[0] == L':' && lpszDisplayName[1] == L':')
|
2020-05-09 12:54:15 +00:00
|
|
|
{
|
2024-04-14 18:25:20 +00:00
|
|
|
return m_regFolder->ParseDisplayName(hwndOwner, pbc, lpszDisplayName, pchEaten, ppidl,
|
|
|
|
pdwAttributes);
|
2020-05-09 12:54:15 +00:00
|
|
|
}
|
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
if (lpszDisplayName[0] &&
|
|
|
|
((L'A' <= lpszDisplayName[0] && lpszDisplayName[0] <= L'Z') ||
|
|
|
|
(L'a' <= lpszDisplayName[0] && lpszDisplayName[0] <= L'z')) &&
|
|
|
|
lpszDisplayName[1] == L':' && lpszDisplayName[2] == L'\\')
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2024-04-14 18:25:20 +00:00
|
|
|
// "C:\..."
|
|
|
|
WCHAR szRoot[8];
|
|
|
|
PathBuildRootW(szRoot, ((*lpszDisplayName - 1) & 0x1F));
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
if (SHIsFileSysBindCtx(pbc, NULL) != S_OK && !(BindCtx_GetMode(pbc, 0) & STGM_CREATE))
|
2015-08-30 14:28:07 +00:00
|
|
|
{
|
2024-04-14 18:25:20 +00:00
|
|
|
if (::GetDriveType(szRoot) == DRIVE_NO_ROOT_DIR)
|
|
|
|
return HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND);
|
|
|
|
}
|
2023-07-20 07:53:04 +00:00
|
|
|
|
2024-04-14 18:25:20 +00:00
|
|
|
CComHeapPtr<ITEMIDLIST> pidlTemp(_ILCreateDrive(szRoot));
|
|
|
|
if (!pidlTemp)
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
if (lpszDisplayName[3])
|
|
|
|
{
|
|
|
|
CComPtr<IShellFolder> pChildFolder;
|
|
|
|
hr = BindToObject(pidlTemp, pbc, IID_PPV_ARG(IShellFolder, &pChildFolder));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
ULONG chEaten;
|
|
|
|
CComHeapPtr<ITEMIDLIST> pidlChild;
|
|
|
|
hr = pChildFolder->ParseDisplayName(hwndOwner, pbc, &lpszDisplayName[3], &chEaten,
|
|
|
|
&pidlChild, pdwAttributes);
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
hr = SHILCombine(pidlTemp, pidlChild, ppidl);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*ppidl = pidlTemp.Detach();
|
|
|
|
if (pdwAttributes && *pdwAttributes)
|
|
|
|
GetAttributesOf(1, (PCUITEMID_CHILD_ARRAY)ppidl, pdwAttributes);
|
|
|
|
hr = S_OK;
|
2015-08-30 14:28:07 +00:00
|
|
|
}
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)->(-- ret=0x%08x)\n", this, hr);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::EnumObjects
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2011-12-18 21:54:04 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::EnumObjects(HWND hwndOwner, DWORD dwFlags, LPENUMIDLIST *ppEnumIDList)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2017-07-16 15:13:02 +00:00
|
|
|
CComPtr<IEnumIDList> pRegEnumerator;
|
|
|
|
m_regFolder->EnumObjects(hwndOwner, dwFlags, &pRegEnumerator);
|
|
|
|
|
|
|
|
return ShellObjectCreatorInit<CDrivesFolderEnum>(hwndOwner, dwFlags, pRegEnumerator, IID_PPV_ARG(IEnumIDList, ppEnumIDList));
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::BindToObject
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::BindToObject(PCUIDLIST_RELATIVE pidl, LPBC pbcReserved, REFIID riid, LPVOID *ppvOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
TRACE("(%p)->(pidl=%p,%p,%s,%p)\n", this,
|
2011-12-18 21:54:04 +00:00
|
|
|
pidl, pbcReserved, shdebugstr_guid(&riid), ppvOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2019-09-08 13:40:46 +00:00
|
|
|
if (!pidl)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
2015-08-17 11:41:15 +00:00
|
|
|
if (_ILIsSpecialFolder(pidl))
|
2016-08-10 18:56:48 +00:00
|
|
|
return m_regFolder->BindToObject(pidl, pbcReserved, riid, ppvOut);
|
2015-08-17 11:41:15 +00:00
|
|
|
|
2017-07-07 20:34:27 +00:00
|
|
|
CHAR* pchDrive = _ILGetDataPointer(pidl)->u.drive.szDriveName;
|
2021-09-13 01:33:14 +00:00
|
|
|
|
2017-07-07 20:34:27 +00:00
|
|
|
PERSIST_FOLDER_TARGET_INFO pfti = {0};
|
|
|
|
pfti.dwAttributes = -1;
|
|
|
|
pfti.csidl = -1;
|
|
|
|
pfti.szTargetParsingName[0] = *pchDrive;
|
|
|
|
pfti.szTargetParsingName[1] = L':';
|
|
|
|
pfti.szTargetParsingName[2] = L'\\';
|
|
|
|
|
|
|
|
HRESULT hr = SHELL32_BindToSF(pidlRoot,
|
|
|
|
&pfti,
|
|
|
|
pidl,
|
|
|
|
&CLSID_ShellFSFolder,
|
|
|
|
riid,
|
|
|
|
ppvOut);
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
2016-08-09 18:05:50 +00:00
|
|
|
return hr;
|
|
|
|
|
2017-07-07 20:34:27 +00:00
|
|
|
return S_OK;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::BindToStorage
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::BindToStorage(PCUIDLIST_RELATIVE pidl, LPBC pbcReserved, REFIID riid, LPVOID *ppvOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
FIXME("(%p)->(pidl=%p,%p,%s,%p) stub\n", this,
|
|
|
|
pidl, pbcReserved, shdebugstr_guid (&riid), ppvOut);
|
|
|
|
|
|
|
|
*ppvOut = NULL;
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::CompareIDs
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
|
|
|
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::CompareIDs(LPARAM lParam, PCUIDLIST_RELATIVE pidl1, PCUIDLIST_RELATIVE pidl2)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2016-11-04 23:11:29 +00:00
|
|
|
HRESULT hres;
|
|
|
|
|
2016-03-30 11:11:37 +00:00
|
|
|
if (!pidl1 || !pidl2)
|
|
|
|
{
|
|
|
|
ERR("Got null pidl pointer (%Ix %p %p)!\n", lParam, pidl1, pidl2);
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
|
|
|
|
2015-10-26 15:07:44 +00:00
|
|
|
if (_ILIsSpecialFolder(pidl1) || _ILIsSpecialFolder(pidl2))
|
2016-11-04 23:11:29 +00:00
|
|
|
return m_regFolder->CompareIDs(lParam, pidl1, pidl2);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2023-07-15 16:14:35 +00:00
|
|
|
UINT iColumn = LOWORD(lParam);
|
|
|
|
if (!_ILIsDrive(pidl1) || !_ILIsDrive(pidl2) || iColumn >= _countof(MyComputerSFHeader))
|
2015-10-26 15:07:44 +00:00
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
CHAR* pszDrive1 = _ILGetDataPointer(pidl1)->u.drive.szDriveName;
|
|
|
|
CHAR* pszDrive2 = _ILGetDataPointer(pidl2)->u.drive.szDriveName;
|
|
|
|
|
|
|
|
int result;
|
2023-07-15 16:14:35 +00:00
|
|
|
switch (MyComputerSFHeader[iColumn].colnameid)
|
2015-10-26 15:07:44 +00:00
|
|
|
{
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_NAME:
|
2015-10-26 15:07:44 +00:00
|
|
|
{
|
|
|
|
result = stricmp(pszDrive1, pszDrive2);
|
2016-11-04 23:11:29 +00:00
|
|
|
hres = MAKE_COMPARE_HRESULT(result);
|
|
|
|
break;
|
2015-10-26 15:07:44 +00:00
|
|
|
}
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_TYPE:
|
2015-10-26 15:07:44 +00:00
|
|
|
{
|
2016-11-04 23:11:29 +00:00
|
|
|
/* We want to return immediately because SHELL32_CompareDetails also compares children. */
|
2015-10-26 15:07:44 +00:00
|
|
|
return SHELL32_CompareDetails(this, lParam, pidl1, pidl2);
|
|
|
|
}
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_DISK_CAPACITY:
|
|
|
|
case IDS_SHV_COLUMN_DISK_AVAILABLE:
|
2015-10-26 15:07:44 +00:00
|
|
|
{
|
|
|
|
ULARGE_INTEGER Drive1Available, Drive1Total, Drive2Available, Drive2Total;
|
|
|
|
|
|
|
|
if (GetVolumeInformationA(pszDrive1, NULL, 0, NULL, NULL, NULL, NULL, 0))
|
|
|
|
GetDiskFreeSpaceExA(pszDrive1, &Drive1Available, &Drive1Total, NULL);
|
|
|
|
else
|
|
|
|
Drive1Available.QuadPart = Drive1Total.QuadPart = 0;
|
|
|
|
|
|
|
|
if (GetVolumeInformationA(pszDrive2, NULL, 0, NULL, NULL, NULL, NULL, 0))
|
|
|
|
GetDiskFreeSpaceExA(pszDrive2, &Drive2Available, &Drive2Total, NULL);
|
|
|
|
else
|
|
|
|
Drive2Available.QuadPart = Drive2Total.QuadPart = 0;
|
|
|
|
|
|
|
|
LARGE_INTEGER Diff;
|
2017-06-12 18:38:26 +00:00
|
|
|
if (lParam == 3) /* Size */
|
2015-10-26 15:07:44 +00:00
|
|
|
Diff.QuadPart = Drive1Total.QuadPart - Drive2Total.QuadPart;
|
|
|
|
else /* Size available */
|
|
|
|
Diff.QuadPart = Drive1Available.QuadPart - Drive2Available.QuadPart;
|
|
|
|
|
2016-11-04 23:11:29 +00:00
|
|
|
hres = MAKE_COMPARE_HRESULT(Diff.QuadPart);
|
|
|
|
break;
|
2015-10-26 15:07:44 +00:00
|
|
|
}
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_COMMENTS:
|
2022-05-05 15:05:18 +00:00
|
|
|
hres = MAKE_COMPARE_HRESULT(0);
|
|
|
|
break;
|
2023-07-15 16:14:35 +00:00
|
|
|
DEFAULT_UNREACHABLE;
|
2015-10-26 15:07:44 +00:00
|
|
|
}
|
2016-11-04 23:11:29 +00:00
|
|
|
|
|
|
|
if (HRESULT_CODE(hres) == 0)
|
|
|
|
return SHELL32_CompareChildren(this, lParam, pidl1, pidl2);
|
|
|
|
|
|
|
|
return hres;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::CreateViewObject
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2011-12-18 21:54:04 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::CreateViewObject(HWND hwndOwner, REFIID riid, LPVOID * ppvOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2014-08-20 00:39:40 +00:00
|
|
|
CComPtr<IShellView> pShellView;
|
2011-05-15 15:55:49 +00:00
|
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
|
|
|
|
TRACE("(%p)->(hwnd=%p,%s,%p)\n", this,
|
|
|
|
hwndOwner, shdebugstr_guid (&riid), ppvOut);
|
|
|
|
|
|
|
|
if (!ppvOut)
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
*ppvOut = NULL;
|
|
|
|
|
2011-12-18 21:54:04 +00:00
|
|
|
if (IsEqualIID(riid, IID_IDropTarget))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2011-12-18 21:54:04 +00:00
|
|
|
WARN("IDropTarget not implemented\n");
|
2011-05-15 15:55:49 +00:00
|
|
|
hr = E_NOTIMPL;
|
|
|
|
}
|
2011-12-18 21:54:04 +00:00
|
|
|
else if (IsEqualIID(riid, IID_IContextMenu))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2018-07-08 15:01:12 +00:00
|
|
|
HKEY hKeys[16];
|
|
|
|
UINT cKeys = 0;
|
|
|
|
AddClassKeyToArray(L"Directory\\Background", hKeys, &cKeys);
|
|
|
|
|
|
|
|
DEFCONTEXTMENU dcm;
|
|
|
|
dcm.hwnd = hwndOwner;
|
|
|
|
dcm.pcmcb = this;
|
|
|
|
dcm.pidlFolder = pidlRoot;
|
|
|
|
dcm.psf = this;
|
|
|
|
dcm.cidl = 0;
|
|
|
|
dcm.apidl = NULL;
|
|
|
|
dcm.cKeys = cKeys;
|
|
|
|
dcm.aKeys = hKeys;
|
|
|
|
dcm.punkAssociationInfo = NULL;
|
|
|
|
hr = SHCreateDefaultContextMenu(&dcm, riid, ppvOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
2011-12-18 21:54:04 +00:00
|
|
|
else if (IsEqualIID(riid, IID_IShellView))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2017-07-05 21:36:20 +00:00
|
|
|
SFV_CREATE sfvparams = {sizeof(SFV_CREATE), this};
|
|
|
|
hr = SHCreateShellFolderView(&sfvparams, (IShellView**)ppvOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("-- (%p)->(interface=%p)\n", this, ppvOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 23:41:49 +00:00
|
|
|
* CDrivesFolder::GetAttributesOf
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetAttributesOf(UINT cidl, PCUITEMID_CHILD_ARRAY apidl, DWORD * rgfInOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)->(cidl=%d apidl=%p mask=%p (0x%08x))\n",
|
|
|
|
this, cidl, apidl, rgfInOut, rgfInOut ? *rgfInOut : 0);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
if (cidl && !apidl)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
if (*rgfInOut == 0)
|
|
|
|
*rgfInOut = ~0;
|
|
|
|
|
2012-01-16 19:54:24 +00:00
|
|
|
/* FIXME: always add SFGAO_CANLINK */
|
2011-12-17 22:48:16 +00:00
|
|
|
if(cidl == 0)
|
2011-05-15 15:55:49 +00:00
|
|
|
*rgfInOut &= dwComputerAttributes;
|
2011-12-17 22:48:16 +00:00
|
|
|
else
|
|
|
|
{
|
2012-01-16 19:54:24 +00:00
|
|
|
for (UINT i = 0; i < cidl; ++i)
|
2011-12-17 22:48:16 +00:00
|
|
|
{
|
2012-01-16 19:54:24 +00:00
|
|
|
if (_ILIsDrive(apidl[i]))
|
2023-07-20 07:53:04 +00:00
|
|
|
{
|
2012-01-16 19:54:24 +00:00
|
|
|
*rgfInOut &= dwDriveAttributes;
|
2023-07-20 07:53:04 +00:00
|
|
|
|
|
|
|
if (_ILGetDriveType(apidl[i]) == DRIVE_CDROM)
|
|
|
|
*rgfInOut &= ~SFGAO_CANRENAME; // CD-ROM drive cannot rename
|
|
|
|
}
|
2012-01-16 19:54:24 +00:00
|
|
|
else if (_ILIsControlPanel(apidl[i]))
|
|
|
|
*rgfInOut &= dwControlPanelAttributes;
|
2015-04-03 12:18:07 +00:00
|
|
|
else if (_ILIsSpecialFolder(*apidl))
|
2016-08-10 18:56:48 +00:00
|
|
|
m_regFolder->GetAttributesOf(1, &apidl[i], rgfInOut);
|
2015-04-03 12:18:07 +00:00
|
|
|
else
|
|
|
|
ERR("Got unknown pidl type!\n");
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
}
|
2012-01-16 19:54:24 +00:00
|
|
|
|
2011-05-15 15:55:49 +00:00
|
|
|
/* make sure SFGAO_VALIDATE is cleared, some apps depend on that */
|
|
|
|
*rgfInOut &= ~SFGAO_VALIDATE;
|
|
|
|
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("-- result=0x%08x\n", *rgfInOut);
|
2012-01-16 19:54:24 +00:00
|
|
|
return S_OK;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::GetUIObjectOf
|
2011-05-15 15:55:49 +00:00
|
|
|
*
|
|
|
|
* PARAMETERS
|
|
|
|
* hwndOwner [in] Parent window for any output
|
|
|
|
* cidl [in] array size
|
|
|
|
* apidl [in] simple pidl array
|
|
|
|
* riid [in] Requested Interface
|
|
|
|
* prgfInOut [ ] reserved
|
|
|
|
* ppvObject [out] Resulting Interface
|
|
|
|
*
|
|
|
|
*/
|
2011-12-18 21:54:04 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetUIObjectOf(HWND hwndOwner,
|
2014-08-15 10:38:11 +00:00
|
|
|
UINT cidl, PCUITEMID_CHILD_ARRAY apidl,
|
2011-12-18 21:54:04 +00:00
|
|
|
REFIID riid, UINT *prgfInOut, LPVOID *ppvOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2016-05-01 19:54:23 +00:00
|
|
|
LPVOID pObj = NULL;
|
2011-05-15 15:55:49 +00:00
|
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
|
|
|
|
TRACE("(%p)->(%p,%u,apidl=%p,%s,%p,%p)\n", this,
|
|
|
|
hwndOwner, cidl, apidl, shdebugstr_guid (&riid), prgfInOut, ppvOut);
|
|
|
|
|
|
|
|
if (!ppvOut)
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
*ppvOut = NULL;
|
|
|
|
|
|
|
|
if (IsEqualIID (riid, IID_IContextMenu) && (cidl >= 1))
|
|
|
|
{
|
2016-11-15 20:14:56 +00:00
|
|
|
if (_ILIsDrive(apidl[0]))
|
|
|
|
hr = CDrivesContextMenu_CreateInstance(pidlRoot, hwndOwner, cidl, apidl, static_cast<IShellFolder*>(this), (IContextMenu**)&pObj);
|
|
|
|
else
|
|
|
|
hr = m_regFolder->GetUIObjectOf(hwndOwner, cidl, apidl, riid, prgfInOut, &pObj);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
else if (IsEqualIID (riid, IID_IDataObject) && (cidl >= 1))
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
hr = IDataObject_Constructor(hwndOwner,
|
|
|
|
pidlRoot, apidl, cidl, TRUE, (IDataObject **)&pObj);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
2016-05-01 19:54:23 +00:00
|
|
|
else if ((IsEqualIID (riid, IID_IExtractIconA) || IsEqualIID (riid, IID_IExtractIconW)) && (cidl == 1))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2016-08-10 18:56:48 +00:00
|
|
|
if (_ILIsDrive(apidl[0]))
|
|
|
|
hr = CDrivesExtractIcon_CreateInstance(this, apidl[0], riid, &pObj);
|
|
|
|
else
|
|
|
|
hr = m_regFolder->GetUIObjectOf(hwndOwner, cidl, apidl, riid, prgfInOut, &pObj);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
2017-02-01 13:31:55 +00:00
|
|
|
else if (IsEqualIID (riid, IID_IDropTarget) && (cidl == 1))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2017-02-01 13:31:55 +00:00
|
|
|
CComPtr<IShellFolder> psfChild;
|
|
|
|
hr = this->BindToObject(apidl[0], NULL, IID_PPV_ARG(IShellFolder, &psfChild));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
return psfChild->CreateViewObject(NULL, riid, ppvOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
hr = E_NOINTERFACE;
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr) && !pObj)
|
|
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
*ppvOut = pObj;
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)->hr=0x%08x\n", this, hr);
|
2011-05-15 15:55:49 +00:00
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::GetDisplayNameOf
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetDisplayNameOf(PCUITEMID_CHILD pidl, DWORD dwFlags, LPSTRRET strRet)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
LPWSTR pszPath;
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)->(pidl=%p,0x%08x,%p)\n", this, pidl, dwFlags, strRet);
|
2011-05-15 15:55:49 +00:00
|
|
|
pdump (pidl);
|
|
|
|
|
|
|
|
if (!strRet)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
2015-08-31 11:28:24 +00:00
|
|
|
if (!_ILIsPidlSimple (pidl))
|
|
|
|
{
|
|
|
|
return SHELL32_GetDisplayNameOfChild(this, pidl, dwFlags, strRet);
|
|
|
|
}
|
2016-08-10 18:56:48 +00:00
|
|
|
else if (_ILIsSpecialFolder(pidl))
|
2015-09-23 15:02:05 +00:00
|
|
|
{
|
2016-08-10 18:56:48 +00:00
|
|
|
return m_regFolder->GetDisplayNameOf(pidl, dwFlags, strRet);
|
2015-09-23 15:02:05 +00:00
|
|
|
}
|
2016-08-10 18:56:48 +00:00
|
|
|
else if (!_ILIsDrive(pidl))
|
2015-09-23 15:02:05 +00:00
|
|
|
{
|
|
|
|
ERR("Wrong pidl type\n");
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
2015-08-31 11:28:24 +00:00
|
|
|
|
2011-12-17 22:53:44 +00:00
|
|
|
pszPath = (LPWSTR)CoTaskMemAlloc((MAX_PATH + 1) * sizeof(WCHAR));
|
2011-05-15 15:55:49 +00:00
|
|
|
if (!pszPath)
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
pszPath[0] = 0;
|
|
|
|
|
2016-08-10 18:56:48 +00:00
|
|
|
_ILSimpleGetTextW(pidl, pszPath, MAX_PATH); /* append my own path */
|
|
|
|
/* long view "lw_name (C:)" */
|
|
|
|
if (!(dwFlags & SHGDN_FORPARSING))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2016-08-10 18:56:48 +00:00
|
|
|
WCHAR wszDrive[18] = {0};
|
|
|
|
|
|
|
|
lstrcpynW(wszDrive, pszPath, 4);
|
|
|
|
pszPath[0] = L'\0';
|
2022-12-20 19:30:09 +00:00
|
|
|
|
|
|
|
if (!SUCCEEDED(getLabelForDrive(wszDrive, pszPath)))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2022-12-20 19:30:09 +00:00
|
|
|
DWORD dwVolumeSerialNumber, dwMaximumComponentLength, dwFileSystemFlags;
|
|
|
|
|
|
|
|
GetVolumeInformationW(wszDrive, pszPath,
|
2024-02-20 01:08:56 +00:00
|
|
|
MAX_PATH - 7,
|
|
|
|
&dwVolumeSerialNumber,
|
|
|
|
&dwMaximumComponentLength, &dwFileSystemFlags, NULL, 0);
|
2022-12-20 19:30:09 +00:00
|
|
|
pszPath[MAX_PATH-1] = L'\0';
|
|
|
|
|
|
|
|
if (!wcslen(pszPath))
|
2016-08-10 18:56:48 +00:00
|
|
|
{
|
2022-12-20 19:30:09 +00:00
|
|
|
UINT DriveType, ResourceId;
|
|
|
|
DriveType = GetDriveTypeW(wszDrive);
|
|
|
|
|
|
|
|
switch (DriveType)
|
|
|
|
{
|
|
|
|
case DRIVE_FIXED:
|
|
|
|
ResourceId = IDS_DRIVE_FIXED;
|
|
|
|
break;
|
|
|
|
case DRIVE_REMOTE:
|
|
|
|
ResourceId = IDS_DRIVE_NETWORK;
|
|
|
|
break;
|
|
|
|
case DRIVE_CDROM:
|
|
|
|
ResourceId = IDS_DRIVE_CDROM;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ResourceId = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ResourceId)
|
|
|
|
{
|
|
|
|
dwFileSystemFlags = LoadStringW(shell32_hInstance, ResourceId, pszPath, MAX_PATH);
|
|
|
|
if (dwFileSystemFlags > MAX_PATH - 7)
|
|
|
|
pszPath[MAX_PATH-7] = L'\0';
|
|
|
|
}
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
}
|
2024-02-20 01:08:56 +00:00
|
|
|
wcscat(pszPath, L" (");
|
2016-08-10 18:56:48 +00:00
|
|
|
wszDrive[2] = L'\0';
|
2024-02-20 01:08:56 +00:00
|
|
|
wcscat(pszPath, wszDrive);
|
|
|
|
wcscat(pszPath, L")");
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
2011-12-18 21:54:04 +00:00
|
|
|
if (SUCCEEDED(hr))
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
strRet->uType = STRRET_WSTR;
|
|
|
|
strRet->pOleStr = pszPath;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
CoTaskMemFree(pszPath);
|
|
|
|
|
2011-12-18 21:54:04 +00:00
|
|
|
TRACE("-- (%p)->(%s)\n", this, strRet->uType == STRRET_CSTR ? strRet->cStr : debugstr_w(strRet->pOleStr));
|
2011-05-15 15:55:49 +00:00
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 23:41:49 +00:00
|
|
|
* CDrivesFolder::SetNameOf
|
2011-05-15 15:55:49 +00:00
|
|
|
* Changes the name of a file object or subfolder, possibly changing its item
|
|
|
|
* identifier in the process.
|
|
|
|
*
|
|
|
|
* PARAMETERS
|
|
|
|
* hwndOwner [in] Owner window for output
|
|
|
|
* pidl [in] simple pidl of item to change
|
|
|
|
* lpszName [in] the items new display name
|
|
|
|
* dwFlags [in] SHGNO formatting flags
|
|
|
|
* ppidlOut [out] simple pidl returned
|
|
|
|
*/
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::SetNameOf(HWND hwndOwner, PCUITEMID_CHILD pidl,
|
|
|
|
LPCOLESTR lpName, DWORD dwFlags, PITEMID_CHILD *pPidlOut)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
WCHAR szName[30];
|
|
|
|
|
|
|
|
if (_ILIsDrive(pidl))
|
|
|
|
{
|
2011-12-18 21:54:04 +00:00
|
|
|
if (_ILSimpleGetTextW(pidl, szName, _countof(szName)))
|
2011-12-17 22:53:44 +00:00
|
|
|
SetVolumeLabelW(szName, lpName);
|
|
|
|
if (pPidlOut)
|
|
|
|
*pPidlOut = _ILCreateDrive(szName);
|
|
|
|
return S_OK;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
2016-08-10 18:56:48 +00:00
|
|
|
return m_regFolder->SetNameOf(hwndOwner, pidl, lpName, dwFlags, pPidlOut);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::GetDefaultSearchGUID(GUID * pguid)
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
FIXME("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::EnumSearches(IEnumExtraSearch ** ppenum)
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
FIXME("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::GetDefaultColumn (DWORD dwRes, ULONG *pSort, ULONG *pDisplay)
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
if (pSort)
|
2011-12-17 22:53:44 +00:00
|
|
|
*pSort = 0;
|
2011-05-15 15:55:49 +00:00
|
|
|
if (pDisplay)
|
|
|
|
*pDisplay = 0;
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2024-04-25 22:20:28 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetDefaultColumnState(UINT iColumn, SHCOLSTATEF * pcsFlags)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2023-07-15 16:14:35 +00:00
|
|
|
if (!pcsFlags || iColumn >= _countof(MyComputerSFHeader))
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_INVALIDARG;
|
2024-04-25 22:20:28 +00:00
|
|
|
*pcsFlags = MyComputerSFHeader[iColumn].colstate;
|
2011-05-15 15:55:49 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetDetailsEx(PCUITEMID_CHILD pidl, const SHCOLUMNID * pscid, VARIANT * pv)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
FIXME("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
2014-08-15 10:38:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetDetailsOf(PCUITEMID_CHILD pidl, UINT iColumn, SHELLDETAILS *psd)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)->(%p %i %p)\n", this, pidl, iColumn, psd);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2023-07-15 16:14:35 +00:00
|
|
|
if (!psd || iColumn >= _countof(MyComputerSFHeader))
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
if (!pidl)
|
|
|
|
{
|
|
|
|
psd->fmt = MyComputerSFHeader[iColumn].fmt;
|
|
|
|
psd->cxChar = MyComputerSFHeader[iColumn].cxChar;
|
2016-04-30 14:05:10 +00:00
|
|
|
return SHSetStrRet(&psd->str, MyComputerSFHeader[iColumn].colnameid);
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
2017-06-11 14:02:14 +00:00
|
|
|
else if (!_ILIsDrive(pidl))
|
2015-09-23 15:02:05 +00:00
|
|
|
{
|
2023-07-15 16:14:35 +00:00
|
|
|
switch (MyComputerSFHeader[iColumn].colnameid)
|
|
|
|
{
|
|
|
|
case IDS_SHV_COLUMN_NAME:
|
|
|
|
case IDS_SHV_COLUMN_TYPE:
|
|
|
|
return m_regFolder->GetDetailsOf(pidl, iColumn, psd);
|
|
|
|
case IDS_SHV_COLUMN_DISK_CAPACITY:
|
|
|
|
case IDS_SHV_COLUMN_DISK_AVAILABLE:
|
|
|
|
return SHSetStrRet(&psd->str, ""); /* blank col */
|
|
|
|
case IDS_SHV_COLUMN_COMMENTS:
|
|
|
|
return m_regFolder->GetDetailsOf(pidl, 2, psd); /* 2 = comments */
|
|
|
|
DEFAULT_UNREACHABLE;
|
|
|
|
}
|
2015-09-23 15:02:05 +00:00
|
|
|
}
|
2011-05-15 15:55:49 +00:00
|
|
|
else
|
|
|
|
{
|
2017-06-11 14:02:14 +00:00
|
|
|
ULARGE_INTEGER ulTotalBytes, ulFreeBytes;
|
|
|
|
CHAR* pszDrive = _ILGetDataPointer(pidl)->u.drive.szDriveName;
|
|
|
|
UINT DriveType = GetDriveTypeA(pszDrive);
|
|
|
|
if (DriveType > DRIVE_RAMDISK)
|
|
|
|
DriveType = DRIVE_FIXED;
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2023-07-15 16:14:35 +00:00
|
|
|
switch (MyComputerSFHeader[iColumn].colnameid)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_NAME:
|
2015-09-27 15:09:28 +00:00
|
|
|
hr = GetDisplayNameOf(pidl, SHGDN_NORMAL | SHGDN_INFOLDER, &psd->str);
|
2011-12-17 22:53:44 +00:00
|
|
|
break;
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_TYPE:
|
2020-05-05 15:35:01 +00:00
|
|
|
if (DriveType == DRIVE_REMOVABLE && !IsDriveFloppyA(pszDrive))
|
|
|
|
hr = SHSetStrRet(&psd->str, IDS_DRIVE_REMOVABLE);
|
|
|
|
else
|
|
|
|
hr = SHSetStrRet(&psd->str, iDriveTypeIds[DriveType]);
|
2011-12-17 22:53:44 +00:00
|
|
|
break;
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_DISK_CAPACITY:
|
|
|
|
case IDS_SHV_COLUMN_DISK_AVAILABLE:
|
2017-06-11 14:02:14 +00:00
|
|
|
psd->str.cStr[0] = 0x00;
|
|
|
|
psd->str.uType = STRRET_CSTR;
|
|
|
|
if (GetVolumeInformationA(pszDrive, NULL, 0, NULL, NULL, NULL, NULL, 0))
|
2015-09-27 15:09:28 +00:00
|
|
|
{
|
2017-06-11 14:02:14 +00:00
|
|
|
GetDiskFreeSpaceExA(pszDrive, &ulFreeBytes, &ulTotalBytes, NULL);
|
2022-05-05 15:05:18 +00:00
|
|
|
if (iColumn == 2)
|
2017-06-11 14:02:14 +00:00
|
|
|
StrFormatByteSize64A(ulTotalBytes.QuadPart, psd->str.cStr, MAX_PATH);
|
|
|
|
else
|
|
|
|
StrFormatByteSize64A(ulFreeBytes.QuadPart, psd->str.cStr, MAX_PATH);
|
2015-09-27 15:09:28 +00:00
|
|
|
}
|
2017-06-11 14:02:14 +00:00
|
|
|
hr = S_OK;
|
2011-12-17 22:53:44 +00:00
|
|
|
break;
|
2023-07-15 16:14:35 +00:00
|
|
|
case IDS_SHV_COLUMN_COMMENTS:
|
|
|
|
hr = SHSetStrRet(&psd->str, ""); /* FIXME: comments */
|
2022-05-05 15:05:18 +00:00
|
|
|
break;
|
2023-07-15 16:14:35 +00:00
|
|
|
DEFAULT_UNREACHABLE;
|
2011-05-15 15:55:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::MapColumnToSCID(UINT column, SHCOLUMNID * pscid)
|
|
|
|
{
|
2011-12-18 21:54:04 +00:00
|
|
|
FIXME("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::GetClassID
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
|
|
|
HRESULT WINAPI CDrivesFolder::GetClassID(CLSID *lpClassId)
|
|
|
|
{
|
2024-02-20 01:08:56 +00:00
|
|
|
TRACE("(%p)\n", this);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
if (!lpClassId)
|
2011-12-17 22:53:44 +00:00
|
|
|
return E_POINTER;
|
2011-05-15 15:55:49 +00:00
|
|
|
|
2011-12-18 21:54:04 +00:00
|
|
|
*lpClassId = CLSID_MyComputer;
|
2011-05-15 15:55:49 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::Initialize
|
2011-05-15 15:55:49 +00:00
|
|
|
*
|
|
|
|
* NOTES: it makes no sense to change the pidl
|
|
|
|
*/
|
2019-09-22 14:38:06 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::Initialize(PCIDLIST_ABSOLUTE pidl)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2011-12-18 21:54:04 +00:00
|
|
|
* CDrivesFolder::GetCurFolder
|
2011-05-15 15:55:49 +00:00
|
|
|
*/
|
2019-09-10 10:24:11 +00:00
|
|
|
HRESULT WINAPI CDrivesFolder::GetCurFolder(PIDLIST_ABSOLUTE *pidl)
|
2011-05-15 15:55:49 +00:00
|
|
|
{
|
2011-12-18 21:54:04 +00:00
|
|
|
TRACE("(%p)->(%p)\n", this, pidl);
|
2011-05-15 15:55:49 +00:00
|
|
|
|
|
|
|
if (!pidl)
|
2016-11-15 20:14:56 +00:00
|
|
|
return E_INVALIDARG; /* xp doesn't have this check and crashes on NULL */
|
2011-12-18 21:54:04 +00:00
|
|
|
|
|
|
|
*pidl = ILClone(pidlRoot);
|
2011-05-15 15:55:49 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
2018-07-08 15:01:12 +00:00
|
|
|
|
|
|
|
/************************************************************************/
|
|
|
|
/* IContextMenuCB interface */
|
|
|
|
|
|
|
|
HRESULT WINAPI CDrivesFolder::CallBack(IShellFolder *psf, HWND hwndOwner, IDataObject *pdtobj, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
2024-05-09 17:52:05 +00:00
|
|
|
enum { IDC_PROPERTIES };
|
2018-07-08 15:01:12 +00:00
|
|
|
/* no data object means no selection */
|
|
|
|
if (!pdtobj)
|
|
|
|
{
|
2024-05-09 17:52:05 +00:00
|
|
|
if (uMsg == DFM_INVOKECOMMAND && wParam == IDC_PROPERTIES)
|
2018-07-08 15:01:12 +00:00
|
|
|
{
|
|
|
|
// "System" properties
|
2024-05-09 17:52:05 +00:00
|
|
|
return SHELL_ExecuteControlPanelCPL(hwndOwner, L"sysdm.cpl") ? S_OK : E_FAIL;
|
2018-07-08 15:01:12 +00:00
|
|
|
}
|
|
|
|
else if (uMsg == DFM_MERGECONTEXTMENU)
|
|
|
|
{
|
|
|
|
QCMINFO *pqcminfo = (QCMINFO *)lParam;
|
|
|
|
HMENU hpopup = CreatePopupMenu();
|
2024-05-09 17:52:05 +00:00
|
|
|
_InsertMenuItemW(hpopup, 0, TRUE, IDC_PROPERTIES, MFT_STRING, MAKEINTRESOURCEW(IDS_PROPERTIES), MFS_ENABLED);
|
|
|
|
pqcminfo->idCmdFirst = Shell_MergeMenus(pqcminfo->hmenu, hpopup, pqcminfo->indexMenu, pqcminfo->idCmdFirst, pqcminfo->idCmdLast, MM_ADDSEPARATOR);
|
2018-07-08 15:01:12 +00:00
|
|
|
DestroyMenu(hpopup);
|
2024-05-09 17:52:05 +00:00
|
|
|
return S_OK;
|
2018-07-08 15:01:12 +00:00
|
|
|
}
|
|
|
|
}
|
2024-05-09 17:52:05 +00:00
|
|
|
return SHELL32_DefaultContextMenuCallBack(psf, pdtobj, uMsg);
|
2018-07-08 15:01:12 +00:00
|
|
|
}
|