2020-02-28 23:13:52 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS api tests
|
|
|
|
* LICENSE: LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
|
|
|
|
* PURPOSE: Test for SHChangeNotify
|
2024-04-24 00:10:29 +00:00
|
|
|
* COPYRIGHT: Copyright 2020-2024 Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
|
2020-02-28 23:13:52 +00:00
|
|
|
*/
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
// This program is used in SHChangeNotify and ShellExecCmdLine testcases.
|
|
|
|
|
2020-02-28 23:13:52 +00:00
|
|
|
#include "shelltest.h"
|
2022-02-13 08:45:08 +00:00
|
|
|
#include "shell32_apitest_sub.h"
|
2024-04-24 00:10:29 +00:00
|
|
|
#include <assert.h>
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
typedef enum DIRTYPE
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
DIRTYPE_DESKTOP = 0,
|
2024-04-24 09:25:15 +00:00
|
|
|
DIRTYPE_DESKTOP_DIR,
|
2024-04-24 00:10:29 +00:00
|
|
|
DIRTYPE_DRIVES,
|
2024-04-24 09:25:15 +00:00
|
|
|
DIRTYPE_PRINTERS,
|
2024-04-24 00:10:29 +00:00
|
|
|
DIRTYPE_DIR1,
|
|
|
|
DIRTYPE_MAX
|
|
|
|
} DIRTYPE;
|
|
|
|
|
|
|
|
static HWND s_hMainWnd = NULL, s_hSubWnd = NULL;
|
|
|
|
static LPITEMIDLIST s_pidl[DIRTYPE_MAX];
|
|
|
|
static UINT s_uRegID = 0;
|
|
|
|
static INT s_iStage = -1;
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
#define EVENTS (SHCNE_CREATE | SHCNE_DELETE | SHCNE_MKDIR | SHCNE_RMDIR | \
|
|
|
|
SHCNE_RENAMEFOLDER | SHCNE_RENAMEITEM | SHCNE_UPDATEDIR | SHCNE_UPDATEITEM)
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
inline LPITEMIDLIST DoGetPidl(INT iDir)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
LPITEMIDLIST ret = NULL;
|
|
|
|
|
|
|
|
switch (iDir)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
case DIRTYPE_DESKTOP:
|
|
|
|
{
|
|
|
|
SHGetSpecialFolderLocation(NULL, CSIDL_DESKTOP, &ret);
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 09:25:15 +00:00
|
|
|
case DIRTYPE_DESKTOP_DIR:
|
|
|
|
{
|
|
|
|
WCHAR szPath1[MAX_PATH];
|
|
|
|
SHGetSpecialFolderPathW(NULL, szPath1, CSIDL_DESKTOPDIRECTORY, FALSE);
|
|
|
|
ret = ILCreateFromPathW(szPath1);
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 00:10:29 +00:00
|
|
|
case DIRTYPE_DRIVES:
|
|
|
|
{
|
|
|
|
SHGetSpecialFolderLocation(NULL, CSIDL_DRIVES, &ret);
|
2020-02-28 23:13:52 +00:00
|
|
|
break;
|
2024-04-24 00:10:29 +00:00
|
|
|
}
|
2024-04-24 09:25:15 +00:00
|
|
|
case DIRTYPE_PRINTERS:
|
|
|
|
{
|
|
|
|
SHGetSpecialFolderLocation(NULL, CSIDL_PRINTERS, &ret);
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 00:10:29 +00:00
|
|
|
case DIRTYPE_DIR1:
|
|
|
|
{
|
|
|
|
WCHAR szPath1[MAX_PATH];
|
|
|
|
SHGetSpecialFolderPathW(NULL, szPath1, CSIDL_PERSONAL, FALSE); // My Documents
|
|
|
|
PathAppendW(szPath1, L"_TESTDIR_1_");
|
|
|
|
ret = ILCreateFromPathW(szPath1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
2020-02-28 23:13:52 +00:00
|
|
|
}
|
2024-04-24 00:10:29 +00:00
|
|
|
|
|
|
|
return ret;
|
2020-02-28 23:13:52 +00:00
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static BOOL OnCreate(HWND hwnd)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
s_hSubWnd = hwnd;
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
for (INT i = 0; i < DIRTYPE_MAX; ++i)
|
|
|
|
s_pidl[i] = DoGetPidl(i);
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
return TRUE;
|
2020-02-28 23:13:52 +00:00
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static BOOL InitSHCN(HWND hwnd)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
assert(0 <= s_iStage);
|
|
|
|
assert(s_iStage < NUM_STAGE);
|
|
|
|
|
|
|
|
SHChangeNotifyEntry entry;
|
|
|
|
INT sources;
|
|
|
|
LONG events;
|
|
|
|
switch (s_iStage)
|
2021-12-27 00:45:44 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
case 0:
|
2024-04-24 09:25:15 +00:00
|
|
|
{
|
|
|
|
entry.fRecursive = FALSE;
|
|
|
|
entry.pidl = NULL;
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1:
|
2024-04-24 00:10:29 +00:00
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
2024-04-24 09:25:15 +00:00
|
|
|
entry.pidl = NULL;
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
entry.fRecursive = FALSE;
|
2024-04-24 00:10:29 +00:00
|
|
|
entry.pidl = s_pidl[DIRTYPE_DESKTOP];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 09:25:15 +00:00
|
|
|
case 3:
|
2024-04-24 00:10:29 +00:00
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
2024-04-24 09:25:15 +00:00
|
|
|
entry.pidl = s_pidl[DIRTYPE_DESKTOP];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 4:
|
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
|
|
|
entry.pidl = s_pidl[DIRTYPE_DESKTOP_DIR];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 5:
|
|
|
|
{
|
|
|
|
entry.fRecursive = FALSE;
|
2024-04-24 00:10:29 +00:00
|
|
|
entry.pidl = s_pidl[DIRTYPE_DRIVES];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 09:25:15 +00:00
|
|
|
case 6:
|
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
|
|
|
entry.pidl = s_pidl[DIRTYPE_DRIVES];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 7:
|
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
|
|
|
entry.pidl = s_pidl[DIRTYPE_PRINTERS];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 8:
|
2024-04-24 00:10:29 +00:00
|
|
|
{
|
|
|
|
entry.fRecursive = FALSE;
|
|
|
|
entry.pidl = s_pidl[DIRTYPE_DIR1];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
2024-04-24 09:25:15 +00:00
|
|
|
case 9:
|
2024-04-24 00:10:29 +00:00
|
|
|
{
|
|
|
|
entry.fRecursive = TRUE;
|
|
|
|
entry.pidl = s_pidl[DIRTYPE_DIR1];
|
|
|
|
sources = SHCNRF_NewDelivery | SHCNRF_ShellLevel | SHCNRF_InterruptLevel |
|
|
|
|
SHCNRF_RecursiveInterrupt;
|
|
|
|
events = EVENTS;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
2021-12-27 00:45:44 +00:00
|
|
|
}
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
s_uRegID = SHChangeNotifyRegister(hwnd, sources, events, WM_SHELL_NOTIFY, 1, &entry);
|
|
|
|
if (s_uRegID == 0)
|
2021-12-27 00:45:44 +00:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static void UnInitSHCN(HWND hwnd)
|
2021-12-27 00:45:44 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
if (s_uRegID)
|
2021-12-27 00:45:44 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
SHChangeNotifyDeregister(s_uRegID);
|
|
|
|
s_uRegID = 0;
|
2021-12-27 00:45:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static void OnCommand(HWND hwnd, UINT id)
|
2021-12-27 00:45:44 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
switch (id)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2024-04-24 00:10:29 +00:00
|
|
|
case IDYES: // Start testing
|
|
|
|
{
|
|
|
|
s_hMainWnd = ::FindWindow(MAIN_CLASSNAME, MAIN_CLASSNAME);
|
|
|
|
if (!s_hMainWnd)
|
|
|
|
{
|
|
|
|
::DestroyWindow(hwnd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
s_iStage = 0;
|
|
|
|
InitSHCN(hwnd);
|
|
|
|
::PostMessageW(s_hMainWnd, WM_COMMAND, IDYES, 0);
|
2020-02-28 23:13:52 +00:00
|
|
|
break;
|
2024-04-24 00:10:29 +00:00
|
|
|
}
|
|
|
|
case IDRETRY: // New stage
|
|
|
|
{
|
|
|
|
UnInitSHCN(hwnd);
|
|
|
|
++s_iStage;
|
|
|
|
InitSHCN(hwnd);
|
|
|
|
::PostMessageW(s_hMainWnd, WM_COMMAND, IDRETRY, 0);
|
2020-02-28 23:13:52 +00:00
|
|
|
break;
|
2024-04-24 00:10:29 +00:00
|
|
|
}
|
|
|
|
case IDNO: // Quit
|
|
|
|
{
|
|
|
|
s_iStage = -1;
|
|
|
|
UnInitSHCN(hwnd);
|
|
|
|
::DestroyWindow(hwnd);
|
2020-02-28 23:13:52 +00:00
|
|
|
break;
|
2024-04-24 00:10:29 +00:00
|
|
|
}
|
2020-02-28 23:13:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static void OnDestroy(HWND hwnd)
|
|
|
|
{
|
|
|
|
UnInitSHCN(hwnd);
|
|
|
|
|
|
|
|
for (auto& pidl : s_pidl)
|
|
|
|
{
|
|
|
|
CoTaskMemFree(pidl);
|
|
|
|
pidl = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
::PostMessageW(s_hMainWnd, WM_COMMAND, IDNO, 0);
|
|
|
|
|
|
|
|
PostQuitMessage(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL DoSendData(LONG lEvent, LPCITEMIDLIST pidl1, LPCITEMIDLIST pidl2)
|
|
|
|
{
|
|
|
|
DWORD cbPidl1 = ILGetSize(pidl1), cbPidl2 = ILGetSize(pidl2);
|
|
|
|
DWORD cbTotal = sizeof(lEvent) + sizeof(cbPidl1) + sizeof(cbPidl2) + cbPidl1 + cbPidl2;
|
|
|
|
LPBYTE pbData = (LPBYTE)::LocalAlloc(LPTR, cbTotal);
|
|
|
|
if (!pbData)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
LPBYTE pb = pbData;
|
|
|
|
|
|
|
|
*(LONG*)pb = lEvent;
|
|
|
|
pb += sizeof(lEvent);
|
|
|
|
|
|
|
|
*(DWORD*)pb = cbPidl1;
|
|
|
|
pb += sizeof(cbPidl1);
|
|
|
|
|
|
|
|
*(DWORD*)pb = cbPidl2;
|
|
|
|
pb += sizeof(cbPidl2);
|
|
|
|
|
|
|
|
CopyMemory(pb, pidl1, cbPidl1);
|
|
|
|
pb += cbPidl1;
|
|
|
|
|
|
|
|
CopyMemory(pb, pidl2, cbPidl2);
|
|
|
|
pb += cbPidl2;
|
|
|
|
|
|
|
|
assert(INT(pb - pbData) == INT(cbTotal));
|
|
|
|
|
|
|
|
COPYDATASTRUCT CopyData;
|
|
|
|
CopyData.dwData = 0xBEEFCAFE;
|
|
|
|
CopyData.cbData = cbTotal;
|
|
|
|
CopyData.lpData = pbData;
|
|
|
|
BOOL ret = (BOOL)::SendMessageW(s_hMainWnd, WM_COPYDATA, (WPARAM)s_hSubWnd, (LPARAM)&CopyData);
|
|
|
|
|
|
|
|
::LocalFree(pbData);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void DoShellNotify(HWND hwnd, PIDLIST_ABSOLUTE pidl1, PIDLIST_ABSOLUTE pidl2, LONG lEvent)
|
|
|
|
{
|
|
|
|
if (s_iStage < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DoSendData(lEvent, pidl1, pidl2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static INT_PTR OnShellNotify(HWND hwnd, WPARAM wParam, LPARAM lParam)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
|
|
|
LONG lEvent;
|
|
|
|
PIDLIST_ABSOLUTE *pidlAbsolute;
|
|
|
|
HANDLE hLock = SHChangeNotification_Lock((HANDLE)wParam, (DWORD)lParam, &pidlAbsolute, &lEvent);
|
|
|
|
if (hLock)
|
|
|
|
{
|
|
|
|
DoShellNotify(hwnd, pidlAbsolute[0], pidlAbsolute[1], lEvent);
|
|
|
|
SHChangeNotification_Unlock(hLock);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pidlAbsolute = (PIDLIST_ABSOLUTE *)wParam;
|
|
|
|
DoShellNotify(hwnd, pidlAbsolute[0], pidlAbsolute[1], lParam);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
static LRESULT CALLBACK SubWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
|
|
|
switch (uMsg)
|
|
|
|
{
|
|
|
|
case WM_CREATE:
|
|
|
|
return (OnCreate(hwnd) ? 0 : -1);
|
|
|
|
|
|
|
|
case WM_COMMAND:
|
|
|
|
OnCommand(hwnd, LOWORD(wParam));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_SHELL_NOTIFY:
|
|
|
|
return OnShellNotify(hwnd, wParam, lParam);
|
|
|
|
|
|
|
|
case WM_DESTROY:
|
|
|
|
OnDestroy(hwnd);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2024-04-24 00:10:29 +00:00
|
|
|
return ::DefWindowProcW(hwnd, uMsg, wParam, lParam);
|
2020-02-28 23:13:52 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
INT APIENTRY
|
2024-04-24 00:10:29 +00:00
|
|
|
wWinMain(
|
|
|
|
HINSTANCE hInstance,
|
|
|
|
HINSTANCE hPrevInstance,
|
|
|
|
LPWSTR lpCmdLine,
|
|
|
|
INT nCmdShow)
|
2020-02-28 23:13:52 +00:00
|
|
|
{
|
2021-06-07 23:07:37 +00:00
|
|
|
if (lstrcmpiW(lpCmdLine, L"") == 0 || lstrcmpiW(lpCmdLine, L"TEST") == 0)
|
2021-06-05 21:51:42 +00:00
|
|
|
return 0;
|
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
WNDCLASSW wc = { 0, SubWindowProc };
|
|
|
|
wc.hInstance = hInstance;
|
|
|
|
wc.hIcon = LoadIconW(NULL, IDI_APPLICATION);
|
|
|
|
wc.hCursor = LoadCursorW(NULL, IDC_ARROW);
|
2020-02-28 23:13:52 +00:00
|
|
|
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
|
2024-04-24 00:10:29 +00:00
|
|
|
wc.lpszClassName = SUB_CLASSNAME;
|
2020-02-28 23:13:52 +00:00
|
|
|
if (!RegisterClassW(&wc))
|
2024-04-24 09:25:15 +00:00
|
|
|
{
|
|
|
|
assert(0);
|
2020-02-28 23:13:52 +00:00
|
|
|
return -1;
|
2024-04-24 09:25:15 +00:00
|
|
|
}
|
2020-02-28 23:13:52 +00:00
|
|
|
|
2024-04-24 00:10:29 +00:00
|
|
|
HWND hwnd = CreateWindowW(SUB_CLASSNAME, SUB_CLASSNAME, WS_OVERLAPPEDWINDOW,
|
2021-12-27 00:45:44 +00:00
|
|
|
CW_USEDEFAULT, CW_USEDEFAULT, 400, 100,
|
2024-04-24 00:10:29 +00:00
|
|
|
NULL, NULL, hInstance, NULL);
|
2020-02-28 23:13:52 +00:00
|
|
|
if (!hwnd)
|
2024-04-24 09:25:15 +00:00
|
|
|
{
|
|
|
|
assert(0);
|
2024-04-24 00:10:29 +00:00
|
|
|
return -2;
|
2024-04-24 09:25:15 +00:00
|
|
|
}
|
2020-02-28 23:13:52 +00:00
|
|
|
|
|
|
|
ShowWindow(hwnd, SW_SHOWNORMAL);
|
|
|
|
UpdateWindow(hwnd);
|
|
|
|
|
|
|
|
MSG msg;
|
|
|
|
while (GetMessageW(&msg, NULL, 0, 0))
|
|
|
|
{
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageW(&msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|