mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 06:45:24 +00:00
298 lines
7 KiB
C++
298 lines
7 KiB
C++
/*
|
|
* PROJECT: ReactOS api tests
|
|
* LICENSE: LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
|
|
* PURPOSE: Test for SHChangeNotify
|
|
* COPYRIGHT: Copyright 2020 Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
|
|
*/
|
|
|
|
#include "shelltest.h"
|
|
#include "shell32_apitest_sub.h"
|
|
|
|
static HWND s_hwnd = NULL;
|
|
static UINT s_uRegID = 0;
|
|
static BOOL s_fRecursive = FALSE;
|
|
static DIRTYPE s_iWatchDir = DIRTYPE_NULL;
|
|
static INT s_nSources = 0;
|
|
static LPITEMIDLIST s_pidl = NULL;
|
|
static WCHAR s_path1[MAX_PATH], s_path2[MAX_PATH];
|
|
static BYTE s_counters[TYPE_MAX + 1];
|
|
static HANDLE s_hEvent = NULL;
|
|
|
|
static BOOL
|
|
OnCreate(HWND hwnd)
|
|
{
|
|
s_hwnd = hwnd;
|
|
s_pidl = DoGetPidl(s_iWatchDir);
|
|
|
|
SHChangeNotifyEntry entry;
|
|
entry.pidl = s_pidl;
|
|
entry.fRecursive = s_fRecursive;
|
|
s_uRegID = SHChangeNotifyRegister(hwnd, s_nSources, SHCNE_ALLEVENTS, WM_SHELL_NOTIFY, 1, &entry);
|
|
return s_uRegID != 0;
|
|
}
|
|
|
|
static void
|
|
OnCommand(HWND hwnd, UINT id)
|
|
{
|
|
switch (id)
|
|
{
|
|
case IDOK:
|
|
case IDCANCEL:
|
|
DestroyWindow(hwnd);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
OnDestroy(HWND hwnd)
|
|
{
|
|
SHChangeNotifyDeregister(s_uRegID);
|
|
s_uRegID = 0;
|
|
|
|
CoTaskMemFree(s_pidl);
|
|
s_pidl = NULL;
|
|
|
|
PostQuitMessage(0);
|
|
s_hwnd = NULL;
|
|
}
|
|
|
|
static BOOL DoPathes(PIDLIST_ABSOLUTE pidl1, PIDLIST_ABSOLUTE pidl2)
|
|
{
|
|
WCHAR path[MAX_PATH];
|
|
if (!SHGetPathFromIDListW(pidl1, path))
|
|
{
|
|
s_path1[0] = s_path2[0] = 0;
|
|
return FALSE;
|
|
}
|
|
|
|
if (wcsstr(path, L"Recent") != NULL)
|
|
return FALSE;
|
|
|
|
StringCchCopyW(s_path1, _countof(s_path1), path);
|
|
|
|
if (!SHGetPathFromIDListW(pidl2, s_path2))
|
|
s_path2[0] = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static VOID DoPathesAndFlags(UINT type, PIDLIST_ABSOLUTE pidl1, PIDLIST_ABSOLUTE pidl2)
|
|
{
|
|
if (DoPathes(pidl1, pidl2))
|
|
{
|
|
s_counters[type] = 1;
|
|
SetEvent(s_hEvent);
|
|
}
|
|
}
|
|
|
|
static void
|
|
DoShellNotify(HWND hwnd, PIDLIST_ABSOLUTE pidl1, PIDLIST_ABSOLUTE pidl2, LONG lEvent)
|
|
{
|
|
switch (lEvent)
|
|
{
|
|
case SHCNE_RENAMEITEM:
|
|
DoPathesAndFlags(TYPE_RENAMEITEM, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_CREATE:
|
|
DoPathesAndFlags(TYPE_CREATE, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_DELETE:
|
|
DoPathesAndFlags(TYPE_DELETE, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_MKDIR:
|
|
DoPathesAndFlags(TYPE_MKDIR, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_RMDIR:
|
|
DoPathesAndFlags(TYPE_RMDIR, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_MEDIAINSERTED:
|
|
break;
|
|
case SHCNE_MEDIAREMOVED:
|
|
break;
|
|
case SHCNE_DRIVEREMOVED:
|
|
break;
|
|
case SHCNE_DRIVEADD:
|
|
break;
|
|
case SHCNE_NETSHARE:
|
|
break;
|
|
case SHCNE_NETUNSHARE:
|
|
break;
|
|
case SHCNE_ATTRIBUTES:
|
|
break;
|
|
case SHCNE_UPDATEDIR:
|
|
DoPathesAndFlags(TYPE_UPDATEDIR, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_UPDATEITEM:
|
|
break;
|
|
case SHCNE_SERVERDISCONNECT:
|
|
break;
|
|
case SHCNE_UPDATEIMAGE:
|
|
break;
|
|
case SHCNE_DRIVEADDGUI:
|
|
break;
|
|
case SHCNE_RENAMEFOLDER:
|
|
DoPathesAndFlags(TYPE_RENAMEFOLDER, pidl1, pidl2);
|
|
break;
|
|
case SHCNE_FREESPACE:
|
|
break;
|
|
case SHCNE_EXTENDED_EVENT:
|
|
break;
|
|
case SHCNE_ASSOCCHANGED:
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static INT_PTR
|
|
OnShellNotify(HWND hwnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
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;
|
|
}
|
|
|
|
static LRESULT
|
|
OnGetNotifyFlags(HWND hwnd)
|
|
{
|
|
if (s_uRegID == 0)
|
|
return 0xFFFFFFFF;
|
|
|
|
DWORD dwFlags = 0;
|
|
for (size_t i = 0; i < _countof(s_counters); ++i)
|
|
{
|
|
if (s_counters[i])
|
|
dwFlags |= (1 << i);
|
|
}
|
|
return dwFlags;
|
|
}
|
|
|
|
static void
|
|
DoSetPaths(HWND hwnd)
|
|
{
|
|
WCHAR szText[MAX_PATH * 2];
|
|
StringCchCopyW(szText, _countof(szText), s_path1);
|
|
StringCchCatW(szText, _countof(szText), L"|");
|
|
StringCchCatW(szText, _countof(szText), s_path2);
|
|
|
|
FILE *fp = _wfopen(TEMP_FILE, L"wb");
|
|
if (fp)
|
|
{
|
|
fwrite(szText, (wcslen(szText) + 1) * sizeof(WCHAR), 1, fp);
|
|
fflush(fp);
|
|
fclose(fp);
|
|
}
|
|
}
|
|
|
|
static LRESULT CALLBACK
|
|
WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
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;
|
|
|
|
case WM_GET_NOTIFY_FLAGS:
|
|
return OnGetNotifyFlags(hwnd);
|
|
|
|
case WM_CLEAR_FLAGS:
|
|
ZeroMemory(&s_counters, sizeof(s_counters));
|
|
s_path1[0] = s_path2[0] = 0;
|
|
break;
|
|
|
|
case WM_SET_PATHS:
|
|
DoSetPaths(hwnd);
|
|
break;
|
|
|
|
default:
|
|
return DefWindowProcW(hwnd, uMsg, wParam, lParam);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static BOOL ParseCommandLine(LPWSTR lpCmdLine)
|
|
{
|
|
LPWSTR pch = lpCmdLine; // fRecursive,iWatchDir,nSources
|
|
s_fRecursive = !!wcstoul(pch, NULL, 0);
|
|
pch = wcschr(pch, L',');
|
|
if (!pch)
|
|
return FALSE;
|
|
++pch;
|
|
|
|
s_iWatchDir = (DIRTYPE)wcstoul(pch, NULL, 0);
|
|
pch = wcschr(pch, L',');
|
|
if (!pch)
|
|
return FALSE;
|
|
++pch;
|
|
|
|
s_nSources = wcstoul(pch, NULL, 0);
|
|
return TRUE;
|
|
}
|
|
|
|
INT APIENTRY
|
|
wWinMain(HINSTANCE hInstance,
|
|
HINSTANCE hPrevInstance,
|
|
LPWSTR lpCmdLine,
|
|
INT nCmdShow)
|
|
{
|
|
if (lstrcmpiW(lpCmdLine, L"") == 0 || lstrcmpiW(lpCmdLine, L"TEST") == 0)
|
|
return 0;
|
|
|
|
if (!ParseCommandLine(lpCmdLine))
|
|
return -1;
|
|
|
|
s_hEvent = OpenEventW(EVENT_ALL_ACCESS, TRUE, EVENT_NAME);
|
|
|
|
WNDCLASSW wc;
|
|
ZeroMemory(&wc, sizeof(wc));
|
|
wc.lpfnWndProc = WindowProc;
|
|
wc.hInstance = GetModuleHandleW(NULL);
|
|
wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
|
|
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
|
|
wc.lpszClassName = CLASSNAME;
|
|
if (!RegisterClassW(&wc))
|
|
return -1;
|
|
|
|
HWND hwnd = CreateWindowW(CLASSNAME, CLASSNAME, WS_OVERLAPPEDWINDOW,
|
|
CW_USEDEFAULT, CW_USEDEFAULT, 400, 100,
|
|
NULL, NULL, GetModuleHandleW(NULL), NULL);
|
|
if (!hwnd)
|
|
return -1;
|
|
|
|
ShowWindow(hwnd, SW_SHOWNORMAL);
|
|
UpdateWindow(hwnd);
|
|
|
|
MSG msg;
|
|
while (GetMessageW(&msg, NULL, 0, 0))
|
|
{
|
|
TranslateMessage(&msg);
|
|
DispatchMessageW(&msg);
|
|
}
|
|
|
|
CloseHandle(s_hEvent);
|
|
|
|
return 0;
|
|
}
|