1 /* 2 * PROJECT: ReactOS api tests 3 * LICENSE: LGPLv2.1+ - See COPYING.LIB in the top level directory 4 * PURPOSE: Test for CShellLink 5 * PROGRAMMER: Andreas Maier 6 */ 7 8 #include "shelltest.h" 9 10 #define NDEBUG 11 #include <debug.h> 12 #include <stdio.h> 13 #include <shellutils.h> 14 15 /* Test IShellLink::SetPath with environment-variables, existing, non-existing, ...*/ 16 typedef struct 17 { 18 PCWSTR pathIn; 19 HRESULT hrSetPath; 20 21 /* Test 1 - hrGetPathX = IShellLink::GetPath(pathOutX, ... , flagsX); */ 22 PCWSTR pathOut1; 23 DWORD flags1; 24 HRESULT hrGetPath1; 25 BOOL expandPathOut1; 26 27 /* Test 2 */ 28 PCWSTR pathOut2; 29 DWORD flags2; 30 HRESULT hrGetPath2; 31 BOOL expandPathOut2; 32 } TEST_SHELL_LINK_DEF; 33 34 static TEST_SHELL_LINK_DEF linkTestList[] = 35 { 36 { 37 L"%comspec%", S_OK, 38 L"%comspec%", SLGP_SHORTPATH, S_OK, TRUE, 39 L"%comspec%", SLGP_RAWPATH, S_OK, FALSE 40 }, 41 { 42 L"%anyvar%", E_INVALIDARG, 43 L"", SLGP_SHORTPATH, S_FALSE, FALSE, 44 L"", SLGP_RAWPATH, S_FALSE, FALSE 45 }, 46 { 47 L"%anyvar%%comspec%", S_OK, 48 L"c:\\%anyvar%%comspec%", SLGP_SHORTPATH, S_OK, TRUE, 49 L"%anyvar%%comspec%", SLGP_RAWPATH, S_OK, FALSE 50 }, 51 { 52 L"%temp%", S_OK, 53 L"%temp%", SLGP_SHORTPATH, S_OK, TRUE, 54 L"%temp%", SLGP_RAWPATH, S_OK, FALSE 55 }, 56 { 57 L"%shell%", S_OK, 58 L"%systemroot%\\system32\\%shell%", SLGP_SHORTPATH, S_OK, TRUE, 59 L"%shell%", SLGP_RAWPATH, S_OK, FALSE 60 }, 61 { 62 L"u:\\anypath\\%anyvar%", S_OK, 63 L"u:\\anypath\\%anyvar%", SLGP_SHORTPATH, S_OK, TRUE, 64 L"u:\\anypath\\%anyvar%", SLGP_RAWPATH, S_OK, FALSE 65 }, 66 { 67 L"c:\\temp", S_OK, 68 L"c:\\temp", SLGP_SHORTPATH, S_OK, FALSE, 69 L"c:\\temp", SLGP_RAWPATH, S_OK, FALSE 70 }, 71 { 72 L"cmd.exe", S_OK, 73 L"%comspec%", SLGP_SHORTPATH, S_OK, TRUE, 74 L"%comspec%", SLGP_RAWPATH, S_OK, TRUE 75 }, 76 { 77 L"%systemroot%\\non-existent-file", S_OK, 78 L"%systemroot%\\non-existent-file", SLGP_SHORTPATH, S_OK, TRUE, 79 L"%systemroot%\\non-existent-file", SLGP_RAWPATH, S_OK, FALSE 80 }, 81 { 82 L"c:\\non-existent-path\\non-existent-file", S_OK, 83 L"c:\\non-existent-path\\non-existent-file", SLGP_SHORTPATH, S_OK, FALSE, 84 L"c:\\non-existent-path\\non-existent-file", SLGP_RAWPATH, S_OK, FALSE 85 }, 86 { 87 L"non-existent-file", E_INVALIDARG, 88 L"", SLGP_SHORTPATH, S_FALSE, FALSE, 89 L"", SLGP_RAWPATH, S_FALSE, FALSE 90 }, 91 }; 92 93 static 94 VOID 95 test_checklinkpath(UINT i, TEST_SHELL_LINK_DEF* testDef) 96 { 97 static WCHAR evVar[MAX_PATH]; 98 99 HRESULT hr, expectedHr; 100 WCHAR wPathOut[MAX_PATH]; 101 BOOL expandPathOut; 102 PCWSTR expectedPathOut; 103 CComPtr<IShellLinkW> psl; 104 UINT i1; 105 DWORD flags; 106 107 hr = CoCreateInstance(CLSID_ShellLink, 108 NULL, 109 CLSCTX_INPROC_SERVER, 110 IID_PPV_ARG(IShellLinkW, &psl)); 111 ok(hr == S_OK, "CoCreateInstance, hr = 0x%lx\n", hr); 112 if (FAILED(hr)) 113 { 114 skip("Could not instantiate CShellLink\n"); 115 return; 116 } 117 118 hr = psl->SetPath(testDef->pathIn); 119 ok(hr == testDef->hrSetPath, "IShellLink::SetPath(%d), got hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrSetPath); 120 121 expectedPathOut = NULL; 122 for (i1 = 0; i1 <= 1; i1++) 123 { 124 if (i1 == 0) /* Usually SLGP_SHORTPATH */ 125 { 126 flags = testDef->flags1; 127 expandPathOut = testDef->expandPathOut1; 128 expectedPathOut = testDef->pathOut1; 129 expectedHr = testDef->hrGetPath1; 130 } 131 else // if (i1 == 1) /* Usually SLGP_RAWPATH */ 132 { 133 flags = testDef->flags2; 134 expandPathOut = testDef->expandPathOut2; 135 expectedPathOut = testDef->pathOut2; 136 expectedHr = testDef->hrGetPath2; 137 } 138 139 /* Patch some variables */ 140 if (expandPathOut) 141 { 142 ExpandEnvironmentStringsW(expectedPathOut, evVar, _countof(evVar)); 143 DPRINT("** %S **\n",evVar); 144 expectedPathOut = evVar; 145 } 146 147 hr = psl->GetPath(wPathOut, _countof(wPathOut), NULL, flags); 148 ok(hr == expectedHr, 149 "IShellLink::GetPath(%d), flags 0x%lx, got hr = 0x%lx, expected 0x%lx\n", 150 i, flags, hr, expectedHr); 151 ok(wcsicmp(wPathOut, expectedPathOut) == 0, 152 "IShellLink::GetPath(%d), flags 0x%lx, in %S, got %S, expected %S\n", 153 i, flags, testDef->pathIn, wPathOut, expectedPathOut); 154 } 155 } 156 157 static 158 VOID 159 TestShellLink(void) 160 { 161 UINT i; 162 163 /* Needed for test */ 164 SetEnvironmentVariableW(L"shell", L"cmd.exe"); 165 166 for (i = 0; i < _countof(linkTestList); ++i) 167 { 168 DPRINT("IShellLink-Test(%d): %S\n", i, linkTestList[i].pathIn); 169 test_checklinkpath(i, &linkTestList[i]); 170 } 171 172 SetEnvironmentVariableW(L"shell",NULL); 173 } 174 175 static 176 VOID 177 TestDescription(void) 178 { 179 HRESULT hr; 180 CComPtr<IShellLinkW> psl; 181 WCHAR buffer[64]; 182 PCWSTR testDescription = L"This is a test description"; 183 184 /* Test SetDescription */ 185 hr = CoCreateInstance(CLSID_ShellLink, 186 NULL, 187 CLSCTX_INPROC_SERVER, 188 IID_PPV_ARG(IShellLinkW, &psl)); 189 ok(hr == S_OK, "CoCreateInstance, hr = 0x%lx\n", hr); 190 if (FAILED(hr)) 191 { 192 skip("Could not instantiate CShellLink\n"); 193 return; 194 } 195 196 memset(buffer, 0x55, sizeof(buffer)); 197 hr = psl->GetDescription(buffer, RTL_NUMBER_OF(buffer)); 198 ok(hr == S_OK, "IShellLink::GetDescription returned hr = 0x%lx\n", hr); 199 ok(buffer[0] == 0, "buffer[0] = %x\n", buffer[0]); 200 ok(buffer[1] == 0x5555, "buffer[1] = %x\n", buffer[1]); 201 202 hr = psl->SetDescription(testDescription); 203 ok(hr == S_OK, "IShellLink::SetDescription returned hr = 0x%lx\n", hr); 204 205 memset(buffer, 0x55, sizeof(buffer)); 206 hr = psl->GetDescription(buffer, RTL_NUMBER_OF(buffer)); 207 ok(hr == S_OK, "IShellLink::GetDescription returned hr = 0x%lx\n", hr); 208 ok(buffer[wcslen(testDescription)] == 0, "buffer[n] = %x\n", buffer[wcslen(testDescription)]); 209 ok(buffer[wcslen(testDescription) + 1] == 0x5555, "buffer[n+1] = %x\n", buffer[wcslen(testDescription) + 1]); 210 ok(!wcscmp(buffer, testDescription), "buffer = '%ls'\n", buffer); 211 212 hr = psl->SetDescription(NULL); 213 ok(hr == S_OK, "IShellLink::SetDescription returned hr = 0x%lx\n", hr); 214 215 memset(buffer, 0x55, sizeof(buffer)); 216 hr = psl->GetDescription(buffer, RTL_NUMBER_OF(buffer)); 217 ok(hr == S_OK, "IShellLink::GetDescription returned hr = 0x%lx\n", hr); 218 ok(buffer[0] == 0, "buffer[0] = %x\n", buffer[0]); 219 ok(buffer[1] == 0x5555, "buffer[1] = %x\n", buffer[1]); 220 } 221 222 223 /* Test IShellLink::Get/SetIconLocation and IExtractIcon::GetIconLocation */ 224 typedef struct 225 { 226 PCWSTR FilePath; 227 228 /* Expected results */ 229 HRESULT hrDefIcon; // Return value for GIL_DEFAULTICON 230 HRESULT hrForShrt; // Return value for GIL_FORSHORTCUT 231 /* Return values for GIL_FORSHELL */ 232 HRESULT hrForShell; 233 PCWSTR IconPath; 234 UINT Flags; 235 } TEST_SHELL_ICON; 236 237 static TEST_SHELL_ICON ShIconTests[] = 238 { 239 /* Executable with icons */ 240 {L"%SystemRoot%\\system32\\cmd.exe", S_FALSE, E_INVALIDARG, 241 S_OK, L"%SystemRoot%\\system32\\cmd.exe", GIL_NOTFILENAME | GIL_PERINSTANCE}, 242 243 /* Executable without icon */ 244 {L"%SystemRoot%\\system32\\autochk.exe", S_FALSE, E_INVALIDARG, 245 S_OK, L"%SystemRoot%\\system32\\autochk.exe", GIL_NOTFILENAME | GIL_PERINSTANCE}, 246 247 /* Existing file */ 248 {L"%SystemRoot%\\system32\\shell32.dll", S_FALSE, E_INVALIDARG, 249 S_OK, L"*", GIL_NOTFILENAME | GIL_PERCLASS}, 250 251 /* Non-existing files */ 252 {L"%SystemRoot%\\non-existent-file.sdf", S_FALSE, E_INVALIDARG, 253 S_OK, L"*", GIL_NOTFILENAME | GIL_PERCLASS}, 254 {L"c:\\non-existent-path\\non-existent-file.sdf", S_FALSE, E_INVALIDARG, 255 S_OK, L"*", GIL_NOTFILENAME | GIL_PERCLASS}, 256 }; 257 258 static 259 VOID 260 test_iconlocation(UINT i, TEST_SHELL_ICON* testDef) 261 { 262 HRESULT hr; 263 CComPtr<IShellLinkW> psl; 264 CComPtr<IExtractIconW> pei; 265 INT iIcon; 266 UINT wFlags; 267 PCWSTR pszExplorer = L"%SystemRoot%\\explorer.exe"; 268 WCHAR szPath[MAX_PATH]; 269 WCHAR szPath2[MAX_PATH]; 270 271 hr = CoCreateInstance(CLSID_ShellLink, 272 NULL, 273 CLSCTX_INPROC_SERVER, 274 IID_PPV_ARG(IShellLinkW, &psl)); 275 ok(hr == S_OK, "CoCreateInstance, hr = 0x%lx\n", hr); 276 if (FAILED(hr)) 277 { 278 skip("Could not instantiate CShellLink\n"); 279 return; 280 } 281 282 /* Set the path to a file */ 283 ExpandEnvironmentStringsW(testDef->FilePath, szPath, _countof(szPath)); 284 hr = psl->SetPath(szPath); 285 ok(hr == S_OK, "IShellLink::SetPath failed, hr = 0x%lx\n", hr); 286 287 /* 288 * This test shows that this does not imply that the icon is automatically 289 * set and be retrieved naively by a call to IShellLink::GetIconLocation. 290 */ 291 iIcon = 0xdeadbeef; 292 wcscpy(szPath, L"garbage"); 293 hr = psl->GetIconLocation(szPath, _countof(szPath), &iIcon); 294 ok(hr == S_OK, "IShellLink::GetIconLocation(%d) failed, hr = 0x%lx\n", i, hr); 295 ok(*szPath == L'\0', "IShellLink::GetIconLocation(%d) returned '%S'\n", i, szPath); 296 ok(iIcon == 0, "IShellLink::GetIconLocation(%d) returned %d, expected %d\n", i, iIcon, 0); 297 298 /* Try to grab the IExtractIconW interface */ 299 hr = psl->QueryInterface(IID_PPV_ARG(IExtractIconW, &pei)); 300 ok(hr == S_OK, "IShellLink::QueryInterface(IExtractIconW)(%d) failed, hr = 0x%lx\n", i, hr); 301 if (!pei) 302 { 303 win_skip("No IExtractIconW interface\n"); 304 return; 305 } 306 307 iIcon = wFlags = 0xdeadbeef; 308 wcscpy(szPath, L"garbage"); 309 hr = pei->GetIconLocation(GIL_DEFAULTICON, szPath, _countof(szPath), &iIcon, &wFlags); 310 ok(hr == testDef->hrDefIcon, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrDefIcon); 311 ok(*szPath == L'\0', "IExtractIcon::GetIconLocation(%d) returned '%S'\n", i, szPath); 312 // ok(iIcon == 0, "IExtractIcon::GetIconLocation(%d) returned %d\n", i, iIcon); 313 314 iIcon = wFlags = 0xdeadbeef; 315 wcscpy(szPath, L"garbage"); 316 hr = pei->GetIconLocation(GIL_FORSHORTCUT, szPath, _countof(szPath), &iIcon, &wFlags); 317 ok(hr == testDef->hrForShrt, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrForShrt); 318 // Here, both szPath and iIcon are untouched... 319 320 iIcon = wFlags = 0xdeadbeef; 321 wcscpy(szPath, L"garbage"); 322 hr = pei->GetIconLocation(GIL_FORSHELL, szPath, _countof(szPath), &iIcon, &wFlags); 323 ok(hr == testDef->hrForShell, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrForShell); 324 ok(wFlags == testDef->Flags, "IExtractIcon::GetIconLocation(%d) returned wFlags = 0x%x, expected 0x%x\n", i, wFlags, testDef->Flags); 325 /* 326 * Actually, even if wFlags specifies GIL_NOTFILENAME, a correct file name is returned 327 * for executables only (at least...), otherwise we can get an asterix '*'. 328 */ 329 ExpandEnvironmentStringsW(testDef->IconPath, szPath2, _countof(szPath2)); 330 ok(_wcsicmp(szPath2, szPath) == 0, "IExtractIcon::GetIconLocation(%d) returned '%S', expected '%S'\n", i, szPath, szPath2); 331 332 // ok(*szPath == L'\0', "IExtractIcon::GetIconLocation returned '%S'\n", szPath); 333 // ok(iIcon == 0, "IExtractIcon::GetIconLocation returned %d\n", iIcon); 334 // ok(FALSE, "hr = 0x%lx, szPath = '%S', iIcon = %d, wFlags = %d\n", hr, szPath, iIcon, wFlags); 335 336 337 /* 338 * Now we test what happens when we explicitly set an icon to the shortcut. 339 * Note that actually, SetIconLocation() does not verify whether the file 340 * really exists. 341 */ 342 hr = psl->SetIconLocation(pszExplorer, 1); 343 ok(hr == S_OK, "IShellLink::SetIconLocation(%d) failed, hr = 0x%lx\n", i, hr); 344 345 /* 346 * First, we call IShellLink::GetIconLocation. We retrieve 347 * exactly what we specified with SetIconLocation. 348 */ 349 iIcon = 0xdeadbeef; 350 wcscpy(szPath, L"garbage"); 351 hr = psl->GetIconLocation(szPath, _countof(szPath), &iIcon); 352 ok(hr == S_OK, "IShellLink::GetIconLocation(%d) failed, hr = 0x%lx\n", i, hr); 353 ok(wcscmp(szPath, pszExplorer) == 0, "IShellLink::GetIconLocation(%d) returned '%S', expected '%S'\n", i, szPath, pszExplorer); 354 ok(iIcon == 1, "IShellLink::GetIconLocation(%d) returned %d, expected %d\n", i, iIcon, 1); 355 356 /* 357 * Now we test what happens with IExtractIcon::GetIconLocation. 358 * We see that it retrieves the icon of the shortcut's underlying file. 359 */ 360 iIcon = wFlags = 0xdeadbeef; 361 wcscpy(szPath, L"garbage"); 362 hr = pei->GetIconLocation(GIL_DEFAULTICON, szPath, _countof(szPath), &iIcon, &wFlags); 363 ok(hr == testDef->hrDefIcon, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrDefIcon); 364 ok(*szPath == L'\0', "IExtractIcon::GetIconLocation(%d) returned '%S'\n", i, szPath); 365 // ok(iIcon == 0, "IExtractIcon::GetIconLocation(%d) returned %d\n", i, iIcon); 366 367 iIcon = wFlags = 0xdeadbeef; 368 wcscpy(szPath, L"garbage"); 369 hr = pei->GetIconLocation(GIL_FORSHORTCUT, szPath, _countof(szPath), &iIcon, &wFlags); 370 ok(hr == testDef->hrForShrt, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrForShrt); 371 // Here, both szPath and iIcon are untouched... 372 373 iIcon = wFlags = 0xdeadbeef; 374 wcscpy(szPath, L"garbage"); 375 hr = pei->GetIconLocation(GIL_FORSHELL, szPath, _countof(szPath), &iIcon, &wFlags); 376 ok(hr == testDef->hrForShell, "IExtractIcon::GetIconLocation(%d) returned hr = 0x%lx, expected 0x%lx\n", i, hr, testDef->hrForShell); 377 ok(wFlags == testDef->Flags, "IExtractIcon::GetIconLocation(%d) returned wFlags = 0x%x, expected 0x%x\n", i, wFlags, testDef->Flags); 378 /* 379 * Actually, even if wFlags specifies GIL_NOTFILENAME, a correct file name is returned 380 * for executables only (at least...), otherwise we can get an asterix '*'. 381 */ 382 ExpandEnvironmentStringsW(testDef->IconPath, szPath2, _countof(szPath2)); 383 ok(_wcsicmp(szPath2, szPath) == 0, "IExtractIcon::GetIconLocation(%d) returned '%S', expected '%S'\n", i, szPath, szPath2); 384 385 // ok(*szPath == L'\0', "IExtractIcon::GetIconLocation returned '%S'\n", szPath); 386 // ok(iIcon == 0, "IExtractIcon::GetIconLocation returned %d\n", iIcon); 387 // ok(FALSE, "hr = 0x%lx, szPath = '%S', iIcon = %d, wFlags = %d\n", hr, szPath, iIcon, wFlags); 388 } 389 390 static 391 VOID 392 TestIconLocation(void) 393 { 394 UINT i; 395 396 for (i = 0; i < _countof(ShIconTests); ++i) 397 { 398 test_iconlocation(i, &ShIconTests[i]); 399 } 400 } 401 402 403 START_TEST(CShellLink) 404 { 405 CoInitializeEx(NULL, COINIT_APARTMENTTHREADED); 406 407 TestShellLink(); 408 TestDescription(); 409 TestIconLocation(); 410 411 CoUninitialize(); 412 } 413