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.

2727 lines
75 KiB

  1. #include "shellprv.h"
  2. #include "common.h"
  3. #include "sftbar.h"
  4. #include "resource.h"
  5. #include "dpastuff.h"
  6. #include "shlwapi.h"
  7. #include "cobjsafe.h"
  8. #include <iimgctx.h>
  9. #include "uemapp.h"
  10. #include "util.h"
  11. #include "brutil.h"
  12. #include "dobjutil.h"
  13. #include "idlcomm.h"
  14. extern UINT g_idFSNotify;
  15. #define TF_SFTBAR TF_MENUBAND
  16. #define PGMP_RECALCSIZE 200
  17. CSFToolbar::CSFToolbar()
  18. {
  19. #ifdef CASCADE_DEBUG
  20. _fCascadeFolder = TRUE;
  21. #endif
  22. _dwStyle = TBSTYLE_TOOLTIPS;
  23. _fDirty = TRUE; // we havn't enumerated, so our state is dirty
  24. _fRegisterChangeNotify = TRUE;
  25. _fAllowReorder = TRUE;
  26. _tbim.iButton = -1;
  27. _iDragSource = -1;
  28. _lEvents = SHCNE_DRIVEADD|SHCNE_CREATE|SHCNE_MKDIR|SHCNE_DRIVEREMOVED|
  29. SHCNE_DELETE|SHCNE_RMDIR|SHCNE_RENAMEITEM|SHCNE_RENAMEFOLDER|
  30. SHCNE_MEDIAINSERTED|SHCNE_MEDIAREMOVED|SHCNE_NETUNSHARE|SHCNE_NETSHARE|
  31. SHCNE_UPDATEITEM|SHCNE_UPDATEIMAGE|SHCNE_ASSOCCHANGED|
  32. SHCNE_UPDATEDIR|SHCNE_EXTENDED_EVENT;
  33. #define SHCNE_PIDL1ISCHILD \
  34. (SHCNE_DRIVEADD|SHCNE_CREATE|SHCNE_MKDIR|SHCNE_DRIVEREMOVED|\
  35. SHCNE_DELETE|SHCNE_RMDIR|SHCNE_NETUNSHARE|SHCNE_NETSHARE|\
  36. SHCNE_UPDATEITEM)
  37. }
  38. CSFToolbar::~CSFToolbar()
  39. {
  40. ATOMICRELEASE(_pcmSF);
  41. ATOMICRELEASE(_piml);
  42. _ReleaseShellFolder();
  43. ILFree(_pidl);
  44. ASSERT(NULL == _hdpa);
  45. if (_hwndWorkerWindow)
  46. DestroyWindow(_hwndWorkerWindow);
  47. OrderList_Destroy(&_hdpaOrder);
  48. }
  49. HRESULT CSFToolbar::QueryInterface(REFIID riid, void **ppvObj)
  50. {
  51. static const QITAB qit[] = {
  52. QITABENT(CSFToolbar, IWinEventHandler),
  53. QITABENT(CSFToolbar, IShellChangeNotify),
  54. QITABENT(CSFToolbar, IDropTarget),
  55. QITABENT(CSFToolbar, IContextMenu),
  56. QITABENT(CSFToolbar, IShellFolderBand),
  57. { 0 },
  58. };
  59. return QISearch(this, qit, riid, ppvObj);
  60. }
  61. HRESULT CSFToolbar::SetShellFolder(IShellFolder* psf, LPCITEMIDLIST pidl)
  62. {
  63. HRESULT hr = E_INVALIDARG;
  64. // Save the old values
  65. LPITEMIDLIST pidlSave = _pidl;
  66. IShellFolder *psfSave = _psf;
  67. ITranslateShellChangeNotify *ptscnSave = _ptscn;
  68. _psf = NULL;
  69. _pidl = NULL;
  70. _ptscn = NULL;
  71. ASSERT(NULL == psf || IS_VALID_CODE_PTR(psf, IShellFolder));
  72. ASSERT(NULL == pidl || IS_VALID_PIDL(pidl));
  73. if (psf || pidl)
  74. {
  75. if (psf)
  76. {
  77. _psf = psf;
  78. _psf->AddRef();
  79. _psf->QueryInterface(IID_PPV_ARG(ITranslateShellChangeNotify, &_ptscn));
  80. }
  81. if (pidl)
  82. _pidl = ILClone(pidl);
  83. hr = S_OK;
  84. }
  85. if (SUCCEEDED(hr))
  86. {
  87. ILFree(pidlSave);
  88. if (psfSave)
  89. psfSave->Release();
  90. if (ptscnSave)
  91. ptscnSave->Release();
  92. }
  93. else
  94. {
  95. ASSERT(_psf == NULL);
  96. ASSERT(_pidl == NULL);
  97. ASSERT(_ptscn == NULL);
  98. // we failed -- restore the old values
  99. _psf = psfSave;
  100. _pidl = pidlSave;
  101. _ptscn = ptscnSave;
  102. }
  103. // This code is here for ShellFolderToolbar reuse. When setting a new shell folder
  104. // into an existing band, we will refresh. Note that this is a noop on a new band.
  105. _RememberOrder();
  106. _SetDirty(TRUE);
  107. if (_fShow)
  108. _FillToolbar();
  109. return hr;
  110. }
  111. HWND CSFToolbar::_CreatePager(HWND hwndParent)
  112. {
  113. if (!_fMulticolumn)
  114. {
  115. _hwndPager = CreateWindowEx(0, WC_PAGESCROLLER, NULL,
  116. WS_CHILD | WS_TABSTOP |
  117. WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
  118. 0, 0, 0, 0, hwndParent, (HMENU) 0, HINST_THISDLL, NULL);
  119. if (_hwndPager)
  120. {
  121. hwndParent = _hwndPager;
  122. }
  123. }
  124. return hwndParent;
  125. }
  126. HRESULT CSFToolbar::_CreateToolbar(HWND hwndParent)
  127. {
  128. if (!_hwndTB)
  129. {
  130. hwndParent = _CreatePager(hwndParent);
  131. _hwndTB = CreateWindowEx(WS_EX_TOOLWINDOW, TOOLBARCLASSNAME, NULL,
  132. WS_VISIBLE | WS_CHILD | TBSTYLE_FLAT |
  133. WS_CLIPCHILDREN | WS_CLIPSIBLINGS |
  134. CCS_NODIVIDER | CCS_NOPARENTALIGN |
  135. CCS_NORESIZE | _dwStyle,
  136. 0, 0, 0, 0, hwndParent, (HMENU) 0, HINST_THISDLL, NULL);
  137. if (!_hwndTB)
  138. {
  139. TraceMsg(TF_ERROR, "_hwndTB failed");
  140. return HRESULT_FROM_WIN32(GetLastError());
  141. }
  142. if (_hwndPager)
  143. SendMessage(_hwndPager, PGM_SETCHILD, 0, (LPARAM)_hwndTB);
  144. SendMessage(_hwndTB, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
  145. // Set the format to ANSI or UNICODE as appropriate.
  146. ToolBar_SetUnicodeFormat(_hwndTB, DLL_IS_UNICODE);
  147. if (_hwndPager)
  148. {
  149. // Set the format to ANSI or UNICODE as appropriate.
  150. ToolBar_SetUnicodeFormat(_hwndPager, DLL_IS_UNICODE);
  151. }
  152. // Make sure we're on the same wavelength.
  153. SendMessage(_hwndTB, CCM_SETVERSION, COMCTL32_VERSION, 0);
  154. SendMessage(_hwndTB, TB_SETEXTENDEDSTYLE, TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_DOUBLEBUFFER);
  155. RECT rc;
  156. SIZE size;
  157. SystemParametersInfoA(SPI_GETWORKAREA, sizeof(RECT), &rc, FALSE);
  158. if (!_hwndPager)
  159. {
  160. size.cx = RECTWIDTH(rc);
  161. size.cy = GetSystemMetrics(SM_CYSCREEN) - (2 * GetSystemMetrics(SM_CYEDGE)); // Need to subrtact off the borders
  162. }
  163. else
  164. {
  165. //HACKHACK: THIS WILL FORCE NO WRAP TO HAPPEN FOR PROPER WIDTH CALC WHEN PAGER IS PRESENT.
  166. size.cx = RECTWIDTH(rc);
  167. size.cy = 32000;
  168. }
  169. ToolBar_SetBoundingSize(_hwndTB, &size);
  170. if (!_SubclassWindow(_hwndTB))
  171. {
  172. _fRegisterChangeNotify = FALSE;
  173. }
  174. }
  175. else
  176. {
  177. if (_hwndPager && GetParent(_hwndPager) != hwndParent)
  178. SetParent(_hwndPager, hwndParent);
  179. }
  180. if (FAILED(_GetTopBrowserWindow(&_hwndDD)))
  181. _hwndDD = GetParent(_hwndTB);
  182. return S_OK;
  183. }
  184. #define MAX_COMMANDID 0xFFFF // We're allowed one word of command ids (tested at 5)
  185. int CSFToolbar::_GetCommandID()
  186. {
  187. int id = -1;
  188. if (!_fCheckIds)
  189. {
  190. id = _nNextCommandID++;
  191. }
  192. else
  193. {
  194. // We are reusing command ids and must verify that
  195. // the current one is not in use. This is slow, but
  196. // I assume the number of buttons on one of these
  197. // bands is relatively few.
  198. //
  199. for (int i = 0 ; i <= MAX_COMMANDID ; i++)
  200. {
  201. TBBUTTONINFO tbbiDummy;
  202. tbbiDummy.cbSize = sizeof(tbbiDummy);
  203. tbbiDummy.dwMask = 0; // we don't care about data, just existence
  204. if (-1 != ToolBar_GetButtonInfo(_hwndTB, _nNextCommandID, &tbbiDummy))
  205. {
  206. // A button by this id wasn't found, so the id must be free
  207. //
  208. id = _nNextCommandID++;
  209. break;
  210. }
  211. _nNextCommandID++;
  212. _nNextCommandID %= MAX_COMMANDID;
  213. }
  214. }
  215. if (_nNextCommandID > MAX_COMMANDID)
  216. {
  217. _nNextCommandID = 0;
  218. _fCheckIds = TRUE;
  219. }
  220. return id;
  221. }
  222. /*----------------------------------------------------------
  223. Purpose: This function determines the toolbar button style for the
  224. given pidl.
  225. Returns S_OK if pdwMIFFlags is also set (i.e., the object
  226. supported IMenuBandItem to provide more info). S_FALSE if only
  227. *pdwTBStyle is set.
  228. */
  229. HRESULT CSFToolbar::_TBStyleForPidl(LPCITEMIDLIST pidl, DWORD *pdwTBStyle, DWORD *pdwTBState, DWORD *pdwMIFFlags, int* piIcon)
  230. {
  231. DWORD dwStyle = TBSTYLE_BUTTON;
  232. if (!_fAccelerators)
  233. dwStyle |= TBSTYLE_NOPREFIX;
  234. *pdwMIFFlags = 0;
  235. *pdwTBStyle = dwStyle;
  236. *piIcon = -1;
  237. *pdwTBState = TBSTATE_ENABLED;
  238. return S_FALSE;
  239. }
  240. PIBDATA CSFToolbar::_CreateItemData(PORDERITEM poi)
  241. {
  242. return new IBDATA(poi);
  243. }
  244. //
  245. // poiOrIndex can be...
  246. //
  247. // A valid pointer (which will be treated as a PORDERITEM)
  248. // A MAKEINTRESOURCE value (which will be treated as a sentinel value)
  249. //
  250. PIBDATA CSFToolbar::_AddOrderItemTB(PORDERITEM poiOrIndex, int index, TBBUTTON* ptbb)
  251. {
  252. TCHAR szName[MAX_PATH];
  253. LPCITEMIDLIST pidlOI;
  254. PORDERITEM poi;
  255. if (IS_INTRESOURCE(poiOrIndex))
  256. {
  257. poi = NULL;
  258. pidlOI = (LPCITEMIDLIST)poiOrIndex;
  259. }
  260. else
  261. {
  262. poi = poiOrIndex;
  263. pidlOI = poi->pidl;
  264. }
  265. // We need to do this even for NULL because _ObtainPIDLName cooks
  266. // up the word "(Empty)" as necessary.
  267. _ObtainPIDLName(pidlOI, szName, SIZECHARS(szName));
  268. TBBUTTON tbb = {0};
  269. DWORD dwMIFFlags;
  270. DWORD dwStyle;
  271. DWORD dwState;
  272. int iIcon;
  273. int iCommandID = _GetCommandID();
  274. BOOL bNoIcon = FALSE;
  275. if (!ptbb)
  276. ptbb = &tbb;
  277. if (S_OK == _TBStyleForPidl(pidlOI, &dwStyle, &dwState, &dwMIFFlags,&iIcon) &&
  278. !(dwMIFFlags & SMIF_ICON))
  279. {
  280. bNoIcon = TRUE;
  281. }
  282. PIBDATA pibdata = _CreateItemData(poi);
  283. if (pibdata)
  284. {
  285. pibdata->SetFlags(dwMIFFlags);
  286. pibdata->SetNoIcon(bNoIcon);
  287. if (!bNoIcon && iIcon != -1)
  288. ptbb->iBitmap = iIcon;
  289. else
  290. ptbb->iBitmap = I_IMAGECALLBACK;
  291. ptbb->idCommand = iCommandID;
  292. ptbb->fsState = (BYTE)dwState;
  293. ptbb->fsStyle = (BYTE)dwStyle;
  294. ptbb->dwData = (DWORD_PTR)pibdata;
  295. ptbb->iString = (INT_PTR)szName;
  296. // Disregard variablewidth if we are vertical
  297. if (_fVariableWidth && !_fVertical)
  298. ptbb->fsStyle |= TBSTYLE_AUTOSIZE;
  299. if (ptbb->idCommand != -1)
  300. {
  301. if (SendMessage(_hwndTB, TB_INSERTBUTTON, index, (LPARAM)ptbb))
  302. {
  303. TraceMsg(TF_BAND, "SFToolbar::_AddPidl %d 0x%x [%s]", ptbb->idCommand, ptbb->dwData, ptbb->iString);
  304. }
  305. else
  306. {
  307. delete pibdata;
  308. pibdata = NULL;
  309. }
  310. }
  311. }
  312. return pibdata;
  313. }
  314. void CSFToolbar::_ObtainPIDLName(LPCITEMIDLIST pidl, LPTSTR psz, int cchMax)
  315. {
  316. DisplayNameOf(_psf, pidl, SHGDN_NORMAL, psz, cchMax);
  317. }
  318. int CSFToolbar::_GetBitmap(int iCommandID, PIBDATA pibdata, BOOL fUseCache)
  319. {
  320. int iBitmap;
  321. if (_fNoIcons || !pibdata || pibdata->GetNoIcon())
  322. {
  323. iBitmap = -1;
  324. }
  325. else
  326. {
  327. iBitmap = OrderItem_GetSystemImageListIndex(pibdata->GetOrderItem(), _psf, fUseCache);
  328. }
  329. return iBitmap;
  330. }
  331. void CSFToolbar::_OnGetDispInfo(LPNMHDR pnm, BOOL fUnicode)
  332. {
  333. LPNMTBDISPINFO pdi = (LPNMTBDISPINFO)pnm;
  334. PIBDATA pibdata = (PIBDATA)pdi->lParam;
  335. if (pdi->dwMask & TBNF_IMAGE)
  336. {
  337. pdi->iImage = _GetBitmap(pdi->idCommand, pibdata, TRUE);
  338. }
  339. if (pdi->dwMask & TBNF_TEXT)
  340. {
  341. if (pdi->pszText)
  342. {
  343. if (fUnicode)
  344. {
  345. pdi->pszText[0] = TEXT('\0');
  346. }
  347. else
  348. {
  349. pdi->pszText[0] = 0;
  350. }
  351. }
  352. }
  353. pdi->dwMask |= TBNF_DI_SETITEM;
  354. }
  355. // Adds pidl as a new button, handles ILFree(pidl) for the caller
  356. //
  357. BOOL CSFToolbar::_AddPidl(LPITEMIDLIST pidl, DWORD dwFlags, int index)
  358. {
  359. if (_hdpa)
  360. {
  361. PORDERITEM poi = OrderItem_Create(pidl, index);
  362. if (poi)
  363. {
  364. int iPos = DPA_InsertPtr(_hdpa, index, poi);
  365. if (-1 != iPos)
  366. {
  367. // If we did not load an order, then new items should
  368. // show up alphabetically in the list, not at the bottom.
  369. if (!_fHasOrder && !(dwFlags & FSNA_BULKADD))
  370. {
  371. // Sort by name
  372. _SortDPA(_hdpa);
  373. // Find the index of the order item. We use this index as
  374. // the toolbar insert index.
  375. index = DPA_GetPtrIndex(_hdpa, poi);
  376. }
  377. if (_AddOrderItemTB(poi, index, NULL))
  378. {
  379. return TRUE;
  380. }
  381. DPA_DeletePtr(_hdpa, iPos);
  382. }
  383. OrderItem_Free(poi);
  384. return FALSE;
  385. }
  386. }
  387. ILFree(pidl);
  388. return FALSE;
  389. }
  390. BOOL CSFToolbar::_FilterPidl(LPCITEMIDLIST pidl)
  391. {
  392. return FALSE;
  393. }
  394. void CSFToolbar::s_NewItem(void *pvParam, LPCITEMIDLIST pidl)
  395. {
  396. CSFToolbar* psft = (CSFToolbar*)pvParam;
  397. psft->v_NewItem(pidl);
  398. }
  399. HRESULT CSFToolbar::_GetIEnumIDList(DWORD dwEnumFlags, IEnumIDList **ppenum)
  400. {
  401. ASSERT(_psf);
  402. // Pass in a NULL hwnd so the enumerator does not show any UI while
  403. // we're filling a band.
  404. return IShellFolder_EnumObjects(_psf, NULL, dwEnumFlags, ppenum);
  405. }
  406. void CSFToolbar::_FillDPA(HDPA hdpa, HDPA hdpaSort, DWORD dwEnumFlags)
  407. {
  408. IEnumIDList* penum;
  409. int cItems = 0;
  410. if (!_psf)
  411. return;
  412. if (S_OK == _GetIEnumIDList(dwEnumFlags, &penum))
  413. {
  414. LPITEMIDLIST pidl;
  415. ULONG ul;
  416. while (S_OK == penum->Next(1, &pidl, &ul))
  417. {
  418. cItems++;
  419. if (_FilterPidl(pidl) || !OrderList_Append(hdpa, pidl, -1))
  420. {
  421. TraceMsg(TF_MENUBAND, "SFToolbar (0x%x)::_FillDPA : Did not Add Pidl (0x%x).", this, pidl);
  422. ILFree(pidl);
  423. }
  424. }
  425. penum->Release();
  426. }
  427. ORDERINFO oinfo;
  428. int iInsertIndex = _tbim.iButton + 1; // This is the button where the cursor sat.
  429. // So, We want to insert after that
  430. if (iInsertIndex >= ToolBar_ButtonCount(_hwndTB)) // But, if it's at the end,
  431. iInsertIndex = -1; // Convert the insert to an append.
  432. // - Comments in rhyme by lamadio
  433. oinfo.psf = _psf;
  434. (oinfo.psf)->AddRef();
  435. oinfo.dwSortBy = (_fHasOrder || _fDropping)? ((_fNoNameSort ? OI_SORTBYORDINAL : OI_SORTBYNAME)): OI_MERGEBYNAME;
  436. OrderList_Merge(hdpa, hdpaSort, _fDropping ? iInsertIndex : _DefaultInsertIndex(), (LPARAM) &oinfo,
  437. s_NewItem, (void *)this);
  438. ATOMICRELEASE(oinfo.psf);
  439. }
  440. // This function re-enumerates the IShellFolder, keeping things ordered correctly.
  441. //
  442. void CSFToolbar::_FillToolbar()
  443. {
  444. HDPA hdpaSort;
  445. HDPA hdpa;
  446. if (!_fDirty || !_psf)
  447. return;
  448. // If we have an order array, use that, otherwise
  449. // use the currently viewed items
  450. // remove the ref for the member variable since we can get reentered
  451. // this would be better with real refcounting but this'll do
  452. BOOL fTakeOrderRef = FALSE;
  453. if (_hdpaOrder)
  454. {
  455. hdpaSort = _hdpaOrder; // already sorted by name
  456. // we set it to null, so we have complete ownership of it.
  457. // at the end we're going to nuke _hdpaOrder anyway in _RememberOrder.
  458. _hdpaOrder = NULL;
  459. fTakeOrderRef = TRUE;
  460. }
  461. else
  462. {
  463. hdpaSort = _hdpa;
  464. _SortDPA(hdpaSort);
  465. }
  466. hdpa = DPA_Create(hdpaSort ? DPA_GetPtrCount(hdpaSort) : 12);
  467. if (hdpa)
  468. {
  469. _FillDPA(hdpa, hdpaSort, SHCONTF_FOLDERS|SHCONTF_NONFOLDERS);
  470. HDPA hdpaToRemove = DPA_Create(4);
  471. if (hdpaToRemove)
  472. {
  473. HDPA hdpaToAdd = DPA_Create(4);
  474. if (hdpaToAdd)
  475. {
  476. int i, j;
  477. BOOL fReleaseAdd = TRUE;
  478. if (_hdpa)
  479. {
  480. // if there isn't anything in the hdpaSort list (which is all the stuff that's already there),
  481. // add a null element. this is so it'll generate a "remove this null element" later.
  482. // otherwise, we might end up with an (Empty) item left over.
  483. if (DPA_GetPtrCount(hdpaSort) == 0)
  484. {
  485. OrderList_Append(hdpaSort, NULL, 0);
  486. }
  487. ORDERINFO oi = {0};
  488. oi.dwSortBy = OI_SORTBYNAME;
  489. oi.psf = _psf;
  490. _psf->AddRef();
  491. DPA_Sort(hdpaSort, OrderItem_Compare, (LPARAM) &oi);
  492. DPA_Sort(hdpa, OrderItem_Compare, (LPARAM) &oi);
  493. i = 0;
  494. j = 0;
  495. while ((i < DPA_GetPtrCount(hdpaSort)) && (j < DPA_GetPtrCount(hdpa)))
  496. {
  497. void *pv1 = DPA_FastGetPtr(hdpaSort, i);
  498. void *pv2 = DPA_FastGetPtr(hdpa, j);
  499. int nCmp = OrderItem_Compare(pv1, pv2, (LPARAM) &oi);
  500. if (nCmp > 0)
  501. {
  502. DPA_AppendPtr(hdpaToAdd, pv2);
  503. j++;
  504. }
  505. else if (nCmp < 0)
  506. {
  507. DPA_AppendPtr(hdpaToRemove, pv1);
  508. i++;
  509. }
  510. else
  511. {
  512. i++;
  513. j++;
  514. }
  515. }
  516. while (i < DPA_GetPtrCount(hdpaSort))
  517. {
  518. DPA_AppendPtr(hdpaToRemove, DPA_FastGetPtr(hdpaSort, i));
  519. i++;
  520. }
  521. while (j < DPA_GetPtrCount(hdpa))
  522. {
  523. DPA_AppendPtr(hdpaToAdd, DPA_FastGetPtr(hdpa, j));
  524. j++;
  525. }
  526. _psf->Release();
  527. }
  528. else
  529. {
  530. DPA_Destroy(hdpaToAdd);
  531. hdpaToAdd = hdpa;
  532. fReleaseAdd = FALSE;
  533. _hdpa = DPA_Create(DPA_GetPtrCount(hdpa));
  534. }
  535. SendMessage(_hwndTB, WM_SETREDRAW, FALSE, 0);
  536. if (_hdpa)
  537. {
  538. _NotifyBulkOperation(TRUE);
  539. // add buttons back in
  540. for (i = 0; i < DPA_GetPtrCount(hdpaToAdd); i++)
  541. {
  542. PORDERITEM poi = (PORDERITEM)DPA_FastGetPtr(hdpaToAdd, i);
  543. _OnFSNotifyAdd(poi->pidl, FSNA_BULKADD, poi->nOrder);
  544. }
  545. // remove buttons that die
  546. for (i = 0; i < DPA_GetPtrCount(hdpaToRemove); i++)
  547. {
  548. PORDERITEM poi = (PORDERITEM)DPA_FastGetPtr(hdpaToRemove, i);
  549. _OnFSNotifyRemove(poi->pidl);
  550. }
  551. _NotifyBulkOperation(FALSE);
  552. }
  553. if (fReleaseAdd)
  554. {
  555. DPA_Destroy(hdpaToAdd);
  556. }
  557. }
  558. DPA_Destroy(hdpaToRemove);
  559. }
  560. OrderList_Destroy(&hdpa);
  561. }
  562. SendMessage(_hwndTB, WM_SETREDRAW, TRUE, 0);
  563. if (fTakeOrderRef)
  564. {
  565. OrderList_Destroy(&hdpaSort);
  566. }
  567. _RememberOrder();
  568. _UpdateButtons();
  569. _SetDirty(FALSE);
  570. if (!SHIsTempDisplayMode())
  571. {
  572. _ToolbarChanged();
  573. }
  574. TraceMsg(TF_BAND, "SFToolbar::_FillToolbar found %d items", DPA_GetPtrCount(_hdpa));
  575. }
  576. void CSFToolbar::EmptyToolbar()
  577. {
  578. if (_hwndTB)
  579. {
  580. TraceMsg(TF_BAND, "SFToolbar::EmptyToolbar %d items", _hdpa ? DPA_GetPtrCount(_hdpa) : 0);
  581. while (InlineDeleteButton(0))
  582. {
  583. // delete the buttons
  584. }
  585. }
  586. OrderList_Destroy(&_hdpa);
  587. _fDirty = TRUE;
  588. _nNextCommandID = 0;
  589. }
  590. void CSFToolbar::_SetDirty(BOOL fDirty)
  591. {
  592. _fDirty = fDirty;
  593. }
  594. UINT CSFToolbar::_IndexToID(int iTBIndex)
  595. {
  596. TBBUTTON tbb;
  597. if (SendMessage(_hwndTB, TB_GETBUTTON, iTBIndex, (LPARAM)&tbb))
  598. {
  599. return tbb.idCommand;
  600. }
  601. return (UINT)-1;
  602. }
  603. // if ptbbi is specified, dwMask must be filled in
  604. //
  605. // if pIndex is specified, it receives the DPA index, not the toolbar index
  606. HRESULT CSFToolbar::_GetButtonFromPidl(LPCITEMIDLIST pidl, TBBUTTONINFO * ptbbi, int * pIndex, LPITEMIDLIST *ppidlOut)
  607. {
  608. if (ppidlOut)
  609. *ppidlOut = NULL;
  610. if (!_hdpa)
  611. return E_FAIL;
  612. for (int i = DPA_GetPtrCount(_hdpa) - 1 ; i >= 0 ; i--)
  613. {
  614. HRESULT hr;
  615. PORDERITEM poi = (PORDERITEM)DPA_FastGetPtr(_hdpa, i);
  616. ASSERT(poi);
  617. if (pidl == poi->pidl)
  618. {
  619. hr = 0;
  620. }
  621. else
  622. {
  623. hr = (!pidl || !poi->pidl) ? E_FAIL : _psf->CompareIDs(0, pidl, poi->pidl);
  624. }
  625. if (ResultFromShort(0) == hr)
  626. {
  627. if (pIndex)
  628. *pIndex = i;
  629. if (ptbbi)
  630. {
  631. int id = _IndexToID(v_DPAIndexToTBIndex(i));
  632. if (id != -1)
  633. {
  634. ptbbi->cbSize = sizeof(*ptbbi);
  635. if (-1 == ToolBar_GetButtonInfo(_hwndTB, id, ptbbi))
  636. {
  637. ZeroMemory(ptbbi, sizeof(*ptbbi));
  638. }
  639. }
  640. else
  641. {
  642. ZeroMemory(ptbbi, sizeof(*ptbbi));
  643. }
  644. }
  645. if (ppidlOut)
  646. *ppidlOut = poi->pidl;
  647. return S_OK;
  648. }
  649. }
  650. return E_FAIL;
  651. }
  652. // On an add, tack the new button on the end
  653. void CSFToolbar::_OnFSNotifyAdd(LPCITEMIDLIST pidl, DWORD dwFlags, int nIndex)
  654. {
  655. // be paranoid and make sure we don't duplicate an item
  656. //
  657. if ((dwFlags & FSNA_BULKADD) || FAILED(_GetButtonFromPidl(pidl, NULL, NULL, NULL)))
  658. {
  659. LPITEMIDLIST pidlNew;
  660. if (_fFSNotify && !_ptscn && pidl && !(dwFlags & FSNA_BULKADD))
  661. {
  662. if (FAILED(SHGetRealIDL(_psf, pidl, &pidlNew)))
  663. pidlNew = NULL;
  664. }
  665. else
  666. {
  667. pidlNew = pidl ? ILClone(pidl) : NULL;
  668. }
  669. if ((dwFlags & FSNA_BULKADD) || !_FilterPidl(pidlNew))
  670. {
  671. int index = (dwFlags & FSNA_ADDDEFAULT) ? _DefaultInsertIndex() : nIndex;
  672. if (_fDropping)
  673. {
  674. if (-1 == _tbim.iButton)
  675. index = 0; // if qlinks has no items, _tbim.iButton is -1, but you can't insert there...
  676. else if (_tbim.dwFlags & TBIMHT_AFTER)
  677. index = _tbim.iButton + 1;
  678. else
  679. index = _tbim.iButton;
  680. // We need to store this as the new order because a drag and drop has occured.
  681. // We will store this order and use it until the end of time.
  682. _fHasOrder = TRUE;
  683. _fChangedOrder = TRUE;
  684. }
  685. _AddPidl(pidlNew, dwFlags, index);
  686. if (!(dwFlags & FSNA_BULKADD))
  687. {
  688. OrderList_Reorder(_hdpa);
  689. }
  690. if (_fDropping)
  691. {
  692. _Dropped(index, FALSE);
  693. _fDropping = FALSE;
  694. }
  695. // NOTE: i'm nuking this call to SetDirty as it doesn't seem
  696. // necessary and we don't have a matching call to _SetDirty(FALSE);
  697. // mismatch of those calls causes nt5 bug #173868. [tjgreen 5-15-98]
  698. //_SetDirty(TRUE);
  699. }
  700. else
  701. {
  702. ILFree(pidlNew);
  703. }
  704. }
  705. }
  706. // This function syncronously removes the button, and deletes it's contents.
  707. // iTBIndex is a toolbar index, not a DPA index.
  708. // This avoids Reentrancy problems, as well as Leaks caused by unhooked toolbars
  709. BOOL_PTR CSFToolbar::InlineDeleteButton(int iTBIndex)
  710. {
  711. BOOL_PTR fRet = FALSE;
  712. TBBUTTONINFO tbbi = {0};
  713. tbbi.cbSize = sizeof(tbbi);
  714. tbbi.dwMask = TBIF_LPARAM | TBIF_BYINDEX;
  715. if (ToolBar_GetButtonInfo(_hwndTB, iTBIndex, &tbbi) >= 0)
  716. {
  717. PIBDATA pibdata = (PIBDATA)tbbi.lParam;
  718. tbbi.lParam = NULL;
  719. ToolBar_SetButtonInfo(_hwndTB, iTBIndex, &tbbi);
  720. fRet = SendMessage(_hwndTB, TB_DELETEBUTTON, iTBIndex, 0);
  721. if (pibdata)
  722. delete pibdata;
  723. }
  724. return fRet;
  725. }
  726. // On a remove, rip out the old button and adjust existing ones
  727. void CSFToolbar::_OnFSNotifyRemove(LPCITEMIDLIST pidl)
  728. {
  729. int i;
  730. LPITEMIDLIST pidlButton;
  731. if (SUCCEEDED(_GetButtonFromPidl(pidl, NULL, &i, &pidlButton)))
  732. {
  733. // remove it from the DPA before nuking the button. There is a rentrancy issue here.
  734. DPA_DeletePtr(_hdpa, i);
  735. InlineDeleteButton(v_DPAIndexToTBIndex(i));
  736. ILFree(pidlButton);
  737. _fChangedOrder = TRUE;
  738. }
  739. }
  740. // On a rename, just change the text of the old button
  741. //
  742. void CSFToolbar::_OnFSNotifyRename(LPCITEMIDLIST pidlFrom, LPCITEMIDLIST pidlTo)
  743. {
  744. TBBUTTONINFO tbbi;
  745. LPITEMIDLIST pidlButton;
  746. int i;
  747. tbbi.dwMask = TBIF_COMMAND | TBIF_LPARAM;
  748. if (SUCCEEDED(_GetButtonFromPidl(pidlFrom, &tbbi, &i, &pidlButton)))
  749. {
  750. LPITEMIDLIST pidlNew;
  751. if (_fFSNotify && !_ptscn)
  752. {
  753. if (FAILED(SHGetRealIDL(_psf, pidlTo, &pidlNew)))
  754. pidlNew = NULL;
  755. }
  756. else
  757. {
  758. pidlNew = ILClone(pidlTo);
  759. }
  760. if (pidlNew)
  761. {
  762. LPITEMIDLIST pidlFree = pidlNew;
  763. PORDERITEM poi = (PORDERITEM)DPA_FastGetPtr(_hdpa, i);
  764. if (EVAL(poi))
  765. {
  766. pidlFree = poi->pidl;
  767. poi->pidl = pidlNew;
  768. TCHAR szName[MAX_PATH];
  769. if (SUCCEEDED(DisplayNameOf(_psf, pidlNew, SHGDN_NORMAL, szName, ARRAYSIZE(szName))))
  770. {
  771. // _GetButtonFromPidl filled in tbbi.cbSize and tbbi.idCommand
  772. //
  773. PIBDATA pibdata = (PIBDATA)tbbi.lParam;
  774. if (pibdata)
  775. pibdata->SetOrderItem(poi);
  776. tbbi.dwMask = TBIF_TEXT;
  777. tbbi.pszText = szName;
  778. EVAL(ToolBar_SetButtonInfo(_hwndTB, tbbi.idCommand, &tbbi));
  779. // Just so that it's new location gets persisted
  780. _fChangedOrder = TRUE;
  781. // sync up the orderlist right now so if an updatedir comes in
  782. // it won't think the renamed pidl is new
  783. _RememberOrder();
  784. }
  785. }
  786. ILFree(pidlFree);
  787. }
  788. }
  789. }
  790. // On a complete update remove the old button and add it again
  791. //
  792. void CSFToolbar::_OnFSNotifyUpdate(LPCITEMIDLIST pidl)
  793. {
  794. TBBUTTONINFO tbbi;
  795. tbbi.dwMask = TBIF_COMMAND;
  796. LPITEMIDLIST pidlButton;
  797. if (SUCCEEDED(_GetButtonFromPidl(pidl, &tbbi, NULL, &pidlButton)))
  798. {
  799. TCHAR szName[MAX_PATH];
  800. if (SUCCEEDED(DisplayNameOf(_psf, pidlButton, SHGDN_NORMAL, szName, ARRAYSIZE(szName))))
  801. {
  802. int iBitmap = _GetBitmap(tbbi.idCommand, _IDToPibData(tbbi.idCommand, NULL), FALSE);
  803. if (iBitmap >= 0)
  804. {
  805. tbbi.dwMask = TBIF_IMAGE | TBIF_TEXT;
  806. tbbi.iImage = iBitmap;
  807. tbbi.pszText = szName;
  808. ToolBar_SetButtonInfo(_hwndTB, tbbi.idCommand, &tbbi);
  809. }
  810. }
  811. }
  812. }
  813. void CSFToolbar::_Refresh()
  814. {
  815. if (!_hdpa)
  816. return;
  817. _RememberOrder();
  818. EmptyToolbar();
  819. if (_fShow)
  820. {
  821. _FillToolbar();
  822. }
  823. }
  824. LRESULT CSFToolbar::_OnTimer(WPARAM wParam)
  825. {
  826. return 0;
  827. }
  828. LRESULT CSFToolbar::_DefWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  829. {
  830. switch (uMsg)
  831. {
  832. case WM_DRAWITEM:
  833. case WM_MEASUREITEM:
  834. case WM_INITMENUPOPUP:
  835. case WM_MENUSELECT:
  836. if (_pcm2)
  837. _pcm2->HandleMenuMsg(uMsg, wParam, lParam);
  838. break;
  839. case WM_MENUCHAR:
  840. {
  841. LRESULT lres = 0;
  842. IContextMenu3* pcm3;
  843. if (_pcm2 && SUCCEEDED(_pcm2->QueryInterface(IID_PPV_ARG(IContextMenu3, &pcm3))))
  844. {
  845. pcm3->HandleMenuMsg2(uMsg, wParam, lParam, &lres);
  846. pcm3->Release();
  847. }
  848. return lres;
  849. }
  850. break;
  851. case WM_TIMER:
  852. if (_OnTimer(wParam))
  853. {
  854. return 1;
  855. }
  856. break;
  857. }
  858. return CNotifySubclassWndProc::_DefWindowProc(hwnd, uMsg, wParam, lParam);
  859. }
  860. /*----------------------------------------------------------
  861. Purpose:
  862. For future use. when renaming a parent of this shell folder
  863. we should rebind to it and refill us.
  864. S_OK Indicates successful handling of this notification
  865. S_FALSE Indicates the notification is not a handled situation.
  866. The caller should handle the notification in this case.
  867. Other Failure code indicates a problem. Caller should abort
  868. operation or handle the notification itself.
  869. */
  870. HRESULT CSFToolbar::_OnRenameFolder(LPCITEMIDLIST pidl1, LPCITEMIDLIST pidl2)
  871. {
  872. HRESULT hr = S_FALSE;
  873. #if 0
  874. // This code is just busted. It was for the case when we rename the parent. We don't support this
  875. if (!_IsChildID(pidl1, FALSE) || !_IsChildID(pidl2, FALSE))
  876. {
  877. // Then this must be a parent of us. At this point we should rebind. The code that
  878. // was here did not work. I've removed it so that we can recode it in the future, but
  879. // now, we're just going to live with it
  880. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: RenameFolder: This is not a child folder.");
  881. hr = S_OK;
  882. }
  883. #endif
  884. return hr;
  885. }
  886. HRESULT CSFToolbar::OnChange(LONG lEvent, LPCITEMIDLIST pidlOrg1, LPCITEMIDLIST pidlOrg2)
  887. {
  888. HRESULT hr = S_OK;
  889. LPITEMIDLIST pidl1 = (LPITEMIDLIST)pidlOrg1;
  890. LPITEMIDLIST pidl2 = (LPITEMIDLIST)pidlOrg2;
  891. LPITEMIDLIST pidl1ToFree = NULL; // Used if we allocate a pidl that needs to be freed. (::TranslateIDs())
  892. LPITEMIDLIST pidl2ToFree = NULL;
  893. LPITEMIDLIST pidlOut1Event2 = NULL; // Used if we allocate a pidl that needs to be freed. (::TranslateIDs())
  894. LPITEMIDLIST pidlOut2Event2 = NULL;
  895. LONG lEvent2 = (LONG)-1;
  896. AddRef(); // This object could be released during this call
  897. if (_ptscn)
  898. {
  899. hr = _ptscn->TranslateIDs(&lEvent, pidlOrg1, pidlOrg2, &pidl1, &pidl2,
  900. &lEvent2, &pidlOut1Event2, &pidlOut2Event2);
  901. if (SUCCEEDED(hr))
  902. {
  903. // if pidl1 doesn't equal pidlOrg1, then pidl1 was allocated and needs to be freed.
  904. pidl1ToFree = ((pidlOrg1 == pidl1) ? NULL : pidl1);
  905. pidl2ToFree = ((pidlOrg2 == pidl2) ? NULL : pidl2);
  906. ASSERT(NULL == pidl1 || IS_VALID_PIDL(pidl1));
  907. ASSERT(NULL == pidl2 || IS_VALID_PIDL(pidl2));
  908. }
  909. }
  910. if (SUCCEEDED(hr))
  911. {
  912. hr = OnTranslatedChange(lEvent, pidl1, pidl2);
  913. // Do we have a second event to process?
  914. if (SUCCEEDED(hr) && lEvent2 != (LONG)-1)
  915. {
  916. // Yes, then go do it.
  917. hr = OnTranslatedChange(lEvent2, pidlOut1Event2, pidlOut2Event2);
  918. }
  919. ILFree(pidlOut1Event2);
  920. ILFree(pidlOut2Event2);
  921. ILFree(pidl1ToFree);
  922. ILFree(pidl2ToFree);
  923. }
  924. Release();
  925. return hr;
  926. }
  927. #ifdef DEBUG
  928. void DBPrPidl(LPCSTR szPre, LPCITEMIDLIST pidl)
  929. {
  930. TCHAR szName[MAX_PATH];
  931. szName[0] = '\0';
  932. if (pidl)
  933. SHGetNameAndFlags(pidl, SHGDN_FORPARSING, szName, SIZECHARS(szName), NULL);
  934. TraceMsg(TF_WARNING, "%hs%s", szPre, szName);
  935. return;
  936. }
  937. #endif
  938. HRESULT CSFToolbar::OnTranslatedChange(LONG lEvent, LPCITEMIDLIST pidl1, LPCITEMIDLIST pidl2)
  939. {
  940. HRESULT hr = S_OK;
  941. BOOL fSizeChanged = FALSE;
  942. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: lEvent=%x", lEvent);
  943. // If we weren't given a pidl we won't register for
  944. // SHChangeNotify calls, but our IShellChange interface
  945. // can still be QI()d so someone could errantly call us.
  946. //
  947. // If we change to using QS() for IShellChange interface
  948. // then we can put this check there...
  949. //
  950. if (NULL == _pidl)
  951. {
  952. // HACKHACK (scotth): resource-based menus (CMenuISF) don't set _pidl.
  953. // Right now allow SHCNE_UPDATEDIR thru...
  954. if (SHCNE_UPDATEDIR == lEvent)
  955. goto HandleUpdateDir;
  956. TraceMsg(TF_WARNING, "CSFToolbar::OnChange - _pidl is NULL");
  957. hr = E_FAIL;
  958. goto CleanUp;
  959. }
  960. if (lEvent & SHCNE_PIDL1ISCHILD)
  961. {
  962. // We only handle notifications for immediate kids. (except SHCNE_RENAMEFOLDER)
  963. //
  964. if (!_IsChildID(pidl1, TRUE))
  965. {
  966. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Not a child. Bailing");
  967. hr = E_FAIL;
  968. goto CleanUp;
  969. }
  970. }
  971. // This is no longer required. It also hinders us for several different notifications (Such as reorder notifies)
  972. #if 0
  973. // If we're hidden we do not want to respond to any change notify
  974. // messages, so we unregister the toolbar. We mark the toolbar as
  975. // dirty so the next time we are shown, we will reenumerate the filesystem.
  976. if (!_fShow && !_fDropping)
  977. {
  978. _SetDirty(TRUE);
  979. _UnregisterChangeNotify();
  980. hr = E_FAIL;
  981. goto CleanUp;
  982. }
  983. #endif
  984. // Have we been shown yet?
  985. if (_hdpa == NULL)
  986. {
  987. // No. Well, then punt this. We'll catch it on the first enum.
  988. hr = E_FAIL;
  989. goto CleanUp;
  990. }
  991. switch (lEvent)
  992. {
  993. case SHCNE_EXTENDED_EVENT:
  994. {
  995. SHChangeDWORDAsIDList UNALIGNED * pdwidl = (SHChangeDWORDAsIDList UNALIGNED *)pidl1;
  996. if (pdwidl->dwItem1 == SHCNEE_ORDERCHANGED)
  997. {
  998. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Reorder event");
  999. // Do this first so that we can say "We can handle it". This prevents the
  1000. // mnfolder code that works around a bug in some installers where they don't
  1001. // send a Create Folder before the create item in that folder. It causes an
  1002. // update dir...
  1003. if (!pidl2 || ILIsEqual(_pidl, pidl2))
  1004. {
  1005. // if this reorder came from us, blow it off
  1006. if (!SHChangeMenuWasSentByMe(this, pidl1))
  1007. {
  1008. // load new order stream
  1009. _LoadOrderStream();
  1010. // rebuild toolbar
  1011. _SetDirty(TRUE);
  1012. if (_fShow)
  1013. _FillToolbar();
  1014. }
  1015. hr = S_OK;
  1016. }
  1017. }
  1018. }
  1019. break;
  1020. case SHCNE_DRIVEADD:
  1021. case SHCNE_CREATE:
  1022. case SHCNE_MKDIR:
  1023. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Adding item");
  1024. pidl1 = ILFindLastID(pidl1);
  1025. _OnFSNotifyAdd(pidl1, FSNA_ADDDEFAULT, 0);
  1026. _RememberOrder();
  1027. fSizeChanged = TRUE;
  1028. break;
  1029. case SHCNE_DRIVEREMOVED:
  1030. case SHCNE_DELETE:
  1031. case SHCNE_RMDIR:
  1032. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Removing item");
  1033. pidl1 = ILFindLastID(pidl1);
  1034. _OnFSNotifyRemove(pidl1);
  1035. _RememberOrder();
  1036. fSizeChanged = TRUE;
  1037. break;
  1038. case SHCNE_RENAMEFOLDER:
  1039. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: RenameFolder");
  1040. // Break if notif is handled or if this is not for our kid.
  1041. //
  1042. hr = _OnRenameFolder(pidl1, pidl2);
  1043. if (S_OK == hr)
  1044. {
  1045. fSizeChanged = TRUE;
  1046. break;
  1047. }
  1048. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: RenameFolder Falling through to RenameItem");
  1049. // fall through
  1050. case SHCNE_RENAMEITEM:
  1051. {
  1052. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: RenameItem");
  1053. BOOL fOurKid1, fOurKid2;
  1054. LPCITEMIDLIST p1 = pidl1;
  1055. LPCITEMIDLIST p2 = pidl2;
  1056. pidl1 = ILFindLastID(pidl1);
  1057. pidl2 = ILFindLastID(pidl2);
  1058. // An item can be renamed out of this folder.
  1059. // Convert that into a remove.
  1060. //
  1061. fOurKid1 = _IsChildID(p1, TRUE);
  1062. fOurKid2 = _IsChildID(p2, TRUE);
  1063. if (fOurKid1 && fOurKid2)
  1064. {
  1065. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Rename: Both are children");
  1066. _OnFSNotifyRename(pidl1, pidl2);
  1067. fSizeChanged = TRUE;
  1068. hr = S_OK;
  1069. break;
  1070. }
  1071. else if (fOurKid1)
  1072. {
  1073. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Rename: Only one is a child. Removing pidl 1");
  1074. _OnFSNotifyRemove(pidl1);
  1075. fSizeChanged = TRUE;
  1076. break;
  1077. }
  1078. else if (fOurKid2)
  1079. {
  1080. // An item can be renamed into this folder.
  1081. // Convert that into an add.
  1082. //
  1083. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Rename: Only one is a child. Adding pidl2");
  1084. _OnFSNotifyAdd(pidl2, FSNA_ADDDEFAULT, 0);
  1085. fSizeChanged = TRUE;
  1086. break;
  1087. }
  1088. else
  1089. {
  1090. // (we get here for guys below us who we don't care about,
  1091. // and also for the fallthru from SHCNE_RENAMEFOLDER)
  1092. TraceMsg(TF_SFTBAR, "CSFTBar::OnTranslateChange: Rename: Not our children");
  1093. /*NOTHING*/
  1094. hr = E_FAIL;
  1095. }
  1096. break;
  1097. }
  1098. case SHCNE_MEDIAINSERTED:
  1099. case SHCNE_MEDIAREMOVED:
  1100. case SHCNE_NETUNSHARE:
  1101. if (_IsEqualID(pidl1))
  1102. goto HandleUpdateDir;
  1103. case SHCNE_NETSHARE:
  1104. case SHCNE_UPDATEITEM:
  1105. if (_IsChildID(pidl1, TRUE))
  1106. {
  1107. pidl1 = ILFindLastID(pidl1);
  1108. _OnFSNotifyUpdate(pidl1);
  1109. fSizeChanged = TRUE;
  1110. }
  1111. break;
  1112. case SHCNE_UPDATEDIR:
  1113. // in OnChange we picked off update dir notify and we didn't translate ids
  1114. // now we can use ILIsEqual -- translate ids won't translate pidls in case
  1115. // of update dir because it looks for immediate child of its, and fails when
  1116. // it receives its own pidl
  1117. // NOTE: When sftbar is registered recursivly, we only get the pidl of the
  1118. // top pane. It is forwarded down to the children. Since this is now a "Child"
  1119. // of the top pane, we check to see if this pidl is a child of that pidl, hence the
  1120. // ILIsParent(pidl1, _pidl)
  1121. // HACKHACK, HUGE HACK: normaly w/ update dir pidl2 is NULL but in start menu
  1122. // augmergeisf can change some other notify (e.g. rename folder) to update dir
  1123. // in which case pidl2 is not null and we have to see if it is our child to do the
  1124. // update (11/18/98) reljai
  1125. if (_IsEqualID(pidl1) || // Calling UpdateDir on _THIS_ folder
  1126. _IsChildID(pidl1, FALSE) || // FEATURE (lamadio) Is this needed?
  1127. _IsChildID(pidl2, FALSE) || // A changed to update (see comment)
  1128. _IsParentID(pidl1)) // Some parent in the chain (because it's recursive)
  1129. {
  1130. HandleUpdateDir:
  1131. // NOTE: if a series of UPDATEIMAGE notifies gets
  1132. // translated to UPDATEDIR and we flicker-perf
  1133. // _FillToolbar, we may lose image updates
  1134. // (in which case, _Refresh would fix it)
  1135. //
  1136. _SetDirty(TRUE);
  1137. _FillToolbar();
  1138. }
  1139. break;
  1140. case SHCNE_ASSOCCHANGED:
  1141. IEInvalidateImageList(); // We may need to use different icons.
  1142. _Refresh(); // full refresh for now.
  1143. break;
  1144. case SHCNE_UPDATEIMAGE: // global
  1145. if (pidl1)
  1146. {
  1147. int iImage = *(int UNALIGNED *)((BYTE *)pidl1 + 2);
  1148. IEInvalidateImageList(); // We may need to use different icons.
  1149. if (pidl2)
  1150. {
  1151. iImage = SHHandleUpdateImage(pidl2);
  1152. if (iImage == -1)
  1153. {
  1154. break;
  1155. }
  1156. }
  1157. if (iImage == -1 || TBHasImage(_hwndTB, iImage))
  1158. {
  1159. _UpdateIconSize(_uIconSize, TRUE);
  1160. _Refresh();
  1161. }
  1162. fSizeChanged = TRUE;
  1163. }
  1164. else
  1165. {
  1166. _Refresh();
  1167. }
  1168. break;
  1169. default:
  1170. hr = E_FAIL;
  1171. break;
  1172. }
  1173. if (fSizeChanged)
  1174. {
  1175. if (_hwndPager)
  1176. SendMessage(_hwndPager, PGMP_RECALCSIZE, (WPARAM) 0, (LPARAM) 0);
  1177. _ToolbarChanged();
  1178. }
  1179. CleanUp:
  1180. return hr;
  1181. }
  1182. BOOL TBHasImage(HWND hwnd, int iImageIndex)
  1183. {
  1184. BOOL fRefresh = FALSE;
  1185. for (int i = ToolBar_ButtonCount(hwnd) - 1 ; i >= 0 ; i--)
  1186. {
  1187. TBBUTTON tbb;
  1188. if (SendMessage(hwnd, TB_GETBUTTON, i, (LPARAM)&tbb))
  1189. {
  1190. if (tbb.iBitmap == iImageIndex)
  1191. {
  1192. fRefresh = TRUE;
  1193. break;
  1194. }
  1195. }
  1196. }
  1197. return fRefresh;
  1198. }
  1199. void CSFToolbar::_SetToolbarState()
  1200. {
  1201. SHSetWindowBits(_hwndTB, GWL_STYLE, TBSTYLE_LIST,
  1202. (_uIconSize != ISFBVIEWMODE_SMALLICONS || _fNoShowText) ? 0 : TBSTYLE_LIST);
  1203. }
  1204. int CSFToolbar::_DefaultInsertIndex()
  1205. {
  1206. return DA_LAST;
  1207. }
  1208. BOOL CSFToolbar::_IsParentID(LPCITEMIDLIST pidl)
  1209. {
  1210. // Is the pidl passed in a parent of one of the IDs in the namespace
  1211. // or the only one i've got?
  1212. if (_ptscn)
  1213. return S_OK == _ptscn->IsEqualID(NULL, pidl);
  1214. else
  1215. return ILIsParent(pidl, _pidl, FALSE);
  1216. }
  1217. BOOL CSFToolbar::_IsEqualID(LPCITEMIDLIST pidl)
  1218. {
  1219. if (_ptscn)
  1220. return S_OK == _ptscn->IsEqualID(pidl, NULL);
  1221. else
  1222. return ILIsEqual(_pidl, pidl);
  1223. }
  1224. BOOL CSFToolbar::_IsChildID(LPCITEMIDLIST pidlChild, BOOL fImmediate)
  1225. {
  1226. BOOL fRet = FALSE;
  1227. if (pidlChild)
  1228. {
  1229. if (_ptscn)
  1230. fRet = S_OK == _ptscn->IsChildID(pidlChild, fImmediate);
  1231. else
  1232. fRet = ILIsParent(_pidl, pidlChild, fImmediate);
  1233. }
  1234. return fRet;
  1235. }
  1236. void CSFToolbar::v_CalcWidth(int* pcxMin, int* pcxMax)
  1237. {
  1238. ASSERT(IS_VALID_WRITE_PTR(pcxMin, int));
  1239. ASSERT(IS_VALID_WRITE_PTR(pcxMax, int));
  1240. // Calculate a decent button width given current state
  1241. HIMAGELIST himl;
  1242. int cxMax = 0;
  1243. int cxMin = 0;
  1244. himl = (HIMAGELIST)SendMessage(_hwndTB, TB_GETIMAGELIST, 0, 0);
  1245. if (himl)
  1246. {
  1247. int cy;
  1248. // Start with the width of the button
  1249. ImageList_GetIconSize(himl, &cxMax, &cy);
  1250. // We want at least a bit of space around the icon
  1251. if (_uIconSize != ISFBVIEWMODE_SMALLICONS)
  1252. cxMax += 20;
  1253. else
  1254. cxMax += 4 * GetSystemMetrics(SM_CXEDGE);
  1255. }
  1256. // Add in any additional space needed
  1257. // Text takes up a bit more space
  1258. if (!_fNoShowText)
  1259. {
  1260. cxMax += 20;
  1261. // Horizontal text takes up a lot
  1262. // if we're smallicon with text (horizontal button)
  1263. // mode, use the minimized metric to mimic the taskbar
  1264. if (_uIconSize == ISFBVIEWMODE_SMALLICONS)
  1265. cxMax = GetSystemMetrics(SM_CXMINIMIZED);
  1266. }
  1267. *pcxMin = cxMin;
  1268. *pcxMax = cxMax;
  1269. }
  1270. // Adjust buttons based on current state.
  1271. //
  1272. void CSFToolbar::_UpdateButtons()
  1273. {
  1274. if (_hwndTB)
  1275. {
  1276. // set "list" (text on right) or not (text underneath)
  1277. // NOTE: list mode always displays some text, don't do it if no text
  1278. _SetToolbarState();
  1279. v_CalcWidth(&_cxMin, &_cxMax);
  1280. SendMessage(_hwndTB, TB_SETBUTTONWIDTH, 0, MAKELONG(_cxMin, _cxMax));
  1281. // We just changed the layout
  1282. //
  1283. SendMessage(_hwndTB, TB_AUTOSIZE, 0, 0);
  1284. if (_hwndPager)
  1285. {
  1286. LRESULT lButtonSize = SendMessage(_hwndTB, TB_GETBUTTONSIZE, 0, 0);
  1287. Pager_SetScrollInfo(_hwndPager, 50, 1, HIWORD(lButtonSize));
  1288. SendMessage(_hwndPager, PGMP_RECALCSIZE, (WPARAM) 0, (LPARAM) 0);
  1289. }
  1290. }
  1291. }
  1292. /*----------------------------------------------------------
  1293. Purpose: Helper function that calls IShellFolder::GetUIObjectOf().
  1294. Returns: pointer to the requested interface
  1295. NULL if failed
  1296. */
  1297. void *CSFToolbar::_GetUIObjectOfPidl(LPCITEMIDLIST pidl, REFIID riid)
  1298. {
  1299. LPCITEMIDLIST * apidl = &pidl;
  1300. void *pv;
  1301. if (FAILED(_psf->GetUIObjectOf(GetHWNDForUIObject(), 1, apidl, riid, 0, &pv)))
  1302. {
  1303. pv = NULL;
  1304. }
  1305. return pv;
  1306. }
  1307. INT_PTR CALLBACK CSFToolbar::_RenameDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
  1308. {
  1309. switch (uMsg)
  1310. {
  1311. case WM_INITDIALOG:
  1312. {
  1313. ASSERT(lParam);
  1314. SetWindowLongPtr(hDlg, DWLP_USER, lParam);
  1315. EnableWindow(GetDlgItem(hDlg, IDOK), FALSE);
  1316. // cross-lang platform support
  1317. SHSetDefaultDialogFont(hDlg, IDD_NAME);
  1318. HWND hwndEdit = GetDlgItem(hDlg, IDD_NAME);
  1319. SendMessage(hwndEdit, EM_LIMITTEXT, MAX_PATH - 1, 0);
  1320. TCHAR szText[MAX_PATH + 80];
  1321. TCHAR szTemplate[80];
  1322. HWND hwndLabel = GetDlgItem(hDlg, IDD_PROMPT);
  1323. GetWindowText(hwndLabel, szTemplate, ARRAYSIZE(szTemplate));
  1324. wnsprintf(szText, ARRAYSIZE(szText), szTemplate, lParam);
  1325. SetWindowText(hwndLabel, szText);
  1326. SetWindowText(hwndEdit, (LPTSTR)lParam);
  1327. break;
  1328. }
  1329. case WM_DESTROY:
  1330. SHRemoveDefaultDialogFont(hDlg);
  1331. return FALSE;
  1332. case WM_COMMAND:
  1333. switch (GET_WM_COMMAND_ID(wParam, lParam))
  1334. {
  1335. case IDD_NAME:
  1336. {
  1337. if (GET_WM_COMMAND_CMD(wParam, lParam) == EN_UPDATE)
  1338. {
  1339. LPTSTR lpstrName = (LPTSTR) GetWindowLongPtr(hDlg, DWLP_USER);
  1340. EnableOKButtonFromID(hDlg, IDD_NAME);
  1341. GetDlgItemText(hDlg, IDD_NAME, lpstrName, MAX_PATH);
  1342. }
  1343. break;
  1344. }
  1345. case IDOK:
  1346. {
  1347. TCHAR pszTmp[MAX_PATH];
  1348. StrCpy(pszTmp, (LPTSTR) GetWindowLongPtr(hDlg, DWLP_USER));
  1349. if (PathCleanupSpec(NULL,pszTmp))
  1350. {
  1351. HWND hwnd;
  1352. ShellMessageBox(HINST_THISDLL, hDlg,
  1353. MAKEINTRESOURCE(IDS_FAVS_INVALIDFN),
  1354. MAKEINTRESOURCE(IDS_FAVS_ADDTOFAVORITES), MB_OK | MB_ICONHAND);
  1355. hwnd = GetDlgItem(hDlg, IDD_NAME);
  1356. SetWindowText(hwnd, TEXT("\0"));
  1357. EnableWindow(GetDlgItem(hDlg, IDOK), FALSE);
  1358. SetFocus(hwnd);
  1359. break;
  1360. }
  1361. }
  1362. // fall through
  1363. case IDCANCEL:
  1364. EndDialog(hDlg, GET_WM_COMMAND_ID(wParam, lParam));
  1365. break;
  1366. default:
  1367. return FALSE;
  1368. }
  1369. break;
  1370. default:
  1371. return FALSE;
  1372. }
  1373. return TRUE;
  1374. }
  1375. // This window proc is used for a temporary worker window that is used to position dialogs
  1376. // as well as maintain the correct Z-Order
  1377. // NOTE: This is used in mnfolder as well.
  1378. LRESULT CALLBACK HiddenWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  1379. {
  1380. switch(uMsg)
  1381. {
  1382. // Make sure activation tracks back to the parent.
  1383. case WM_ACTIVATE:
  1384. {
  1385. if (WA_ACTIVE != LOWORD(wParam))
  1386. goto DefWnd;
  1387. SetActiveWindow(GetParent(hwnd));
  1388. return FALSE;
  1389. }
  1390. case WM_WINDOWPOSCHANGING:
  1391. {
  1392. WINDOWPOS* pwp = (WINDOWPOS*)lParam;
  1393. pwp->flags |= SWP_NOOWNERZORDER;
  1394. }
  1395. break;
  1396. }
  1397. DefWnd:
  1398. return DefWindowProc(hwnd, uMsg, wParam, lParam);
  1399. }
  1400. HWND CSFToolbar::CreateWorkerWindow()
  1401. {
  1402. if (!_hwndWorkerWindow)
  1403. {
  1404. _hwndWorkerWindow = SHCreateWorkerWindow(HiddenWndProc, GetHWNDForUIObject(), WS_EX_TOOLWINDOW /*| WS_EX_TOPMOST */, WS_POPUP, 0, _hwndTB);
  1405. }
  1406. return _hwndWorkerWindow;
  1407. }
  1408. HRESULT CSFToolbar::_OnRename(POINT *ppt, int id)
  1409. {
  1410. ASSERT(_psf);
  1411. TCHAR szName[MAX_PATH];
  1412. LPITEMIDLIST pidl = ILClone(_IDToPidl(id));
  1413. if (!pidl)
  1414. return E_OUTOFMEMORY;
  1415. _ObtainPIDLName(pidl, szName, ARRAYSIZE(szName));
  1416. // create a temp window so that placement of the dialog will be close to the point.
  1417. // do this so that we'll use USER's code to get placement correctly w/ respect to multimon and work area
  1418. _hwndWorkerWindow = CreateWorkerWindow();
  1419. SetWindowPos(_hwndWorkerWindow, NULL, ppt->x, ppt->y, 0, 0, SWP_NOZORDER | SWP_NOSIZE | SWP_NOACTIVATE);
  1420. // Now the horrible work of disabling our UI parent window so we can go modal.
  1421. // In an ideal world, we would pass our true parent window and USER will do
  1422. // all the work of modality, but we have to use our worker window thingie
  1423. // to get the dialog positioned correctly with respect to multimon,
  1424. // so we have to find the modal parent and disable him the hard way.
  1425. //
  1426. IUnknown *punkSite;
  1427. IUnknown *punkTLB;
  1428. // Doesn't matter what we SAFECAST "this" to; just pick something to keep the compiler happy
  1429. IUnknown_GetSite(SAFECAST(this, IWinEventHandler*), IID_PPV_ARG(IUnknown, &punkSite));
  1430. IUnknown_QueryService(punkSite, SID_STopLevelBrowser, IID_PPV_ARG(IUnknown, &punkTLB));
  1431. // Tell OLE to go modal
  1432. HRESULT hrModeless = IUnknown_EnableModless(punkTLB, FALSE);
  1433. // Tell USER to go modal
  1434. HWND hwndDisable;
  1435. IUnknown_GetWindow(punkTLB, &hwndDisable);
  1436. BOOL bPrevEnabled = FALSE;
  1437. while (hwndDisable && (GetWindowLong(hwndDisable, GWL_STYLE) & WS_CHILD))
  1438. hwndDisable = GetParent(hwndDisable);
  1439. if (hwndDisable)
  1440. bPrevEnabled = !EnableWindow(hwndDisable, FALSE); // return value of EnableWindow needs to be negated.
  1441. while (1)
  1442. {
  1443. if (DialogBoxParam(HINST_THISDLL, MAKEINTRESOURCE(DLG_ISFBANDRENAME), _hwndWorkerWindow, _RenameDlgProc, (LPARAM)szName) != IDOK)
  1444. break;
  1445. WCHAR wsz[MAX_PATH];
  1446. SHTCharToUnicode(szName, wsz, ARRAYSIZE(wsz));
  1447. // Must re-assert TOPMOSTness so SetNameOf UI will be visible.
  1448. // (We lose it when the user dismisses the dialog box above.)
  1449. // Curiously, the worker window is owned by the app's window, not the
  1450. // menu, so the worker window ends up fighting with the menu to see who is on top
  1451. SetWindowPos(_hwndWorkerWindow, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOOWNERZORDER | SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
  1452. if (SUCCEEDED(_psf->SetNameOf(_hwndWorkerWindow, pidl, wsz, 0, NULL)))
  1453. {
  1454. SHChangeNotifyHandleEvents();
  1455. _SaveOrderStream();
  1456. break;
  1457. }
  1458. }
  1459. // (must undo modality in reverse order)
  1460. // Tell USER to return to modeless (as appropriate)
  1461. if (hwndDisable)
  1462. EnableWindow(hwndDisable, bPrevEnabled);
  1463. // Tell OLE to return to modeless (as appropriate)
  1464. if (SUCCEEDED(hrModeless))
  1465. IUnknown_EnableModless(punkTLB, TRUE);
  1466. ATOMICRELEASE(punkTLB);
  1467. ATOMICRELEASE(punkSite);
  1468. ILFree(pidl);
  1469. return S_OK;
  1470. }
  1471. BOOL CSFToolbar::_UpdateIconSize(UINT uIconSize, BOOL fUpdateButtons)
  1472. {
  1473. BOOL fChanged = (_uIconSize != uIconSize);
  1474. _uIconSize = uIconSize;
  1475. TraceMsg(TF_BAND, "ISFBand::_UpdateIconSize going %hs", (_uIconSize == ISFBVIEWMODE_LARGEICONS ? "LARGE" : (_uIconSize == ISFBVIEWMODE_SMALLICONS ? "SMALL" : "LOGOS")));
  1476. if (_hwndTB)
  1477. {
  1478. ATOMICRELEASE(_piml);
  1479. if (!_fNoIcons)
  1480. {
  1481. int iImageList = (_uIconSize == ISFBVIEWMODE_LARGEICONS) ? SHIL_LARGE : SHIL_SYSSMALL;
  1482. SHGetImageList(iImageList, IID_PPV_ARG(IImageList, &_piml));
  1483. }
  1484. // sending a null himl is significant.. it means no image list
  1485. SendMessage(_hwndTB, TB_SETIMAGELIST, 0, (LPARAM)_piml);
  1486. if (fUpdateButtons)
  1487. _UpdateButtons();
  1488. }
  1489. return fChanged;
  1490. }
  1491. HMENU CSFToolbar::_GetContextMenu(IContextMenu* pcm, int* pid)
  1492. {
  1493. HMENU hmenu = CreatePopupMenu();
  1494. if (hmenu)
  1495. {
  1496. UINT fFlags = CMF_CANRENAME;
  1497. if (0 > GetKeyState(VK_SHIFT))
  1498. fFlags |= CMF_EXTENDEDVERBS;
  1499. pcm->QueryContextMenu(hmenu, 0, *pid, CONTEXTMENU_IDCMD_LAST, fFlags);
  1500. }
  1501. return hmenu;
  1502. }
  1503. void CSFToolbar::_OnDefaultContextCommand(int idCmd)
  1504. {
  1505. }
  1506. HRESULT CSFToolbar::_GetTopBrowserWindow(HWND* phwnd)
  1507. {
  1508. IUnknown * punkSite;
  1509. HRESULT hr = IUnknown_GetSite(SAFECAST(this, IWinEventHandler*), IID_PPV_ARG(IUnknown, &punkSite));
  1510. if (SUCCEEDED(hr))
  1511. {
  1512. hr = SHGetTopBrowserWindow(punkSite, phwnd);
  1513. punkSite->Release();
  1514. }
  1515. return hr;
  1516. }
  1517. HRESULT CSFToolbar::_OnOpen(int id, BOOL fExplore)
  1518. {
  1519. HRESULT hr = E_FAIL;
  1520. LPCITEMIDLIST pidl = _IDToPidl(id);
  1521. if (pidl)
  1522. {
  1523. IUnknown* punkSite;
  1524. hr = IUnknown_GetSite(SAFECAST(this, IWinEventHandler*), IID_PPV_ARG(IUnknown, &punkSite));
  1525. if (SUCCEEDED(hr))
  1526. {
  1527. DWORD dwFlags = SBSP_DEFBROWSER | SBSP_DEFMODE;
  1528. if (fExplore)
  1529. dwFlags |= SBSP_EXPLOREMODE;
  1530. hr = SHNavigateToFavorite(_psf, pidl, punkSite, dwFlags);
  1531. punkSite->Release();
  1532. }
  1533. }
  1534. return hr;
  1535. }
  1536. HRESULT CSFToolbar::_HandleSpecialCommand(IContextMenu* pcm, PPOINT ppt, int id, int idCmd)
  1537. {
  1538. TCHAR szCommandString[40];
  1539. HRESULT hr = ContextMenu_GetCommandStringVerb(pcm, idCmd, szCommandString, ARRAYSIZE(szCommandString));
  1540. if (SUCCEEDED(hr))
  1541. {
  1542. if (lstrcmpi(szCommandString, TEXT("rename")) == 0)
  1543. return _OnRename(ppt, id);
  1544. else if (lstrcmpi(szCommandString, TEXT("open")) == 0)
  1545. return _OnOpen(id, FALSE);
  1546. else if (lstrcmpi(szCommandString, TEXT("explore")) == 0)
  1547. return _OnOpen(id, TRUE);
  1548. }
  1549. return S_FALSE;
  1550. }
  1551. LRESULT CSFToolbar::_DoContextMenu(IContextMenu* pcm, LPPOINT ppt, int id, LPRECT prcExclude)
  1552. {
  1553. LRESULT lres = 0;
  1554. int idCmdFirst = CONTEXTMENU_IDCMD_FIRST;
  1555. HMENU hmContext = _GetContextMenu(pcm, &idCmdFirst);
  1556. if (hmContext)
  1557. {
  1558. int idCmd;
  1559. if (_hwndToolTips)
  1560. SendMessage(_hwndToolTips, TTM_ACTIVATE, FALSE, 0L);
  1561. TPMPARAMS tpm;
  1562. TPMPARAMS * ptpm = NULL;
  1563. if (prcExclude)
  1564. {
  1565. tpm.cbSize = sizeof(tpm);
  1566. tpm.rcExclude = *((LPRECT)prcExclude);
  1567. ptpm = &tpm;
  1568. }
  1569. idCmd = TrackPopupMenuEx(hmContext,
  1570. TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTALIGN,
  1571. ppt->x, ppt->y, _hwndTB, ptpm);
  1572. if (_hwndToolTips)
  1573. SendMessage(_hwndToolTips, TTM_ACTIVATE, TRUE, 0L);
  1574. if (idCmd)
  1575. {
  1576. if (idCmd < idCmdFirst)
  1577. {
  1578. _OnDefaultContextCommand(idCmd);
  1579. }
  1580. else
  1581. {
  1582. idCmd -= idCmdFirst;
  1583. if (_HandleSpecialCommand(pcm, ppt, id, idCmd) != S_OK)
  1584. {
  1585. _hwndWorkerWindow = CreateWorkerWindow();
  1586. SetWindowPos(_hwndWorkerWindow, NULL, ppt->x, ppt->y, 0, 0, SWP_NOZORDER | SWP_NOSIZE | SWP_NOACTIVATE);
  1587. CMINVOKECOMMANDINFO ici = {
  1588. sizeof(CMINVOKECOMMANDINFO),
  1589. 0,
  1590. _hwndWorkerWindow,
  1591. MAKEINTRESOURCEA(idCmd),
  1592. NULL, NULL,
  1593. SW_NORMAL,
  1594. };
  1595. pcm->InvokeCommand(&ici);
  1596. }
  1597. }
  1598. }
  1599. // if we get this far
  1600. // we need to return handled so that WM_CONTEXTMENU doesn't come through
  1601. lres = 1;
  1602. DestroyMenu(hmContext);
  1603. }
  1604. return lres;
  1605. }
  1606. LRESULT CSFToolbar::_OnContextMenu(WPARAM wParam, LPARAM lParam)
  1607. {
  1608. LRESULT lres = 0;
  1609. RECT rc;
  1610. LPRECT prcExclude = NULL;
  1611. POINT pt;
  1612. int i;
  1613. if (lParam != (LPARAM)-1)
  1614. {
  1615. pt.x = GET_X_LPARAM(lParam);
  1616. pt.y = GET_Y_LPARAM(lParam);
  1617. POINT pt2 = pt;
  1618. MapWindowPoints(HWND_DESKTOP, _hwndTB, &pt2, 1);
  1619. i = ToolBar_HitTest(_hwndTB, &pt2);
  1620. }
  1621. else
  1622. {
  1623. // keyboard context menu.
  1624. i = (int)SendMessage(_hwndTB, TB_GETHOTITEM, 0, 0);
  1625. if (i >= 0)
  1626. {
  1627. SendMessage(_hwndTB, TB_GETITEMRECT, i, (LPARAM)&rc);
  1628. MapWindowPoints(_hwndTB, HWND_DESKTOP, (LPPOINT)&rc, 2);
  1629. pt.x = rc.left;
  1630. pt.y = rc.bottom;
  1631. prcExclude = &rc;
  1632. }
  1633. }
  1634. TraceMsg(TF_BAND, "NM_RCLICK %d,%d = %d", pt.x, pt.y, i);
  1635. if (i >= 0)
  1636. {
  1637. UINT id = _IndexToID(i);
  1638. if (-1 != id)
  1639. {
  1640. LPCITEMIDLIST pidl = _IDToPidl(id, NULL);
  1641. if (pidl)
  1642. {
  1643. LPCONTEXTMENU pcm = (LPCONTEXTMENU)_GetUIObjectOfPidl(pidl, IID_IContextMenu);
  1644. if (pcm)
  1645. {
  1646. // grab pcm2 for owner draw support
  1647. pcm->QueryInterface(IID_PPV_ARG(IContextMenu2, &_pcm2));
  1648. ToolBar_MarkButton(_hwndTB, id, TRUE);
  1649. lres = _DoContextMenu(pcm, &pt, id, prcExclude);
  1650. ToolBar_MarkButton(_hwndTB, id, FALSE);
  1651. if (lres)
  1652. _FlushNotifyMessages(_hwndTB);
  1653. ATOMICRELEASE(_pcm2);
  1654. pcm->Release();
  1655. }
  1656. }
  1657. }
  1658. }
  1659. return lres;
  1660. }
  1661. LRESULT CSFToolbar::_OnCustomDraw(NMCUSTOMDRAW* pnmcd)
  1662. {
  1663. return CDRF_DODEFAULT;
  1664. }
  1665. void CSFToolbar::_OnDragBegin(int iItem, DWORD dwPreferredEffect)
  1666. {
  1667. LPCITEMIDLIST pidl = _IDToPidl(iItem, &_iDragSource);
  1668. ToolBar_SetHotItem(_hwndTB, _iDragSource);
  1669. if (_hwndTB && pidl)
  1670. DragDrop(_hwndTB, _psf, pidl, dwPreferredEffect, NULL);
  1671. _iDragSource = -1;
  1672. }
  1673. LRESULT CSFToolbar::_OnHotItemChange(NMTBHOTITEM * pnm)
  1674. {
  1675. LPNMTBHOTITEM lpnmhi = (LPNMTBHOTITEM)pnm;
  1676. if (_hwndPager && (lpnmhi->dwFlags & (HICF_ARROWKEYS | HICF_ACCELERATOR)))
  1677. {
  1678. int iOldPos, iNewPos;
  1679. RECT rc, rcPager;
  1680. int heightPager;
  1681. int iSelected = lpnmhi->idNew;
  1682. iOldPos = (int)SendMessage(_hwndPager, PGM_GETPOS, (WPARAM)0, (LPARAM)0);
  1683. iNewPos = iOldPos;
  1684. SendMessage(_hwndTB, TB_GETITEMRECT, (WPARAM)iSelected, (LPARAM)&rc);
  1685. if (rc.top < iOldPos)
  1686. {
  1687. iNewPos =rc.top;
  1688. }
  1689. GetClientRect(_hwndPager, &rcPager);
  1690. heightPager = RECTHEIGHT(rcPager);
  1691. if (rc.top >= iOldPos + heightPager)
  1692. {
  1693. iNewPos += (rc.bottom - (iOldPos + heightPager)) ;
  1694. }
  1695. if (iNewPos != iOldPos)
  1696. SendMessage(_hwndPager, PGM_SETPOS, (WPARAM)0, (LPARAM)iNewPos);
  1697. }
  1698. return 0;
  1699. }
  1700. void CSFToolbar::_OnToolTipsCreated(NMTOOLTIPSCREATED* pnm)
  1701. {
  1702. _hwndToolTips = pnm->hwndToolTips;
  1703. SHSetWindowBits(_hwndToolTips, GWL_STYLE, TTS_ALWAYSTIP | TTS_TOPMOST | TTS_NOPREFIX, TTS_ALWAYSTIP | TTS_TOPMOST | TTS_NOPREFIX);
  1704. // set the AutoPopTime (the duration of showing the tooltip) to a large value
  1705. SendMessage(_hwndToolTips, TTM_SETDELAYTIME, TTDT_AUTOPOP, (LPARAM)MAXSHORT);
  1706. }
  1707. LRESULT CSFToolbar::_OnNotify(LPNMHDR pnm)
  1708. {
  1709. LRESULT lres = 0;
  1710. //The following statement traps all pager control notification messages.
  1711. if ((pnm->code <= PGN_FIRST) && (pnm->code >= PGN_LAST))
  1712. {
  1713. return SendMessage(_hwndTB, WM_NOTIFY, (WPARAM)0, (LPARAM)pnm);
  1714. }
  1715. switch (pnm->code)
  1716. {
  1717. case TBN_DRAGOUT:
  1718. {
  1719. TBNOTIFY *ptbn = (TBNOTIFY*)pnm;
  1720. _OnDragBegin(ptbn->iItem, 0);
  1721. lres = 1;
  1722. break;
  1723. }
  1724. case TBN_HOTITEMCHANGE:
  1725. _OnHotItemChange((LPNMTBHOTITEM)pnm);
  1726. break;
  1727. case TBN_GETINFOTIP:
  1728. {
  1729. LPNMTBGETINFOTIP pnmTT = (LPNMTBGETINFOTIP)pnm;
  1730. UINT uiCmd = pnmTT->iItem;
  1731. DWORD dwFlags = _fNoShowText ? QITIPF_USENAME | QITIPF_LINKNOTARGET : QITIPF_LINKNOTARGET;
  1732. if (!GetInfoTipEx(_psf, dwFlags, _IDToPidl(uiCmd), pnmTT->pszText, pnmTT->cchTextMax))
  1733. {
  1734. TBBUTTONINFO tbbi;
  1735. tbbi.cbSize = sizeof(tbbi);
  1736. tbbi.dwMask = TBIF_TEXT;
  1737. tbbi.pszText = pnmTT->pszText;
  1738. tbbi.cchText = pnmTT->cchTextMax;
  1739. lres = (-1 != ToolBar_GetButtonInfo(_hwndTB, uiCmd, &tbbi));
  1740. }
  1741. break;
  1742. }
  1743. //WARNING: Right now I am calling the same function for both A and W version if this notification supports
  1744. // Strings then it needs to thunk. Right now its only used for image
  1745. case TBN_GETDISPINFOA:
  1746. _OnGetDispInfo(pnm, FALSE);
  1747. break;
  1748. case TBN_GETDISPINFOW:
  1749. _OnGetDispInfo(pnm, TRUE);
  1750. break;
  1751. case NM_TOOLTIPSCREATED:
  1752. _OnToolTipsCreated((NMTOOLTIPSCREATED*)pnm);
  1753. break;
  1754. case NM_RCLICK:
  1755. lres = _OnContextMenu(NULL, GetMessagePos());
  1756. break;
  1757. case NM_CUSTOMDRAW:
  1758. return _OnCustomDraw((NMCUSTOMDRAW*)pnm);
  1759. }
  1760. return lres;
  1761. }
  1762. DWORD CSFToolbar::_GetAttributesOfPidl(LPCITEMIDLIST pidl, DWORD dwAttribs)
  1763. {
  1764. if (FAILED(_psf->GetAttributesOf(1, &pidl, &dwAttribs)))
  1765. dwAttribs = 0;
  1766. return dwAttribs;
  1767. }
  1768. PIBDATA CSFToolbar::_PosToPibData(UINT iPos)
  1769. {
  1770. ASSERT(IsWindow(_hwndTB));
  1771. // Initialize to NULL in case the GetButton Fails.
  1772. TBBUTTON tbb = {0};
  1773. PIBDATA pibData = NULL;
  1774. if (ToolBar_GetButton(_hwndTB, iPos, &tbb))
  1775. {
  1776. pibData = (PIBDATA)tbb.dwData;
  1777. }
  1778. return pibData;
  1779. }
  1780. PIBDATA CSFToolbar::_IDToPibData(UINT uiCmd, int * piPos)
  1781. {
  1782. PIBDATA pibdata = NULL;
  1783. // Initialize to NULL in case the GetButtonInfo Fails.
  1784. TBBUTTONINFO tbbi = {0};
  1785. tbbi.cbSize = sizeof(tbbi);
  1786. tbbi.dwMask = TBIF_LPARAM;
  1787. int iPos = ToolBar_GetButtonInfo(_hwndTB, uiCmd, &tbbi);
  1788. if (iPos >= 0)
  1789. pibdata = (PIBDATA)tbbi.lParam;
  1790. if (piPos)
  1791. *piPos = iPos;
  1792. return pibdata;
  1793. }
  1794. LPCITEMIDLIST CSFToolbar::_IDToPidl(UINT uiCmd, int *piPos)
  1795. {
  1796. LPCITEMIDLIST pidl;
  1797. PIBDATA pibdata = _IDToPibData(uiCmd, piPos);
  1798. if (pibdata)
  1799. pidl = pibdata->GetPidl();
  1800. else
  1801. pidl = NULL;
  1802. return pidl;
  1803. }
  1804. /*----------------------------------------------------------
  1805. Purpose: IWinEventHandler::OnWinEvent method
  1806. Processes messages passed on from the bandsite.
  1807. */
  1808. HRESULT CSFToolbar::OnWinEvent(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *plres)
  1809. {
  1810. *plres = 0;
  1811. // We are addref'n here because during the course of the
  1812. // Context menu, the view could be changed which free's the menu.
  1813. // We will release after we're sure the this pointer is no longer needed.
  1814. AddRef();
  1815. switch (uMsg)
  1816. {
  1817. case WM_SYSCOLORCHANGE:
  1818. SendMessage(_hwndTB, uMsg, wParam, lParam);
  1819. InvalidateRect(_hwndTB, NULL, TRUE);
  1820. break;
  1821. case WM_PALETTECHANGED:
  1822. InvalidateRect(_hwndTB, NULL, FALSE);
  1823. SendMessage(_hwndTB, uMsg, wParam, lParam);
  1824. break;
  1825. case WM_COMMAND:
  1826. *plres = _OnCommand(wParam, lParam);
  1827. break;
  1828. case WM_NOTIFY:
  1829. *plres = _OnNotify((LPNMHDR)lParam);
  1830. break;
  1831. case WM_CONTEXTMENU:
  1832. *plres = _OnContextMenu(wParam, lParam);
  1833. break;
  1834. }
  1835. Release();
  1836. return S_OK;
  1837. }
  1838. // Map the information loaded (or ctor) into _psf, [_pidl]
  1839. //
  1840. HRESULT CSFToolbar::_AfterLoad()
  1841. {
  1842. HRESULT hr = S_OK;
  1843. // if we have a pidl then we need to get ready
  1844. // for notifications...
  1845. //
  1846. if (_pidl)
  1847. {
  1848. // pidls must be rooted off the desktop
  1849. //
  1850. _fFSNotify = TRUE;
  1851. // shortcut -- just specifying a pidl is good enough
  1852. //
  1853. if (!_psf)
  1854. {
  1855. _fPSFBandDesktop = TRUE;
  1856. hr = IEBindToObject(_pidl, &_psf);
  1857. }
  1858. }
  1859. return hr;
  1860. }
  1861. // IDropTarget implementation
  1862. HRESULT CSFToolbar::GetWindowsDDT(HWND * phwndLock, HWND * phwndScroll)
  1863. {
  1864. *phwndLock = _hwndTB;
  1865. *phwndScroll = _hwndTB;
  1866. return S_OK;
  1867. }
  1868. HRESULT CSFToolbar::HitTestDDT(UINT nEvent, LPPOINT ppt, DWORD_PTR * pdwId, DWORD *pdwEffect)
  1869. {
  1870. TBINSERTMARK tbim;
  1871. switch (nEvent)
  1872. {
  1873. case HTDDT_ENTER:
  1874. return S_OK;
  1875. case HTDDT_OVER:
  1876. {
  1877. int iButton = IBHT_BACKGROUND; // assume we hit the background
  1878. // if we're the source, this may be a move operation
  1879. //
  1880. *pdwEffect = (_iDragSource >= 0) ? DROPEFFECT_MOVE : DROPEFFECT_NONE;
  1881. if (!ToolBar_InsertMarkHitTest(_hwndTB, ppt, &tbim))
  1882. {
  1883. if (tbim.dwFlags & TBIMHT_BACKGROUND)
  1884. {
  1885. RECT rc;
  1886. GetClientRect(_hwndTB, &rc);
  1887. // are we outside the toolbar window entirely?
  1888. if (!PtInRect(&rc, *ppt))
  1889. {
  1890. // rebar already did the hittesting so we are on the rebar
  1891. // but not the toolbar => we are in the title part
  1892. if (!_AllowDropOnTitle())
  1893. {
  1894. // yes; don't allow drop here
  1895. iButton = IBHT_OUTSIDEWINDOW;
  1896. *pdwEffect = DROPEFFECT_NONE;
  1897. }
  1898. // set tbim.iButton to invalid value so we don't draw insert mark
  1899. tbim.iButton = -1;
  1900. }
  1901. }
  1902. else
  1903. {
  1904. // nope, we hit a real button
  1905. //
  1906. if (tbim.iButton == _iDragSource)
  1907. {
  1908. iButton = IBHT_SOURCE; // don't drop on the source button
  1909. }
  1910. else
  1911. {
  1912. iButton = tbim.iButton;
  1913. }
  1914. tbim.iButton = IBHT_BACKGROUND;
  1915. // we never force a move operation if we're on a real button
  1916. *pdwEffect = DROPEFFECT_NONE;
  1917. }
  1918. }
  1919. *pdwId = iButton;
  1920. }
  1921. break;
  1922. case HTDDT_LEAVE:
  1923. // Reset
  1924. tbim.iButton = IBHT_BACKGROUND;
  1925. tbim.dwFlags = 0;
  1926. break;
  1927. default:
  1928. return E_INVALIDARG;
  1929. }
  1930. // update ui
  1931. if (tbim.iButton != _tbim.iButton || tbim.dwFlags != _tbim.dwFlags)
  1932. {
  1933. if (ppt)
  1934. _tbim = tbim;
  1935. // for now I don't want to rely on non-filesystem IShellFolder
  1936. // implementations to call our OnChange method when a drop occurs,
  1937. // so don't even show the insert mark.
  1938. //
  1939. if (_fFSNotify || _iDragSource >= 0)
  1940. {
  1941. DAD_ShowDragImage(FALSE);
  1942. ToolBar_SetInsertMark(_hwndTB, &tbim);
  1943. DAD_ShowDragImage(TRUE);
  1944. }
  1945. }
  1946. return S_OK;
  1947. }
  1948. HRESULT CSFToolbar::GetObjectDDT(DWORD_PTR dwId, REFIID riid, void ** ppvObj)
  1949. {
  1950. HRESULT hr = E_NOINTERFACE;
  1951. *ppvObj = NULL;
  1952. if ((IBHT_SOURCE == dwId) || (IBHT_OUTSIDEWINDOW == dwId))
  1953. {
  1954. // do nothing
  1955. }
  1956. else if (IBHT_BACKGROUND == dwId)
  1957. {
  1958. // nash:41937: not sure how, but _psf can be NULL...
  1959. if (EVAL(_psf))
  1960. hr = _psf->CreateViewObject(_hwndTB, riid, ppvObj);
  1961. }
  1962. else
  1963. {
  1964. LPCITEMIDLIST pidl = _IDToPidl((UINT)dwId, NULL);
  1965. if (pidl)
  1966. {
  1967. *ppvObj = _GetUIObjectOfPidl(pidl, riid);
  1968. if (*ppvObj)
  1969. hr = S_OK;
  1970. }
  1971. }
  1972. //TraceMsg(TF_BAND, "SFToolbar::GetObject(%d) returns %x", dwId, hr);
  1973. return hr;
  1974. }
  1975. HRESULT CSFToolbar::_SaveOrderStream()
  1976. {
  1977. if (_fChangedOrder)
  1978. {
  1979. // Notify everyone that the order changed
  1980. SHSendChangeMenuNotify(this, SHCNEE_ORDERCHANGED, 0, _pidl);
  1981. _fChangedOrder = FALSE;
  1982. return S_OK;
  1983. }
  1984. else
  1985. return S_FALSE;
  1986. }
  1987. void CSFToolbar::_Dropped(int nIndex, BOOL fDroppedOnSource)
  1988. {
  1989. _fDropped = TRUE;
  1990. _fChangedOrder = TRUE;
  1991. // Save new order stream
  1992. _SaveOrderStream();
  1993. if (fDroppedOnSource)
  1994. _FlushNotifyMessages(_hwndTB);
  1995. }
  1996. /*----------------------------------------------------------
  1997. Purpose: CDelegateDropTarget::OnDropDDT
  1998. */
  1999. HRESULT CSFToolbar::OnDropDDT(IDropTarget *pdt, IDataObject *pdtobj, DWORD * pgrfKeyState, POINTL pt, DWORD *pdwEffect)
  2000. {
  2001. // Are we NOT the drag source?
  2002. if (_iDragSource == -1)
  2003. {
  2004. // No, we're not. Well, then the source may be the chevron menu
  2005. // representing the hidden items in this menu. Let's check
  2006. LPITEMIDLIST pidl;
  2007. if (SUCCEEDED(SHPidlFromDataObject2(pdtobj, &pidl)))
  2008. {
  2009. // We've got a pidl, Are we the parent? Do we have a button?
  2010. int iIndex;
  2011. if (ILIsParent(_pidl, pidl, TRUE) &&
  2012. SUCCEEDED(_GetButtonFromPidl(ILFindLastID(pidl), NULL, &iIndex, NULL)))
  2013. {
  2014. // We are the parent! Then let's copy that down and set it
  2015. // as the drag source so that down below we reorder.
  2016. _iDragSource = iIndex;
  2017. }
  2018. ILFree(pidl);
  2019. }
  2020. }
  2021. if (_iDragSource >= 0)
  2022. {
  2023. if (_fAllowReorder)
  2024. {
  2025. TraceMsg(TF_BAND, "SFToolbar::OnDrop reorder %d to %d %s", _iDragSource, _tbim.iButton, _tbim.dwFlags & TBIMHT_AFTER ? "A" : "B");
  2026. int iNewLocation = _tbim.iButton;
  2027. if (_tbim.dwFlags & TBIMHT_AFTER)
  2028. iNewLocation++;
  2029. if (iNewLocation > _iDragSource)
  2030. iNewLocation--;
  2031. if (ToolBar_MoveButton(_hwndTB, _iDragSource, iNewLocation))
  2032. {
  2033. PORDERITEM poi = (PORDERITEM)DPA_DeletePtr(_hdpa, v_TBIndexToDPAIndex(_iDragSource));
  2034. if (poi)
  2035. {
  2036. DPA_InsertPtr(_hdpa, v_TBIndexToDPAIndex(iNewLocation), poi);
  2037. OrderList_Reorder(_hdpa);
  2038. // If we're dropping again, then we don't need the _hdpaOrder...
  2039. OrderList_Destroy(&_hdpaOrder);
  2040. // A reorder has occurred. We need to use the order stream as the order...
  2041. _fHasOrder = TRUE;
  2042. _fDropping = TRUE;
  2043. _Dropped(iNewLocation, TRUE);
  2044. _fDropping = FALSE;
  2045. _RememberOrder();
  2046. _SetDirty(TRUE);
  2047. }
  2048. }
  2049. }
  2050. // Don't forget to reset this!
  2051. _iDragSource = -1;
  2052. DragLeave();
  2053. }
  2054. else
  2055. {
  2056. // We want to override the default to be LINK (SHIFT+CONTROL)
  2057. if (0 == DataObj_GetDWORD(pdtobj, g_cfPreferredDropEffect, 0))
  2058. {
  2059. if (!(*pgrfKeyState & (MK_CONTROL | MK_SHIFT | MK_ALT)))
  2060. {
  2061. // NOTE: not all data objects will allow us to call SetData()
  2062. DataObj_SetDWORD(pdtobj, g_cfPreferredDropEffect, DROPEFFECT_LINK);
  2063. }
  2064. }
  2065. _fDropping = TRUE;
  2066. return S_OK;
  2067. }
  2068. return S_FALSE;
  2069. }
  2070. void CSFToolbar::_SortDPA(HDPA hdpa)
  2071. {
  2072. // If we don't have a _psf, then we certainly can't sort it
  2073. // If we don't have a hdpa, then we certainly can't sort it
  2074. // If the hdpa is empty, then there's no point in sorting it
  2075. if (_psf && hdpa && DPA_GetPtrCount(hdpa))
  2076. {
  2077. ORDERINFO oinfo;
  2078. oinfo.psf = _psf;
  2079. oinfo.psf->AddRef();
  2080. oinfo.dwSortBy = (_fNoNameSort ? OI_SORTBYORDINAL : OI_SORTBYNAME);
  2081. DPA_Sort(hdpa, OrderItem_Compare, (LPARAM)&oinfo);
  2082. oinfo.psf->Release();
  2083. }
  2084. }
  2085. void CSFToolbar::_RememberOrder()
  2086. {
  2087. OrderList_Destroy(&_hdpaOrder);
  2088. if (_hdpa)
  2089. {
  2090. _hdpaOrder = OrderList_Clone(_hdpa);
  2091. _SortDPA(_hdpaOrder);
  2092. }
  2093. }
  2094. HMENU CSFToolbar::_GetBaseContextMenu()
  2095. {
  2096. HMENU hmenu = SHLoadMenuPopup(HINST_THISDLL, MENU_ISFBAND);
  2097. // no logo view, remove the menu item...
  2098. HMENU hView = GetSubMenu(hmenu, 0);
  2099. DeleteMenu(hView, ISFBIDM_LOGOS, MF_BYCOMMAND);
  2100. return hmenu;
  2101. }
  2102. HMENU CSFToolbar::_GetContextMenu()
  2103. {
  2104. HMENU hmenuSrc = _GetBaseContextMenu();
  2105. if (hmenuSrc)
  2106. {
  2107. MENUITEMINFO mii;
  2108. mii.cbSize = sizeof(mii);
  2109. mii.fMask = MIIM_STATE;
  2110. mii.fState = MF_CHECKED;
  2111. UINT uCmdId = ISFBIDM_LOGOS;
  2112. if (_uIconSize != ISFBVIEWMODE_LOGOS)
  2113. uCmdId = (_uIconSize == ISFBVIEWMODE_LARGEICONS ? ISFBIDM_LARGE : ISFBIDM_SMALL);
  2114. SetMenuItemInfo(hmenuSrc, uCmdId, MF_BYCOMMAND, &mii);
  2115. if (!_fNoShowText)
  2116. SetMenuItemInfo(hmenuSrc, ISFBIDM_SHOWTEXT, MF_BYCOMMAND, &mii);
  2117. if (!_fFSNotify || !_pidl || ILIsEmpty(_pidl))
  2118. DeleteMenu(hmenuSrc, ISFBIDM_OPEN, MF_BYCOMMAND);
  2119. HMENU hView = GetSubMenu(hmenuSrc, 0);
  2120. DeleteMenu(hView, ISFBIDM_LOGOS, MF_BYCOMMAND);
  2121. }
  2122. return hmenuSrc;
  2123. }
  2124. // IContextMenu implementation
  2125. //
  2126. HRESULT CSFToolbar::QueryContextMenu(HMENU hmenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
  2127. {
  2128. HMENU hmenuSrc = _GetContextMenu();
  2129. int i = 0;
  2130. if (hmenuSrc)
  2131. {
  2132. i += Shell_MergeMenus(hmenu, hmenuSrc, indexMenu, idCmdFirst, idCmdLast, 0);
  2133. DestroyMenu(hmenuSrc);
  2134. }
  2135. if (!_pcmSF && _fAllowRename && _psf)
  2136. {
  2137. _psf->CreateViewObject(_hwndTB, IID_PPV_ARG(IContextMenu, &_pcmSF));
  2138. }
  2139. if (_pcmSF)
  2140. {
  2141. _idCmdSF = i - idCmdFirst;
  2142. HRESULT hrT = _pcmSF->QueryContextMenu(hmenu, indexMenu + i, i, 0x7fff, CMF_BANDCMD);
  2143. if (SUCCEEDED(hrT))
  2144. i += HRESULT_CODE(hrT);
  2145. }
  2146. return i;
  2147. }
  2148. BOOL CSFToolbar::_UpdateShowText(BOOL fNoShowText)
  2149. {
  2150. BOOL fChanged = (!_fNoShowText != !fNoShowText);
  2151. _fNoShowText = (fNoShowText != 0);
  2152. TraceMsg(TF_BAND, "ISFBand::_UpdateShowText turning text %hs", _fNoShowText ? "OFF" : "ON");
  2153. if (_hwndTB)
  2154. {
  2155. SendMessage(_hwndTB, TB_SETMAXTEXTROWS, _fNoShowText ? 0 : 1, 0L);
  2156. _UpdateButtons();
  2157. }
  2158. return fChanged;
  2159. }
  2160. HRESULT CSFToolbar::InvokeCommand(LPCMINVOKECOMMANDINFO lpici)
  2161. {
  2162. BOOL fChanged = FALSE;
  2163. int idCmd = -1;
  2164. if (!HIWORD(lpici->lpVerb))
  2165. idCmd = LOWORD(lpici->lpVerb);
  2166. switch (idCmd)
  2167. {
  2168. case ISFBIDM_REFRESH:
  2169. _Refresh();
  2170. break;
  2171. case ISFBIDM_OPEN:
  2172. OpenFolderPidl(_pidl);
  2173. break;
  2174. case ISFBIDM_LARGE:
  2175. fChanged = _UpdateIconSize(ISFBVIEWMODE_LARGEICONS, TRUE);
  2176. break;
  2177. case ISFBIDM_SMALL:
  2178. fChanged = _UpdateIconSize(ISFBVIEWMODE_SMALLICONS, TRUE);
  2179. break;
  2180. case ISFBIDM_SHOWTEXT:
  2181. fChanged = _UpdateShowText(!_fNoShowText);
  2182. break;
  2183. default:
  2184. if (_pcmSF && idCmd >= _idCmdSF)
  2185. {
  2186. LPCSTR lpOldVerb = lpici->lpVerb;
  2187. lpici->lpVerb = MAKEINTRESOURCEA(idCmd -= _idCmdSF);
  2188. _pcmSF->InvokeCommand(lpici);
  2189. _FlushNotifyMessages(_hwndTB);
  2190. lpici->lpVerb = lpOldVerb;
  2191. }
  2192. else
  2193. TraceMsg(TF_BAND, "SFToolbar::InvokeCommand %d not handled", idCmd);
  2194. break;
  2195. }
  2196. // Our minimum sizes have changed, notify the bandsite
  2197. //
  2198. if (fChanged)
  2199. _ToolbarChanged();
  2200. return S_OK;
  2201. }
  2202. HRESULT CSFToolbar::GetCommandString(UINT_PTR idCmd, UINT uType, UINT *pwReserved, LPSTR pszName, UINT cchMax)
  2203. {
  2204. return E_NOTIMPL;
  2205. }
  2206. void CSFToolbar::_RegisterToolbar()
  2207. {
  2208. // Since _SubclassWindow protects against multiply subclassing,
  2209. // This call is safe, and ensures that the toolbar is subclassed before
  2210. // even trying to register it for change notify.
  2211. if (_hwndTB && _fRegisterChangeNotify)
  2212. _RegisterChangeNotify();
  2213. CDelegateDropTarget::Init();
  2214. }
  2215. void CSFToolbar::_UnregisterToolbar()
  2216. {
  2217. if (_hwndTB)
  2218. {
  2219. if (_fRegisterChangeNotify)
  2220. _UnregisterChangeNotify();
  2221. _UnsubclassWindow(_hwndTB);
  2222. }
  2223. }
  2224. void CSFToolbar::_RegisterChangeNotify()
  2225. {
  2226. // Since we want to register for change notify ONLY once,
  2227. // and only if this is a file system toolbar.
  2228. if (!_fFSNRegistered && _fFSNotify)
  2229. {
  2230. if (_ptscn)
  2231. _ptscn->Register(_hwndTB, g_idFSNotify, _lEvents);
  2232. else
  2233. _RegisterWindow(_hwndTB, _pidl, _lEvents);
  2234. _fFSNRegistered = TRUE;
  2235. }
  2236. }
  2237. void CSFToolbar::_UnregisterChangeNotify()
  2238. {
  2239. // Only unregister if we have been registered.
  2240. if (_hwndTB && _fFSNRegistered && _fFSNotify)
  2241. {
  2242. _fFSNRegistered = FALSE;
  2243. if (_ptscn)
  2244. _ptscn->Unregister();
  2245. else
  2246. _UnregisterWindow(_hwndTB);
  2247. }
  2248. }
  2249. void CSFToolbar::_ReleaseShellFolder()
  2250. {
  2251. if (_psf)
  2252. {
  2253. IUnknown_SetOwner(_psf, NULL);
  2254. ATOMICRELEASE(_psf);
  2255. }
  2256. ATOMICRELEASE(_ptscn);
  2257. }
  2258. // IWinEventHandler::IsWindowOwner
  2259. HRESULT CSFToolbar::IsWindowOwner(HWND hwnd)
  2260. {
  2261. if (hwnd == _hwndTB ||
  2262. hwnd == _hwndToolTips ||
  2263. hwnd == _hwndPager)
  2264. return S_OK;
  2265. return S_FALSE;
  2266. }