Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3383 lines
110 KiB

  1. //-------------------------------------------------------------------------//
  2. //
  3. // AugMisf.cpp - Augmented Merge IShellFolder class implementation.
  4. //
  5. //-------------------------------------------------------------------------//
  6. #include "priv.h"
  7. #include "augmisf.h"
  8. #include "resource.h"
  9. #include "mluisupp.h"
  10. #define TF_AUGM 0x10000000
  11. //-------------------------------------------------------------------------//
  12. // BUGBUG: Shell allocator bullchit, inserted here because SHRealloc
  13. // isn't imported into browseui, this module's hosting executable.
  14. // If we get SHRealloc, the following block can be removed:
  15. #define _EXPL_SHELL_ALLOCATOR_
  16. #ifdef _EXPL_SHELL_ALLOCATOR_
  17. #define SHRealloc( pv, cb ) shrealloc( pv, cb )
  18. void* shrealloc( void* pv, size_t cb )
  19. {
  20. IMalloc* pMalloc ;
  21. void* pvRet = NULL ;
  22. if( SUCCEEDED( SHGetMalloc( &pMalloc ) ) ) {
  23. pvRet = pMalloc->Realloc( pv, cb ) ;
  24. ATOMICRELEASE( pMalloc ) ;
  25. }
  26. return pvRet ;
  27. }
  28. #endif _EXPL_SHELL_ALLOCATOR_
  29. BOOL AffectAllUsers(HWND hwnd);
  30. // id - verb mappings for IContextMenu impl
  31. const struct
  32. {
  33. UINT id;
  34. LPCSTR pszVerb;
  35. } c_sIDVerbMap[] =
  36. {
  37. {SMIDM_DELETE, "delete"},
  38. {SMIDM_RENAME, "rename"},
  39. {SMIDM_PROPERTIES, "properties"},
  40. //{SMIDM_OPEN, "open"},
  41. //{SMIDM_EXPLORE, "explore"},
  42. };
  43. // augmisf context menu
  44. class CAugMergeISFContextMenu : public IContextMenu2
  45. {
  46. public:
  47. // *** IUnknown methods ***
  48. STDMETHOD (QueryInterface)(REFIID, void**);
  49. STDMETHOD_(ULONG, AddRef)();
  50. STDMETHOD_(ULONG, Release)();
  51. // *** IContextMenu methods ***
  52. STDMETHOD(QueryContextMenu)(HMENU hmenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags);
  53. STDMETHOD(InvokeCommand)(LPCMINVOKECOMMANDINFO lpici);
  54. STDMETHOD(GetCommandString)(UINT_PTR idCmd, UINT uType, UINT* pwReserved, LPSTR pszName, UINT cchMax);
  55. // *** IContextMenu2 methods ***
  56. STDMETHOD(HandleMenuMsg)(UINT uMsg, WPARAM wParam, LPARAM lParam);
  57. protected:
  58. CAugMergeISFContextMenu(IShellFolder *psfCommon, LPCITEMIDLIST pidlCommon,
  59. IShellFolder *psfUser, LPCITEMIDLIST pidlUser, LPITEMIDLIST pidl,
  60. HWND hwnd, UINT * prgfInOut);
  61. ~CAugMergeISFContextMenu();
  62. friend class CAugmentedMergeISF;
  63. friend CAugMergeISFContextMenu* CreateMergeISFContextMenu(
  64. IShellFolder *psfCommon, LPCITEMIDLIST pidlCommon,
  65. IShellFolder *psfUser, LPCITEMIDLIST pidlUser, LPITEMIDLIST pidl,
  66. HWND hwnd, UINT * prgfInOut);
  67. protected:
  68. LPITEMIDLIST _pidlItem;
  69. IShellFolder * _psfCommon;
  70. IShellFolder * _psfUser;
  71. IContextMenu * _pcmCommon;
  72. IContextMenu * _pcmUser;
  73. LPITEMIDLIST _pidlCommon;
  74. LPITEMIDLIST _pidlUser;
  75. UINT _idFirst;
  76. LONG _cRef;
  77. HWND _hwnd;
  78. };
  79. CAugMergeISFContextMenu* CreateMergeISFContextMenu(
  80. IShellFolder *psfCommon, LPCITEMIDLIST pidlCommon,
  81. IShellFolder *psfUser, LPCITEMIDLIST pidlUser, LPITEMIDLIST pidl,
  82. HWND hwnd, UINT * prgfInOut)
  83. {
  84. CAugMergeISFContextMenu* pcm = new CAugMergeISFContextMenu(psfCommon, pidlCommon,
  85. psfUser, pidlUser,
  86. pidl, hwnd, prgfInOut);
  87. if (pcm)
  88. {
  89. if (!pcm->_pidlItem)
  90. {
  91. delete pcm;
  92. pcm = NULL;
  93. }
  94. }
  95. return pcm;
  96. }
  97. CAugMergeISFContextMenu::CAugMergeISFContextMenu(IShellFolder *psfCommon, LPCITEMIDLIST pidlCommon,
  98. IShellFolder *psfUser, LPCITEMIDLIST pidlUser,
  99. LPITEMIDLIST pidl, HWND hwnd, UINT * prgfInOut)
  100. {
  101. _cRef = 1;
  102. HRESULT hres;
  103. _hwnd = hwnd;
  104. _psfCommon = psfCommon;
  105. if (_psfCommon)
  106. {
  107. _psfCommon->AddRef();
  108. hres = _psfCommon->GetUIObjectOf(hwnd, 1, (LPCITEMIDLIST*)&pidl, IID_IContextMenu, prgfInOut, (void **)&_pcmCommon);
  109. ASSERT(SUCCEEDED(hres) || !_pcmCommon);
  110. _pidlCommon = ILClone(pidlCommon);
  111. }
  112. _psfUser = psfUser;
  113. if (_psfUser)
  114. {
  115. _psfUser->AddRef();
  116. hres = _psfUser->GetUIObjectOf(hwnd, 1, (LPCITEMIDLIST*)&pidl, IID_IContextMenu, prgfInOut, (void **)&_pcmUser);
  117. ASSERT(SUCCEEDED(hres) || !_pcmUser);
  118. _pidlUser = ILClone(pidlUser);
  119. }
  120. _pidlItem = ILClone(pidl);
  121. ASSERT(_psfCommon || _psfUser);
  122. }
  123. CAugMergeISFContextMenu::~CAugMergeISFContextMenu()
  124. {
  125. ATOMICRELEASE(_psfCommon);
  126. ATOMICRELEASE(_pcmCommon);
  127. ATOMICRELEASE(_psfUser);
  128. ATOMICRELEASE(_pcmUser);
  129. ILFree(_pidlCommon);
  130. ILFree(_pidlUser);
  131. ILFree(_pidlItem);
  132. }
  133. STDMETHODIMP CAugMergeISFContextMenu::QueryInterface(REFIID riid, LPVOID *ppvOut)
  134. {
  135. static const QITAB qit[] = {
  136. QITABENTMULTI(CAugMergeISFContextMenu, IContextMenu, IContextMenu2),
  137. QITABENT(CAugMergeISFContextMenu, IContextMenu2),
  138. { 0 },
  139. };
  140. return QISearch(this, qit, riid, ppvOut);
  141. }
  142. STDMETHODIMP_(ULONG) CAugMergeISFContextMenu::AddRef()
  143. {
  144. return InterlockedIncrement(&_cRef);
  145. }
  146. STDMETHODIMP_(ULONG) CAugMergeISFContextMenu::Release()
  147. {
  148. if (InterlockedDecrement(&_cRef))
  149. return _cRef;
  150. delete this;
  151. return 0;
  152. }
  153. HRESULT CAugMergeISFContextMenu::QueryContextMenu(HMENU hmenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
  154. {
  155. HRESULT hres = E_INVALIDARG;
  156. if (hmenu)
  157. {
  158. HMENU hmContext = LoadMenuPopup(MENU_SM_CONTEXTMENU);
  159. if (hmContext)
  160. {
  161. if (!_psfCommon || !_psfUser)
  162. {
  163. DeleteMenu(hmContext, SMIDM_OPENCOMMON, MF_BYCOMMAND);
  164. DeleteMenu(hmContext, SMIDM_EXPLORECOMMON, MF_BYCOMMAND);
  165. }
  166. _idFirst = idCmdFirst;
  167. Shell_MergeMenus(hmenu, hmContext, -1, idCmdFirst, idCmdLast, MM_ADDSEPARATOR);
  168. DestroyMenu(hmContext);
  169. // Make it look "Shell Like"
  170. SetMenuDefaultItem(hmenu, 0, MF_BYPOSITION);
  171. hres = S_OK;
  172. }
  173. else
  174. hres = E_OUTOFMEMORY;
  175. }
  176. return hres;
  177. }
  178. HRESULT CAugMergeISFContextMenu::InvokeCommand(LPCMINVOKECOMMANDINFO pici)
  179. {
  180. UINT id = -1;
  181. HRESULT hres = E_FAIL;
  182. CMINVOKECOMMANDINFO ici = *pici;
  183. if (pici->cbSize < SIZEOF(CMINVOKECOMMANDINFO))
  184. return E_INVALIDARG;
  185. if (HIWORD(pici->lpVerb))
  186. {
  187. for (int i=0; i < ARRAYSIZE(c_sIDVerbMap); i++)
  188. {
  189. if (lstrcmpiA(pici->lpVerb, c_sIDVerbMap[i].pszVerb) == 0)
  190. {
  191. id = c_sIDVerbMap[i].id;
  192. break;
  193. }
  194. }
  195. }
  196. else
  197. id = (UINT) PtrToUlong( pici->lpVerb ); // Win64: should be ok since MAKEINTRESOURCE assumed
  198. switch (id)
  199. {
  200. case -1:
  201. hres = E_INVALIDARG;
  202. break;
  203. case SMIDM_OPEN:
  204. case SMIDM_EXPLORE:
  205. case SMIDM_OPENCOMMON:
  206. case SMIDM_EXPLORECOMMON:
  207. {
  208. IShellFolder * psf;
  209. LPITEMIDLIST pidl;
  210. if (id == SMIDM_OPEN || id == SMIDM_EXPLORE)
  211. {
  212. if (_psfUser)
  213. {
  214. psf = _psfUser;
  215. pidl = _pidlUser;
  216. }
  217. else
  218. {
  219. psf = _psfCommon;
  220. pidl = _pidlCommon;
  221. }
  222. }
  223. else
  224. {
  225. psf = _psfCommon;
  226. pidl = _pidlCommon;
  227. }
  228. if (psf && pidl)
  229. {
  230. SHELLEXECUTEINFO shei = {0};
  231. shei.lpIDList = ILCombine(pidl, _pidlItem);
  232. if (shei.lpIDList)
  233. {
  234. shei.cbSize = sizeof(shei);
  235. shei.fMask = SEE_MASK_IDLIST;
  236. shei.nShow = SW_SHOWNORMAL;
  237. if (id == SMIDM_EXPLORE || id == SMIDM_EXPLORECOMMON)
  238. shei.lpVerb = TEXT("explore");
  239. else
  240. shei.lpVerb = TEXT("open");
  241. hres = ShellExecuteEx(&shei) ? S_OK : E_FAIL;
  242. ILFree((LPITEMIDLIST)shei.lpIDList);
  243. }
  244. }
  245. }
  246. break;
  247. case SMIDM_PROPERTIES:
  248. {
  249. IContextMenu * pcm = _pcmUser ? _pcmUser : _pcmCommon;
  250. if (pcm)
  251. {
  252. ici.lpVerb = "properties";
  253. hres = pcm->InvokeCommand(&ici);
  254. }
  255. }
  256. break;
  257. case SMIDM_DELETE:
  258. ici.lpVerb = "delete";
  259. hres = S_OK;
  260. if (_pcmUser)
  261. {
  262. hres = _pcmUser->InvokeCommand(&ici);
  263. }
  264. else if (_pcmCommon)
  265. {
  266. ici.fMask |= CMIC_MASK_FLAG_NO_UI;
  267. if (AffectAllUsers(_hwnd))
  268. hres = _pcmCommon->InvokeCommand(&ici);
  269. else
  270. hres = E_FAIL;
  271. }
  272. break;
  273. case SMIDM_RENAME:
  274. ASSERT(0);
  275. hres = E_NOTIMPL; // sftbar picks this off
  276. break;
  277. }
  278. return hres;
  279. }
  280. HRESULT CAugMergeISFContextMenu::GetCommandString(UINT_PTR idCmd, UINT uType, UINT* pwReserved, LPSTR pszName, UINT cchMax)
  281. {
  282. HRESULT hres = E_NOTIMPL;
  283. // if hiword in not null then a string is passed to us. we don't handle that case (yet?)
  284. if (!HIWORD(idCmd) && (uType == GCS_VERBA || uType == GCS_VERBW))
  285. {
  286. hres = E_INVALIDARG;
  287. for (int i = 0; hres != S_OK && i < ARRAYSIZE(c_sIDVerbMap); i++)
  288. {
  289. if (c_sIDVerbMap[i].id == idCmd)
  290. {
  291. if (uType == GCS_VERBA)
  292. lstrcpynA(pszName, c_sIDVerbMap[i].pszVerb, cchMax);
  293. else
  294. SHAnsiToUnicode(c_sIDVerbMap[i].pszVerb, (LPWSTR)pszName, cchMax);
  295. hres = S_OK;
  296. }
  297. }
  298. }
  299. return hres;
  300. }
  301. // we need IContextMenu2 although HandleMenuMsg is not impl because of the way sftbar
  302. // works -- it caches only IContextMenu2 so if we don't have ICM2 sftbar will think
  303. // that it does not have context menu so it will eat the messages intended for the hmenu
  304. // that way, with context menu up, if user presses esc it will kill start menu sub menu
  305. // not the context menu.
  306. HRESULT CAugMergeISFContextMenu::HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam)
  307. {
  308. return E_NOTIMPL;
  309. }
  310. //-------------------------------------------------------------------------//
  311. // Augmented Merge Shell Folder's pidl wrapper package consists of a versioned
  312. // header followed by n 'source namespace' pidl wrappers.
  313. // Each individual pidl wrapper consists of a header containing a
  314. // collection lookup index followed by the source pidl itself. The
  315. // source pidl's mkid.cb member is used to seek the next pidl wrap in
  316. // the package.
  317. //-------------------------------------------------------------------------//
  318. //-------------------------------------------------------------------------//
  319. //--- Augmented Merge Shell Folder's pidl wrapper header
  320. typedef struct tagAUGM_IDWRAP {
  321. USHORT cb ; // pidl wrap length
  322. USHORT Reserved ; // reserved.
  323. ULONG tag ; // AugMergeISF pidl signature
  324. ULONG version ; // AugMergeISF pidl version
  325. ULONG cSrcs ; // Number of source namespace objects backing this composite pidl
  326. } AUGM_IDWRAP;
  327. typedef UNALIGNED AUGM_IDWRAP *PAUGM_IDWRAP;
  328. //--- Source pidl header. One or more of these records will be concatenated
  329. // within the wrap following the wrap header.
  330. typedef struct tagAUGM_IDSRC {
  331. UINT nID ; // source namespace index
  332. BYTE pidl[0] ; // source pidl
  333. } AUGM_IDSRC;
  334. typedef UNALIGNED AUGM_IDSRC *PAUGM_IDSRC;
  335. //-------------------------------------------------------------------------//
  336. // Constants
  337. //-------------------------------------------------------------------------//
  338. #define AUGM_WRAPTAG MAKELONG( MAKEWORD('A','u'), MAKEWORD('g','M') )
  339. #define AUGM_WRAPVERSION_1_0 MAKELONG( 1, 0 )
  340. #define AUGM_WRAPCURRENTVERSION AUGM_WRAPVERSION_1_0
  341. #define INVALID_NAMESPACE_INDEX ((UINT)-1)
  342. #define CB_IDLIST_TERMINATOR sizeof(USHORT)
  343. //-------------------------------------------------------------------------//
  344. // Augmented Merge shell folder pidl wrap utilities
  345. //-------------------------------------------------------------------------//
  346. //-------------------------------------------------------------------------//
  347. // Resolves the wrap header from the indicated pidl.
  348. #define AugMergeISF_GetWrap( p ) ((PAUGM_IDWRAP)(p))
  349. //-------------------------------------------------------------------------//
  350. // Determines whether the indicated pidl is an Augmented Merge
  351. // shell folder pidl wrapper.
  352. HRESULT AugMergeISF_IsWrap(
  353. IN LPCITEMIDLIST pidlTest,
  354. IN ULONG nVersion = AUGM_WRAPCURRENTVERSION )
  355. {
  356. ASSERT(IS_VALID_PIDL( pidlTest ));
  357. if (pidlTest)
  358. {
  359. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pidlTest ) ;
  360. return (pWrap->cb >= sizeof(AUGM_IDWRAP) &&
  361. pWrap->tag == AUGM_WRAPTAG &&
  362. pWrap->version == nVersion) ?
  363. S_OK : E_UNEXPECTED ; //BUGBUG: better error code for version mismatch?
  364. }
  365. else
  366. {
  367. return E_INVALIDARG;
  368. }
  369. }
  370. //-------------------------------------------------------------------------//
  371. // Retrieves the number of source namespace pidls in the wrap.
  372. // If the pidl was not wrapped, the return value is -1.
  373. ULONG AugMergeISF_GetSourceCount( IN LPCITEMIDLIST pidl )
  374. {
  375. ASSERT(SUCCEEDED(AugMergeISF_IsWrap(pidl)));
  376. return AugMergeISF_GetWrap(pidl)->cSrcs;
  377. }
  378. //-------------------------------------------------------------------------//
  379. // Creates an IDLIST wrapper object based on the indicated source pidl.
  380. HRESULT AugMergeISF_CreateWrap(
  381. IN LPCITEMIDLIST pidlSrc,
  382. IN UINT nSrcID,
  383. OUT LPITEMIDLIST* ppidlWrap )
  384. {
  385. ASSERT( ppidlWrap ) ;
  386. ASSERT( IS_VALID_PIDL( pidlSrc ) && INVALID_NAMESPACE_INDEX != nSrcID ) ;
  387. *ppidlWrap = NULL ;
  388. // Allocate a header and terminator
  389. LPBYTE pBuf = NULL ;
  390. WORD cbAlloc = sizeof(AUGM_IDWRAP) +
  391. sizeof(AUGM_IDSRC) + pidlSrc->mkid.cb +
  392. // we need two terminators, one for pidlSrc and one for the wrap
  393. // the one for pidlSrc is necessary for the ILClone to work
  394. // because it gets confused with the nSrcID that follows the pidl
  395. CB_IDLIST_TERMINATOR +
  396. CB_IDLIST_TERMINATOR ;
  397. if( NULL == (pBuf = (LPBYTE)IEILCreate( cbAlloc )) )
  398. return E_OUTOFMEMORY ;
  399. // Initialize wrap header members
  400. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pBuf ) ;
  401. pWrap->cb = cbAlloc - CB_IDLIST_TERMINATOR ;
  402. pWrap->tag = AUGM_WRAPTAG ;
  403. pWrap->version = AUGM_WRAPCURRENTVERSION ;
  404. if( pidlSrc )
  405. {
  406. PAUGM_IDSRC pSrc = (PAUGM_IDSRC)(pBuf + sizeof(AUGM_IDWRAP)) ;
  407. pSrc->nID = nSrcID ;
  408. memcpy( pSrc->pidl, pidlSrc, pidlSrc->mkid.cb ) ;
  409. pWrap->cSrcs = 1 ;
  410. }
  411. *ppidlWrap = (LPITEMIDLIST)pWrap ;
  412. return S_OK ;
  413. }
  414. BOOL WrappedPidlContainsSrcID(LPCITEMIDLIST pidlWrap, UINT uSrcID)
  415. {
  416. ASSERT(SUCCEEDED(AugMergeISF_IsWrap(pidlWrap)));
  417. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pidlWrap ) ;
  418. if( pWrap->cSrcs > 0 )
  419. {
  420. LPBYTE p = ((LPBYTE)pWrap) + sizeof(AUGM_IDWRAP) ; // position of first pidl header.
  421. PAUGM_IDSRC pSrc = (PAUGM_IDSRC)p ;
  422. // offset to next pidl header, needs terminator so that ILClone below can work
  423. UINT cbPidl= ((LPITEMIDLIST)pSrc->pidl)->mkid.cb + CB_IDLIST_TERMINATOR;
  424. if (pSrc->nID != uSrcID && pWrap->cSrcs > 1)
  425. {
  426. pSrc = (PAUGM_IDSRC)(p + sizeof(AUGM_IDSRC) + cbPidl) ;
  427. }
  428. if (pSrc->nID == uSrcID)
  429. return TRUE;
  430. }
  431. return FALSE;
  432. }
  433. HRESULT AugMergeISF_WrapRemovePidl(
  434. IN LPITEMIDLIST pidlWrap,
  435. IN UINT nSrcID,
  436. OUT LPITEMIDLIST* ppidlRet )
  437. {
  438. ASSERT( IS_VALID_WRITE_PTR( ppidlRet, LPITEMIDLIST )) ;
  439. *ppidlRet = NULL ;
  440. HRESULT hr = AugMergeISF_IsWrap(pidlWrap);
  441. if (SUCCEEDED(hr))
  442. {
  443. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pidlWrap ) ;
  444. ASSERT(pWrap->cSrcs > 1);
  445. LPBYTE p = ((LPBYTE)pWrap) + sizeof(AUGM_IDWRAP) ; // position of first pidl header.
  446. PAUGM_IDSRC pSrc = (PAUGM_IDSRC)p ;
  447. // offset to next pidl header, needs terminator so that ILClone below can work
  448. UINT cbPidl= ((LPITEMIDLIST)pSrc->pidl)->mkid.cb + CB_IDLIST_TERMINATOR;
  449. // We want to look for the Other SrcID. So we loop while the source id we're removing is
  450. // equal. When it's not equal, we've got the ID.
  451. if (pSrc->nID == nSrcID)
  452. {
  453. pSrc = (PAUGM_IDSRC)(p + sizeof(AUGM_IDSRC) + cbPidl) ;
  454. }
  455. if (pSrc->nID != nSrcID)
  456. {
  457. hr = AugMergeISF_CreateWrap((LPITEMIDLIST)pSrc->pidl, pSrc->nID, ppidlRet);
  458. ILFree(pidlWrap);
  459. }
  460. else
  461. {
  462. hr = E_FAIL;
  463. }
  464. }
  465. return hr;
  466. }
  467. //-------------------------------------------------------------------------//
  468. // Adds a source pidl to the indicated pidl wrap.
  469. HRESULT AugMergeISF_WrapAddPidl(
  470. IN LPCITEMIDLIST pidlSrc,
  471. IN UINT nSrcID,
  472. IN OUT LPITEMIDLIST* ppidlWrap )
  473. {
  474. ASSERT (ppidlWrap && IS_VALID_PIDL( *ppidlWrap ));
  475. ASSERT (IS_VALID_PIDL( pidlSrc ));
  476. ASSERT (INVALID_NAMESPACE_INDEX != nSrcID );
  477. HRESULT hr ;
  478. if (FAILED((hr = AugMergeISF_IsWrap(*ppidlWrap))))
  479. return hr ;
  480. // AHHHHHHHHHHH Rewrite this.
  481. if (WrappedPidlContainsSrcID(*ppidlWrap, nSrcID))
  482. {
  483. if (AugMergeISF_GetSourceCount(*ppidlWrap) > 1)
  484. {
  485. hr = AugMergeISF_WrapRemovePidl((LPITEMIDLIST)*ppidlWrap, nSrcID, ppidlWrap);
  486. }
  487. else
  488. {
  489. ILFree(*ppidlWrap);
  490. return AugMergeISF_CreateWrap(pidlSrc, nSrcID, ppidlWrap);
  491. }
  492. if (FAILED(hr))
  493. {
  494. return hr;
  495. }
  496. }
  497. // Retrieve wrap header
  498. PAUGM_IDWRAP pWrap = (PAUGM_IDWRAP)*ppidlWrap ;
  499. // Reallocate a block large enough to contain our new record.
  500. WORD offTerm0 = pWrap->cb, // offset to end of existing wrap
  501. offTerm1 = offTerm0 + sizeof(AUGM_IDSRC) + pidlSrc->mkid.cb, // offset to end of next record
  502. cbRealloc= offTerm1 + 2*CB_IDLIST_TERMINATOR ; // total bytes to reallocate
  503. LPBYTE pRealloc ;
  504. if( NULL == (pRealloc = (LPBYTE)SHRealloc( pWrap, cbRealloc )) )
  505. return E_OUTOFMEMORY ;
  506. // Adjust our pointers if memory moved
  507. pWrap = (PAUGM_IDWRAP)pRealloc ;
  508. // Initialize new record in the wrap
  509. UNALIGNED AUGM_IDSRC* pSrc = (PAUGM_IDSRC)(pRealloc + offTerm0 ) ;
  510. pSrc->nID = nSrcID ;
  511. memcpy( pSrc->pidl, pidlSrc, pidlSrc->mkid.cb ) ;
  512. // Terminate new record
  513. ZeroMemory( pRealloc + offTerm1, 2*CB_IDLIST_TERMINATOR ) ;
  514. // Update our header
  515. pWrap->cb = cbRealloc - CB_IDLIST_TERMINATOR ;
  516. pWrap->cSrcs++ ;
  517. *ppidlWrap = (LPITEMIDLIST)pWrap ;
  518. return S_OK ;
  519. }
  520. //-------------------------------------------------------------------------//
  521. // Private pidl enumeration block (GetFirst/GetNext)
  522. typedef struct tagAUGM_IDWRAP_ENUM
  523. {
  524. ULONG cbStruct ; // structure size
  525. PAUGM_IDWRAP pWrap ; // wrap header.
  526. PAUGM_IDSRC pSrcNext ; // pointer to next src header
  527. } AUGM_IDWRAP_ENUM, *PAUGM_IDWRAP_ENUM ;
  528. //-------------------------------------------------------------------------//
  529. // Begins enumeration of source pidls in the indicated pidl wrap.
  530. HANDLE AugMergeISF_EnumFirstSrcPidl(
  531. IN LPCITEMIDLIST pidlWrap,
  532. OUT UINT* pnSrcID,
  533. OUT LPITEMIDLIST* ppidlRet )
  534. {
  535. ASSERT( IS_VALID_WRITE_PTR( ppidlRet, LPITEMIDLIST ) && IS_VALID_WRITE_PTR( pnSrcID, UINT ) ) ;
  536. PAUGM_IDWRAP_ENUM pEnum = NULL ;
  537. *ppidlRet = NULL ;
  538. *pnSrcID = (UINT)-1 ;
  539. HRESULT hr = AugMergeISF_IsWrap(pidlWrap);
  540. if(SUCCEEDED(hr))
  541. {
  542. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pidlWrap ) ;
  543. if( pWrap->cSrcs > 0 )
  544. {
  545. LPBYTE p = ((LPBYTE)pWrap) + sizeof(AUGM_IDWRAP) ; // position of first pidl header.
  546. PAUGM_IDSRC pSrc = (PAUGM_IDSRC)p ;
  547. // offset to next pidl header, needs terminator so that ILClone below can work
  548. UINT cbPidl= ((LPITEMIDLIST)pSrc->pidl)->mkid.cb + CB_IDLIST_TERMINATOR;
  549. if( NULL != (pEnum = new AUGM_IDWRAP_ENUM) )
  550. {
  551. pEnum->cbStruct = sizeof(*pEnum) ;
  552. pEnum->pWrap = pWrap ;
  553. pEnum->pSrcNext = (PAUGM_IDSRC)(p + sizeof(AUGM_IDSRC) + cbPidl) ;
  554. *pnSrcID = pSrc->nID ;
  555. *ppidlRet = ILClone( (LPITEMIDLIST)pSrc->pidl ) ;
  556. if ( NULL == *ppidlRet )
  557. {
  558. delete pEnum;
  559. pEnum = NULL;
  560. }
  561. }
  562. }
  563. }
  564. return pEnum ;
  565. }
  566. //-------------------------------------------------------------------------//
  567. // Continues source pidl enumeration
  568. BOOL AugMergeISF_EnumNextSrcPidl(
  569. IN HANDLE hEnum,
  570. OUT UINT* pnSrcID,
  571. OUT LPITEMIDLIST* ppidlRet )
  572. {
  573. PAUGM_IDWRAP_ENUM pEnum = (PAUGM_IDWRAP_ENUM)hEnum ;
  574. HRESULT hr = E_UNEXPECTED ;
  575. ASSERT( IS_VALID_WRITE_PTR( pEnum, AUGM_IDWRAP_ENUM ) ) ;
  576. ASSERT( sizeof(*pEnum) == pEnum->cbStruct ) ;
  577. ASSERT( sizeof(*pEnum) == pEnum->cbStruct );
  578. *ppidlRet = NULL ;
  579. *pnSrcID = (UINT)-1 ;
  580. if (SUCCEEDED((hr = AugMergeISF_IsWrap((LPCITEMIDLIST)pEnum->pWrap))))
  581. {
  582. if ((LPBYTE)(pEnum->pWrap) + pEnum->pWrap->cb <= (LPBYTE)pEnum->pSrcNext)
  583. hr = S_FALSE ;
  584. else
  585. {
  586. UNALIGNED AUGM_IDSRC* pualSrcNext = pEnum->pSrcNext;
  587. *pnSrcID = pualSrcNext->nID;
  588. *ppidlRet = ILClone((LPITEMIDLIST)pualSrcNext->pidl);
  589. pEnum->pSrcNext = (PAUGM_IDSRC)(
  590. ((LPBYTE)pualSrcNext) +
  591. sizeof(AUGM_IDSRC) +
  592. ((LPITEMIDLIST)pualSrcNext->pidl)->mkid.cb +
  593. CB_IDLIST_TERMINATOR);
  594. hr = S_OK ;
  595. return TRUE ;
  596. }
  597. }
  598. return FALSE ;
  599. }
  600. //-------------------------------------------------------------------------//
  601. // Terminates source pidl enumeration
  602. void AugMergeISF_EndEnumSrcPidls(
  603. IN OUT HANDLE& hEnum )
  604. {
  605. PAUGM_IDWRAP_ENUM pEnum = (PAUGM_IDWRAP_ENUM)hEnum ;
  606. ASSERT( IS_VALID_WRITE_PTR( pEnum, AUGM_IDWRAP_ENUM ) &&
  607. sizeof(*pEnum) == pEnum->cbStruct );
  608. delete pEnum ;
  609. hEnum = NULL ;
  610. }
  611. //-------------------------------------------------------------------------//
  612. // Allocates and returns a copy of the specified source pidl
  613. // from the wrapped pidl.
  614. HRESULT AugMergeISF_GetSrcPidl(
  615. IN LPCITEMIDLIST pidlWrap,
  616. IN UINT nSrcID,
  617. OUT LPITEMIDLIST* ppidlRet )
  618. {
  619. ASSERT( ppidlRet ) ;
  620. *ppidlRet = NULL ;
  621. HANDLE hEnum ;
  622. BOOL bEnum ;
  623. UINT nSrcIDEnum ;
  624. LPITEMIDLIST pidlEnum ;
  625. for( hEnum = AugMergeISF_EnumFirstSrcPidl( pidlWrap, &nSrcIDEnum, &pidlEnum ), bEnum = TRUE ;
  626. hEnum && bEnum ;
  627. bEnum = AugMergeISF_EnumNextSrcPidl( hEnum, &nSrcIDEnum, &pidlEnum ) )
  628. {
  629. if( nSrcIDEnum == nSrcID )
  630. {
  631. *ppidlRet = pidlEnum ;
  632. AugMergeISF_EndEnumSrcPidls(hEnum);
  633. return S_OK ;
  634. }
  635. ILFree( pidlEnum ) ;
  636. }
  637. AugMergeISF_EndEnumSrcPidls( hEnum ) ;
  638. return E_FAIL ;
  639. }
  640. #ifdef DEBUG
  641. BOOL IsValidWrappedPidl(LPCITEMIDLIST pidlWrap)
  642. {
  643. BOOL fValid = FALSE;
  644. if (pidlWrap == NULL)
  645. return FALSE;
  646. if (FAILED(AugMergeISF_IsWrap(pidlWrap)))
  647. return FALSE;
  648. HANDLE hEnum ;
  649. UINT nSrcIDEnum ;
  650. LPITEMIDLIST pidlEnum ;
  651. hEnum = AugMergeISF_EnumFirstSrcPidl( pidlWrap, &nSrcIDEnum, &pidlEnum );
  652. do
  653. {
  654. fValid = IS_VALID_PIDL(pidlEnum);
  655. ILFree(pidlEnum);
  656. }
  657. while( fValid && AugMergeISF_EnumNextSrcPidl( hEnum, &nSrcIDEnum, &pidlEnum ));
  658. AugMergeISF_EndEnumSrcPidls( hEnum ) ;
  659. return fValid;
  660. }
  661. #endif
  662. //-------------------------------------------------------------------------//
  663. int AugmEnumCompare(void *pv1, void *pv2, LPARAM lParam)
  664. {
  665. CAugISFEnumItem* paugmEnum1 = (CAugISFEnumItem*)pv1;
  666. CAugISFEnumItem* paugmEnum2 = (CAugISFEnumItem*)pv2;
  667. int iRet = -1;
  668. if (paugmEnum1 && paugmEnum2)
  669. {
  670. // Are these two items of different types?
  671. if (BOOLIFY(paugmEnum1->_rgfAttrib & SFGAO_FOLDER) ^ BOOLIFY(paugmEnum2->_rgfAttrib & SFGAO_FOLDER))
  672. {
  673. // Yes. Then Folders sort before items.
  674. iRet = BOOLIFY(paugmEnum1->_rgfAttrib & SFGAO_FOLDER) ? 1 : -1;
  675. }
  676. else // They are of the same type. Then compare by name
  677. {
  678. iRet = lstrcmpi(paugmEnum1->_pszDisplayName, paugmEnum2->_pszDisplayName);
  679. }
  680. }
  681. return iRet;
  682. }
  683. LPVOID AugmEnumMerge(UINT uMsg, void * pv1, void * pv2, LPARAM lParam)
  684. {
  685. void * pvRet = pv1;
  686. switch (uMsg)
  687. {
  688. case DPAMM_MERGE:
  689. {
  690. HANDLE hEnum;
  691. UINT nSrcID;
  692. LPITEMIDLIST pidl;
  693. CAugISFEnumItem* paugmeDest = (CAugISFEnumItem*)pv1;
  694. CAugISFEnumItem* paugmeSrc = (CAugISFEnumItem*)pv2;
  695. ASSERT(paugmeDest && paugmeSrc);
  696. hEnum = AugMergeISF_EnumFirstSrcPidl(paugmeSrc->_pidlWrap, &nSrcID, &pidl);
  697. if (hEnum)
  698. {
  699. // add pidl from src to dest
  700. AugMergeISF_WrapAddPidl(pidl, nSrcID, &paugmeDest->_pidlWrap);
  701. // no longer need hEnum
  702. AugMergeISF_EndEnumSrcPidls(hEnum);
  703. // this was copied to paugmeDest->_pidlWrap
  704. ILFree(pidl);
  705. }
  706. }
  707. break;
  708. case DPAMM_INSERT:
  709. {
  710. CAugISFEnumItem* paugmNew = new CAugISFEnumItem;
  711. CAugISFEnumItem* paugmSrc = (CAugISFEnumItem*)pv1;
  712. if (paugmNew)
  713. {
  714. paugmNew->_pidlWrap = ILClone(paugmSrc->_pidlWrap);
  715. if (paugmNew->_pidlWrap)
  716. {
  717. paugmNew->SetDisplayName(paugmSrc->_pszDisplayName);
  718. paugmNew->_rgfAttrib = paugmSrc->_rgfAttrib;
  719. }
  720. else
  721. {
  722. delete paugmNew;
  723. paugmNew = NULL;
  724. }
  725. }
  726. pvRet = paugmNew;
  727. }
  728. break;
  729. default:
  730. ASSERT(0);
  731. }
  732. return pvRet;
  733. }
  734. typedef struct
  735. {
  736. LPTSTR pszDisplayName;
  737. BOOL fFolder;
  738. } AUGMISFSEARCHFORPIDL;
  739. int CALLBACK AugMISFSearchForOnePidlByDisplayName(LPVOID p1, LPVOID p2, LPARAM lParam)
  740. {
  741. AUGMISFSEARCHFORPIDL* pSearchFor = (AUGMISFSEARCHFORPIDL*)p1;
  742. CAugISFEnumItem* paugmEnum = (CAugISFEnumItem*)p2;
  743. // Are they of different types?
  744. if (BOOLIFY(paugmEnum->_rgfAttrib & SFGAO_FOLDER) ^ pSearchFor->fFolder)
  745. {
  746. // Yes.
  747. return pSearchFor->fFolder ? 1 : -1;
  748. }
  749. else // They are of the same type. Then compare by name
  750. {
  751. return StrCmpI(pSearchFor->pszDisplayName, paugmEnum->_pszDisplayName);
  752. }
  753. }
  754. //-------------------------------------------------------------------------------------------------//
  755. // DPA utilities
  756. #define DPA_GETPTRCOUNT( hdpa ) ((NULL != (hdpa)) ? DPA_GetPtrCount((hdpa)) : 0)
  757. #define DPA_GETPTR( hdpa, i, type ) ((NULL != (hdpa)) ? (type*)DPA_GetPtr((hdpa), i) : (type*)NULL)
  758. #define DPA_DESTROY( hdpa, pfn ) { if( NULL != hdpa ) \
  759. { DPA_DestroyCallback( hdpa, pfn, NULL ) ; \
  760. hdpa = NULL ; }}
  761. //-------------------------------------------------------------------------------------------------//
  762. // Forwards...
  763. class CEnum ;
  764. class CChildObject ;
  765. //-------------------------------------------------------------------------------------------------//
  766. // Augmented Merge Shell Folder source namespace descriptor.
  767. //
  768. // Objects of class CNamespace are created by CAugmentedMergeISF in
  769. // the AddNameSpace() method impl, and are maintained in the collection
  770. // CAugmentedMergeISF::_hdpaNamespaces.
  771. //
  772. class CNamespace
  773. //-------------------------------------------------------------------------------------------------//
  774. {
  775. public:
  776. CNamespace( const GUID * pguidUIObject, IShellFolder* psf, LPCITEMIDLIST pidl, ULONG dwAttrib ) ;
  777. ~CNamespace() ;
  778. IShellFolder* ShellFolder() { return _psf ; }
  779. REFGUID Guid() { return _guid ; }
  780. ULONG Attrib() const { return _dwAttrib ; }
  781. LPITEMIDLIST GetPidl() const { return _pidl; }
  782. void Assign( const GUID * pguidUIObject, IShellFolder* psf, LPCITEMIDLIST pidl, ULONG dwAttrib ) ;
  783. void Unassign() ;
  784. HRESULT RegisterNotify( HWND, UINT, ULONG ) ;
  785. HRESULT UnregisterNotify() ;
  786. BOOL SetOwner( IUnknown *punk ) ;
  787. protected:
  788. IShellFolder* _psf ; // IShellFolder interface pointer
  789. GUID _guid ; // optional GUID for specialized UI handling
  790. LPITEMIDLIST _pidl ; // optional pidl
  791. ULONG _dwAttrib ; // optional flags
  792. UINT _uChangeReg ; // Shell change notify registration ID.
  793. } ;
  794. //-------------------------------------------------------------------------------------------------//
  795. inline CNamespace::CNamespace( const GUID * pguidUIObject, IShellFolder* psf, LPCITEMIDLIST pidl, ULONG dwAttrib )
  796. : _psf(NULL),
  797. _pidl(NULL),
  798. _guid(GUID_NULL),
  799. _dwAttrib(0),
  800. _uChangeReg(0)
  801. {
  802. Assign( pguidUIObject, psf, pidl, dwAttrib ) ;
  803. }
  804. //-------------------------------------------------------------------------------------------------//
  805. inline CNamespace::~CNamespace() {
  806. UnregisterNotify() ;
  807. Unassign() ;
  808. }
  809. //-------------------------------------------------------------------------------------------------//
  810. // Assigns data members.
  811. void CNamespace::Assign( const GUID * pguidUIObject, IShellFolder* psf, LPCITEMIDLIST pidl, ULONG dwAttrib )
  812. {
  813. Unassign() ;
  814. if( NULL != (_psf = psf) )
  815. _psf->AddRef() ;
  816. _pidl = ILClone( pidl ) ;
  817. _guid = pguidUIObject ? *pguidUIObject : GUID_NULL ;
  818. _dwAttrib = dwAttrib ;
  819. }
  820. //-------------------------------------------------------------------------------------------------//
  821. // Unassigns data members.
  822. void CNamespace::Unassign()
  823. {
  824. ATOMICRELEASE( _psf ) ;
  825. ILFree( _pidl ) ;
  826. _pidl = NULL ;
  827. _guid = GUID_NULL ;
  828. _dwAttrib = 0L ;
  829. }
  830. //-------------------------------------------------------------------------------------------------//
  831. // Register change notification for the namespace
  832. HRESULT CNamespace::RegisterNotify( HWND hwnd, UINT uMsg, ULONG lEvents )
  833. {
  834. if( 0 == _uChangeReg )
  835. _uChangeReg = ::RegisterNotify(hwnd,
  836. uMsg,
  837. _pidl,
  838. lEvents,
  839. SHCNRF_ShellLevel | SHCNRF_InterruptLevel | SHCNRF_RecursiveInterrupt,
  840. TRUE);
  841. return 0 != _uChangeReg ? S_OK : E_FAIL ;
  842. }
  843. //-------------------------------------------------------------------------------------------------//
  844. // Unregister change notification for the namespace
  845. HRESULT CNamespace::UnregisterNotify()
  846. {
  847. if( 0 != _uChangeReg )
  848. {
  849. UINT uID = _uChangeReg;
  850. _uChangeReg = 0;
  851. ::SHChangeNotifyDeregister(uID);
  852. }
  853. return S_OK;
  854. }
  855. //-------------------------------------------------------------------------------------------------//
  856. inline BOOL CNamespace::SetOwner(IUnknown *punkOwner)
  857. {
  858. if (_psf)
  859. {
  860. IUnknown_SetOwner(_psf, punkOwner);
  861. return TRUE ;
  862. }
  863. return FALSE ;
  864. }
  865. //-------------------------------------------------------------------------//
  866. CAugmentedMergeISF::CAugmentedMergeISF() : _cRef(1)
  867. {
  868. ASSERT(_hdpaNamespaces == NULL);
  869. ASSERT(_punkOwner == NULL);
  870. ASSERT(_pdt == NULL);
  871. DllAddRef() ;
  872. }
  873. //-------------------------------------------------------------------------//
  874. CAugmentedMergeISF::~CAugmentedMergeISF()
  875. {
  876. SetOwner(NULL);
  877. FreeNamespaces();
  878. DllRelease();
  879. }
  880. //-------------------------------------------------------------------------//
  881. // CAugmentedMergeISF global CreateInstance method for da class factory
  882. //-------------------------------------------------------------------------//
  883. STDAPI CAugmentedISF2_CreateInstance( IUnknown* pUnkOuter, IUnknown** ppunk, LPCOBJECTINFO poi )
  884. {
  885. // aggregation checking is handled in class factory
  886. CAugmentedMergeISF* pObj;
  887. if( NULL == (pObj = new CAugmentedMergeISF) )
  888. return E_OUTOFMEMORY ;
  889. *ppunk = SAFECAST( pObj, IAugmentedShellFolder2 * ) ;
  890. return S_OK;
  891. }
  892. //-------------------------------------------------------------------------//
  893. // CAugmentedMergeISF - IUnknown methods
  894. //-------------------------------------------------------------------------//
  895. //-------------------------------------------------------------------------//
  896. STDMETHODIMP CAugmentedMergeISF::QueryInterface(REFIID riid, void **ppvObj)
  897. {
  898. static const QITAB qit[] = {
  899. QITABENTMULTI( CAugmentedMergeISF, IShellFolder, IAugmentedShellFolder ),
  900. QITABENT( CAugmentedMergeISF, IAugmentedShellFolder ),
  901. QITABENT( CAugmentedMergeISF, IAugmentedShellFolder2 ),
  902. QITABENT( CAugmentedMergeISF, IShellFolder2 ),
  903. QITABENT( CAugmentedMergeISF, IShellService ),
  904. QITABENT( CAugmentedMergeISF, ITranslateShellChangeNotify ),
  905. QITABENT( CAugmentedMergeISF, IDropTarget ),
  906. { 0 },
  907. };
  908. return QISearch( this, qit, riid, ppvObj ) ;
  909. }
  910. //-------------------------------------------------------------------------//
  911. STDMETHODIMP_(ULONG) CAugmentedMergeISF::AddRef()
  912. {
  913. return InterlockedIncrement(&_cRef);
  914. }
  915. //-------------------------------------------------------------------------//
  916. STDMETHODIMP_(ULONG) CAugmentedMergeISF::Release()
  917. {
  918. if (InterlockedDecrement(&_cRef))
  919. return _cRef;
  920. delete this;
  921. return 0;
  922. }
  923. //-------------------------------------------------------------------------//
  924. // CAugmentedMergeISF - IShellFolder methods
  925. //-------------------------------------------------------------------------//
  926. //-------------------------------------------------------------------------//
  927. STDMETHODIMP CAugmentedMergeISF::EnumObjects(HWND hwnd, DWORD grfFlags, IEnumIDList **ppenumIDList)
  928. {
  929. HRESULT hr = E_FAIL;
  930. if (_hdpaNamespaces)
  931. {
  932. // BUGBUG (lamadio): This does not work if you have 2 enumerators. But,
  933. // when asking for a new enumerator, we should flush the cache.
  934. FreeObjects();
  935. *ppenumIDList = new CEnum(this, grfFlags);
  936. if (NULL == *ppenumIDList)
  937. return E_OUTOFMEMORY ;
  938. hr = S_OK ;
  939. }
  940. return hr;
  941. }
  942. //-------------------------------------------------------------------------//
  943. STDMETHODIMP CAugmentedMergeISF::BindToObject( LPCITEMIDLIST pidlWrap, LPBC pbc, REFIID riid, LPVOID *ppvOut )
  944. {
  945. ASSERT(IS_VALID_PIDL( pidlWrap ) && NULL != ppvOut);
  946. *ppvOut = NULL ;
  947. if (SUCCEEDED(AugMergeISF_IsWrap(pidlWrap)))
  948. {
  949. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap( pidlWrap ) ;
  950. ASSERT(IsValidWrappedPidl(pidlWrap));
  951. ASSERT( pWrap ) ;
  952. ASSERT( pWrap->cSrcs > 0 ) ; // should never, never happen
  953. HANDLE hEnum ;
  954. BOOL bEnum ;
  955. UINT nIDSrc = -1 ;
  956. DEBUG_CODE(int iNumBound = 0);
  957. LPITEMIDLIST pidlSrc ;
  958. HRESULT hr = E_UNEXPECTED ;
  959. CNamespace* pSrc = NULL ;
  960. CAugmentedMergeISF* pISF ;
  961. if (NULL == (pISF = new CAugmentedMergeISF))
  962. return E_OUTOFMEMORY ;
  963. for (hEnum = AugMergeISF_EnumFirstSrcPidl( pidlWrap, &nIDSrc, &pidlSrc ), bEnum = TRUE ;
  964. hEnum && bEnum ;
  965. bEnum = AugMergeISF_EnumNextSrcPidl( hEnum, &nIDSrc, &pidlSrc))
  966. {
  967. if (SUCCEEDED((hr = QueryNameSpace(nIDSrc, (PVOID*)&pSrc))) && pSrc)
  968. {
  969. IShellFolder *psf;
  970. hr = S_FALSE;
  971. if (SUCCEEDED(pSrc->ShellFolder()->BindToObject(pidlSrc, NULL, IID_IShellFolder, (void **)&psf)))
  972. {
  973. LPCITEMIDLIST pidlParent = pSrc->GetPidl();
  974. LPITEMIDLIST pidlFull = ILCombine(pidlParent, pidlSrc);
  975. hr = pISF->AddNameSpace(NULL, psf, pidlFull, pSrc->Attrib());
  976. #ifdef DEBUG
  977. if (SUCCEEDED(hr))
  978. iNumBound++;
  979. #endif
  980. ILFree(pidlFull);
  981. psf->Release();
  982. }
  983. ASSERT(SUCCEEDED(hr));
  984. }
  985. ILFree(pidlSrc);
  986. }
  987. // If this hits, then something is terribly wrong. Either we were unable to bind to the
  988. // ShellFolders, or the add failed. This could be caused by a bad wrapped pidl.
  989. ASSERT(iNumBound > 0);
  990. AugMergeISF_EndEnumSrcPidls( hEnum ) ;
  991. hr = pISF->QueryInterface(riid, ppvOut);
  992. pISF->Release();
  993. return hr ;
  994. }
  995. return E_UNEXPECTED ;
  996. }
  997. //-------------------------------------------------------------------------//
  998. STDMETHODIMP CAugmentedMergeISF::BindToStorage( LPCITEMIDLIST, LPBC, REFIID, void ** )
  999. {
  1000. return E_NOTIMPL ;
  1001. }
  1002. //-------------------------------------------------------------------------//
  1003. STDMETHODIMP CAugmentedMergeISF::CompareIDs(
  1004. LPARAM lParam,
  1005. LPCITEMIDLIST pidl1,
  1006. LPCITEMIDLIST pidl2)
  1007. {
  1008. IShellFolder *psf1 = NULL, *psf2 = NULL;
  1009. LPITEMIDLIST pidlItem1 = NULL, pidlItem2 = NULL;
  1010. int iRet = 0 ;
  1011. HRESULT hr1, hr2, hr ;
  1012. hr1 = GetDefNamespace( pidl1, ASFF_DEFNAMESPACE_DISPLAYNAME, &psf1, &pidlItem1 ) ;
  1013. hr2 = GetDefNamespace( pidl2, ASFF_DEFNAMESPACE_DISPLAYNAME, &psf2, &pidlItem2 ) ;
  1014. if( SUCCEEDED( hr1 ) && SUCCEEDED( hr2 ) )
  1015. {
  1016. ULONG dwAttrib1 = SFGAO_FOLDER, dwAttrib2 = SFGAO_FOLDER;
  1017. // Same namespace? Just forward the request.
  1018. if( psf1 == psf2 )
  1019. {
  1020. hr = psf1->CompareIDs( lParam, pidlItem1, pidlItem2 ) ;
  1021. ILFree( pidlItem1 ) ;
  1022. ILFree( pidlItem2 ) ;
  1023. return hr ;
  1024. }
  1025. hr1 = psf1->GetAttributesOf( 1, (LPCITEMIDLIST*)&pidlItem1, &dwAttrib1 ) ;
  1026. hr2 = psf2->GetAttributesOf( 1, (LPCITEMIDLIST*)&pidlItem2, &dwAttrib2 ) ;
  1027. if( SUCCEEDED( hr1 ) && SUCCEEDED( hr2 ) )
  1028. {
  1029. // Comparison heuristics:
  1030. // (1) folders take precedence over nonfolders, (2) alphanum comparison
  1031. if( 0 != (dwAttrib1 & SFGAO_FOLDER) &&
  1032. 0 == (dwAttrib2 & SFGAO_FOLDER) )
  1033. iRet = -1 ;
  1034. else if( 0 == (dwAttrib1 & SFGAO_FOLDER) &&
  1035. 0 != (dwAttrib2 & SFGAO_FOLDER) )
  1036. iRet = 1 ;
  1037. else
  1038. {
  1039. STRRET strName1, strName2;
  1040. HRESULT hres1 = E_FAIL;
  1041. HRESULT hres2 = E_FAIL;
  1042. TCHAR szName1[MAX_PATH], szName2[MAX_PATH];
  1043. hr1 = psf1->GetDisplayNameOf(pidlItem1, SHGDN_FORPARSING | SHGDN_INFOLDER, &strName1);
  1044. hr2 = psf2->GetDisplayNameOf(pidlItem2, SHGDN_FORPARSING | SHGDN_INFOLDER, &strName2);
  1045. if (SUCCEEDED(hr1) && SUCCEEDED(hr2))
  1046. {
  1047. // must call StrRetToBuf because it frees StrRet strings if allocated
  1048. hres1 = StrRetToBuf(&strName1, pidlItem1, szName1, ARRAYSIZE(szName1));
  1049. hres2 = StrRetToBuf(&strName2, pidlItem2, szName2, ARRAYSIZE(szName2));
  1050. }
  1051. // if the names match we return -1 because they are different pidls with
  1052. // the same name
  1053. if (SUCCEEDED(hr1) && SUCCEEDED(hr2) && SUCCEEDED(hres1) && SUCCEEDED(hres2))
  1054. {
  1055. iRet = lstrcmp(szName1, szName2); // Comparisons are by name with items of the same type.
  1056. }
  1057. }
  1058. }
  1059. }
  1060. hr = FAILED( hr1 ) ? hr1 :
  1061. FAILED( hr2 ) ? hr2 :
  1062. S_OK ;
  1063. if( pidlItem1 )
  1064. ILFree( pidlItem1 ) ;
  1065. if( pidlItem2 )
  1066. ILFree( pidlItem2 ) ;
  1067. return MAKE_HRESULT( HRESULT_SEVERITY( hr ), HRESULT_FACILITY( hr ), iRet ) ;
  1068. }
  1069. //-------------------------------------------------------------------------//
  1070. STDMETHODIMP CAugmentedMergeISF::CreateViewObject(
  1071. HWND hwndOwner,
  1072. REFIID riid,
  1073. LPVOID * ppvOut )
  1074. {
  1075. HRESULT hr ;
  1076. CNamespace *pSrc, *pSrc0 ;
  1077. pSrc = pSrc0 = NULL ;
  1078. // TODO: Handle IDropTarget here, delegate for all others.
  1079. if (IsEqualIID(riid, IID_IDropTarget))
  1080. {
  1081. hr = QueryInterface(riid, ppvOut);
  1082. if (SUCCEEDED(hr))
  1083. _hwnd = hwndOwner;
  1084. return hr;
  1085. }
  1086. // Search for default namespace for CreateViewObj()
  1087. if( FAILED( (hr = GetDefNamespace( ASFF_DEFNAMESPACE_VIEWOBJ, (PVOID*)&pSrc, NULL, (PVOID*)&pSrc0 )) ) )
  1088. return hr ;
  1089. if( NULL == pSrc )
  1090. pSrc = pSrc0 ;
  1091. if( NULL != pSrc )
  1092. {
  1093. ASSERT( pSrc->ShellFolder() ) ;
  1094. hr = pSrc->ShellFolder()->CreateViewObject( hwndOwner, riid, ppvOut ) ;
  1095. }
  1096. return hr ;
  1097. }
  1098. //-------------------------------------------------------------------------//
  1099. STDMETHODIMP CAugmentedMergeISF::GetAttributesOf(
  1100. UINT cidl,
  1101. LPCITEMIDLIST * apidl,
  1102. ULONG * rgfInOut )
  1103. {
  1104. IShellFolder* pISF ;
  1105. LPITEMIDLIST pidlItem ;
  1106. HRESULT hr ;
  1107. if( cidl > 1 ) // support 1 only.
  1108. return E_NOTIMPL ;
  1109. if( !apidl )
  1110. return E_INVALIDARG ;
  1111. // Forward to default namespace for item attributes
  1112. if( FAILED( (hr = GetDefNamespace(
  1113. apidl[0], ASFF_DEFNAMESPACE_ATTRIB, &pISF, &pidlItem )) ) )
  1114. return hr ;
  1115. hr = pISF->GetAttributesOf( 1, (LPCITEMIDLIST*)&pidlItem, rgfInOut ) ;
  1116. ILFree( pidlItem ) ;
  1117. return hr ;
  1118. }
  1119. //-------------------------------------------------------------------------//
  1120. STDMETHODIMP CAugmentedMergeISF::GetUIObjectOf(
  1121. HWND hwndOwner,
  1122. UINT cidl,
  1123. LPCITEMIDLIST * apidl,
  1124. REFIID riid,
  1125. UINT * prgfInOut,
  1126. LPVOID * ppvOut )
  1127. {
  1128. IShellFolder* pISF ;
  1129. LPITEMIDLIST pidlItem ;
  1130. HRESULT hr ;
  1131. if (cidl > 1) // support 1 only.
  1132. return E_NOTIMPL ;
  1133. if (!apidl)
  1134. return E_INVALIDARG ;
  1135. if (IsEqualGUID(riid, IID_IContextMenu))
  1136. {
  1137. hr = _GetContextMenu(hwndOwner, cidl, apidl, prgfInOut, ppvOut);
  1138. }
  1139. else
  1140. {
  1141. // Forward to default namespace for UI object
  1142. if (FAILED((hr = GetDefNamespace(apidl[0], ASFF_DEFNAMESPACE_UIOBJ, &pISF, &pidlItem))))
  1143. return hr ;
  1144. hr = pISF->GetUIObjectOf(hwndOwner, 1, (LPCITEMIDLIST*)&pidlItem, riid, prgfInOut, ppvOut);
  1145. ILFree(pidlItem);
  1146. }
  1147. return hr ;
  1148. }
  1149. //-------------------------------------------------------------------------//
  1150. STDMETHODIMP CAugmentedMergeISF::GetDisplayNameOf(
  1151. LPCITEMIDLIST pidl,
  1152. DWORD grfFlags,
  1153. LPSTRRET pstrName )
  1154. {
  1155. IShellFolder* pISF ;
  1156. LPITEMIDLIST pidlItem ;
  1157. HRESULT hr ;
  1158. // Forward to default namespace for display name
  1159. if (FAILED((hr = GetDefNamespace(
  1160. pidl, ASFF_DEFNAMESPACE_DISPLAYNAME, &pISF, &pidlItem))))
  1161. return hr ;
  1162. if (SUCCEEDED((hr = pISF->GetDisplayNameOf(pidlItem, grfFlags, pstrName))))
  1163. {
  1164. // STRRET_OFFSET has no meaning in context of the pidl wrapper.
  1165. // We can either calculate the offset into the wrapper, or allocate
  1166. // a wide char for the name. For expedience, we'll allocate the name.
  1167. if (pstrName->uType == STRRET_OFFSET)
  1168. {
  1169. UINT cch = lstrlenA( STRRET_OFFPTR( pidlItem, pstrName ) ) ;
  1170. LPWSTR pwszName = (LPWSTR)SHAlloc( (cch + 1) * sizeof(WCHAR));
  1171. if (NULL != pwszName)
  1172. {
  1173. SHAnsiToUnicode( STRRET_OFFPTR( pidlItem, pstrName ), pwszName, cch+1 );
  1174. pwszName[cch] = (WCHAR)0 ;
  1175. }
  1176. pstrName->pOleStr = pwszName ;
  1177. pstrName->uType = STRRET_WSTR ;
  1178. }
  1179. #ifdef DEBUG
  1180. // If the trace flags are set, and this is not comming from an internal query,
  1181. // Then append the location where this name came from
  1182. if (g_qwTraceFlags & TF_AUGM && _fInternalGDNO == FALSE)
  1183. {
  1184. if (pstrName->uType == STRRET_WSTR)
  1185. {
  1186. LPWSTR wszOldName = pstrName->pOleStr;
  1187. UINT cch = lstrlenW(wszOldName);
  1188. pstrName->pOleStr = (LPWSTR)SHAlloc( (cch + 50) * sizeof(WCHAR));
  1189. if (pstrName->pOleStr)
  1190. {
  1191. StrCpyW(pstrName->pOleStr, wszOldName);
  1192. if (AugMergeISF_GetSourceCount(pidl) > 1)
  1193. StrCatW(pstrName->pOleStr, L" (Merged)");
  1194. else if (WrappedPidlContainsSrcID(pidl, 0))
  1195. StrCatW(pstrName->pOleStr, L" (1)");
  1196. else
  1197. StrCatW(pstrName->pOleStr, L" (2)");
  1198. SHFree(wszOldName);
  1199. }
  1200. else
  1201. {
  1202. pstrName->pOleStr = wszOldName;
  1203. }
  1204. }
  1205. else if (pstrName->uType == STRRET_CSTR)
  1206. {
  1207. if (AugMergeISF_GetSourceCount(pidl) > 1)
  1208. StrCatA(pstrName->cStr, " (Merged)");
  1209. else if (WrappedPidlContainsSrcID(pidl, 0))
  1210. StrCatA(pstrName->cStr, " (1)");
  1211. else
  1212. StrCatA(pstrName->cStr, " (2)");
  1213. }
  1214. }
  1215. #endif
  1216. }
  1217. ILFree( pidlItem ) ;
  1218. return hr ;
  1219. }
  1220. //-------------------------------------------------------------------------//
  1221. STDMETHODIMP CAugmentedMergeISF::ParseDisplayName(
  1222. HWND hwndOwner,
  1223. LPBC pbcReserved,
  1224. LPOLESTR pwszName,
  1225. ULONG * pchEaten,
  1226. LPITEMIDLIST * ppidl,
  1227. ULONG * pdwAttrib )
  1228. {
  1229. int iIndex;
  1230. LPITEMIDLIST pidl;
  1231. *ppidl = NULL;
  1232. // This ParseDisplayName should iterate through all our delegates until one works.
  1233. for (iIndex = NamespaceCount() - 1; iIndex >=0 ; iIndex--)
  1234. {
  1235. CNamespace* pSrc = Namespace(iIndex) ;
  1236. if (pSrc)
  1237. {
  1238. if (SUCCEEDED(pSrc->ShellFolder()->ParseDisplayName(hwndOwner, pbcReserved, pwszName, pchEaten,
  1239. &pidl, pdwAttrib)))
  1240. {
  1241. ASSERT(pidl); // Make sure a valid pidl comes out.
  1242. if (*ppidl == NULL)
  1243. AugMergeISF_CreateWrap(pidl, iIndex, ppidl);
  1244. else
  1245. AugMergeISF_WrapAddPidl(pidl, iIndex, ppidl);
  1246. ILFree(pidl);
  1247. }
  1248. }
  1249. }
  1250. return *ppidl? S_OK : E_FAIL;
  1251. }
  1252. //-------------------------------------------------------------------------//
  1253. STDMETHODIMP CAugmentedMergeISF::SetNameOf(
  1254. HWND hwndOwner,
  1255. LPCITEMIDLIST pidl,
  1256. LPCOLESTR pwszName,
  1257. DWORD uFlags,
  1258. LPITEMIDLIST *ppidlOut )
  1259. {
  1260. CNamespace* pnsCommon;
  1261. CNamespace* pnsUser;
  1262. LPITEMIDLIST pidlItem;
  1263. HRESULT hres;
  1264. UINT uiUser;
  1265. UINT uiCommon;
  1266. hres = _GetNamespaces(pidl, &pnsCommon, &uiCommon, &pnsUser, &uiUser, &pidlItem, NULL);
  1267. if (SUCCEEDED(hres))
  1268. {
  1269. LPITEMIDLIST pidlNew = NULL;
  1270. UINT uiNamespace = INVALID_NAMESPACE_INDEX;
  1271. if (pnsUser)
  1272. {
  1273. hres = pnsUser->ShellFolder()->SetNameOf(hwndOwner, pidlItem, pwszName, uFlags, &pidlNew);
  1274. uiNamespace = uiUser;
  1275. }
  1276. else if (pnsCommon)
  1277. {
  1278. hres = E_FAIL;
  1279. if (AffectAllUsers(hwndOwner))
  1280. {
  1281. hres = pnsCommon->ShellFolder()->SetNameOf(hwndOwner, pidlItem, pwszName, uFlags, &pidlNew);
  1282. uiNamespace = uiCommon;
  1283. }
  1284. }
  1285. if (ppidlOut)
  1286. {
  1287. *ppidlOut = NULL;
  1288. // wrap the pidl
  1289. if (SUCCEEDED(hres) && pidlNew)
  1290. AugMergeISF_CreateWrap(pidlNew, uiNamespace, ppidlOut);
  1291. }
  1292. ILFree(pidlNew);
  1293. ILFree(pidlItem);
  1294. }
  1295. return hres;
  1296. }
  1297. //-------------------------------------------------------------------------//
  1298. // IAugmentedShellFolder methods
  1299. //-------------------------------------------------------------------------//
  1300. //-------------------------------------------------------------------------//
  1301. // Adds a source namespace to the Augmented Merge shell folder object.
  1302. STDMETHODIMP CAugmentedMergeISF::AddNameSpace(
  1303. const GUID * pguidObject,
  1304. IShellFolder * psf,
  1305. LPCITEMIDLIST pidl,
  1306. DWORD dwFlags )
  1307. {
  1308. ASSERT (IS_VALID_CODE_PTR(psf, IShellFolder*));
  1309. ASSERT (IS_VALID_PIDL(pidl));
  1310. // Check for duplicate via full display name
  1311. for( int i=0, max = NamespaceCount() ; i < max; i++ )
  1312. {
  1313. CNamespace* pSrc = Namespace( i ) ;
  1314. if (pSrc)
  1315. {
  1316. if (ILIsEqual(pSrc->GetPidl(), pidl))
  1317. {
  1318. // Found! Reassign attributes
  1319. pSrc->Assign( pguidObject, psf, pidl, dwFlags ) ;
  1320. return S_OK ;
  1321. }
  1322. }
  1323. }
  1324. // No match; safe to append it to collection, creating DPA if necessary.
  1325. if( NULL == _hdpaNamespaces &&
  1326. NULL == (_hdpaNamespaces= DPA_Create( 2 )) )
  1327. return E_OUTOFMEMORY ;
  1328. CNamespace *pSrc = new CNamespace( pguidObject, psf, pidl, dwFlags );
  1329. if( NULL == pSrc )
  1330. return E_OUTOFMEMORY ;
  1331. return DPA_AppendPtr( _hdpaNamespaces, pSrc ) >= 0 ? S_OK : E_FAIL;
  1332. }
  1333. //-------------------------------------------------------------------------//
  1334. // Retrieves the primary namespace iid for the wrapped pidl.
  1335. STDMETHODIMP CAugmentedMergeISF::GetNameSpaceID(
  1336. LPCITEMIDLIST pidl,
  1337. GUID * pguidOut )
  1338. {
  1339. HRESULT hr ;
  1340. if (FAILED((hr = AugMergeISF_IsWrap( pidl ))))
  1341. return hr ;
  1342. // BUGBUG: need to enumerate wrapped source pidls
  1343. return E_NOTIMPL ;
  1344. }
  1345. //-------------------------------------------------------------------------//
  1346. // Retrieves a pointer to a source namespace descriptor associated with
  1347. // the specified lookup index.
  1348. STDMETHODIMP CAugmentedMergeISF::QueryNameSpace( ULONG nID, PVOID* ppSrc )
  1349. {
  1350. if (!ppSrc)
  1351. return E_INVALIDARG;
  1352. *ppSrc = NULL;
  1353. LONG cSrcs;
  1354. if ((cSrcs = NamespaceCount()) <=0)
  1355. return E_FAIL;
  1356. if(nID >= (ULONG)cSrcs)
  1357. return E_INVALIDARG;
  1358. if (NULL == (*ppSrc = Namespace(nID)))
  1359. return E_UNEXPECTED;
  1360. return S_OK;
  1361. }
  1362. //-------------------------------------------------------------------------//
  1363. // Retrieves data for the namespace identified by dwID.
  1364. STDMETHODIMP CAugmentedMergeISF::QueryNameSpace(
  1365. ULONG nID,
  1366. GUID * pguidOut,
  1367. IShellFolder ** ppsf )
  1368. {
  1369. CNamespace* pSrc = NULL ;
  1370. HRESULT hr = QueryNameSpace( nID, (PVOID*)&pSrc ) ;
  1371. if( pguidOut )
  1372. *pguidOut = NULL != pSrc ? pSrc->Guid() : GUID_NULL ;
  1373. if( ppsf )
  1374. {
  1375. if( (*ppsf = (NULL != pSrc) ? pSrc->ShellFolder() : NULL) != NULL )
  1376. (*ppsf)->AddRef() ;
  1377. }
  1378. return hr ;
  1379. }
  1380. //-------------------------------------------------------------------------//
  1381. STDMETHODIMP CAugmentedMergeISF::EnumNameSpace(
  1382. DWORD uNameSpace,
  1383. DWORD * pdwID )
  1384. {
  1385. return E_NOTIMPL ;
  1386. }
  1387. //-------------------------------------------------------------------------//
  1388. // IAugmentedShellFolder2 methods
  1389. //-------------------------------------------------------------------------//
  1390. //GetNameSpaceCount and GetIDListWrapCount are not used anywhere
  1391. #if 0
  1392. //-------------------------------------------------------------------------//
  1393. STDMETHODIMP CAugmentedMergeISF::GetNameSpaceCount( OUT LONG* pcNamespaces )
  1394. {
  1395. if( !pcNamespaces )
  1396. return E_INVALIDARG ;
  1397. *pcNamespaces = (LONG)NamespaceCount() ;
  1398. return S_OK ;
  1399. }
  1400. //-------------------------------------------------------------------------//
  1401. STDMETHODIMP CAugmentedMergeISF::GetIDListWrapCount(
  1402. LPCITEMIDLIST pidlWrap,
  1403. OUT LONG * pcPidls)
  1404. {
  1405. if( NULL == pidlWrap || NULL == pcPidls )
  1406. return E_INVALIDARG ;
  1407. *pcPidls = 0 ;
  1408. HRESULT hr ;
  1409. if (SUCCEEDED((hr = AugMergeISF_IsWrap(pidlWrap))))
  1410. {
  1411. PAUGM_IDWRAP pWrap = AugMergeISF_GetWrap(pidlWrap);
  1412. *pcPidls = pWrap->cSrcs;
  1413. hr = S_OK;
  1414. }
  1415. return hr;
  1416. }
  1417. #endif // #if 0
  1418. //-------------------------------------------------------------------------//
  1419. STDMETHODIMP CAugmentedMergeISF::UnWrapIDList(
  1420. LPCITEMIDLIST pidlWrap,
  1421. LONG cPidls,
  1422. IShellFolder** apsf,
  1423. LPITEMIDLIST* apidlFolder,
  1424. LPITEMIDLIST* apidlItems,
  1425. LONG* pcFetched )
  1426. {
  1427. HRESULT hr ;
  1428. HANDLE hEnum ;
  1429. BOOL bEnum = TRUE ;
  1430. UINT nSrcID ;
  1431. LPITEMIDLIST pidlItem ;
  1432. LONG cFetched = 0;
  1433. if (NULL == pidlWrap || cPidls <= 0)
  1434. return E_INVALIDARG ;
  1435. if (FAILED((hr = AugMergeISF_IsWrap(pidlWrap))))
  1436. return hr ;
  1437. // Enumerate pidls in wrap
  1438. for (hEnum = AugMergeISF_EnumFirstSrcPidl( pidlWrap, &nSrcID, &pidlItem);
  1439. cFetched < cPidls && hEnum && bEnum ;
  1440. bEnum = AugMergeISF_EnumNextSrcPidl( hEnum, &nSrcID, &pidlItem))
  1441. {
  1442. // Retrieve namespace data
  1443. CNamespace* pSrc ;
  1444. if (SUCCEEDED((hr = QueryNameSpace(nSrcID, (PVOID*)&pSrc))))
  1445. {
  1446. if (apsf)
  1447. {
  1448. apsf[cFetched] = pSrc->ShellFolder() ;
  1449. if (apsf[cFetched])
  1450. apsf[cFetched]->AddRef();
  1451. }
  1452. if (apidlFolder)
  1453. apidlFolder[cFetched] = ILClone(pSrc->GetPidl());
  1454. if (apidlItems)
  1455. {
  1456. apidlItems[cFetched] = pidlItem;
  1457. pidlItem = NULL; // paranoia -- just making sure we, somehow, don't free this guy at the end of the for loop
  1458. }
  1459. cFetched++ ;
  1460. }
  1461. else
  1462. {
  1463. ILFree( pidlItem ) ;
  1464. }
  1465. }
  1466. ILFree(pidlItem); // AugMergeISF_EnumNextSrcPidl is called (if there are 2 wrapped pidls and we ask for only one)
  1467. // right before we exit the for loop so we have to free pidl if allocated.
  1468. if (hEnum)
  1469. AugMergeISF_EndEnumSrcPidls( hEnum );
  1470. if( pcFetched )
  1471. *pcFetched = cFetched ;
  1472. return cFetched == cPidls ? S_OK : S_FALSE ;
  1473. }
  1474. //-------------------------------------------------------------------------//
  1475. STDMETHODIMP CAugmentedMergeISF::SetOwner( IUnknown* punkOwner )
  1476. {
  1477. HRESULT hr = S_OK ;
  1478. int cSrcs = NamespaceCount() ;
  1479. if( cSrcs > 0 )
  1480. DPA_EnumCallback( _hdpaNamespaces, SetOwnerProc, NULL ) ;
  1481. ATOMICRELEASE( _punkOwner ) ;
  1482. if( punkOwner )
  1483. {
  1484. hr = punkOwner->QueryInterface(IID_IUnknown, (LPVOID *)&_punkOwner ) ;
  1485. if( cSrcs )
  1486. DPA_EnumCallback( _hdpaNamespaces, SetOwnerProc, (void *)_punkOwner);
  1487. }
  1488. return hr ;
  1489. }
  1490. //-------------------------------------------------------------------------//
  1491. int CAugmentedMergeISF::SetOwnerProc( LPVOID pv, LPVOID pvParam )
  1492. {
  1493. CNamespace* pSrc = (CNamespace*) pv ;
  1494. ASSERT( pSrc ) ;
  1495. return pSrc->SetOwner( (IUnknown*)pvParam ) ;
  1496. }
  1497. //-------------------------------------------------------------------------//
  1498. // ITranslateShellChangeNotify methods
  1499. //-------------------------------------------------------------------------//
  1500. LPITEMIDLIST ILCombineBase(LPCITEMIDLIST pidlContainingBase, LPCITEMIDLIST pidlRel)
  1501. {
  1502. // This routine differs from ILCombine in that it takes the First pidl's base, and
  1503. // cats on the last id of the second pidl. We need this so Wrapped pidls
  1504. // end up with the same base, and we get a valid full pidl.
  1505. LPITEMIDLIST pidlRet = NULL;
  1506. LPITEMIDLIST pidlBase = ILClone(pidlContainingBase);
  1507. if (pidlBase)
  1508. {
  1509. ILRemoveLastID(pidlBase);
  1510. pidlRet = ILCombine(pidlBase, pidlRel);
  1511. ILFree(pidlBase);
  1512. }
  1513. return pidlRet;
  1514. }
  1515. BOOL IsFolderEvent(LONG lEvent)
  1516. {
  1517. return lEvent == SHCNE_MKDIR || lEvent == SHCNE_RMDIR || lEvent == SHCNE_RENAMEFOLDER;
  1518. }
  1519. #ifdef DEBUG
  1520. void CAugmentedMergeISF::DumpObjects()
  1521. {
  1522. if (g_dwDumpFlags & TF_AUGM)
  1523. {
  1524. ASSERT(_hdpaObjects);
  1525. int iObjectCount = DPA_GetPtrCount(_hdpaObjects);
  1526. TraceMsg(TF_AUGM, "CAugMISF::DumpObjects: Number of items: %d", iObjectCount);
  1527. CNamespace* pns = (CNamespace *)DPA_FastGetPtr(_hdpaNamespaces, 0);
  1528. if (pns)
  1529. DebugDumpPidl(TF_AUGM, TEXT("CAugMISF::DumpObjects Namespace 1"), pns->GetPidl());
  1530. pns = (CNamespace *)DPA_FastGetPtr(_hdpaNamespaces, 1);
  1531. if (pns)
  1532. DebugDumpPidl(TF_AUGM, TEXT("CAugMISF::DumpObjects Namespace 2"), pns->GetPidl());
  1533. for (int i = 0; i < iObjectCount; i++)
  1534. {
  1535. CAugISFEnumItem* pEnumItem = (CAugISFEnumItem*)DPA_FastGetPtr(_hdpaObjects, i);
  1536. TraceMsg(TF_ALWAYS, "CAugMISF::DumpObjects: %s, Folder: %s Merged: %s",
  1537. pEnumItem->_pszDisplayName,
  1538. BOOLIFY(pEnumItem->_rgfAttrib & SFGAO_FOLDER) ? TEXT("Yes") : TEXT("No"),
  1539. (AugMergeISF_GetSourceCount(pEnumItem->_pidlWrap) > 1)? TEXT("Yes") : TEXT("No"));
  1540. }
  1541. }
  1542. }
  1543. #endif
  1544. BOOL GetRealPidlFromSimple(LPCITEMIDLIST pidlSimple, LPITEMIDLIST* ppidlReal)
  1545. {
  1546. // Similar to SHGetRealIDL in Function, but SHGetRealIDL does SHGDN_FORPARSING | INFOLDER.
  1547. // I need the parsing name. I can't rev SHGetRealIDL very easily, so here's this one!
  1548. TCHAR szFullName[MAX_PATH];
  1549. if (SUCCEEDED(SHGetNameAndFlags(pidlSimple, SHGDN_FORPARSING, szFullName, SIZECHARS(szFullName), NULL)))
  1550. {
  1551. *ppidlReal = ILCreateFromPath(szFullName);
  1552. }
  1553. if (*ppidlReal == NULL) // Unable to create? Then use the simple pidl. This is because it does not exist any more
  1554. { // For say, a Delete Notify
  1555. *ppidlReal = ILClone(pidlSimple);
  1556. }
  1557. return *ppidlReal != NULL;
  1558. }
  1559. //-------------------------------------------------------------------------------------------------//
  1560. STDMETHODIMP CAugmentedMergeISF::TranslateIDs(
  1561. LONG *plEvent,
  1562. LPCITEMIDLIST pidl1,
  1563. LPCITEMIDLIST pidl2,
  1564. LPITEMIDLIST * ppidlOut1,
  1565. LPITEMIDLIST * ppidlOut2,
  1566. LONG *plEvent2, LPITEMIDLIST *ppidlOut1Event2,
  1567. LPITEMIDLIST *ppidlOut2Event2)
  1568. {
  1569. HRESULT hres = E_FAIL;
  1570. switch (*plEvent)
  1571. {
  1572. case SHCNE_EXTENDED_EVENT:
  1573. case SHCNE_ASSOCCHANGED:
  1574. case SHCNE_UPDATEIMAGE:
  1575. return S_OK;
  1576. case SHCNE_UPDATEDIR:
  1577. FreeObjects();
  1578. return S_OK;
  1579. }
  1580. ASSERT(ppidlOut1);
  1581. ASSERT(ppidlOut2);
  1582. LONG lEvent = *plEvent;
  1583. *plEvent2 = (LONG)-1;
  1584. *ppidlOut1Event2 = NULL;
  1585. *ppidlOut2Event2 = NULL;
  1586. *ppidlOut1 = (LPITEMIDLIST)pidl1;
  1587. *ppidlOut2 = (LPITEMIDLIST)pidl2;
  1588. if (!plEvent)
  1589. return E_FAIL;
  1590. // If they are already wrapped, don't wrap twice.
  1591. if ((pidl1 && SUCCEEDED(AugMergeISF_IsWrap(ILFindLastID(pidl1)))) ||
  1592. (pidl2 && SUCCEEDED(AugMergeISF_IsWrap(ILFindLastID(pidl2)))))
  1593. {
  1594. // We don't want to wrap twice.
  1595. return E_FAIL;
  1596. }
  1597. if (!_hdpaNamespaces)
  1598. return E_FAIL;
  1599. if (!_hdpaObjects)
  1600. return E_FAIL;
  1601. CAugISFEnumItem* pEnumItem;
  1602. int iIndex;
  1603. int iShellFolder1 = -1;
  1604. int iShellFolder2 = -1;
  1605. IShellFolder* psf1 = NULL;
  1606. IShellFolder* psf2 = NULL;
  1607. LPITEMIDLIST pidlReal1 = NULL;
  1608. LPITEMIDLIST pidlReal2 = NULL;
  1609. LPITEMIDLIST pidlRealRel1 = NULL;
  1610. LPITEMIDLIST pidlRealRel2 = NULL;
  1611. BOOL fFolder = IsFolderEvent(*plEvent);
  1612. // Get the information about these Simple pidls: Are they our Children? If so, what namespace?
  1613. BOOL fChild1 = IsChildIDInternal(pidl1, TRUE, &iShellFolder1);
  1614. BOOL fChild2 = IsChildIDInternal(pidl2, TRUE, &iShellFolder2);
  1615. // Is either a child?
  1616. if (!(fChild1 || fChild2))
  1617. return hres;
  1618. // Ok, pidl1 is a child, can we get the Real pidl from the simple one?
  1619. if (pidl1 && !GetRealPidlFromSimple(pidl1, &pidlReal1))
  1620. goto Cleanup;
  1621. // Ok, pidl2 is a child, can we get the Real pidl from the simple one?
  1622. if (pidl2 && !GetRealPidlFromSimple(pidl2, &pidlReal2))
  1623. goto Cleanup;
  1624. // These are for code clarity later on. We deal with Relative pidls from here until the very end,
  1625. // when we combine the base of the in pidls with the outgoing wrapped pidls.
  1626. if (pidlReal1)
  1627. pidlRealRel1 = ILFindLastID(pidlReal1);
  1628. if (pidlReal2)
  1629. pidlRealRel2 = ILFindLastID(pidlReal2);
  1630. // Is Pidl1 in our namespaces?
  1631. if (iShellFolder1 != -1)
  1632. {
  1633. // Yes, lets get the non-refcounted shell folder that know's about this pidl.
  1634. CNamespace * pns = (CNamespace *)DPA_GetPtr(_hdpaNamespaces, iShellFolder1);
  1635. psf1 = pns->ShellFolder(); // Non ref counted.
  1636. }
  1637. // Is Pidl2 in our namespaces?
  1638. if (iShellFolder2 != -1)
  1639. {
  1640. // Yes, lets get the non-refcounted shell folder that know's about this pidl.
  1641. CNamespace * pns = (CNamespace *)DPA_GetPtr(_hdpaNamespaces, iShellFolder2);
  1642. psf2 = pns->ShellFolder(); // Non ref counted.
  1643. }
  1644. hres = S_OK;
  1645. DEBUG_CODE(_fInternalGDNO = TRUE);
  1646. switch(*plEvent)
  1647. {
  1648. case 0: // Just look up the pidls and return.
  1649. {
  1650. DWORD rgfAttrib = SFGAO_FOLDER;
  1651. if (iShellFolder1 != -1)
  1652. {
  1653. psf1->GetAttributesOf(1, (LPCITEMIDLIST*)&pidlRealRel1, &rgfAttrib);
  1654. if (S_OK == _SearchForPidl(psf1, pidlRealRel1, BOOLIFY(rgfAttrib & SFGAO_FOLDER), &iIndex, &pEnumItem))
  1655. {
  1656. *ppidlOut1 = ILCombineBase(pidlReal1, pEnumItem->_pidlWrap);
  1657. if (!*ppidlOut1)
  1658. hres = E_OUTOFMEMORY;
  1659. }
  1660. }
  1661. rgfAttrib = SFGAO_FOLDER;
  1662. if (iShellFolder2 != -1 && SUCCEEDED(hres))
  1663. {
  1664. psf2->GetAttributesOf(1, (LPCITEMIDLIST*)&pidlRealRel2, &rgfAttrib);
  1665. if (S_OK == _SearchForPidl(psf2, pidlRealRel2, BOOLIFY(rgfAttrib & SFGAO_FOLDER), &iIndex, &pEnumItem))
  1666. {
  1667. *ppidlOut2 = ILCombineBase(pidlReal2, pEnumItem->_pidlWrap);
  1668. if (!*ppidlOut2)
  1669. hres = E_OUTOFMEMORY;
  1670. }
  1671. }
  1672. }
  1673. break;
  1674. case SHCNE_CREATE:
  1675. case SHCNE_MKDIR:
  1676. {
  1677. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s", fFolder?
  1678. TEXT("SHCNE_MKDIR") : TEXT("SHCNE_CREATE"));
  1679. // Is there a thing of this name already?
  1680. if (S_OK == _SearchForPidl(psf1, pidlRealRel1, fFolder, &iIndex, &pEnumItem))
  1681. {
  1682. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s needs to be merged. Converting to Rename", pEnumItem->_pszDisplayName);
  1683. // Yes; Then we need to merge this new pidl into the wrapped pidl, and change this
  1684. // to a rename, passing the Old wrapped pidl as the first arg, and the new wrapped pidl
  1685. // as the second arg. I have to be careful about the freeing:
  1686. // Free *ppidlOut1
  1687. // Clone pEnumItem->_pidlWrap -> *ppidlOut1.
  1688. // Add pidl1 to pEnumItem->_pidlWrap.
  1689. // Clone new pEnumItem->_pidlWrap -> *ppidlOut2. ASSERT(*ppidlOut2 == NULL)
  1690. *ppidlOut1 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1691. if (*ppidlOut1)
  1692. {
  1693. AugMergeISF_WrapAddPidl(pidlRealRel1, iShellFolder1, &pEnumItem->_pidlWrap);
  1694. *ppidlOut2 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1695. if (!*ppidlOut2)
  1696. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl2");
  1697. *plEvent = fFolder? SHCNE_RENAMEFOLDER : SHCNE_RENAMEITEM;
  1698. }
  1699. else
  1700. {
  1701. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl1");
  1702. }
  1703. }
  1704. else
  1705. {
  1706. LPITEMIDLIST pidlWrap;
  1707. CAugISFEnumItem* paugmEnum = new CAugISFEnumItem;
  1708. if (paugmEnum)
  1709. {
  1710. if (SUCCEEDED(AugMergeISF_CreateWrap(pidlRealRel1, (UINT)iShellFolder1, &pidlWrap)) &&
  1711. paugmEnum->InitWithWrappedToOwn(SAFECAST(this, IAugmentedShellFolder2*),
  1712. iShellFolder1, pidlWrap))
  1713. {
  1714. AUGMISFSEARCHFORPIDL AugMSearch;
  1715. AugMSearch.pszDisplayName = paugmEnum->_pszDisplayName;
  1716. AugMSearch.fFolder = fFolder;
  1717. int iInsertIndex = DPA_Search(_hdpaObjects, (LPVOID)&AugMSearch, 0,
  1718. AugMISFSearchForOnePidlByDisplayName, NULL, DPAS_SORTED | DPAS_INSERTAFTER);
  1719. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Creating new unmerged %s at %d",
  1720. paugmEnum->_pszDisplayName, iInsertIndex);
  1721. if (iInsertIndex < 0)
  1722. iInsertIndex = DA_LAST;
  1723. if (DPA_InsertPtr(_hdpaObjects, iInsertIndex, paugmEnum) == -1)
  1724. {
  1725. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Was unable to add %s for some reason. Bailing",
  1726. paugmEnum->_pszDisplayName);
  1727. DestroyObjectsProc(paugmEnum, NULL);
  1728. }
  1729. else
  1730. {
  1731. *ppidlOut1 = ILCombineBase(pidl1, paugmEnum->_pidlWrap);
  1732. }
  1733. }
  1734. else
  1735. DestroyObjectsProc(paugmEnum, NULL);
  1736. }
  1737. }
  1738. }
  1739. break;
  1740. case SHCNE_DELETE:
  1741. case SHCNE_RMDIR:
  1742. {
  1743. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s", fFolder?
  1744. TEXT("SHCNE_RMDIR") : TEXT("SHCNE_DELETE"));
  1745. int iDeleteIndex;
  1746. // Is there a folder of this name already?
  1747. if (S_OK == _SearchForPidl(psf1, pidlRealRel1,
  1748. fFolder, &iDeleteIndex, &pEnumItem))
  1749. {
  1750. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Found %s checking merge state.", pEnumItem->_pszDisplayName);
  1751. // Yes; Then we need to unmerge this pidl from the wrapped pidl, and change this
  1752. // to a rename, passing the Old wrapped pidl as the first arg, and the new wrapped pidl
  1753. // as the second arg. I have to be careful about the freeing:
  1754. // Free *ppidlOut1
  1755. // Clone pEnumItem->_pidlWrap -> *ppidlOut1.
  1756. // Remove pidl1 from pEnumItem->_pidlWrap
  1757. // Convert to rename, pass new wrapped as second arg.
  1758. if (AugMergeISF_GetSourceCount( pEnumItem->_pidlWrap ) > 1 )
  1759. {
  1760. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is Merged. Removing pidl, convert to rename", pEnumItem->_pszDisplayName);
  1761. *ppidlOut1 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1762. if (*ppidlOut1)
  1763. {
  1764. EVAL(SUCCEEDED(AugMergeISF_WrapRemovePidl(pEnumItem->_pidlWrap,
  1765. iShellFolder1, &pEnumItem->_pidlWrap)));
  1766. *ppidlOut2 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1767. if (!*ppidlOut2)
  1768. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl2");
  1769. *plEvent = fFolder? SHCNE_RENAMEFOLDER : SHCNE_RENAMEITEM;
  1770. }
  1771. else
  1772. {
  1773. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl1");
  1774. }
  1775. }
  1776. else
  1777. {
  1778. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is not Merged. deleteing", pEnumItem->_pszDisplayName);
  1779. pEnumItem = (CAugISFEnumItem*)DPA_DeletePtr(_hdpaObjects, iDeleteIndex);
  1780. if (EVAL(pEnumItem))
  1781. {
  1782. *ppidlOut1 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1783. DestroyObjectsProc(pEnumItem, NULL);
  1784. }
  1785. else
  1786. {
  1787. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to get %d from DPA", iDeleteIndex);
  1788. }
  1789. }
  1790. }
  1791. }
  1792. break;
  1793. case SHCNE_RENAMEITEM:
  1794. case SHCNE_RENAMEFOLDER:
  1795. {
  1796. // BUGBUG (lamadio): When renaming an item in the menu, this code will split it into
  1797. // a Delete and a Create. We need to detect this situation and convert it to 1 rename. This
  1798. // will solve the problem of the lost order during a rename....
  1799. BOOL fEvent1Set = FALSE;
  1800. BOOL fFirstPidlInNamespace = FALSE;
  1801. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s", fFolder?
  1802. TEXT("SHCNE_RENAMEFOLDER") : TEXT("SHCNE_RENAMEITEM"));
  1803. // Is this item being renamed FROM the Folder?
  1804. if (iShellFolder1 != -1 && // Is this pidl a child of the Folder?
  1805. S_OK == _SearchForPidl(psf1, pidlRealRel1,
  1806. fFolder, &iIndex, &pEnumItem)) // Is it found?
  1807. {
  1808. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Old pidl %s is in the Folder", pEnumItem->_pszDisplayName);
  1809. // Yes.
  1810. // Then we need to see if the item that it's being renamed from was Merged
  1811. // Need this for reentrancy
  1812. if (WrappedPidlContainsSrcID(pEnumItem->_pidlWrap, iShellFolder1))
  1813. {
  1814. // Was it merged?
  1815. if (AugMergeISF_GetSourceCount(pEnumItem->_pidlWrap) > 1) // Case 3)
  1816. {
  1817. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is Merged. Removing pidl. Convert to rename for event 1",
  1818. pEnumItem->_pszDisplayName);
  1819. // Yes;
  1820. // Then we need to unmerge that item.
  1821. *ppidlOut1 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1822. if (*ppidlOut1)
  1823. {
  1824. // UnWrap
  1825. AugMergeISF_WrapRemovePidl(pEnumItem->_pidlWrap, iShellFolder1, &pEnumItem->_pidlWrap);
  1826. *ppidlOut2 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1827. if (!*ppidlOut2)
  1828. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl2");
  1829. // This We need to "Rename" the old wrapped pidl, to this new one
  1830. // that does not contain the old item.
  1831. fEvent1Set = TRUE;
  1832. }
  1833. else
  1834. {
  1835. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to create new pidl1");
  1836. }
  1837. }
  1838. else
  1839. {
  1840. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is not merged. Nuking item Convert to Delete for event 1.",
  1841. pEnumItem->_pszDisplayName);
  1842. // No, This was not a wrapped pidl. Then, convert to a delete:
  1843. pEnumItem = (CAugISFEnumItem*)DPA_DeletePtr(_hdpaObjects, iIndex);
  1844. if (EVAL(pEnumItem))
  1845. {
  1846. // If we're renaming from this folder, into this folder, Then the first event stays a rename.
  1847. if (iShellFolder2 == -1)
  1848. {
  1849. fEvent1Set = TRUE;
  1850. *plEvent = fFolder? SHCNE_RMDIR : SHCNE_DELETE;
  1851. }
  1852. else
  1853. {
  1854. fFirstPidlInNamespace = TRUE;
  1855. }
  1856. *ppidlOut1 = ILCombineBase(pidl1, pEnumItem->_pidlWrap);
  1857. DestroyObjectsProc(pEnumItem, NULL);
  1858. }
  1859. else
  1860. {
  1861. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Failure. Was unable to find Item at %d", iIndex);
  1862. }
  1863. }
  1864. }
  1865. else
  1866. {
  1867. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Skipping this because we already processed it."
  1868. "Dragging To Desktop?");
  1869. hres = E_FAIL;
  1870. }
  1871. }
  1872. // Is this item is being rename INTO the Start Menu?
  1873. if (iShellFolder2 != -1)
  1874. {
  1875. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: New pidl is in the Folder");
  1876. LPITEMIDLIST* ppidlNewWrapped1 = ppidlOut1;
  1877. LPITEMIDLIST* ppidlNewWrapped2 = ppidlOut2;
  1878. LONG* plNewEvent = plEvent;
  1879. if (fEvent1Set)
  1880. {
  1881. plNewEvent = plEvent2;
  1882. ppidlNewWrapped1 = ppidlOut1Event2;
  1883. ppidlNewWrapped2 = ppidlOut2Event2;
  1884. }
  1885. if (S_OK == _SearchForPidl(psf2, pidlRealRel2,
  1886. fFolder, &iIndex, &pEnumItem))
  1887. {
  1888. // If we're renaming from this folder, into this folder, Check to see if the destination has a
  1889. // conflict. If there is a confict (This case), then convert first event to a remove,
  1890. // and the second event to the rename.
  1891. if (fFirstPidlInNamespace)
  1892. {
  1893. fEvent1Set = TRUE;
  1894. *plEvent = fFolder? SHCNE_RMDIR : SHCNE_DELETE;
  1895. plNewEvent = plEvent2;
  1896. ppidlNewWrapped1 = ppidlOut1Event2;
  1897. ppidlNewWrapped2 = ppidlOut2Event2;
  1898. }
  1899. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is in Folder", pEnumItem->_pszDisplayName);
  1900. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Adding pidl to %s. Convert to Rename for event %s",
  1901. pEnumItem->_pszDisplayName, fEvent1Set? TEXT("2") : TEXT("1"));
  1902. // Then the destination needs to be merged.
  1903. *ppidlNewWrapped1 = ILCombineBase(pidl2, pEnumItem->_pidlWrap);
  1904. if (*ppidlNewWrapped1)
  1905. {
  1906. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Successfully created out pidl1");
  1907. AugMergeISF_WrapAddPidl(pidlRealRel2, iShellFolder2, &pEnumItem->_pidlWrap);
  1908. *ppidlNewWrapped2 = ILCombineBase(pidl2, pEnumItem->_pidlWrap);
  1909. *plNewEvent = fFolder? SHCNE_RENAMEFOLDER : SHCNE_RENAMEITEM;
  1910. }
  1911. }
  1912. else
  1913. {
  1914. LPITEMIDLIST pidlWrap;
  1915. CAugISFEnumItem* paugmEnum = new CAugISFEnumItem;
  1916. if (paugmEnum)
  1917. {
  1918. if (SUCCEEDED(AugMergeISF_CreateWrap(pidlRealRel2, (UINT)iShellFolder2, &pidlWrap)) &&
  1919. paugmEnum->InitWithWrappedToOwn(SAFECAST(this, IAugmentedShellFolder2*),
  1920. iShellFolder2, pidlWrap))
  1921. {
  1922. AUGMISFSEARCHFORPIDL AugMSearch;
  1923. AugMSearch.pszDisplayName = paugmEnum->_pszDisplayName;
  1924. AugMSearch.fFolder = BOOLIFY(paugmEnum->_rgfAttrib & SFGAO_FOLDER);
  1925. int iInsertIndex = DPA_Search(_hdpaObjects, &AugMSearch, 0,
  1926. AugMISFSearchForOnePidlByDisplayName, NULL, DPAS_SORTED | DPAS_INSERTAFTER);
  1927. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: %s is a new item. Converting to Create",
  1928. paugmEnum->_pszDisplayName);
  1929. if (iInsertIndex < 0)
  1930. iInsertIndex = DA_LAST;
  1931. if (DPA_InsertPtr(_hdpaObjects, iInsertIndex, paugmEnum) == -1)
  1932. {
  1933. TraceMsg(TF_ERROR, "CAugMISF::TranslateIDs: Was unable to add %s for some reason. Bailing",
  1934. paugmEnum->_pszDisplayName);
  1935. DestroyObjectsProc(paugmEnum, NULL);
  1936. }
  1937. else
  1938. {
  1939. TraceMsg(TF_AUGM, "CAugMISF::TranslateIDs: Creating new item %s at %d for event %s",
  1940. paugmEnum->_pszDisplayName, iInsertIndex, fEvent1Set? TEXT("2") : TEXT("1"));
  1941. // If we're renaming from this folder, into this folder, Then the first event stays
  1942. // a rename.
  1943. if (!fFirstPidlInNamespace)
  1944. {
  1945. *plNewEvent = fFolder ? SHCNE_MKDIR : SHCNE_CREATE;
  1946. *ppidlNewWrapped1 = ILCombineBase(pidl2, pidlWrap);
  1947. *ppidlNewWrapped2 = NULL;
  1948. }
  1949. else
  1950. *ppidlOut2 = ILCombineBase(pidl2, pidlWrap);
  1951. }
  1952. }
  1953. else
  1954. DestroyObjectsProc(paugmEnum, NULL);
  1955. }
  1956. }
  1957. }
  1958. }
  1959. break;
  1960. default:
  1961. break;
  1962. }
  1963. Cleanup:
  1964. ILFree(pidlReal1);
  1965. ILFree(pidlReal2);
  1966. #ifdef DEBUG
  1967. DumpObjects();
  1968. _fInternalGDNO = FALSE;
  1969. #endif
  1970. return hres;
  1971. }
  1972. BOOL CAugmentedMergeISF::IsChildIDInternal(LPCITEMIDLIST pidlKid, BOOL fImmediate, int* piShellFolder)
  1973. {
  1974. // This is basically the same Method as the interface method, but returns the shell folder
  1975. // that it came from.
  1976. BOOL fChild = FALSE;
  1977. //At this point we should have a translated pidl
  1978. if (pidlKid)
  1979. {
  1980. if (SUCCEEDED(AugMergeISF_IsWrap(pidlKid)))
  1981. {
  1982. LPCITEMIDLIST pidlRelKid = ILFindLastID(pidlKid);
  1983. if (pidlRelKid)
  1984. {
  1985. UINT uiId;
  1986. LPITEMIDLIST pidl;
  1987. HANDLE hEnum = AugMergeISF_EnumFirstSrcPidl(pidlRelKid, &uiId, &pidl);
  1988. if (hEnum)
  1989. {
  1990. do
  1991. {
  1992. ILFree(pidl);
  1993. for (int i = 0; fChild == FALSE && i < DPA_GetPtrCount(_hdpaNamespaces); i++)
  1994. {
  1995. CNamespace * pns = (CNamespace *)DPA_GetPtr(_hdpaNamespaces, i);
  1996. // reuse pidl
  1997. if (pns && (pidl = pns->GetPidl()) != NULL)
  1998. {
  1999. if (ILIsParent(pidl, pidlKid, fImmediate) &&
  2000. !ILIsEqual(pidl, pidlKid))
  2001. {
  2002. fChild = TRUE;
  2003. if (piShellFolder)
  2004. *piShellFolder = i;
  2005. }
  2006. }
  2007. }
  2008. }
  2009. while (fChild == FALSE && AugMergeISF_EnumNextSrcPidl(hEnum, &uiId, &pidl));
  2010. AugMergeISF_EndEnumSrcPidls(hEnum);
  2011. }
  2012. }
  2013. }
  2014. else
  2015. {
  2016. int cSrcs = NamespaceCount();
  2017. for(int i = 0; fChild == FALSE && i < cSrcs ; i++)
  2018. {
  2019. CNamespace* pSrc = Namespace(i);
  2020. if (pSrc && ILIsParent(pSrc->GetPidl(), pidlKid, fImmediate) &&
  2021. !ILIsEqual(pSrc->GetPidl(), pidlKid))
  2022. {
  2023. fChild = TRUE;
  2024. if (piShellFolder)
  2025. *piShellFolder = i;
  2026. }
  2027. }
  2028. }
  2029. }
  2030. return fChild;
  2031. }
  2032. //-------------------------------------------------------------------------------------------------//
  2033. STDMETHODIMP CAugmentedMergeISF::IsChildID( LPCITEMIDLIST pidlKid, BOOL fImmediate)
  2034. {
  2035. return IsChildIDInternal(pidlKid, fImmediate, NULL) ? S_OK : S_FALSE;
  2036. }
  2037. //-------------------------------------------------------------------------------------------------//
  2038. STDMETHODIMP CAugmentedMergeISF::IsEqualID( LPCITEMIDLIST pidl1, LPCITEMIDLIST pidl2 )
  2039. {
  2040. // This used to return E_NOTIMPL. I'm kinda overloading the interface to mean:
  2041. // is this equal tp any of your namespaces.
  2042. HRESULT hres = S_FALSE;
  2043. int cSrcs = NamespaceCount();
  2044. for(int i = 0; hres == S_FALSE && i < cSrcs ; i++)
  2045. {
  2046. CNamespace* pSrc = Namespace(i);
  2047. if (pidl1)
  2048. {
  2049. if (pSrc && ILIsEqual(pSrc->GetPidl(), pidl1))
  2050. hres = S_OK;
  2051. }
  2052. else if (pidl2) // If you pass a pidl2 it means: Is pidl2 a parent of one of my namespaces?
  2053. {
  2054. if (pSrc && ILIsParent(pidl2, pSrc->GetPidl(), FALSE))
  2055. hres = S_OK;
  2056. }
  2057. }
  2058. return hres;
  2059. }
  2060. //-------------------------------------------------------------------------------------------------//
  2061. STDMETHODIMP CAugmentedMergeISF::Register(
  2062. HWND hwnd,
  2063. UINT uMsg,
  2064. long lEvents )
  2065. {
  2066. int i, cSrcs ;
  2067. if( 0 >= (cSrcs = NamespaceCount()) )
  2068. return E_FAIL ;
  2069. for( i = 0; i < cSrcs ; i++ )
  2070. {
  2071. CNamespace* pSrc ;
  2072. if( NULL != (pSrc = Namespace( i )) )
  2073. pSrc->RegisterNotify( hwnd, uMsg, lEvents ) ;
  2074. }
  2075. return S_OK ;
  2076. }
  2077. //-------------------------------------------------------------------------------------------------//
  2078. STDMETHODIMP CAugmentedMergeISF::Unregister ()
  2079. {
  2080. int i, cSrcs = NamespaceCount() ;
  2081. if( cSrcs <= 0 )
  2082. return E_FAIL ;
  2083. for( i = 0; i < cSrcs ; i++ )
  2084. {
  2085. CNamespace* pSrc ;
  2086. if( NULL != (pSrc = Namespace( i )) )
  2087. pSrc->UnregisterNotify() ;
  2088. }
  2089. return S_OK ;
  2090. }
  2091. // *** IDropTarget methods ***
  2092. #define HIDA_GetPIDLFolder(pida) (LPCITEMIDLIST)(((LPBYTE)pida)+(pida)->aoffset[0])
  2093. #define HIDA_GetPIDLItem(pida, i) (LPCITEMIDLIST)(((LPBYTE)pida)+(pida)->aoffset[i+1])
  2094. HRESULT CAugmentedMergeISF::DragEnter(IDataObject *pDataObj, DWORD grfKeyState, POINTL pt, DWORD *pdwEffect)
  2095. {
  2096. ASSERT(!_fCommon);
  2097. ASSERT(_pdt == NULL);
  2098. if (pDataObj)
  2099. {
  2100. InitClipboardFormats();
  2101. FORMATETC fmte = {g_cfHIDA, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL};
  2102. STGMEDIUM medium;
  2103. medium.pUnkForRelease = NULL;
  2104. medium.hGlobal = NULL;
  2105. if (SUCCEEDED(pDataObj->GetData(&fmte, &medium)))
  2106. {
  2107. LPIDA pida = (LPIDA)GlobalLock(medium.hGlobal);
  2108. if (pida)
  2109. {
  2110. LPCITEMIDLIST pidlItem = HIDA_GetPIDLFolder(pida);
  2111. _fCommon = BOOLIFY(_IsCommonPidl(pidlItem));
  2112. GlobalUnlock(medium.hGlobal);
  2113. }
  2114. ReleaseStgMedium(&medium);
  2115. }
  2116. CNamespace *pSrc = NULL;
  2117. ULONG gdnsAttribs = 0;
  2118. if (!_fCommon)
  2119. gdnsAttribs = ASFF_DEFNAMESPACE_ALL;
  2120. if (SUCCEEDED(GetDefNamespace(gdnsAttribs, (PVOID*)&pSrc, NULL, NULL)))
  2121. {
  2122. if (SUCCEEDED(pSrc->ShellFolder()->CreateViewObject(_hwnd, IID_IDropTarget, (void **)&_pdt)))
  2123. {
  2124. _pdt->DragEnter(pDataObj, grfKeyState, pt, pdwEffect);
  2125. }
  2126. }
  2127. }
  2128. _grfDragEnterKeyState = grfKeyState;
  2129. _dwDragEnterEffect = *pdwEffect;
  2130. return S_OK;
  2131. }
  2132. HRESULT CAugmentedMergeISF::DragOver(DWORD grfKeyState, POINTL pt, DWORD *pdwEffect)
  2133. {
  2134. HRESULT hres = S_OK;
  2135. if (_pdt)
  2136. hres = _pdt->DragOver(grfKeyState, pt, pdwEffect);
  2137. return hres;
  2138. }
  2139. HRESULT CAugmentedMergeISF::DragLeave(void)
  2140. {
  2141. HRESULT hres = S_OK;
  2142. _fCommon = 0;
  2143. if (_pdt)
  2144. {
  2145. hres = _pdt->DragLeave();
  2146. ATOMICRELEASE(_pdt);
  2147. }
  2148. return hres;
  2149. }
  2150. HRESULT CAugmentedMergeISF::Drop(IDataObject *pDataObj, DWORD grfKeyState, POINTL pt, DWORD *pdwEffect)
  2151. {
  2152. HRESULT hres = S_OK;
  2153. BOOL bNoUI = !_fCommon;
  2154. BOOL bConfirmed = !_fCommon;
  2155. if (!_pdt && pDataObj)
  2156. {
  2157. LPITEMIDLIST pidlParent = NULL,
  2158. pidlOther = NULL;
  2159. int csidl = _fCommon ? CSIDL_COMMON_STARTMENU : CSIDL_STARTMENU,
  2160. csidlOther = _fCommon ? CSIDL_STARTMENU : CSIDL_COMMON_STARTMENU;
  2161. if (SUCCEEDED(SHGetSpecialFolderLocation(NULL, csidl, &pidlParent)) &&
  2162. SUCCEEDED(SHGetSpecialFolderLocation(NULL, csidlOther, &pidlOther)))
  2163. {
  2164. FORMATETC fmte = {g_cfHIDA, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL};
  2165. STGMEDIUM medium;
  2166. medium.pUnkForRelease = NULL;
  2167. medium.hGlobal = NULL;
  2168. if (SUCCEEDED(pDataObj->GetData(&fmte, &medium)))
  2169. {
  2170. LPIDA pida = (LPIDA)GlobalLock(medium.hGlobal);
  2171. if (pida)
  2172. {
  2173. IShellFolder *psfParent = NULL,
  2174. *psfOther = NULL;
  2175. if (SUCCEEDED(IEBindToObject(pidlParent, &psfParent)) &&
  2176. SUCCEEDED(IEBindToObject(pidlOther, &psfOther)))
  2177. {
  2178. LPCITEMIDLIST pidlItem, pidl;
  2179. LPITEMIDLIST pidlRel;
  2180. pidlItem = HIDA_GetPIDLItem(pida, 0);
  2181. // we came here because we don't have pdt which means that
  2182. // there is only one folder in our namespace and that's not
  2183. // the one we have to drop IDataObj on.
  2184. CNamespace* pCNamespace = Namespace(0);
  2185. if (pCNamespace)
  2186. {
  2187. pidl = pCNamespace->GetPidl(); // don't need to free pidl.
  2188. if (pidl)
  2189. {
  2190. pidlRel = ILClone(ILFindChild(pidlOther, pidl));
  2191. if (pidlRel)
  2192. {
  2193. STRRET strret;
  2194. TCHAR szDir[MAX_PATH];
  2195. strret.uType = STRRET_CSTR;
  2196. if (SUCCEEDED(psfParent->GetDisplayNameOf(pidlRel, SHGDN_FORPARSING, &strret)) &&
  2197. SUCCEEDED(StrRetToBuf(&strret, pidlRel, szDir, ARRAYSIZE(szDir))))
  2198. {
  2199. if (_fCommon)
  2200. {
  2201. bConfirmed = AffectAllUsers(_hwnd);
  2202. bNoUI = TRUE;
  2203. }
  2204. if (bConfirmed)
  2205. {
  2206. BOOL bCreated = FALSE;
  2207. switch (SHCreateDirectory(_hwnd, szDir))
  2208. {
  2209. case ERROR_FILENAME_EXCED_RANGE:
  2210. case ERROR_FILE_EXISTS:
  2211. case ERROR_ALREADY_EXISTS:
  2212. case 0: // It was created successfully.
  2213. bCreated = TRUE;
  2214. }
  2215. if (bCreated)
  2216. {
  2217. IShellFolder *psf;
  2218. if (SUCCEEDED(psfParent->BindToObject(pidlRel, NULL, IID_IShellFolder, (void **)&psf)))
  2219. {
  2220. psf->CreateViewObject(_hwnd, IID_IDropTarget, (void **)&_pdt);
  2221. // we're going to call drop on it, call dragenter first
  2222. if (_pdt)
  2223. _pdt->DragEnter(pDataObj, _grfDragEnterKeyState, pt, &_dwDragEnterEffect);
  2224. psf->Release();
  2225. }
  2226. }
  2227. }
  2228. }
  2229. ILFree(pidlRel);
  2230. }
  2231. }
  2232. }
  2233. }
  2234. if (psfParent)
  2235. psfParent->Release();
  2236. if (psfOther)
  2237. psfOther->Release();
  2238. GlobalUnlock(medium.hGlobal);
  2239. }
  2240. ReleaseStgMedium(&medium);
  2241. }
  2242. }
  2243. ILFree(pidlParent);
  2244. ILFree(pidlOther);
  2245. }
  2246. if (_pdt)
  2247. {
  2248. hres = E_FAIL;
  2249. if ((bNoUI || (bConfirmed = AffectAllUsers(_hwnd))) && bConfirmed)
  2250. hres = _pdt->Drop(pDataObj, grfKeyState, pt, pdwEffect);
  2251. else
  2252. hres = _pdt->DragLeave();
  2253. ATOMICRELEASE(_pdt);
  2254. }
  2255. _fCommon = 0;
  2256. return hres;
  2257. }
  2258. //-------------------------------------------------------------------------------------------------//
  2259. LPITEMIDLIST CAugmentedMergeISF::GetNativePidl(LPCITEMIDLIST pidlWrap, LPARAM nSrcID /*int nID*/)
  2260. {
  2261. LPITEMIDLIST pidlRet = NULL;
  2262. if (SUCCEEDED(AugMergeISF_GetSrcPidl(pidlWrap, (UINT)nSrcID, &pidlRet)))
  2263. return pidlRet ;
  2264. // not wrapped by me.
  2265. return NULL;
  2266. }
  2267. BOOL AffectAllUsers(HWND hwnd)
  2268. {
  2269. TCHAR szMessage[255];
  2270. TCHAR szTitle[20];
  2271. BOOL bRet = FALSE;
  2272. if (MLLoadShellLangString(IDS_ALLUSER_WARNING, szMessage, ARRAYSIZE(szMessage)) > 0 &&
  2273. MLLoadShellLangString(IDS_ALLUSER_WARNING_TITLE, szTitle, ARRAYSIZE(szTitle)) > 0)
  2274. {
  2275. bRet = IDYES == MessageBox(hwnd, szMessage, szTitle, MB_YESNO | MB_ICONINFORMATION);
  2276. }
  2277. return bRet;
  2278. }
  2279. BOOL CAugmentedMergeISF::_IsCommonPidl(LPCITEMIDLIST pidlItem)
  2280. {
  2281. BOOL bRet = FALSE;
  2282. LPITEMIDLIST pidlCommon;
  2283. if (SUCCEEDED(SHGetSpecialFolderLocation(NULL, CSIDL_COMMON_STARTMENU, &pidlCommon)))
  2284. {
  2285. bRet = ILIsParent(pidlCommon, pidlItem, FALSE);
  2286. ILFree(pidlCommon);
  2287. }
  2288. if (!bRet &&
  2289. SUCCEEDED(SHGetSpecialFolderLocation(NULL, CSIDL_COMMON_PROGRAMS, &pidlCommon)))
  2290. {
  2291. bRet = ILIsParent(pidlCommon, pidlItem, FALSE);
  2292. ILFree(pidlCommon);
  2293. }
  2294. if (!bRet &&
  2295. SUCCEEDED(SHGetSpecialFolderLocation(NULL, CSIDL_COMMON_DESKTOPDIRECTORY, &pidlCommon)))
  2296. {
  2297. bRet = ILIsParent(pidlCommon, pidlItem, FALSE);
  2298. ILFree(pidlCommon);
  2299. }
  2300. return bRet;
  2301. }
  2302. HRESULT CAugmentedMergeISF::_SearchForPidl(IShellFolder* psf, LPCITEMIDLIST pidl, BOOL fFolder, int* piIndex, CAugISFEnumItem** ppEnumItem)
  2303. {
  2304. STRRET str;
  2305. TCHAR szDisplayName[MAX_PATH];
  2306. int iIndex = -1;
  2307. *ppEnumItem = NULL;
  2308. if (SUCCEEDED(psf->GetDisplayNameOf(pidl, SHGDN_FORPARSING | SHGDN_INFOLDER, &str)) &&
  2309. SUCCEEDED(StrRetToBuf(&str, pidl, szDisplayName, ARRAYSIZE(szDisplayName))))
  2310. {
  2311. AUGMISFSEARCHFORPIDL SearchFor;
  2312. SearchFor.pszDisplayName = szDisplayName;
  2313. SearchFor.fFolder = fFolder;
  2314. iIndex = DPA_Search(_hdpaObjects, (LPVOID)&SearchFor, 0,
  2315. AugMISFSearchForOnePidlByDisplayName, NULL, DPAS_SORTED);
  2316. if (iIndex >= 0)
  2317. {
  2318. *ppEnumItem = DPA_GETPTR( _hdpaObjects, iIndex, CAugISFEnumItem);
  2319. }
  2320. }
  2321. if (piIndex)
  2322. *piIndex = iIndex;
  2323. if (*ppEnumItem)
  2324. return S_OK;
  2325. return S_FALSE;
  2326. }
  2327. // given a wrapped pidl
  2328. // f-n returns common and user namespaces (if they are in wrapped pidl) -- note that they are not addrefed
  2329. // unwrapped pidl, and if the unwrapped pidl is folder or not
  2330. HRESULT CAugmentedMergeISF::_GetNamespaces(LPCITEMIDLIST pidlWrap,
  2331. CNamespace** ppnsCommon,
  2332. UINT* pnCommonID,
  2333. CNamespace** ppnsUser,
  2334. UINT* pnUserID,
  2335. LPITEMIDLIST* ppidl,
  2336. BOOL *pbIsFolder)
  2337. {
  2338. HRESULT hres;
  2339. UINT nSrcID;
  2340. CNamespace * pns;
  2341. int cWrapped;
  2342. HANDLE hEnum;
  2343. ASSERT(ppnsCommon && ppnsUser && ppidl);
  2344. *ppnsCommon = NULL;
  2345. *ppnsUser = NULL;
  2346. *ppidl = NULL;
  2347. ASSERT(SUCCEEDED(AugMergeISF_IsWrap(pidlWrap)));
  2348. cWrapped = AugMergeISF_GetSourceCount(pidlWrap);
  2349. if (NULL == _hdpaNamespaces || 0 >= cWrapped ||
  2350. NULL == (hEnum = AugMergeISF_EnumFirstSrcPidl(pidlWrap, &nSrcID, ppidl)))
  2351. return E_FAIL;
  2352. hres = QueryNameSpace(nSrcID, (void **)&pns);
  2353. if (EVAL(SUCCEEDED(hres)))
  2354. {
  2355. IShellFolder * psf;
  2356. ULONG rgf = SFGAO_FOLDER;
  2357. psf = pns->ShellFolder(); // no addref
  2358. ASSERT(psf);
  2359. if (SUCCEEDED(psf->GetAttributesOf(1, (LPCITEMIDLIST*)ppidl, &rgf)))
  2360. {
  2361. if (pbIsFolder)
  2362. *pbIsFolder = rgf & SFGAO_FOLDER;
  2363. LPITEMIDLIST pidlItem;
  2364. UINT nCommonID;
  2365. CNamespace* pnsCommonTemp;
  2366. // get common namespace (attribs = 0)
  2367. hres = GetDefNamespace(0, (void **)&pnsCommonTemp, &nCommonID, NULL);
  2368. ASSERT(NamespaceCount() == 2 && SUCCEEDED(hres) || NamespaceCount() == 1);
  2369. if (FAILED(hres))
  2370. nCommonID = 1;
  2371. if (nCommonID == nSrcID)
  2372. {
  2373. *ppnsCommon = pns;
  2374. if (pnCommonID)
  2375. *pnCommonID = nCommonID;
  2376. }
  2377. else
  2378. {
  2379. *ppnsUser = pns;
  2380. if (pnUserID)
  2381. *pnUserID = nSrcID;
  2382. }
  2383. if (AugMergeISF_EnumNextSrcPidl(hEnum, &nSrcID, &pidlItem))
  2384. {
  2385. ASSERT(ILIsEqual(*ppidl, pidlItem));
  2386. ILFree(pidlItem);
  2387. if (SUCCEEDED(QueryNameSpace(nSrcID, (void **)&pns)))
  2388. {
  2389. ASSERT(pns);
  2390. if (nCommonID == nSrcID)
  2391. {
  2392. *ppnsCommon = pns;
  2393. if (pnCommonID)
  2394. *pnCommonID = nCommonID;
  2395. }
  2396. else
  2397. {
  2398. *ppnsUser = pns;
  2399. if (pnUserID)
  2400. *pnUserID = nSrcID;
  2401. }
  2402. }
  2403. }
  2404. hres = S_OK;
  2405. }
  2406. }
  2407. AugMergeISF_EndEnumSrcPidls(hEnum);
  2408. return hres;
  2409. }
  2410. HRESULT CAugmentedMergeISF::_GetContextMenu(HWND hwnd, UINT cidl, LPCITEMIDLIST * apidl,
  2411. UINT * prgfInOut, void ** ppvOut)
  2412. {
  2413. HRESULT hres;
  2414. LPITEMIDLIST pidl;
  2415. BOOL bIsFolder;
  2416. CNamespace * pnsCommon;
  2417. CNamespace * pnsUser;
  2418. ASSERT(cidl == 1);
  2419. // psfCommon and psfUser are not addrefed
  2420. hres = _GetNamespaces(apidl[0], &pnsCommon, NULL, &pnsUser, NULL, &pidl, &bIsFolder);
  2421. if (SUCCEEDED(hres))
  2422. {
  2423. ASSERT(pnsCommon || pnsUser);
  2424. if (bIsFolder)
  2425. {
  2426. // folder? need our context menu
  2427. IShellFolder * psfCommon = NULL;
  2428. IShellFolder * psfUser = NULL;
  2429. LPCITEMIDLIST pidlCommon = NULL;
  2430. LPCITEMIDLIST pidlUser = NULL;
  2431. if (pnsCommon)
  2432. {
  2433. psfCommon = pnsCommon->ShellFolder();
  2434. pidlCommon = pnsCommon->GetPidl();
  2435. }
  2436. if (pnsUser)
  2437. {
  2438. psfUser = pnsUser->ShellFolder();
  2439. pidlUser = pnsUser->GetPidl();
  2440. }
  2441. CAugMergeISFContextMenu * pcm = CreateMergeISFContextMenu(psfCommon, pidlCommon,
  2442. psfUser, pidlUser,
  2443. pidl, hwnd, prgfInOut);
  2444. if (pcm)
  2445. {
  2446. hres = pcm->QueryInterface(IID_IContextMenu, ppvOut);
  2447. pcm->Release();
  2448. }
  2449. else
  2450. hres = E_OUTOFMEMORY;
  2451. }
  2452. else
  2453. { // it's not a folder
  2454. // delegate to the isf
  2455. IShellFolder * psf = pnsUser ? pnsUser->ShellFolder() : pnsCommon->ShellFolder();
  2456. hres = psf->GetUIObjectOf(hwnd, 1, (LPCITEMIDLIST*)&pidl, IID_IContextMenu, prgfInOut, ppvOut);
  2457. }
  2458. ILFree(pidl);
  2459. }
  2460. return hres;
  2461. }
  2462. //-------------------------------------------------------------------------//
  2463. STDMETHODIMP CAugmentedMergeISF::GetDefNamespace(
  2464. LPCITEMIDLIST pidlWrap,
  2465. ULONG dwAttrib,
  2466. OUT IShellFolder** ppsf,
  2467. OUT LPITEMIDLIST* ppidlItem )
  2468. {
  2469. HRESULT hr ;
  2470. LPITEMIDLIST pidl ;
  2471. CNamespace* pSrc ;
  2472. ULONG dwDefAttrib = dwAttrib & ASFF_DEFNAMESPACE_ALL ;
  2473. int cWrapped ;
  2474. UINT nSrcID ;
  2475. HANDLE hEnum ;
  2476. ASSERT( ppsf ) ;
  2477. *ppsf = NULL ;
  2478. if (ppidlItem)
  2479. *ppidlItem = NULL ;
  2480. if (FAILED((hr = AugMergeISF_IsWrap( pidlWrap ))))
  2481. return hr ;
  2482. cWrapped = AugMergeISF_GetSourceCount( pidlWrap ) ;
  2483. // No namespaces?
  2484. if (NULL == _hdpaNamespaces || 0 >= cWrapped ||
  2485. NULL == (hEnum = AugMergeISF_EnumFirstSrcPidl( pidlWrap, &nSrcID, &pidl)))
  2486. return E_FAIL ;
  2487. // Only one namespace in wrap? Give up the shell folder and item ID.
  2488. if (1 == cWrapped || 0==dwDefAttrib)
  2489. {
  2490. AugMergeISF_EndEnumSrcPidls( hEnum ) ; // no need to go further
  2491. // Retrieve the namespace object identified by nSrcID.
  2492. if( SUCCEEDED( (hr = QueryNameSpace( nSrcID, (PVOID*)&pSrc )) ) )
  2493. {
  2494. *ppsf = pSrc->ShellFolder() ;
  2495. if( ppidlItem )
  2496. *ppidlItem = pidl ;
  2497. return S_OK ;
  2498. }
  2499. ILFree( pidl ) ;
  2500. return hr ;
  2501. }
  2502. // More than one namespace in wrap?
  2503. if( cWrapped > 1 )
  2504. {
  2505. LPITEMIDLIST pidl0 = NULL ;
  2506. CNamespace* pSrc0 = NULL ; // get this below.
  2507. for (BOOL bEnum = TRUE ; bEnum ;
  2508. bEnum = AugMergeISF_EnumNextSrcPidl(hEnum, &nSrcID, &pidl))
  2509. {
  2510. if (SUCCEEDED((hr = QueryNameSpace(nSrcID, (PVOID*)&pSrc))))
  2511. {
  2512. if (dwDefAttrib & pSrc->Attrib())
  2513. {
  2514. // Matched attributes; we're done.
  2515. AugMergeISF_EndEnumSrcPidls(hEnum);
  2516. *ppsf = pSrc->ShellFolder() ;
  2517. if (ppidlItem)
  2518. *ppidlItem = pidl;
  2519. if(pidl0)
  2520. ILFree(pidl0);
  2521. return S_OK ;
  2522. }
  2523. // Stash first namespace object and item pidl.
  2524. // We'll default to these if
  2525. if( NULL == pSrc0 )
  2526. {
  2527. pSrc0 = pSrc ;
  2528. pidl0 = ILClone( pidl ) ;
  2529. }
  2530. }
  2531. ILFree( pidl ) ;
  2532. }
  2533. AugMergeISF_EndEnumSrcPidls( hEnum ) ;
  2534. // Default to first namespace
  2535. if( pSrc0 && pidl0 )
  2536. {
  2537. *ppsf = pSrc0->ShellFolder() ;
  2538. if( ppidlItem )
  2539. *ppidlItem = pidl0 ;
  2540. return S_OK ;
  2541. }
  2542. }
  2543. return E_UNEXPECTED ;
  2544. }
  2545. //-------------------------------------------------------------------------//
  2546. // Retrieves the default namespaces for the indicated attibutes.
  2547. // The dwAttrib arg must be initialized prior to function entry,
  2548. STDMETHODIMP CAugmentedMergeISF::GetDefNamespace(
  2549. ULONG dwAttrib,
  2550. OUT PVOID* ppSrc, UINT *pnSrcID, PVOID* ppSrc0 )
  2551. {
  2552. CNamespace* pSrc ;
  2553. ULONG dwDefAttrib = dwAttrib & ASFF_DEFNAMESPACE_ALL ;
  2554. // this is an internal helper so we better make sure we pass the correct params!
  2555. //if (NULL == ppSrc)
  2556. // return E_INVALIDARG ;
  2557. *ppSrc = NULL ;
  2558. if( ppSrc0 )
  2559. *ppSrc0 = NULL ;
  2560. for( int i = 0, cSrcs = NamespaceCount(); i < cSrcs ; i++ )
  2561. {
  2562. if( NULL != (pSrc = Namespace( i )) )
  2563. {
  2564. if( 0 == i && ppSrc0 )
  2565. *ppSrc0 = pSrc ;
  2566. if( dwDefAttrib & pSrc->Attrib() ||
  2567. dwDefAttrib == 0 && !(pSrc->Attrib() & ASFF_DEFNAMESPACE_ALL))
  2568. {
  2569. *ppSrc = pSrc;
  2570. if (NULL != pnSrcID)
  2571. *pnSrcID = i;
  2572. return S_OK ;
  2573. }
  2574. }
  2575. }
  2576. return E_FAIL ;
  2577. }
  2578. // BUGBUG(lamadio): Move this to a better location, This is a nice generic function
  2579. #ifdef DEBUG
  2580. BOOL DPA_VerifySorted(HDPA hdpa, PFNDPACOMPARE pfn, LPARAM lParam)
  2581. {
  2582. if (!EVAL(hdpa))
  2583. return FALSE;
  2584. for (int i = 0; i < DPA_GetPtrCount(hdpa) - 1; i++)
  2585. {
  2586. if (pfn(DPA_FastGetPtr(hdpa, i), DPA_FastGetPtr(hdpa, i + 1), lParam) > 0)
  2587. return FALSE;
  2588. }
  2589. return TRUE;
  2590. }
  2591. #else
  2592. #define DPA_VerifySorted(hdpa, pfn, lParam)
  2593. #endif
  2594. int CAugmentedMergeISF::AcquireObjects()
  2595. {
  2596. HDPA hdpa2 = NULL;
  2597. DEBUG_CODE(_fInternalGDNO = TRUE);
  2598. for (int i = 0; i < DPA_GETPTRCOUNT(_hdpaNamespaces); i++)
  2599. {
  2600. CNamespace * pns = DPA_GETPTR(_hdpaNamespaces, i, CNamespace);
  2601. IShellFolder * psf;
  2602. IEnumIDList * peid;
  2603. HDPA * phdpa;
  2604. ASSERT(pns);
  2605. psf = pns->ShellFolder(); // no addref here!
  2606. if (i == 0)
  2607. {
  2608. phdpa = &_hdpaObjects;
  2609. _hdpaObjects = DPA_Create(4); // We should always create the DPA
  2610. }
  2611. else
  2612. {
  2613. ASSERT(i == 1); // no support for more than 2 isf's
  2614. phdpa = &hdpa2;
  2615. }
  2616. HRESULT hres = psf->EnumObjects(NULL, SHCONTF_FOLDERS | SHCONTF_NONFOLDERS | SHCONTF_INCLUDEHIDDEN, &peid);
  2617. if (SUCCEEDED(hres))
  2618. {
  2619. if (!*phdpa)
  2620. *phdpa = DPA_Create(4);
  2621. if (*phdpa)
  2622. {
  2623. LPITEMIDLIST pidl;
  2624. ULONG cEnum;
  2625. while (SUCCEEDED(peid->Next(1, &pidl, &cEnum)) && 1 == cEnum)
  2626. {
  2627. CAugISFEnumItem* paugmEnum = new CAugISFEnumItem;
  2628. if (paugmEnum)
  2629. {
  2630. if (paugmEnum->Init(SAFECAST(this, IAugmentedShellFolder2*), i, pidl))
  2631. {
  2632. if (DPA_AppendPtr(*phdpa, paugmEnum) == -1)
  2633. DestroyObjectsProc(paugmEnum, NULL);
  2634. }
  2635. else
  2636. delete paugmEnum;
  2637. }
  2638. ILFree(pidl);
  2639. }
  2640. }
  2641. peid->Release();
  2642. }
  2643. else
  2644. {
  2645. TraceMsg(TF_WARNING, "CAugMISF::AcquireObjects: Failed to get enumerator 0x%x", hres);
  2646. }
  2647. }
  2648. // now that we have both hdpa's (or one) let's merge them.
  2649. if (DPA_GETPTRCOUNT(_hdpaNamespaces) == 2 && hdpa2)
  2650. {
  2651. DPA_Merge(_hdpaObjects, hdpa2, DPAM_UNION, AugmEnumCompare, AugmEnumMerge, (LPARAM)0);
  2652. DPA_DESTROY(hdpa2, DestroyObjectsProc);
  2653. }
  2654. else
  2655. {
  2656. DPA_Sort(_hdpaObjects, AugmEnumCompare, 0);
  2657. }
  2658. ASSERT(DPA_VerifySorted(_hdpaObjects, AugmEnumCompare, 0));
  2659. DEBUG_CODE(_fInternalGDNO = FALSE);
  2660. #ifdef DEBUG
  2661. TraceMsg(TF_AUGM, "CAugMISF::AcquireObjects");
  2662. DumpObjects();
  2663. #endif
  2664. _count = DPA_GETPTRCOUNT(_hdpaObjects);
  2665. return _count;
  2666. }
  2667. //-------------------------------------------------------------------------//
  2668. void CAugmentedMergeISF::FreeObjects()
  2669. {
  2670. DPA_DESTROY( _hdpaObjects, DestroyObjectsProc ) ;
  2671. _hdpaObjects = NULL;
  2672. _count = 0 ;
  2673. }
  2674. //-------------------------------------------------------------------------//
  2675. int CAugmentedMergeISF::DestroyObjectsProc( LPVOID pv, LPVOID pvData )
  2676. {
  2677. CAugISFEnumItem* paugmEnum = (CAugISFEnumItem*)pv;
  2678. if (EVAL(NULL != paugmEnum))
  2679. {
  2680. delete paugmEnum;
  2681. }
  2682. return TRUE ;
  2683. }
  2684. //-------------------------------------------------------------------------//
  2685. void CAugmentedMergeISF::FreeNamespaces()
  2686. {
  2687. DPA_DESTROY( _hdpaNamespaces, DestroyNamespacesProc ) ;
  2688. }
  2689. //-------------------------------------------------------------------------//
  2690. int CAugmentedMergeISF::DestroyNamespacesProc( LPVOID pv, LPVOID pvData )
  2691. {
  2692. CNamespace* p ;
  2693. if( NULL != (p = (CNamespace*)pv) )
  2694. delete p ;
  2695. return TRUE ;
  2696. }
  2697. STDMETHODIMP CAugmentedMergeISF::GetPidl(int* piPos, DWORD grfEnumFlags, LPITEMIDLIST* ppidl)
  2698. {
  2699. *ppidl = NULL;
  2700. if (_hdpaObjects == NULL)
  2701. AcquireObjects();
  2702. if (_hdpaObjects == NULL)
  2703. return E_OUTOFMEMORY;
  2704. BOOL fWantFolders = 0 != (grfEnumFlags & SHCONTF_FOLDERS),
  2705. fWantNonFolders = 0 != (grfEnumFlags & SHCONTF_NONFOLDERS),
  2706. fWantHidden = 0 != (grfEnumFlags & SHCONTF_INCLUDEHIDDEN) ;
  2707. while (*piPos < _count)
  2708. {
  2709. CAugISFEnumItem* paugmEnum = DPA_GETPTR( _hdpaObjects, *piPos, CAugISFEnumItem);
  2710. if ( NULL != paugmEnum )
  2711. {
  2712. BOOL fFolder = 0 != (paugmEnum->_rgfAttrib & SFGAO_FOLDER),
  2713. fHidden = 0 != (paugmEnum->_rgfAttrib & SFGAO_HIDDEN);
  2714. if ((!fHidden || fWantHidden) &&
  2715. ((fFolder && fWantFolders) || (!fFolder && fWantNonFolders)))
  2716. {
  2717. // Copy out the pidl ;
  2718. *ppidl = ILClone(paugmEnum->_pidlWrap);
  2719. break;
  2720. }
  2721. else
  2722. {
  2723. (*piPos)++;
  2724. }
  2725. }
  2726. }
  2727. if (*ppidl)
  2728. return S_OK;
  2729. return S_FALSE;
  2730. }
  2731. //-------------------------------------------------------------------------//
  2732. CEnum::CEnum(IAugmentedMergedShellFolderInternal* psmsfi, DWORD grfEnumFlags, int iPos) :
  2733. _cRef(1),
  2734. _iPos(iPos),
  2735. _psmsfi(psmsfi),
  2736. _grfEnumFlags(grfEnumFlags)
  2737. {
  2738. _psmsfi->AddRef();
  2739. }
  2740. CEnum::~CEnum()
  2741. {
  2742. ATOMICRELEASE(_psmsfi);
  2743. }
  2744. //-------------------------------------------------------------------------//
  2745. // class CEnum - IUnknown methods
  2746. //-------------------------------------------------------------------------//
  2747. //-------------------------------------------------------------------------//
  2748. STDMETHODIMP CEnum::QueryInterface( REFIID riid, LPVOID * ppvObj )
  2749. {
  2750. static const QITAB qit[] = {
  2751. QITABENT(CEnum, IEnumIDList),
  2752. { 0 }
  2753. };
  2754. return QISearch(this, qit, riid, ppvObj);
  2755. }
  2756. //-------------------------------------------------------------------------//
  2757. STDMETHODIMP_(ULONG) CEnum::AddRef ()
  2758. {
  2759. return InterlockedIncrement((LONG*)&_cRef);
  2760. }
  2761. //-------------------------------------------------------------------------//
  2762. STDMETHODIMP_(ULONG) CEnum::Release ()
  2763. {
  2764. if (InterlockedDecrement((LONG*)&_cRef))
  2765. return _cRef;
  2766. delete this ;
  2767. return 0;
  2768. }
  2769. //-------------------------------------------------------------------------//
  2770. // class CEnum - IEnumIDList methods
  2771. //-------------------------------------------------------------------------//
  2772. STDMETHODIMP CEnum::Next(
  2773. ULONG celt,
  2774. LPITEMIDLIST *rgelt,
  2775. ULONG *pceltFetched )
  2776. {
  2777. int iStart = _iPos;
  2778. int cFetched = 0;
  2779. HRESULT hres = S_OK;
  2780. if( !(celt > 0 && rgelt) || (NULL == pceltFetched && celt > 1 ) )
  2781. return E_INVALIDARG ;
  2782. *rgelt = 0;
  2783. while(hres == S_OK && (_iPos - iStart) < (int)celt)
  2784. {
  2785. LPITEMIDLIST pidl;
  2786. hres = _psmsfi->GetPidl(&_iPos, _grfEnumFlags, &pidl);
  2787. if (hres == S_OK)
  2788. {
  2789. rgelt[cFetched] = pidl;
  2790. cFetched++ ;
  2791. }
  2792. _iPos++;
  2793. }
  2794. if( pceltFetched )
  2795. *pceltFetched = cFetched ;
  2796. return celt == (ULONG)cFetched ? S_OK : S_FALSE ;
  2797. }
  2798. //-------------------------------------------------------------------------//
  2799. STDMETHODIMP CEnum::Skip(ULONG celt)
  2800. {
  2801. _iPos += celt;
  2802. return S_OK ;
  2803. }
  2804. //-------------------------------------------------------------------------//
  2805. STDMETHODIMP CEnum::Reset()
  2806. {
  2807. _iPos = 0;
  2808. return S_OK ;
  2809. }
  2810. //-------------------------------------------------------------------------//
  2811. // REVIEW: Can probably be E_NOTIMPL
  2812. STDMETHODIMP CEnum::Clone( IEnumIDList **ppenum )
  2813. {
  2814. if( NULL == (*ppenum = new CEnum( _psmsfi, _grfEnumFlags, _iPos )) )
  2815. return E_OUTOFMEMORY;
  2816. return S_OK;
  2817. }
  2818. BOOL CAugISFEnumItem::Init(IShellFolder* psf, int iShellFolder, LPCITEMIDLIST pidl)
  2819. {
  2820. // This is ok, the memory just gets written to twice.
  2821. if (SUCCEEDED(AugMergeISF_CreateWrap(pidl, iShellFolder, &_pidlWrap)))
  2822. {
  2823. // Takes ownership of passed in pidl.
  2824. return InitWithWrappedToOwn(psf, iShellFolder, _pidlWrap);
  2825. }
  2826. return FALSE;
  2827. }
  2828. BOOL CAugISFEnumItem::InitWithWrappedToOwn(IShellFolder* psf, int iShellFolder, LPITEMIDLIST pidl)
  2829. {
  2830. BOOL fRet = FALSE;
  2831. STRRET str;
  2832. TCHAR szDisplayName[MAX_PATH];
  2833. _pidlWrap = pidl;
  2834. _rgfAttrib = SFGAO_FOLDER | SFGAO_HIDDEN;
  2835. psf->GetAttributesOf(1, (LPCITEMIDLIST*)&pidl, &_rgfAttrib);
  2836. if (SUCCEEDED(psf->GetDisplayNameOf(pidl, SHGDN_FORPARSING | SHGDN_INFOLDER, &str)) &&
  2837. SUCCEEDED(StrRetToBuf(&str, pidl, szDisplayName, ARRAYSIZE(szDisplayName))))
  2838. {
  2839. SetDisplayName(szDisplayName);
  2840. fRet = TRUE;
  2841. }
  2842. return fRet;
  2843. }