1 /*
2  * PROJECT:         ReactOS API tests
3  * LICENSE:         LGPLv2.1+ - See COPYING.LIB in the top level directory
4  * PURPOSE:         Test for SHELLSTATE
5  * PROGRAMMERS:     Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
6  */
7 #include "shelltest.h"
8 
9 #define NDEBUG
10 #include <debug.h>
11 #include <stdio.h>
12 #include <shellutils.h>
13 #include <strsafe.h>
14 #include <shlwapi.h>
15 
16 /* [HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer] */
17 /* The contents of RegValue ShellState. */
18 typedef struct REGSHELLSTATE
19 {
20     DWORD dwSize;
21     SHELLSTATE ss;
22 } REGSHELLSTATE, *PREGSHELLSTATE;
23 
24 static void dump(const char *name, const void *ptr, size_t siz)
25 {
26     char buf[256], sz[16];
27 
28     StringCbCopyA(buf, sizeof(buf), name);
29     StringCbCatA(buf, sizeof(buf), ": ");
30 
31     const BYTE *pb = reinterpret_cast<const BYTE *>(ptr);
32     while (siz--)
33     {
34         StringCbPrintfA(sz, sizeof(sz), "%02X ", *pb++);
35         StringCbCatA(buf, sizeof(buf), sz);
36     }
37 
38     trace("%s\n", buf);
39 }
40 
41 static int read_key(REGSHELLSTATE *prss)
42 {
43     HKEY hKey;
44     LONG result;
45     DWORD cb;
46     static const LPCWSTR s_pszExplorer =
47         L"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer";
48 
49     memset(prss, 0, sizeof(*prss));
50 
51     result = RegOpenKeyExW(HKEY_CURRENT_USER, s_pszExplorer, 0, KEY_READ, &hKey);
52     ok(result == ERROR_SUCCESS, "result was %ld\n", result);
53     ok(hKey != NULL, "hKey was NULL\n");
54 
55     if (result != ERROR_SUCCESS || !hKey)
56     {
57         skip("RegOpenKeyEx failed: %ld\n", result);
58         return 1;
59     }
60 
61     cb = sizeof(*prss);
62     result = RegQueryValueExW(hKey, L"ShellState", NULL, NULL, reinterpret_cast<LPBYTE>(prss), &cb);
63     RegCloseKey(hKey);
64 
65     ok(result == ERROR_SUCCESS, "result was %ld\n", result);
66     if (result != ERROR_SUCCESS)
67     {
68         skip("RegQueryValueEx failed: %ld\n", result);
69         return 2;
70     }
71 
72     return 0;
73 }
74 
75 extern "C" HKEY WINAPI SHGetShellKey(DWORD flags, LPCWSTR sub_key, BOOL create);
76 
77 static int read_advanced_key(SHELLSTATE* pss)
78 {
79     HKEY hKey;
80     DWORD dwValue, dwSize;
81 
82     hKey = SHGetShellKey(1, L"Advanced", FALSE);
83     if (hKey == NULL)
84     {
85         return 0;
86     }
87 
88     dwSize = sizeof(dwValue);
89     if (SHQueryValueExW(hKey, L"Hidden", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
90     {
91         pss->fShowAllObjects = (dwValue == 1);
92         pss->fShowSysFiles = (dwValue == 2);
93     }
94 
95     dwSize = sizeof(dwValue);
96     if (SHQueryValueExW(hKey, L"HideFileExt", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
97     {
98         pss->fShowExtensions = (dwValue == 0);
99     }
100 
101     dwSize = sizeof(dwValue);
102     if (SHQueryValueExW(hKey, L"DontPrettyPath", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
103     {
104         pss->fDontPrettyPath = (dwValue != 0);
105     }
106 
107     dwSize = sizeof(dwValue);
108     if (SHQueryValueExW(hKey, L"MapNetDrvBtn", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
109     {
110         pss->fMapNetDrvBtn = (dwValue != 0);
111     }
112 
113     dwSize = sizeof(dwValue);
114     if (SHQueryValueExW(hKey, L"ShowInfoTip", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
115     {
116         pss->fShowInfoTip = (dwValue != 0);
117     }
118 
119     dwSize = sizeof(dwValue);
120     if (SHQueryValueExW(hKey, L"HideIcons", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
121     {
122         pss->fHideIcons = (dwValue != 0);
123     }
124 
125     dwSize = sizeof(dwValue);
126     if (SHQueryValueExW(hKey, L"WebView", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
127     {
128         pss->fWebView = (dwValue != 0);
129     }
130 
131     dwSize = sizeof(dwValue);
132     if (SHQueryValueExW(hKey, L"Filter", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
133     {
134         pss->fFilter = (dwValue != 0);
135     }
136 
137     dwSize = sizeof(dwValue);
138     if (SHQueryValueExW(hKey, L"ShowSuperHidden", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
139     {
140         pss->fShowSuperHidden = (dwValue != 0);
141     }
142 
143     dwSize = sizeof(dwValue);
144     if (SHQueryValueExW(hKey, L"NoNetCrawling", NULL, NULL, &dwValue, &dwSize) == ERROR_SUCCESS)
145     {
146         pss->fNoNetCrawling = (dwValue != 0);
147     }
148 
149     RegCloseKey(hKey);
150     return 0;
151 }
152 
153 static int dump_pss(SHELLSTATE *pss)
154 {
155     dump("SHELLSTATE", pss, sizeof(*pss));
156     return 0;
157 }
158 
159 START_TEST(ShellState)
160 {
161     OSVERSIONINFO osinfo;
162     REGSHELLSTATE rss;
163     SHELLSTATE ss, *pss;
164     SHELLFLAGSTATE FlagState;
165     LPBYTE pb;
166     int ret;
167 
168     trace("GetVersion(): 0x%08lX\n", GetVersion());
169 
170     osinfo.dwOSVersionInfoSize = sizeof(osinfo);
171     GetVersionEx(&osinfo);
172     trace("osinfo.dwMajorVersion: 0x%08lX\n", osinfo.dwMajorVersion);
173     trace("osinfo.dwMinorVersion: 0x%08lX\n", osinfo.dwMinorVersion);
174     trace("osinfo.dwBuildNumber: 0x%08lX\n", osinfo.dwBuildNumber);
175     trace("osinfo.dwPlatformId: 0x%08lX\n", osinfo.dwPlatformId);
176 
177     trace("WINVER: 0x%04X\n", WINVER);
178     trace("_WIN32_WINNT: 0x%04X\n", _WIN32_WINNT);
179     trace("_WIN32_IE: 0x%04X\n", _WIN32_IE);
180     trace("NTDDI_VERSION: 0x%08X\n", NTDDI_VERSION);
181 
182 #ifdef _MSC_VER
183     trace("_MSC_VER: 0x%08X\n", int(_MSC_VER));
184 #elif defined(__MINGW32__)
185     trace("__MINGW32__: 0x%08X\n", int(__MINGW32__));
186 #elif defined(__clang__)
187     trace("__clang__: 0x%08X\n", int(__clang__));
188 #else
189     #error Unknown compiler.
190 #endif
191 
192     ok(sizeof(REGSHELLSTATE) >= 0x24, "sizeof(REGSHELLSTATE) was %d\n", (int)sizeof(REGSHELLSTATE));
193     trace("sizeof(SHELLSTATE): %d\n", (int)sizeof(SHELLSTATE));
194     trace("__alignof(SHELLSTATE): %d\n", (int)__alignof(SHELLSTATE));
195     trace("sizeof(SHELLFLAGSTATE): %d\n", (int)sizeof(SHELLFLAGSTATE));
196     trace("sizeof(CABINETSTATE): %d\n", (int)sizeof(CABINETSTATE));
197 
198     pss = &rss.ss;
199     pb = reinterpret_cast<LPBYTE>(pss);
200 
201     ret = read_key(&rss);
202     if (ret)
203     {
204         return;
205     }
206 
207     dump_pss(pss);
208     ok(rss.dwSize >= 0x24, "rss.dwSize was %ld (0x%lX).\n", rss.dwSize, rss.dwSize);
209 
210     read_advanced_key(&rss.ss);
211 
212 #define DUMP_LONG(x) trace(#x ": 0x%08X\n", int(x));
213 #define DUMP_BOOL(x) trace(#x ": %d\n", !!int(x));
214     DUMP_BOOL(pss->fShowAllObjects);
215     DUMP_BOOL(pss->fShowExtensions);
216     DUMP_BOOL(pss->fNoConfirmRecycle);
217     DUMP_BOOL(pss->fShowSysFiles);
218     DUMP_BOOL(pss->fShowCompColor);
219     DUMP_BOOL(pss->fDoubleClickInWebView);
220     DUMP_BOOL(pss->fDesktopHTML);
221     DUMP_BOOL(pss->fWin95Classic);
222     DUMP_BOOL(pss->fDontPrettyPath);
223     DUMP_BOOL(pss->fShowAttribCol);
224     DUMP_BOOL(pss->fMapNetDrvBtn);
225     DUMP_BOOL(pss->fShowInfoTip);
226     DUMP_BOOL(pss->fHideIcons);
227     DUMP_BOOL(pss->fWebView);
228     DUMP_BOOL(pss->fFilter);
229     DUMP_BOOL(pss->fShowSuperHidden);
230     DUMP_BOOL(pss->fNoNetCrawling);
231     DUMP_LONG(pss->lParamSort);
232     DUMP_LONG(pss->iSortDirection);
233     DUMP_LONG(pss->version);
234     DUMP_LONG(pss->lParamSort);
235     DUMP_LONG(pss->iSortDirection);
236     DUMP_LONG(pss->version);
237     DUMP_BOOL(pss->fSepProcess);
238     DUMP_BOOL(pss->fStartPanelOn);
239     DUMP_BOOL(pss->fShowStartPage);
240 #if NTDDI_VERSION >= 0x06000000     // for future use
241     DUMP_BOOL(pss->fIconsOnly);
242     DUMP_BOOL(pss->fShowTypeOverlay);
243     DUMP_BOOL(pss->fShowStatusBar);
244 #endif
245 
246 #define SSF_MASK \
247     (SSF_SHOWALLOBJECTS | SSF_SHOWEXTENSIONS | SSF_NOCONFIRMRECYCLE | \
248      SSF_SHOWCOMPCOLOR | SSF_DOUBLECLICKINWEBVIEW | SSF_DESKTOPHTML | \
249      SSF_WIN95CLASSIC | SSF_DONTPRETTYPATH | SSF_SHOWATTRIBCOL | \
250      SSF_MAPNETDRVBUTTON | SSF_SHOWINFOTIP | SSF_HIDEICONS)
251     // For future:
252     // SSF_AUTOCHECKSELECT, SSF_ICONSONLY, SSF_SHOWTYPEOVERLAY, SSF_SHOWSTATUSBAR
253 
254     /* Get the settings */
255     memset(&ss, 0, sizeof(ss));
256     SHGetSetSettings(&ss, SSF_MASK, FALSE);
257 #define CHECK_REG_FLAG(x) ok(pss->x == ss.x, "ss.%s expected %d, was %d\n", #x, (int)pss->x, (int)ss.x)
258     CHECK_REG_FLAG(fShowAllObjects);
259     CHECK_REG_FLAG(fShowExtensions);
260     CHECK_REG_FLAG(fNoConfirmRecycle);
261     CHECK_REG_FLAG(fShowSysFiles);    // No use
262     CHECK_REG_FLAG(fShowCompColor);
263     CHECK_REG_FLAG(fDoubleClickInWebView);
264     CHECK_REG_FLAG(fDesktopHTML);
265     CHECK_REG_FLAG(fWin95Classic);
266     CHECK_REG_FLAG(fDontPrettyPath);
267     CHECK_REG_FLAG(fShowAttribCol);
268     CHECK_REG_FLAG(fMapNetDrvBtn);
269     CHECK_REG_FLAG(fShowInfoTip);
270     CHECK_REG_FLAG(fHideIcons);
271 #if NTDDI_VERSION >= 0x06000000     // for future use
272     CHECK_REG_FLAG(fAutoCheckSelect);
273     CHECK_REG_FLAG(fIconsOnly);
274 #endif
275 
276     /* Get the flag settings */
277     memset(&FlagState, 0, sizeof(FlagState));
278     SHGetSettings(&FlagState, SSF_MASK);
279 #define CHECK_FLAG(x) ok(ss.x == FlagState.x, "FlagState.%s expected %d, was %d\n", #x, (int)ss.x, (int)FlagState.x)
280     CHECK_FLAG(fShowAllObjects);
281     CHECK_FLAG(fShowExtensions);
282     CHECK_FLAG(fNoConfirmRecycle);
283     CHECK_FLAG(fShowSysFiles);    // No use
284     CHECK_FLAG(fShowCompColor);
285     CHECK_FLAG(fDoubleClickInWebView);
286     CHECK_FLAG(fDesktopHTML);
287     CHECK_FLAG(fWin95Classic);
288     CHECK_FLAG(fDontPrettyPath);
289     CHECK_FLAG(fShowAttribCol);
290     CHECK_FLAG(fMapNetDrvBtn);
291     CHECK_FLAG(fShowInfoTip);
292     CHECK_FLAG(fHideIcons);
293 #if NTDDI_VERSION >= 0x06000000     // for future use
294     CHECK_FLAG(fAutoCheckSelect);
295     CHECK_FLAG(fIconsOnly);
296 #endif
297 
298 #if 1
299     #define DO_IT(x) x
300 #else
301     #define DO_IT(x) do { trace(#x ";\n"); x; } while (0)
302 #endif
303 
304     DO_IT(memset(pss, 0, sizeof(*pss)));
305     DO_IT(pss->dwWin95Unused = 1);
306     ok(pb[4] == 0x01 || dump_pss(pss), "Unexpected pss ^\n");
307 
308     DO_IT(memset(pss, 0, sizeof(*pss)));
309     DO_IT(pss->lParamSort = 1);
310     ok(pb[12] == 0x01 || dump_pss(pss), "Unexpected pss ^\n");
311 
312     DO_IT(memset(pss, 0, sizeof(*pss)));
313     DO_IT(pss->iSortDirection = 0xDEADBEEF);
314     ok(*(UNALIGNED DWORD *)(pb + 16) == 0xDEADBEEF || dump_pss(pss), "Unexpected pss ^\n");
315 
316     DO_IT(memset(pss, 0, sizeof(*pss)));
317     DO_IT(pss->version = 0xDEADBEEF);
318     ok(*(UNALIGNED DWORD *)(pb + 20) == 0xDEADBEEF || dump_pss(pss), "Unexpected pss ^\n");
319 
320     DO_IT(memset(pss, 0, sizeof(*pss)));
321     DO_IT(pss->fSepProcess = TRUE);
322     ok(pb[28] == 0x01 || dump_pss(pss), "Unexpected pss ^\n");
323 }
324