1 /* 2 * PROJECT: NT Object Namespace shell extension 3 * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later) 4 * PURPOSE: Folder view class header and implementation 5 * COPYRIGHT: Copyright 2015-2017 David Quintana <gigaherz@gmail.com> 6 */ 7 8 #pragma once 9 10 extern const GUID CLSID_NtObjectFolder; 11 12 class CFolderViewCB : 13 public CComObjectRootEx<CComMultiThreadModelNoCS>, 14 public IShellFolderViewCB 15 { 16 IShellView* m_View; 17 18 public: 19 20 CFolderViewCB() : m_View(NULL) {} 21 virtual ~CFolderViewCB() {} 22 23 STDMETHODIMP MessageSFVCB(UINT uMsg, WPARAM wParam, LPARAM lParam) override 24 { 25 switch (uMsg) 26 { 27 case SFVM_DEFVIEWMODE: 28 { 29 FOLDERVIEWMODE* pViewMode = (FOLDERVIEWMODE*)lParam; 30 *pViewMode = FVM_DETAILS; 31 return S_OK; 32 } 33 34 case SFVM_COLUMNCLICK: 35 return S_FALSE; 36 37 case SFVM_BACKGROUNDENUM: 38 return S_OK; 39 } 40 41 DbgPrint("MessageSFVCB unimplemented %d %08x %08x\n", uMsg, wParam, lParam); 42 return E_NOTIMPL; 43 } 44 45 STDMETHOD(Initialize)(IShellView* psv) 46 { 47 m_View = psv; 48 return S_OK; 49 } 50 51 DECLARE_NOT_AGGREGATABLE(CFolderViewCB) 52 DECLARE_PROTECT_FINAL_CONSTRUCT() 53 54 BEGIN_COM_MAP(CFolderViewCB) 55 COM_INTERFACE_ENTRY_IID(IID_IShellFolderViewCB, IShellFolderViewCB) 56 END_COM_MAP() 57 }; 58 59 template<class TSelf, typename TItemId, class TExtractIcon> 60 class CCommonFolder : 61 public CComObjectRootEx<CComMultiThreadModelNoCS>, 62 public IShellFolder2, 63 public IPersistFolder2 64 { 65 protected: 66 WCHAR m_NtPath[MAX_PATH]; 67 68 LPITEMIDLIST m_shellPidl; 69 70 public: 71 72 CCommonFolder() : 73 m_shellPidl(NULL) 74 { 75 } 76 77 virtual ~CCommonFolder() 78 { 79 if (m_shellPidl) 80 ILFree(m_shellPidl); 81 } 82 83 // IShellFolder 84 STDMETHODIMP ParseDisplayName( 85 HWND hwndOwner, 86 LPBC pbcReserved, 87 LPOLESTR lpszDisplayName, 88 ULONG *pchEaten, 89 LPITEMIDLIST *ppidl, 90 ULONG *pdwAttributes) override 91 { 92 if (!ppidl) 93 return E_POINTER; 94 95 if (pchEaten) 96 *pchEaten = 0; 97 98 if (pdwAttributes) 99 *pdwAttributes = 0; 100 101 TRACE("CCommonFolder::ParseDisplayName name=%S (ntPath=%S)\n", lpszDisplayName, m_NtPath); 102 103 const TItemId * info; 104 IEnumIDList * it; 105 HRESULT hr = EnumObjects(hwndOwner, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS, &it); 106 if (FAILED(hr)) 107 return hr; 108 109 PWSTR end = StrChrW(lpszDisplayName, '\\'); 110 int length = end ? end - lpszDisplayName : wcslen(lpszDisplayName); 111 112 while (TRUE) 113 { 114 hr = it->Next(1, ppidl, NULL); 115 116 if (FAILED(hr)) 117 return hr; 118 119 if (hr != S_OK) 120 return HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND); 121 122 hr = GetInfoFromPidl(*ppidl, &info); 123 if (FAILED_UNEXPECTEDLY(hr)) 124 return hr; 125 126 if (StrCmpNW(info->entryName, lpszDisplayName, length) == 0) 127 break; 128 } 129 130 // if has remaining path to parse (and the path didn't just terminate in a backslash) 131 if (end && wcslen(end) > 1) 132 { 133 CComPtr<IShellFolder> psfChild; 134 hr = BindToObject(*ppidl, pbcReserved, IID_PPV_ARG(IShellFolder, &psfChild)); 135 if (FAILED_UNEXPECTEDLY(hr)) 136 return hr; 137 138 LPITEMIDLIST child; 139 hr = psfChild->ParseDisplayName(hwndOwner, pbcReserved, end + 1, pchEaten, &child, pdwAttributes); 140 if (FAILED(hr)) 141 return hr; 142 143 LPITEMIDLIST old = *ppidl; 144 *ppidl = ILCombine(old, child); 145 ILFree(old); 146 147 // Count the path separator 148 if (pchEaten) 149 (*pchEaten) += 1; 150 } 151 else 152 { 153 if (pdwAttributes) 154 *pdwAttributes = ConvertAttributes(info, pdwAttributes); 155 } 156 157 if (pchEaten) 158 *pchEaten += wcslen(info->entryName); 159 160 return S_OK; 161 } 162 163 STDMETHOD(EnumObjects)( 164 HWND hwndOwner, 165 SHCONTF grfFlags, 166 IEnumIDList **ppenumIDList) PURE; 167 168 STDMETHODIMP BindToObject( 169 LPCITEMIDLIST pidl, 170 LPBC pbcReserved, 171 REFIID riid, 172 void **ppvOut) override 173 { 174 const TItemId * info; 175 176 if (IsEqualIID(riid, IID_IShellFolder)) 177 { 178 HRESULT hr = GetInfoFromPidl(pidl, &info); 179 if (FAILED_UNEXPECTEDLY(hr)) 180 return hr; 181 182 WCHAR path[MAX_PATH]; 183 184 StringCbCopyW(path, sizeof(path), m_NtPath); 185 PathAppendW(path, info->entryName); 186 187 LPITEMIDLIST first = ILCloneFirst(pidl); 188 LPCITEMIDLIST rest = ILGetNext(pidl); 189 190 LPITEMIDLIST fullPidl = ILCombine(m_shellPidl, first); 191 192 CComPtr<IShellFolder> psfChild; 193 hr = InternalBindToObject(path, info, first, rest, fullPidl, pbcReserved, &psfChild); 194 195 ILFree(fullPidl); 196 ILFree(first); 197 198 if (FAILED(hr)) 199 return hr; 200 201 if (hr == S_FALSE) 202 return S_OK; 203 204 if (rest->mkid.cb > 0) 205 { 206 return psfChild->BindToObject(rest, pbcReserved, riid, ppvOut); 207 } 208 209 return psfChild->QueryInterface(riid, ppvOut); 210 } 211 212 return E_NOTIMPL; 213 } 214 215 protected: 216 STDMETHOD(InternalBindToObject)( 217 PWSTR path, 218 const TItemId * info, 219 LPITEMIDLIST first, 220 LPCITEMIDLIST rest, 221 LPITEMIDLIST fullPidl, 222 LPBC pbcReserved, 223 IShellFolder** ppsfChild) PURE; 224 225 STDMETHOD(ResolveSymLink)( 226 const TItemId * info, 227 LPITEMIDLIST * fullPidl) 228 { 229 return E_NOTIMPL; 230 } 231 232 public: 233 STDMETHODIMP BindToStorage( 234 LPCITEMIDLIST pidl, 235 LPBC pbcReserved, 236 REFIID riid, 237 void **ppvObj) override 238 { 239 UNIMPLEMENTED; 240 return E_NOTIMPL; 241 } 242 243 STDMETHODIMP CompareIDs( 244 LPARAM lParam, 245 LPCITEMIDLIST pidl1, 246 LPCITEMIDLIST pidl2) override 247 { 248 HRESULT hr; 249 250 TRACE("CompareIDs %d\n", lParam); 251 252 const TItemId * id1; 253 hr = GetInfoFromPidl(pidl1, &id1); 254 if (FAILED(hr)) 255 return E_INVALIDARG; 256 257 const TItemId * id2; 258 hr = GetInfoFromPidl(pidl2, &id2); 259 if (FAILED(hr)) 260 return E_INVALIDARG; 261 262 hr = CompareIDs(lParam, id1, id2); 263 if (hr != S_EQUAL) 264 return hr; 265 266 // The wollowing snipped is basically SHELL32_CompareChildren 267 268 PUIDLIST_RELATIVE rest1 = ILGetNext(pidl1); 269 PUIDLIST_RELATIVE rest2 = ILGetNext(pidl2); 270 271 bool isEmpty1 = (rest1->mkid.cb == 0); 272 bool isEmpty2 = (rest2->mkid.cb == 0); 273 274 if (isEmpty1 || isEmpty2) 275 return MAKE_COMPARE_HRESULT(isEmpty2 - isEmpty1); 276 277 LPCITEMIDLIST first1 = ILCloneFirst(pidl1); 278 if (!first1) 279 return E_OUTOFMEMORY; 280 281 CComPtr<IShellFolder> psfNext; 282 hr = BindToObject(first1, NULL, IID_PPV_ARG(IShellFolder, &psfNext)); 283 if (FAILED_UNEXPECTEDLY(hr)) 284 return hr; 285 286 return psfNext->CompareIDs(lParam, rest1, rest2); 287 } 288 289 protected: 290 STDMETHOD(CompareName)( 291 LPARAM lParam, 292 const TItemId * first, 293 const TItemId * second) 294 { 295 bool f1 = IsFolder(first); 296 bool f2 = IsFolder(second); 297 298 HRESULT hr = MAKE_COMPARE_HRESULT(f2 - f1); 299 if (hr != S_EQUAL) 300 return hr; 301 302 bool canonical = (lParam & 0xFFFF0000) == SHCIDS_CANONICALONLY; 303 if (canonical) 304 { 305 // Shortcut: avoid comparing contents if not necessary when the results are not for display. 306 hr = MAKE_COMPARE_HRESULT(second->entryNameLength - first->entryNameLength); 307 if (hr != S_EQUAL) 308 return hr; 309 310 int minlength = min(first->entryNameLength, second->entryNameLength); 311 if (minlength > 0) 312 { 313 hr = MAKE_COMPARE_HRESULT(memcmp(first->entryName, second->entryName, minlength)); 314 if (hr != S_EQUAL) 315 return hr; 316 } 317 318 return S_EQUAL; 319 } 320 321 int minlength = min(first->entryNameLength, second->entryNameLength); 322 if (minlength > 0) 323 { 324 hr = MAKE_COMPARE_HRESULT(StrCmpNW(first->entryName, second->entryName, minlength / sizeof(WCHAR))); 325 if (hr != S_EQUAL) 326 return hr; 327 } 328 329 return MAKE_COMPARE_HRESULT(second->entryNameLength - first->entryNameLength); 330 } 331 332 public: 333 STDMETHODIMP CreateViewObject( 334 HWND hwndOwner, 335 REFIID riid, 336 void **ppvOut) override 337 { 338 if (!IsEqualIID(riid, IID_IShellView)) 339 return E_NOINTERFACE; 340 341 _CComObject<CFolderViewCB> *pcb; 342 343 HRESULT hr = _CComObject<CFolderViewCB>::CreateInstance(&pcb); 344 if (FAILED(hr)) 345 return hr; 346 347 pcb->AddRef(); 348 349 SFV_CREATE sfv; 350 sfv.cbSize = sizeof(sfv); 351 sfv.pshf = this; 352 sfv.psvOuter = NULL; 353 sfv.psfvcb = pcb; 354 355 IShellView* view; 356 357 hr = SHCreateShellFolderView(&sfv, &view); 358 if (FAILED(hr)) 359 return hr; 360 361 pcb->Initialize(view); 362 363 pcb->Release(); 364 365 *ppvOut = view; 366 367 return S_OK; 368 } 369 370 STDMETHODIMP GetAttributesOf( 371 UINT cidl, 372 PCUITEMID_CHILD_ARRAY apidl, 373 SFGAOF *rgfInOut) override 374 { 375 const TItemId * info; 376 377 TRACE("GetAttributesOf %d\n", cidl); 378 379 if (cidl == 0) 380 { 381 *rgfInOut &= SFGAO_FOLDER | SFGAO_HASSUBFOLDER | SFGAO_BROWSABLE; 382 return S_OK; 383 } 384 385 for (int i = 0; i < (int)cidl; i++) 386 { 387 PCUITEMID_CHILD pidl = apidl[i]; 388 389 HRESULT hr = GetInfoFromPidl(pidl, &info); 390 if (FAILED_UNEXPECTEDLY(hr)) 391 return hr; 392 393 // Update attributes. 394 *rgfInOut = ConvertAttributes(info, rgfInOut); 395 } 396 397 return S_OK; 398 } 399 400 STDMETHODIMP GetUIObjectOf( 401 HWND hwndOwner, 402 UINT cidl, 403 PCUITEMID_CHILD_ARRAY apidl, 404 REFIID riid, 405 UINT *prgfInOut, 406 void **ppvOut) override 407 { 408 DWORD res; 409 TRACE("GetUIObjectOf\n"); 410 411 if (IsEqualIID(riid, IID_IContextMenu) || 412 IsEqualIID(riid, IID_IContextMenu2) || 413 IsEqualIID(riid, IID_IContextMenu3)) 414 { 415 CComPtr<IContextMenu> pcm; 416 417 HKEY keys[1]; 418 419 LPITEMIDLIST parent = m_shellPidl; 420 421 CComPtr<IShellFolder> psfParent = this; 422 423 LPCITEMIDLIST child; 424 425 int nkeys = _countof(keys); 426 if (cidl == 1 && IsSymLink(apidl[0])) 427 { 428 const TItemId * info; 429 HRESULT hr = GetInfoFromPidl(apidl[0], &info); 430 if (FAILED(hr)) 431 return hr; 432 433 LPITEMIDLIST target; 434 hr = ResolveSymLink(info, &target); 435 if (FAILED(hr)) 436 return hr; 437 438 CComPtr<IShellFolder> psfTarget; 439 hr = ::SHBindToParent(target, IID_PPV_ARG(IShellFolder, &psfTarget), &child); 440 if (FAILED(hr)) 441 { 442 ILFree(target); 443 return hr; 444 } 445 446 parent = ILClone(target); 447 ILRemoveLastID(parent); 448 psfParent = psfTarget; 449 450 apidl = &child; 451 } 452 453 if (cidl == 1 && IsFolder(apidl[0])) 454 { 455 res = RegOpenKey(HKEY_CLASSES_ROOT, L"Folder", keys + 0); 456 if (!NT_SUCCESS(res)) 457 return HRESULT_FROM_NT(res); 458 } 459 else 460 { 461 nkeys = 0; 462 } 463 464 HRESULT hr = CDefFolderMenu_Create2(parent, hwndOwner, cidl, apidl, psfParent, DefCtxMenuCallback, nkeys, keys, &pcm); 465 if (FAILED_UNEXPECTEDLY(hr)) 466 return hr; 467 468 return pcm->QueryInterface(riid, ppvOut); 469 } 470 471 if (IsEqualIID(riid, IID_IExtractIconW)) 472 { 473 return ShellObjectCreatorInit<TExtractIcon>(m_NtPath, m_shellPidl, cidl, apidl, riid, ppvOut); 474 } 475 476 if (IsEqualIID(riid, IID_IDataObject)) 477 { 478 return CIDLData_CreateFromIDArray(m_shellPidl, cidl, apidl, (IDataObject**)ppvOut); 479 } 480 481 if (IsEqualIID(riid, IID_IQueryAssociations)) 482 { 483 if (cidl == 1 && IsFolder(apidl[0])) 484 { 485 CComPtr<IQueryAssociations> pqa; 486 HRESULT hr = AssocCreate(CLSID_QueryAssociations, IID_PPV_ARG(IQueryAssociations, &pqa)); 487 if (FAILED_UNEXPECTEDLY(hr)) 488 return hr; 489 490 hr = pqa->Init(ASSOCF_INIT_DEFAULTTOFOLDER, L"NTObjShEx.NTDirectory", NULL, hwndOwner); 491 if (FAILED_UNEXPECTEDLY(hr)) 492 return hr; 493 494 return pqa->QueryInterface(riid, ppvOut); 495 } 496 } 497 498 return E_NOTIMPL; 499 } 500 501 STDMETHODIMP GetDisplayNameOf( 502 LPCITEMIDLIST pidl, 503 SHGDNF uFlags, 504 STRRET *lpName) override 505 { 506 const TItemId * info; 507 508 TRACE("GetDisplayNameOf %p\n", pidl); 509 510 HRESULT hr = GetInfoFromPidl(pidl, &info); 511 if (FAILED_UNEXPECTEDLY(hr)) 512 return hr; 513 514 if (GET_SHGDN_FOR(uFlags) & SHGDN_FOREDITING) 515 { 516 hr = MakeStrRetFromString(info->entryName, info->entryNameLength, lpName); 517 if (FAILED_UNEXPECTEDLY(hr)) 518 return hr; 519 } 520 521 WCHAR path[MAX_PATH] = { 0 }; 522 523 if (GET_SHGDN_FOR(uFlags) & SHGDN_FORPARSING) 524 { 525 if (GET_SHGDN_RELATION(uFlags) != SHGDN_INFOLDER) 526 { 527 hr = GetFullName(m_shellPidl, uFlags, path, _countof(path)); 528 if (FAILED_UNEXPECTEDLY(hr)) 529 return hr; 530 } 531 } 532 533 PathAppendW(path, info->entryName); 534 535 LPCITEMIDLIST pidlNext = ILGetNext(pidl); 536 if (pidlNext && pidlNext->mkid.cb > 0) 537 { 538 LPITEMIDLIST pidlFirst = ILCloneFirst(pidl); 539 540 CComPtr<IShellFolder> psfChild; 541 hr = BindToObject(pidlFirst, NULL, IID_PPV_ARG(IShellFolder, &psfChild)); 542 if (FAILED_UNEXPECTEDLY(hr)) 543 return hr; 544 545 WCHAR temp[MAX_PATH]; 546 STRRET childName; 547 548 hr = psfChild->GetDisplayNameOf(pidlNext, uFlags | SHGDN_INFOLDER, &childName); 549 if (FAILED_UNEXPECTEDLY(hr)) 550 return hr; 551 552 hr = StrRetToBufW(&childName, pidlNext, temp, _countof(temp)); 553 if (FAILED_UNEXPECTEDLY(hr)) 554 return hr; 555 556 PathAppendW(path, temp); 557 558 ILFree(pidlFirst); 559 } 560 561 hr = MakeStrRetFromString(path, lpName); 562 if (FAILED_UNEXPECTEDLY(hr)) 563 return hr; 564 565 return S_OK; 566 } 567 568 STDMETHODIMP SetNameOf( 569 HWND hwnd, 570 LPCITEMIDLIST pidl, 571 LPCOLESTR lpszName, 572 SHGDNF uFlags, 573 LPITEMIDLIST *ppidlOut) override 574 { 575 UNIMPLEMENTED; 576 return E_NOTIMPL; 577 } 578 579 // IShellFolder2 580 STDMETHODIMP GetDefaultSearchGUID(GUID *lpguid) override 581 { 582 UNIMPLEMENTED; 583 return E_NOTIMPL; 584 } 585 586 STDMETHODIMP EnumSearches(IEnumExtraSearch **ppenum) override 587 { 588 UNIMPLEMENTED; 589 return E_NOTIMPL; 590 } 591 592 STDMETHODIMP GetDefaultColumn( 593 DWORD dwReserved, 594 ULONG *pSort, 595 ULONG *pDisplay) override 596 { 597 if (pSort) 598 *pSort = 0; 599 if (pDisplay) 600 *pDisplay = 0; 601 return S_OK; 602 } 603 604 STDMETHOD(GetDefaultColumnState)( 605 UINT iColumn, 606 SHCOLSTATEF *pcsFlags) PURE; 607 608 STDMETHOD(GetDetailsEx)( 609 LPCITEMIDLIST pidl, 610 const SHCOLUMNID *pscid, 611 VARIANT *pv) PURE; 612 613 STDMETHOD(GetDetailsOf)( 614 LPCITEMIDLIST pidl, 615 UINT iColumn, 616 SHELLDETAILS *psd) PURE; 617 618 STDMETHOD(MapColumnToSCID)( 619 UINT iColumn, 620 SHCOLUMNID *pscid) PURE; 621 622 // IPersist 623 STDMETHODIMP GetClassID(CLSID *lpClassId) override 624 { 625 if (!lpClassId) 626 return E_POINTER; 627 628 *lpClassId = CLSID_NtObjectFolder; 629 return S_OK; 630 } 631 632 // IPersistFolder 633 STDMETHODIMP Initialize(PCIDLIST_ABSOLUTE pidl) override 634 { 635 m_shellPidl = ILClone(pidl); 636 637 StringCbCopyW(m_NtPath, sizeof(m_NtPath), L"\\"); 638 639 return S_OK; 640 } 641 642 // IPersistFolder2 643 STDMETHODIMP GetCurFolder(PIDLIST_ABSOLUTE * pidl) override 644 { 645 if (pidl) 646 *pidl = ILClone(m_shellPidl); 647 if (!m_shellPidl) 648 return S_FALSE; 649 return S_OK; 650 } 651 652 // Internal 653 protected: 654 STDMETHOD(CompareIDs)( 655 LPARAM lParam, 656 const TItemId * first, 657 const TItemId * second) PURE; 658 659 STDMETHOD_(ULONG, ConvertAttributes)( 660 const TItemId * entry, 661 PULONG inMask) PURE; 662 663 STDMETHOD_(BOOL, IsFolder)(LPCITEMIDLIST pcidl) 664 { 665 const TItemId * info; 666 667 HRESULT hr = GetInfoFromPidl(pcidl, &info); 668 if (FAILED(hr)) 669 return hr; 670 671 return IsFolder(info); 672 } 673 674 STDMETHOD_(BOOL, IsFolder)(const TItemId * info) PURE; 675 676 STDMETHOD_(BOOL, IsSymLink)(LPCITEMIDLIST pcidl) 677 { 678 const TItemId * info; 679 680 HRESULT hr = GetInfoFromPidl(pcidl, &info); 681 if (FAILED(hr)) 682 return hr; 683 684 return IsSymLink(info); 685 } 686 687 STDMETHOD_(BOOL, IsSymLink)(const TItemId * info) 688 { 689 return FALSE; 690 } 691 692 virtual HRESULT GetInfoFromPidl(LPCITEMIDLIST pcidl, const TItemId ** pentry) PURE; 693 694 public: 695 static HRESULT CALLBACK DefCtxMenuCallback(IShellFolder * /*psf*/, HWND /*hwnd*/, IDataObject * /*pdtobj*/, UINT uMsg, WPARAM /*wParam*/, LPARAM /*lParam*/) 696 { 697 switch (uMsg) 698 { 699 case DFM_MERGECONTEXTMENU: 700 return S_OK; 701 702 case DFM_INVOKECOMMAND: 703 case DFM_INVOKECOMMANDEX: 704 case DFM_GETDEFSTATICID: // Required for Windows 7 to pick a default 705 return S_FALSE; 706 } 707 return E_NOTIMPL; 708 } 709 710 DECLARE_NOT_AGGREGATABLE(TSelf) 711 DECLARE_PROTECT_FINAL_CONSTRUCT() 712 713 BEGIN_COM_MAP(TSelf) 714 COM_INTERFACE_ENTRY_IID(IID_IShellFolder, IShellFolder) 715 COM_INTERFACE_ENTRY_IID(IID_IShellFolder2, IShellFolder2) 716 COM_INTERFACE_ENTRY_IID(IID_IPersist, IPersist) 717 COM_INTERFACE_ENTRY_IID(IID_IPersistFolder, IPersistFolder) 718 COM_INTERFACE_ENTRY_IID(IID_IPersistFolder2, IPersistFolder2) 719 END_COM_MAP() 720 721 }; 722