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.

810 lines
22 KiB

  1. #include "priv.h"
  2. #include "sccls.h"
  3. #include "resource.h"
  4. #include "mshtmhst.h"
  5. #include "deskbar.h"
  6. #include "bands.h"
  7. #define WANT_CBANDSITE_CLASS
  8. #include "bandsite.h"
  9. #include <trayp.h> // TM_*
  10. #include <desktray.h> // IDeskTray
  11. #include "dbapp.h"
  12. #include "mluisupp.h"
  13. /*
  14. this virtual app implments DeskBars that you have on the desktop.
  15. it has the glue that combines CDeskBar with CBandSite and populates the
  16. bands (as well as persistance and such)
  17. -Chee
  18. */
  19. #define DM_INIT 0
  20. #define DM_PERSIST 0 // trace IPS::Load, ::Save, etc.
  21. #define DM_MENU 0 // menu code
  22. #define DM_DRAG 0 // drag&drop
  23. #define DM_TRAY 0 // tray: marshal, side, etc.
  24. #ifdef DEBUG
  25. extern unsigned long DbStreamTell(IStream *pstm);
  26. #else
  27. #define DbStreamTell(pstm) ((ULONG) 0)
  28. #endif
  29. #define SUPERCLASS CDeskBar
  30. /*
  31. Instead of just 4 Deskbars on the whole desktop, we now have 4 deskbars for
  32. each monitor, however, this brings problem whenever a monitor goes away, we
  33. need to clean up the following datastructure.
  34. - dli
  35. */
  36. // FEATURE: (dli) maybe this should be moved into multimon.h
  37. // however, people should not get into the habbit of depending on this.
  38. // and it's really not used anywhere else, so, keep it here for now.
  39. #define DSA_MONITORSGROW 1
  40. typedef struct DeskBarsPerMonitor {
  41. HMONITOR hMon;
  42. IDeskBar* Deskbars[4];
  43. } DESKBARSPERMONITOR, *LPDESKBARSPERMONITOR;
  44. HDSA g_hdsaDeskBars = NULL;
  45. enum ips_e {
  46. IPS_FALSE, // reserved, must be 0 (FALSE)
  47. IPS_LOAD,
  48. IPS_INITNEW
  49. };
  50. CASSERT(IPS_FALSE == 0);
  51. CDeskBarApp::~CDeskBarApp()
  52. {
  53. _LeaveSide();
  54. if (_pbs)
  55. _pbs->Release();
  56. if (_pcm)
  57. _pcm->Release();
  58. }
  59. LRESULT CDeskBarApp::v_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  60. {
  61. LRESULT lres = SUPERCLASS::v_WndProc(hwnd, uMsg, wParam, lParam);
  62. if (!_hwnd) {
  63. return lres; // destroyed by superclass
  64. }
  65. if (_eMode == WBM_BFLOATING) {
  66. switch (uMsg) {
  67. case WM_NOTIFY:
  68. {
  69. //
  70. // override the hittest value to be HTCAPTION if we're docked browser based
  71. //
  72. NMHDR* pnm = (NMHDR*)lParam;
  73. if (pnm->code == NM_NCHITTEST &&
  74. pnm->hwndFrom == _hwndChild) {
  75. //
  76. // in the floating bug docked int he browser, we don't do
  77. // mdi child stuff, so we make the gripper work as the caption
  78. //
  79. NMMOUSE* pnmm = (NMMOUSE*)pnm;
  80. if (pnmm->dwHitInfo == RBHT_CAPTION ||
  81. pnmm->dwHitInfo == RBHT_GRABBER)
  82. lres = HTTRANSPARENT;
  83. }
  84. }
  85. break;
  86. case WM_NCHITTEST:
  87. // all "client" areas are captions in this mode
  88. if (lres == HTCLIENT)
  89. lres = HTCAPTION;
  90. break;
  91. case WM_SETCURSOR:
  92. DefWindowProcWrap(hwnd, uMsg, wParam, lParam);
  93. return TRUE;
  94. }
  95. }
  96. return lres;
  97. }
  98. BOOL CDeskBarApp::_OnCloseBar(BOOL fConfirm)
  99. {
  100. // if we are closing a bar with no bands in it, don't pop up the dialog
  101. if ((_pbs && (_pbs->EnumBands(-1,NULL)==0)) ||
  102. (!fConfirm || ConfirmRemoveBand(_hwnd, IDS_CONFIRMCLOSEBAR, TEXT(""))) )
  103. return SUPERCLASS::_OnCloseBar(FALSE);
  104. return FALSE;
  105. }
  106. // Gets the Deskbars on a specific monitor
  107. // DBPM -- DeskBars Per Monitor
  108. LPDESKBARSPERMONITOR GetDBPMWithMonitor(HMONITOR hMon, BOOL fCreate)
  109. {
  110. int ihdsa;
  111. LPDESKBARSPERMONITOR pdbpm;
  112. if (!g_hdsaDeskBars) {
  113. if (fCreate)
  114. g_hdsaDeskBars = DSA_Create(SIZEOF(DESKBARSPERMONITOR), DSA_MONITORSGROW);
  115. }
  116. if (!g_hdsaDeskBars)
  117. return NULL;
  118. // If we find the DBPM with this HMONITOR, return it.
  119. for (ihdsa = 0; ihdsa < DSA_GetItemCount(g_hdsaDeskBars); ihdsa++) {
  120. pdbpm = (LPDESKBARSPERMONITOR)DSA_GetItemPtr(g_hdsaDeskBars, ihdsa);
  121. if (pdbpm->hMon == hMon)
  122. return pdbpm;
  123. }
  124. if (fCreate) {
  125. DESKBARSPERMONITOR dbpm = {0};
  126. // This monitor is not setup, so set it, and set us the
  127. // the ownder of _uSide
  128. dbpm.hMon = hMon;
  129. ihdsa = DSA_AppendItem(g_hdsaDeskBars, &dbpm);
  130. pdbpm = (LPDESKBARSPERMONITOR)DSA_GetItemPtr(g_hdsaDeskBars, ihdsa);
  131. return pdbpm;
  132. }
  133. // When all else fails, return NULL
  134. return NULL;
  135. }
  136. void CDeskBarApp::_LeaveSide()
  137. {
  138. if (ISABE_DOCK(_uSide) && !ISWBM_FLOAT(_eMode)) {
  139. // remove ourselves from the array list of where we were
  140. LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(_hMon, FALSE);
  141. if (pdbpm && (pdbpm->Deskbars[_uSide] == this)) {
  142. ASSERT(pdbpm->hMon);
  143. ASSERT(pdbpm->hMon == _hMon);
  144. pdbpm->Deskbars[_uSide] = NULL;
  145. }
  146. }
  147. }
  148. //***
  149. // NOTES
  150. // FEATURE: should we create/use IDeskTray::AppBarGetState?
  151. UINT GetTraySide(HMONITOR * phMon)
  152. {
  153. LRESULT lTmp;
  154. APPBARDATA abd;
  155. abd.cbSize = sizeof(APPBARDATA);
  156. abd.hWnd = GetTrayWindow();
  157. if (phMon)
  158. Tray_GetHMonitor(abd.hWnd, phMon);
  159. abd.uEdge = (UINT)-1;
  160. //lTmp = g_pdtray->AppBarGetTaskBarPos(&abd);
  161. lTmp = SHAppBarMessage(ABM_GETTASKBARPOS, &abd);
  162. ASSERT(lTmp);
  163. TraceMsg(DM_TRAY, "gts: ret=ABE_%d", abd.uEdge);
  164. return abd.uEdge;
  165. }
  166. //***
  167. // ENTRY/EXIT
  168. // fNoMerge is for the IPS::Load case
  169. // NOTES
  170. // warning: be careful of reentrancy! fNoMove is how we guard against it.
  171. void CDeskBarApp::_SetModeSide(UINT eMode, UINT uSide, HMONITOR hMonNew, BOOL fNoMerge)
  172. {
  173. BOOL fNoMove;
  174. // make sure we don't merge etc. on NOOP moves.
  175. // we do such moves to force refresh (e.g. for autohide and IPS::Load);
  176. // also happens w/ drags which end up back where they started
  177. fNoMove = (eMode == _eMode && uSide == _uSide && hMonNew == _hMon);
  178. if (!fNoMove)
  179. _LeaveSide();
  180. // warning: this may call (e.g.) AppBarRegister, which causes a
  181. // resize, which calls back to us. careful of reentrancy!!!
  182. // if we do reenter we end up w/ nt5:155043, where entry #1 has
  183. // fNoMove==0, then we get a recalc, entry #2 has fNoMove==1,
  184. // and we set our side array to us, then return back to entry
  185. // #1 which merges into itself!
  186. SUPERCLASS::_SetModeSide(eMode, uSide, hMonNew, fNoMerge);
  187. if (!fNoMove) {
  188. if (ISABE_DOCK(_uSide) && !ISWBM_FLOAT(_eMode)) {
  189. LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMonNew, TRUE);
  190. HMONITOR hMonTray = NULL;
  191. if (pdbpm) {
  192. if (fNoMerge) {
  193. if (!pdbpm->Deskbars[_uSide]) {
  194. // 1st guy on an edge owns it
  195. // if we don't do this, when we load persisted state on logon
  196. // we end up w/ *no* edge owner (since fNoMerge), so we don't
  197. // merge on subsequent moves.
  198. goto Lsetowner;
  199. }
  200. }
  201. else if (pdbpm->Deskbars[_uSide]) {
  202. // if someone already there, try merging into them
  203. #ifdef DEBUG
  204. // alt+drag suppresses merge
  205. // DEBUG only since don't track >1 per side, but useful
  206. // for testing appbars and toolbars anyway
  207. if (!(GetKeyState(VK_MENU) < 0))
  208. #endif
  209. {
  210. extern IBandSite* _GetBandSite(IDeskBar * pdb);
  211. IBandSite *pbs;
  212. pbs = _GetBandSite(pdbpm->Deskbars[_uSide]);
  213. // nt5:215952: should 'never' have pbs==0 but somehow
  214. // it does happen (during deskbar automation tests).
  215. // call andyp or tjgreen if you hit this assert so
  216. // we can figure out why.
  217. if (TPTR(pbs)) {
  218. _MergeSide(pbs); // dst=pbs, src=this
  219. pbs->Release();
  220. }
  221. }
  222. }
  223. else if ((GetTraySide(&hMonTray) == _uSide) && (hMonTray == _hMon) && !(GetKeyState(VK_SHIFT) < 0)) {
  224. // ditto for tray (but need to marshal/unmarshal)
  225. #ifdef DEBUG
  226. // alt+drag suppresses merge
  227. // DEBUG only since don't track >1 per side, but useful
  228. // for testing appbars and toolbars anyway
  229. if (!(GetKeyState(VK_MENU) < 0))
  230. #endif
  231. {
  232. _MergeSide((IBandSite *)1); // dst=pbs, src=this
  233. }
  234. }
  235. else {
  236. // o.w. nobody there yet, set ourselves as owner
  237. ASSERT(pdbpm->hMon);
  238. ASSERT(pdbpm->hMon == hMonNew);
  239. Lsetowner:
  240. TraceMsg(DM_TRAY, "cdba._sms: 1st side owner this=0x%x", this);
  241. pdbpm->Deskbars[_uSide] = this;
  242. }
  243. }
  244. }
  245. }
  246. }
  247. void CDeskBarApp::_UpdateCaptionTitle()
  248. {
  249. if (ISWBM_FLOAT(_eMode)) {
  250. int iCount = (int)_pbs->EnumBands((UINT)-1, NULL);
  251. if (iCount == 1) {
  252. DWORD dwBandID;
  253. if (SUCCEEDED(_pbs->EnumBands(0, &dwBandID))) {
  254. WCHAR wszTitle[80];
  255. if (SUCCEEDED(_pbs->QueryBand(dwBandID, NULL, NULL, wszTitle, ARRAYSIZE(wszTitle)))) {
  256. USES_CONVERSION;
  257. SetWindowText(_hwnd, W2T(wszTitle));
  258. }
  259. }
  260. }
  261. else {
  262. TCHAR szTitle[80];
  263. szTitle[0] = 0;
  264. MLLoadString(IDS_WEBBARSTITLE,szTitle,ARRAYSIZE(szTitle));
  265. SetWindowText(_hwnd, szTitle);
  266. }
  267. }
  268. }
  269. void CDeskBarApp::_NotifyModeChange(DWORD dwMode)
  270. {
  271. SUPERCLASS::_NotifyModeChange(dwMode);
  272. _UpdateCaptionTitle();
  273. }
  274. //*** GetTrayIface -- get iface from tray (w/ marshal/unmarshal)
  275. //
  276. HRESULT GetTrayIface(REFIID riid, void **ppvObj)
  277. {
  278. HRESULT hr = E_FAIL;
  279. HWND hwndTray;
  280. IStream *pstm;
  281. TraceMsg(DM_TRAY, "gtif: marshal!");
  282. *ppvObj = NULL;
  283. hwndTray = GetTrayWindow();
  284. if (hwndTray) {
  285. pstm = (IStream *) SendMessage(hwndTray, TM_MARSHALBS, (WPARAM)(GUID *)&riid, 0);
  286. if (EVAL(pstm)) {
  287. // paired w/ matching Marshal in explorer (TM_MARSHALBS)
  288. hr = CoGetInterfaceAndReleaseStream(pstm, riid, ppvObj);
  289. ASSERT(SUCCEEDED(hr));
  290. }
  291. }
  292. return hr;
  293. }
  294. //*** _MergeSide -- merge two deskbars into one
  295. // ENTRY/EXIT
  296. // this [INOUT] destination deskbar (ptr:1 if tray)
  297. // pdbSrc [INOUT] source deskbar; deleted if all bands moved successfully
  298. // ret S_OK if all bands moved; S_FALSE if some moved; E_* o.w.
  299. HRESULT CDeskBarApp::_MergeSide(IBandSite *pbsDst)
  300. {
  301. extern HRESULT _MergeBS(IDropTarget *pdtDst, IBandSite *pbsSrc);
  302. HRESULT hr;
  303. IDropTarget *pdtDst;
  304. AddRef(); // make sure we don't disappear partway thru operation
  305. if (pbsDst == (IBandSite *)1) {
  306. // get (marshal'ed) iface from tray
  307. hr = GetTrayIface(IID_IDropTarget, (void **)&pdtDst);
  308. ASSERT(SUCCEEDED(hr));
  309. }
  310. else {
  311. // don't merge into ourself!
  312. ASSERT(pbsDst != _pbs);
  313. ASSERT(!SHIsSameObject(pbsDst, SAFECAST(_pbs, IBandSite*)));
  314. hr = pbsDst->QueryInterface(IID_IDropTarget, (void **)&pdtDst);
  315. ASSERT(SUCCEEDED(hr));
  316. }
  317. ASSERT(SUCCEEDED(hr) || pdtDst == NULL);
  318. if (pdtDst) {
  319. hr = _MergeBS(pdtDst, _pbs);
  320. pdtDst->Release();
  321. }
  322. Release();
  323. return hr;
  324. }
  325. void CDeskBarApp::_CreateBandSiteMenu()
  326. {
  327. CoCreateInstance(CLSID_BandSiteMenu, NULL,CLSCTX_INPROC_SERVER,
  328. IID_PPV_ARG(IContextMenu3, &_pcm));
  329. if (_pcm)
  330. {
  331. IShellService* pss;
  332. _pcm->QueryInterface(IID_IShellService, (LPVOID*)&pss);
  333. if (pss)
  334. {
  335. pss->SetOwner((IBandSite*)_pbs);
  336. pss->Release();
  337. }
  338. }
  339. }
  340. HRESULT CDeskBarApp::QueryInterface(REFIID riid, LPVOID * ppvObj)
  341. {
  342. if (IsEqualIID(riid, IID_IContextMenu) ||
  343. IsEqualIID(riid, IID_IContextMenu2) ||
  344. IsEqualIID(riid, IID_IContextMenu3))
  345. {
  346. if (!_pcm)
  347. {
  348. _CreateBandSiteMenu();
  349. }
  350. // only return out our pointer if we got the one we're going
  351. // to delegate to
  352. if (_pcm)
  353. {
  354. *ppvObj = SAFECAST(this, IContextMenu3*);
  355. AddRef();
  356. return S_OK;
  357. }
  358. }
  359. return SUPERCLASS::QueryInterface(riid, ppvObj);
  360. }
  361. HRESULT CDeskBarApp::QueryService(REFGUID guidService,
  362. REFIID riid, void **ppvObj)
  363. {
  364. if (IsEqualGUID(guidService,SID_SBandSite)) {
  365. return QueryInterface(riid, ppvObj);
  366. }
  367. return SUPERCLASS::QueryService(guidService, riid, ppvObj);
  368. }
  369. HRESULT CDeskBarApp::InvokeCommand(LPCMINVOKECOMMANDINFO pici)
  370. {
  371. int idCmd = -1;
  372. if (!HIWORD(pici->lpVerb))
  373. idCmd = LOWORD(pici->lpVerb);
  374. if (idCmd >= _idCmdDeskBarFirst)
  375. {
  376. _AppBarOnCommand(idCmd - _idCmdDeskBarFirst);
  377. return S_OK;
  378. }
  379. return _pcm->InvokeCommand(pici);
  380. }
  381. HRESULT CDeskBarApp::GetCommandString( UINT_PTR idCmd,
  382. UINT uType,
  383. UINT *pwReserved,
  384. LPSTR pszName,
  385. UINT cchMax)
  386. {
  387. return _pcm->GetCommandString(idCmd, uType, pwReserved, pszName, cchMax);
  388. }
  389. HRESULT CDeskBarApp::HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam)
  390. {
  391. return _pcm->HandleMenuMsg(uMsg, wParam, lParam);
  392. }
  393. HRESULT CDeskBarApp::HandleMenuMsg2(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT* plres)
  394. {
  395. return _pcm->HandleMenuMsg2(uMsg, wParam, lParam, plres);
  396. }
  397. HRESULT CDeskBarApp::QueryContextMenu(HMENU hmenu,
  398. UINT indexMenu,
  399. UINT idCmdFirst,
  400. UINT idCmdLast,
  401. UINT uFlags)
  402. {
  403. HRESULT hr = _pcm->QueryContextMenu(hmenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
  404. if (SUCCEEDED(hr))
  405. {
  406. int i = hr;
  407. HMENU hmenuSrc;
  408. _idCmdDeskBarFirst = i;
  409. hmenuSrc = _GetContextMenu();
  410. // off-by-1 and by idCmdFirst+i, i think...
  411. i += Shell_MergeMenus(hmenu, hmenuSrc, (UINT)-1, idCmdFirst + i, idCmdLast, MM_ADDSEPARATOR) - (idCmdFirst + i);
  412. DestroyMenu(hmenuSrc);
  413. return ResultFromShort(i); // potentially off-by-1, but who cares...
  414. }
  415. return hr;
  416. }
  417. //***
  418. // NOTES
  419. // FEATURE: nuke this, fold it into CDeskBarApp_CreateInstance
  420. HRESULT DeskBarApp_Create(IUnknown** ppunk)
  421. {
  422. HRESULT hres;
  423. *ppunk = NULL;
  424. CDeskBarApp *pdb = new CDeskBarApp();
  425. if (!pdb)
  426. return E_OUTOFMEMORY;
  427. CBandSite *pcbs = new CBandSite(NULL);
  428. if (pcbs)
  429. {
  430. IDeskBarClient *pdbc = SAFECAST(pcbs, IDeskBarClient*);
  431. hres = pdb->SetClient(pdbc);
  432. if (SUCCEEDED(hres))
  433. {
  434. pdb->_pbs = pcbs;
  435. pcbs->AddRef();
  436. *ppunk = SAFECAST(pdb, IDeskBar*);
  437. }
  438. pdbc->Release();
  439. }
  440. else
  441. {
  442. hres = E_OUTOFMEMORY;
  443. }
  444. if (FAILED(hres))
  445. pdb->Release();
  446. return hres;
  447. }
  448. STDAPI CDeskBarApp_CreateInstance(IUnknown* pUnkOuter, IUnknown** ppunk, LPCOBJECTINFO poi)
  449. {
  450. HRESULT hres;
  451. IUnknown *punk;
  452. // aggregation checking is handled in class factory
  453. hres = DeskBarApp_Create(&punk);
  454. if (SUCCEEDED(hres)) {
  455. *ppunk = SAFECAST(punk, IDockingWindow*);
  456. return S_OK;
  457. }
  458. return E_OUTOFMEMORY;
  459. }
  460. //*** CDeskBarApp::IInputObject*::* {
  461. //
  462. HRESULT CDeskBarApp::TranslateAcceleratorIO(LPMSG lpMsg)
  463. {
  464. if (lpMsg->message == WM_SYSKEYDOWN) {
  465. if (lpMsg->wParam == VK_F4) {
  466. // ie4:28819: need to trap VK_F4 here, o.w. CBaseBrowser::TA
  467. // does a last-chance (winsdk)::TA (to IDM_CLOSE) and doing a
  468. // shutdown!
  469. PostMessage(_hwnd, WM_CLOSE, 0, 0);
  470. return S_OK;
  471. }
  472. }
  473. return SUPERCLASS::TranslateAcceleratorIO(lpMsg);
  474. }
  475. // }
  476. //*** CDeskBarApp::IPersistStream*::* {
  477. //
  478. HRESULT CDeskBarApp::GetClassID(CLSID *pClassID)
  479. {
  480. *pClassID = CLSID_DeskBarApp;
  481. return S_OK;
  482. }
  483. HRESULT CDeskBarApp::IsDirty(void)
  484. {
  485. return S_FALSE; // Never be dirty
  486. }
  487. //
  488. // Persisted CDeskBarApp
  489. //
  490. #define STC_VERSION 1
  491. struct SThisClass
  492. {
  493. DWORD cbSize;
  494. DWORD cbVersion;
  495. };
  496. HRESULT CDeskBarApp::Load(IStream *pstm)
  497. {
  498. SThisClass stc;
  499. ULONG cbRead;
  500. HRESULT hres;
  501. TraceMsg(DM_PERSIST, "cdba.l enter(this=%x pstm=%x) tell()=%x", this, pstm, DbStreamTell(pstm));
  502. ASSERT(!_eInitLoaded);
  503. _eInitLoaded = IPS_LOAD;
  504. hres = pstm->Read(&stc, SIZEOF(stc), &cbRead);
  505. #ifdef DEBUG
  506. // just in case we toast ourselves (offscreen or something)...
  507. static BOOL fNoPersist = FALSE;
  508. if (fNoPersist)
  509. hres = E_FAIL;
  510. #endif
  511. if (hres==S_OK && cbRead==SIZEOF(stc)) {
  512. if (stc.cbSize==SIZEOF(SThisClass) && stc.cbVersion==STC_VERSION) {
  513. _eInitLoaded = IPS_LOAD; // FEATURE: what if OLFS of bands fails?
  514. hres = SUPERCLASS::Load(pstm);
  515. TraceMsg(DM_INIT, "cdba::Load succeeded");
  516. } else {
  517. TraceMsg(DM_ERROR, "cdba::Load failed stc.cbSize==SIZEOF(SThisClass) && stc.cbVersion==SWB_VERSION");
  518. hres = E_FAIL;
  519. }
  520. } else {
  521. TraceMsg(DM_ERROR, "cdba::Load failed (hres==S_OK && cbRead==SIZEOF(_adEdge)");
  522. hres = E_FAIL;
  523. }
  524. TraceMsg(DM_PERSIST, "cdba.l leave tell()=%x", DbStreamTell(pstm));
  525. // after loading this, if we find that we're supposed to be browser docked,
  526. // make our bandsite always have a gripper
  527. if (_eMode == WBM_BFLOATING)
  528. {
  529. BANDSITEINFO bsinfo;
  530. bsinfo.dwMask = BSIM_STYLE;
  531. bsinfo.dwStyle = BSIS_ALWAYSGRIPPER;
  532. _pbs->SetBandSiteInfo(&bsinfo);
  533. }
  534. return hres;
  535. }
  536. HRESULT CDeskBarApp::Save(IStream *pstm, BOOL fClearDirty)
  537. {
  538. HRESULT hres;
  539. SThisClass stc;
  540. TraceMsg(DM_PERSIST, "cdba.s enter(this=%x pstm=%x) tell()=%x", this, pstm, DbStreamTell(pstm));
  541. stc.cbSize = SIZEOF(SThisClass);
  542. stc.cbVersion = STC_VERSION;
  543. hres = pstm->Write(&stc, SIZEOF(stc), NULL);
  544. if (SUCCEEDED(hres)) {
  545. SUPERCLASS::Save(pstm, fClearDirty);
  546. }
  547. TraceMsg(DM_PERSIST, "cdba.s leave tell()=%x", DbStreamTell(pstm));
  548. return hres;
  549. }
  550. HRESULT CDeskBarApp::GetSizeMax(ULARGE_INTEGER *pcbSize)
  551. {
  552. ULARGE_INTEGER cbMax = { SIZEOF(SThisClass), 0 };
  553. *pcbSize = cbMax;
  554. return S_OK;
  555. }
  556. HRESULT CDeskBarApp::InitNew(void)
  557. {
  558. HRESULT hres;
  559. ASSERT(!_eInitLoaded);
  560. _eInitLoaded = IPS_INITNEW;
  561. TraceMsg(DM_INIT, "CDeskBarApp::InitNew called");
  562. hres = SUPERCLASS::InitNew();
  563. if (FAILED(hres))
  564. return hres;
  565. // can't call _InitPos4 until set site in SetSite
  566. return hres;
  567. }
  568. HRESULT CDeskBarApp::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt,
  569. VARIANTARG *pvarargIn, VARIANTARG *pvarargOut)
  570. {
  571. if (pguidCmdGroup == NULL) {
  572. /*NOTHING*/
  573. }
  574. else if (IsEqualGUID(CGID_DeskBarClient, *pguidCmdGroup)) {
  575. switch (nCmdID) {
  576. case DBCID_EMPTY:
  577. if (_pbs) {
  578. // if we have no bands left, close
  579. PostMessage(_hwnd, WM_CLOSE, 0, 0);
  580. }
  581. return S_OK;
  582. }
  583. }
  584. else if (IsEqualIID(*pguidCmdGroup, CGID_DeskBand)) {
  585. switch (nCmdID) {
  586. case DBID_BANDINFOCHANGED:
  587. _UpdateCaptionTitle();
  588. return S_OK;
  589. }
  590. }
  591. else if (IsEqualIID(*pguidCmdGroup, CGID_BandSite)) {
  592. switch (nCmdID) {
  593. case BSID_BANDADDED:
  594. case BSID_BANDREMOVED:
  595. _UpdateCaptionTitle();
  596. return S_OK;
  597. }
  598. }
  599. return SUPERCLASS::Exec(pguidCmdGroup, nCmdID, nCmdexecopt, pvarargIn, pvarargOut);
  600. }
  601. HRESULT CDeskBarApp::Load(IPropertyBag *pPropBag, IErrorLog *pErrorLog)
  602. {
  603. HRESULT hres;
  604. ASSERT(!_eInitLoaded);
  605. _eInitLoaded = IPS_LOAD;
  606. TraceMsg(DM_INIT, "CDeskBarApp::Load(bag) called");
  607. hres = SUPERCLASS::Load(pPropBag, pErrorLog);
  608. // after loading this, if we find that we're supposed to be browser docked,
  609. // make our bandsite always have a gripper
  610. if (_eMode == WBM_BFLOATING)
  611. {
  612. BANDSITEINFO bsinfo;
  613. bsinfo.dwMask = BSIM_STYLE;
  614. bsinfo.dwStyle = BSIS_ALWAYSGRIPPER;
  615. _pbs->SetBandSiteInfo(&bsinfo);
  616. }
  617. return hres;
  618. }
  619. IBandSite * _GetBandSite(IDeskBar * pdb)
  620. {
  621. IBandSite* pbs = NULL;
  622. if (pdb) {
  623. IUnknown* punkClient;
  624. pdb->GetClient(&punkClient);
  625. if (punkClient) {
  626. punkClient->QueryInterface(IID_IBandSite, (LPVOID*)&pbs);
  627. punkClient->Release();
  628. }
  629. }
  630. return pbs;
  631. }
  632. IBandSite* DeskBarApp_GetBandSiteOnEdge(UINT uEdge)
  633. {
  634. // APPCOMPAT: (dli) if no HMONITOR is passed in, use the primary monitor
  635. // should make sure that there is always a valid HMONITOR passed in
  636. HMONITOR hMon = GetPrimaryMonitor();
  637. // --------------------------------------------------------------
  638. LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMon, FALSE);
  639. if (pdbpm) {
  640. ASSERT(pdbpm->hMon);
  641. ASSERT(pdbpm->hMon == hMon);
  642. return _GetBandSite(pdbpm->Deskbars[uEdge]);
  643. }
  644. return NULL;
  645. }
  646. IBandSite* DeskBarApp_GetBandSiteAtPoint(LPPOINT ppt)
  647. {
  648. HWND hwnd = WindowFromPoint(*ppt);
  649. HMONITOR hMon = MonitorFromPoint(*ppt, MONITOR_DEFAULTTONULL);
  650. if (hwnd && hMon) {
  651. LPDESKBARSPERMONITOR pdbpm = GetDBPMWithMonitor(hMon, FALSE);
  652. if (pdbpm) {
  653. ASSERT(pdbpm->hMon);
  654. ASSERT(pdbpm->hMon == hMon);
  655. int i;
  656. for (i = 0; i < 4; i++) {
  657. if (pdbpm->Deskbars[i]) {
  658. HWND hwndDeskbar;
  659. pdbpm->Deskbars[i]->GetWindow(&hwndDeskbar);
  660. if (hwndDeskbar == hwnd) {
  661. return _GetBandSite(pdbpm->Deskbars[i]);
  662. }
  663. }
  664. }
  665. }
  666. }
  667. return NULL;
  668. }
  669. // }