[SHELL] IPersistFolder::Initialize takes a PCIDLIST_ABSOLUTE. CORE-16385
[reactos.git] / dll / win32 / shell32 / shellmenu / CMenuBand.cpp
1 /*
2 * Shell Menu Band
3 *
4 * Copyright 2014 David Quintana
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20 #include "shellmenu.h"
21 #include <windowsx.h>
22 #include <commoncontrols.h>
23 #include <shlwapi_undoc.h>
24
25 #include "CMenuBand.h"
26 #include "CMenuToolbars.h"
27 #include "CMenuFocusManager.h"
28
29 WINE_DEFAULT_DEBUG_CHANNEL(CMenuBand);
30
31 #undef UNIMPLEMENTED
32
33 #define UNIMPLEMENTED TRACE("%s is UNIMPLEMENTED!\n", __FUNCTION__)
34
35 CMenuBand::CMenuBand() :
36 m_staticToolbar(NULL),
37 m_SFToolbar(NULL),
38 m_site(NULL),
39 m_psmc(NULL),
40 m_subMenuChild(NULL),
41 m_subMenuParent(NULL),
42 m_childBand(NULL),
43 m_parentBand(NULL),
44 m_hmenu(NULL),
45 m_menuOwner(NULL),
46 m_useBigIcons(FALSE),
47 m_topLevelWindow(NULL),
48 m_hotBar(NULL),
49 m_hotItem(-1),
50 m_popupBar(NULL),
51 m_popupItem(-1),
52 m_Show(FALSE),
53 m_shellBottom(FALSE),
54 m_trackedPopup(NULL),
55 m_trackedHwnd(NULL)
56 {
57 m_focusManager = CMenuFocusManager::AcquireManager();
58 }
59
60 CMenuBand::~CMenuBand()
61 {
62 CMenuFocusManager::ReleaseManager(m_focusManager);
63
64 delete m_staticToolbar;
65 delete m_SFToolbar;
66
67 if (m_hmenu)
68 DestroyMenu(m_hmenu);
69 }
70
71 HRESULT STDMETHODCALLTYPE CMenuBand::Initialize(
72 IShellMenuCallback *psmc,
73 UINT uId,
74 UINT uIdAncestor,
75 DWORD dwFlags)
76 {
77 if (m_psmc != psmc)
78 m_psmc = psmc;
79 m_uId = uId;
80 m_uIdAncestor = uIdAncestor;
81 m_dwFlags = dwFlags;
82
83 if (m_psmc)
84 {
85 _CallCB(SMC_CREATE, 0, reinterpret_cast<LPARAM>(&m_UserData));
86 }
87
88 return S_OK;
89 }
90
91 HRESULT STDMETHODCALLTYPE CMenuBand::GetMenuInfo(
92 IShellMenuCallback **ppsmc,
93 UINT *puId,
94 UINT *puIdAncestor,
95 DWORD *pdwFlags)
96 {
97 if (!pdwFlags) // maybe?
98 return E_INVALIDARG;
99
100 if (ppsmc)
101 {
102 if (m_psmc)
103 m_psmc->AddRef();
104 *ppsmc = m_psmc;
105 }
106
107 if (puId)
108 *puId = m_uId;
109
110 if (puIdAncestor)
111 *puIdAncestor = m_uIdAncestor;
112
113 *pdwFlags = m_dwFlags;
114
115 return S_OK;
116 }
117
118 HRESULT STDMETHODCALLTYPE CMenuBand::SetMenu(
119 HMENU hmenu,
120 HWND hwnd,
121 DWORD dwFlags)
122 {
123 HRESULT hr;
124
125 TRACE("CMenuBand::SetMenu called, hmenu=%p; hwnd=%p, flags=%x\n", hmenu, hwnd, dwFlags);
126
127 BOOL created = FALSE;
128
129 if (m_hmenu && m_hmenu != hmenu)
130 {
131 DestroyMenu(m_hmenu);
132 m_hmenu = NULL;
133 }
134
135 m_hmenu = hmenu;
136 m_menuOwner = hwnd;
137
138 if (m_hmenu && m_staticToolbar == NULL)
139 {
140 m_staticToolbar = new CMenuStaticToolbar(this);
141 created = true;
142 }
143
144 if (m_staticToolbar)
145 {
146 hr = m_staticToolbar->SetMenu(hmenu, hwnd, dwFlags);
147 if (FAILED_UNEXPECTEDLY(hr))
148 return hr;
149 }
150
151 if (m_site)
152 {
153 HWND hwndParent;
154
155 hr = m_site->GetWindow(&hwndParent);
156 if (FAILED_UNEXPECTEDLY(hr))
157 return hr;
158
159 if (created)
160 {
161 hr = m_staticToolbar->CreateToolbar(hwndParent, m_dwFlags);
162 if (FAILED_UNEXPECTEDLY(hr))
163 return hr;
164
165 hr = m_staticToolbar->FillToolbar();
166 }
167 else
168 {
169 hr = m_staticToolbar->FillToolbar(TRUE);
170 }
171 }
172
173 return hr;
174 }
175
176 HRESULT STDMETHODCALLTYPE CMenuBand::GetMenu(
177 HMENU *phmenu,
178 HWND *phwnd,
179 DWORD *pdwFlags)
180 {
181 if (m_staticToolbar == NULL)
182 return E_FAIL;
183
184 return m_staticToolbar->GetMenu(phmenu, phwnd, pdwFlags);
185 }
186
187 HRESULT STDMETHODCALLTYPE CMenuBand::SetSite(IUnknown *pUnkSite)
188 {
189 HWND hwndParent;
190 HRESULT hr;
191
192 m_site = NULL;
193
194 if (pUnkSite == NULL)
195 return S_OK;
196
197 hwndParent = NULL;
198 hr = pUnkSite->QueryInterface(IID_PPV_ARG(IOleWindow, &m_site));
199 if (FAILED_UNEXPECTEDLY(hr))
200 return hr;
201
202 hr = m_site->GetWindow(&hwndParent);
203 if (FAILED_UNEXPECTEDLY(hr))
204 return hr;
205
206 if (!::IsWindow(hwndParent))
207 return E_FAIL;
208
209 if (m_staticToolbar != NULL)
210 {
211 hr = m_staticToolbar->CreateToolbar(hwndParent, m_dwFlags);
212 if (FAILED_UNEXPECTEDLY(hr))
213 return hr;
214
215 hr = m_staticToolbar->FillToolbar();
216 if (FAILED_UNEXPECTEDLY(hr))
217 return hr;
218 }
219
220 if (m_SFToolbar != NULL)
221 {
222 hr = m_SFToolbar->CreateToolbar(hwndParent, m_dwFlags);
223 if (FAILED_UNEXPECTEDLY(hr))
224 return hr;
225
226 hr = m_SFToolbar->FillToolbar();
227 if (FAILED_UNEXPECTEDLY(hr))
228 return hr;
229 }
230
231 hr = IUnknown_QueryService(m_site, SID_SMenuPopup, IID_PPV_ARG(IMenuPopup, &m_subMenuParent));
232 if (hr != E_NOINTERFACE && FAILED_UNEXPECTEDLY(hr))
233 return hr;
234
235 CComPtr<IOleWindow> pTopLevelWindow;
236 hr = IUnknown_QueryService(m_site, SID_STopLevelBrowser, IID_PPV_ARG(IOleWindow, &pTopLevelWindow));
237 if (SUCCEEDED(hr))
238 {
239 hr = pTopLevelWindow->GetWindow(&m_topLevelWindow);
240 if (FAILED_UNEXPECTEDLY(hr))
241 return hr;
242 }
243 else
244 {
245 m_topLevelWindow = hwndParent;
246 }
247
248 return S_OK;
249 }
250
251 HRESULT STDMETHODCALLTYPE CMenuBand::GetSite(REFIID riid, PVOID *ppvSite)
252 {
253 if (m_site == NULL)
254 return E_FAIL;
255
256 return m_site->QueryInterface(riid, ppvSite);
257 }
258
259 HRESULT STDMETHODCALLTYPE CMenuBand::GetWindow(HWND *phwnd)
260 {
261 if (m_SFToolbar != NULL)
262 return m_SFToolbar->GetWindow(phwnd);
263
264 if (m_staticToolbar != NULL)
265 return m_staticToolbar->GetWindow(phwnd);
266
267 if (phwnd) *phwnd = NULL;
268
269 return E_FAIL;
270 }
271
272 HRESULT STDMETHODCALLTYPE CMenuBand::OnPosRectChangeDB(RECT *prc)
273 {
274 SIZE maxStatic = { 0 };
275 SIZE maxShlFld = { 0 };
276 HRESULT hr = S_OK;
277
278 if (m_staticToolbar != NULL)
279 hr = m_staticToolbar->GetSizes(NULL, &maxStatic, NULL);
280 if (FAILED_UNEXPECTEDLY(hr))
281 return hr;
282
283 if (m_SFToolbar != NULL)
284 hr = m_SFToolbar->GetSizes(NULL, &maxShlFld, NULL);
285 if (FAILED_UNEXPECTEDLY(hr))
286 return hr;
287
288 if (m_staticToolbar == NULL && m_SFToolbar == NULL)
289 return E_FAIL;
290
291 int sy = min(prc->bottom - prc->top, maxStatic.cy + maxShlFld.cy);
292
293 int syStatic = maxStatic.cy;
294 int syShlFld = sy - syStatic;
295
296 // TODO: Windows has a more complex system to decide ordering.
297 // Because we only support two toolbars at once, this is enough for us.
298 if (m_shellBottom)
299 {
300 // Static menu on top
301 if (m_SFToolbar)
302 {
303 m_SFToolbar->SetPosSize(
304 prc->left,
305 prc->top + syStatic,
306 prc->right - prc->left,
307 syShlFld);
308 }
309 if (m_staticToolbar)
310 {
311 m_staticToolbar->SetPosSize(
312 prc->left,
313 prc->top,
314 prc->right - prc->left,
315 syStatic);
316 }
317 }
318 else
319 {
320 // Folder menu on top
321 if (m_SFToolbar)
322 {
323 m_SFToolbar->SetPosSize(
324 prc->left,
325 prc->top,
326 prc->right - prc->left,
327 syShlFld);
328 }
329 if (m_staticToolbar)
330 {
331 m_staticToolbar->SetPosSize(
332 prc->left,
333 prc->top + syShlFld,
334 prc->right - prc->left,
335 syStatic);
336 }
337 }
338
339 return S_OK;
340 }
341
342 HRESULT STDMETHODCALLTYPE CMenuBand::GetBandInfo(
343 DWORD dwBandID,
344 DWORD dwViewMode,
345 DESKBANDINFO *pdbi)
346 {
347 SIZE minStatic = { 0 };
348 SIZE minShlFld = { 0 };
349 SIZE maxStatic = { 0 };
350 SIZE maxShlFld = { 0 };
351 SIZE intStatic = { 0 };
352 SIZE intShlFld = { 0 };
353
354 HRESULT hr = S_OK;
355
356 if (m_staticToolbar != NULL)
357 hr = m_staticToolbar->GetSizes(&minStatic, &maxStatic, &intStatic);
358 if (FAILED_UNEXPECTEDLY(hr))
359 return hr;
360
361 if (m_SFToolbar != NULL)
362 hr = m_SFToolbar->GetSizes(&minShlFld, &maxShlFld, &intShlFld);
363 if (FAILED_UNEXPECTEDLY(hr))
364 return hr;
365
366 if (m_staticToolbar == NULL && m_SFToolbar == NULL)
367 return E_FAIL;
368
369 if (m_dwFlags & SMINIT_VERTICAL)
370 {
371 pdbi->ptMinSize.x = max(minStatic.cx, minShlFld.cx) + 20;
372 pdbi->ptMinSize.y = minStatic.cy + minShlFld.cy;
373 pdbi->ptMaxSize.x = max(maxStatic.cx, maxShlFld.cx) + 20;
374 pdbi->ptMaxSize.y = maxStatic.cy + maxShlFld.cy;
375 pdbi->dwModeFlags = DBIMF_VARIABLEHEIGHT;
376 }
377 else
378 {
379 pdbi->ptMinSize.x = minStatic.cx + minShlFld.cx;
380 pdbi->ptMinSize.y = max(minStatic.cy, minShlFld.cy);
381 pdbi->ptMaxSize.x = maxStatic.cx + maxShlFld.cx;
382 pdbi->ptMaxSize.y = max(maxStatic.cy, maxShlFld.cy);
383 }
384 pdbi->ptIntegral.x = max(intStatic.cx, intShlFld.cx);
385 pdbi->ptIntegral.y = max(intStatic.cy, intShlFld.cy);
386 pdbi->ptActual = pdbi->ptMinSize;
387
388 return S_OK;
389 }
390
391 HRESULT STDMETHODCALLTYPE CMenuBand::ShowDW(BOOL fShow)
392 {
393 HRESULT hr = S_OK;
394
395 if (m_Show == fShow)
396 return S_OK;
397
398 m_Show = fShow;
399
400 if (m_staticToolbar != NULL)
401 {
402 hr = m_staticToolbar->ShowDW(fShow);
403 if (FAILED_UNEXPECTEDLY(hr))
404 return hr;
405 }
406
407 if (m_SFToolbar != NULL)
408 {
409 hr = m_SFToolbar->ShowDW(fShow);
410 if (FAILED_UNEXPECTEDLY(hr))
411 return hr;
412 }
413
414 if (fShow)
415 {
416 hr = _CallCB(SMC_INITMENU, 0, 0);
417 if (FAILED_UNEXPECTEDLY(hr))
418 return hr;
419 }
420 else if (m_parentBand)
421 {
422 m_parentBand->SetClient(NULL);
423 }
424
425 if (_IsPopup() == S_OK)
426 {
427 if (fShow)
428 hr = m_focusManager->PushMenuPopup(this);
429 else
430 hr = m_focusManager->PopMenuPopup(this);
431 }
432 else
433 {
434 if (fShow)
435 hr = m_focusManager->PushMenuBar(this);
436 else
437 hr = m_focusManager->PopMenuBar(this);
438 }
439
440 return S_OK;
441 }
442
443 HRESULT STDMETHODCALLTYPE CMenuBand::CloseDW(DWORD dwReserved)
444 {
445 if (m_subMenuChild)
446 {
447 m_subMenuChild->OnSelect(MPOS_CANCELLEVEL);
448 }
449
450 if (m_subMenuChild)
451 {
452 TRACE("Child object should have removed itself.\n");
453 }
454
455 ShowDW(FALSE);
456
457 if (m_staticToolbar != NULL)
458 {
459 m_staticToolbar->Close();
460 }
461
462 if (m_SFToolbar != NULL)
463 {
464 m_SFToolbar->Close();
465 }
466
467 if (m_site) m_site.Release();
468 if (m_subMenuChild) m_subMenuChild.Release();
469 if (m_subMenuParent) m_subMenuParent.Release();
470 if (m_childBand) m_childBand.Release();
471 if (m_parentBand) m_parentBand.Release();
472
473 return S_OK;
474 }
475
476 HRESULT STDMETHODCALLTYPE CMenuBand::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
477 {
478 HRESULT hr;
479
480 if (m_subMenuParent)
481 {
482 hr = m_subMenuParent->SetSubMenu(this, fActivate);
483 if (FAILED_UNEXPECTEDLY(hr))
484 return hr;
485 }
486
487 if (fActivate)
488 {
489 CComPtr<IOleWindow> pTopLevelWindow;
490 hr = IUnknown_QueryService(m_site, SID_SMenuPopup, IID_PPV_ARG(IOleWindow, &pTopLevelWindow));
491 if (FAILED_UNEXPECTEDLY(hr))
492 return hr;
493
494 hr = pTopLevelWindow->GetWindow(&m_topLevelWindow);
495 if (FAILED_UNEXPECTEDLY(hr))
496 return hr;
497 }
498 else
499 {
500 m_topLevelWindow = NULL;
501 }
502
503 return S_FALSE;
504 }
505
506 HRESULT STDMETHODCALLTYPE CMenuBand::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
507 {
508 if (!pguidCmdGroup)
509 return E_FAIL;
510
511 if (IsEqualGUID(*pguidCmdGroup, CLSID_MenuBand))
512 {
513 if (nCmdID == 16) // set (big) icon size
514 {
515 this->m_useBigIcons = nCmdexecopt == 2;
516 return S_OK;
517 }
518 else if (nCmdID == 19) // popup-related
519 {
520 return S_FALSE;
521 }
522 else if (nCmdID == 5) // select an item
523 {
524 if (nCmdexecopt == 0) // first
525 {
526 _KeyboardItemChange(VK_HOME);
527 }
528 else // last
529 {
530 _KeyboardItemChange(VK_END);
531 }
532 return S_FALSE;
533 }
534
535 return S_FALSE;
536 }
537
538 UNIMPLEMENTED;
539 return S_OK;
540 }
541
542 HRESULT STDMETHODCALLTYPE CMenuBand::QueryService(REFGUID guidService, REFIID riid, void **ppvObject)
543 {
544 if (IsEqualIID(guidService, SID_SMenuBandChild) ||
545 IsEqualIID(guidService, SID_SMenuBandBottom) ||
546 IsEqualIID(guidService, SID_SMenuBandBottomSelected))
547 return this->QueryInterface(riid, ppvObject);
548 WARN("Unknown service requested %s\n", wine_dbgstr_guid(&guidService));
549 return E_NOINTERFACE;
550 }
551
552 HRESULT STDMETHODCALLTYPE CMenuBand::Popup(POINTL *ppt, RECTL *prcExclude, MP_POPUPFLAGS dwFlags)
553 {
554 UNIMPLEMENTED;
555 return S_OK;
556 }
557
558 HRESULT STDMETHODCALLTYPE CMenuBand::OnSelect(DWORD dwSelectType)
559 {
560 // When called from outside, this is straightforward:
561 // Things that a submenu needs to know, are spread down, and
562 // things that the parent needs to know, are spread up. No drama.
563 // The fun is in _MenuItemSelect (internal method).
564 switch (dwSelectType)
565 {
566 case MPOS_CHILDTRACKING:
567 if (!m_subMenuParent)
568 break;
569 // TODO: Cancel timers?
570 return m_subMenuParent->OnSelect(dwSelectType);
571 case MPOS_SELECTLEFT:
572 if (m_subMenuChild)
573 m_subMenuChild->OnSelect(MPOS_CANCELLEVEL);
574 if (!m_subMenuParent)
575 break;
576 return m_subMenuParent->OnSelect(dwSelectType);
577 case MPOS_SELECTRIGHT:
578 if (!m_subMenuParent)
579 break;
580 return m_subMenuParent->OnSelect(dwSelectType);
581 case MPOS_EXECUTE:
582 case MPOS_FULLCANCEL:
583 if (m_subMenuChild)
584 m_subMenuChild->OnSelect(dwSelectType);
585 if (!m_subMenuParent)
586 break;
587 return m_subMenuParent->OnSelect(dwSelectType);
588 case MPOS_CANCELLEVEL:
589 if (m_subMenuChild)
590 m_subMenuChild->OnSelect(dwSelectType);
591 break;
592 }
593 return S_FALSE;
594 }
595
596 HRESULT STDMETHODCALLTYPE CMenuBand::SetSubMenu(IMenuPopup *pmp, BOOL fSet)
597 {
598 UNIMPLEMENTED;
599 return S_OK;
600 }
601
602 // Used by the focus manager to update the child band pointer
603 HRESULT CMenuBand::_SetChildBand(CMenuBand * child)
604 {
605 m_childBand = child;
606 if (!child)
607 {
608 _ChangePopupItem(NULL, -1);
609 }
610 return S_OK;
611 }
612
613 // User by the focus manager to update the parent band pointer
614 HRESULT CMenuBand::_SetParentBand(CMenuBand * parent)
615 {
616 m_parentBand = parent;
617 return S_OK;
618 }
619
620 HRESULT CMenuBand::_IsPopup()
621 {
622 return !(m_dwFlags & SMINIT_VERTICAL);
623 }
624
625 HRESULT CMenuBand::_IsTracking()
626 {
627 return m_popupBar != NULL;
628 }
629
630 HRESULT STDMETHODCALLTYPE CMenuBand::SetClient(IUnknown *punkClient)
631 {
632 CComPtr<IMenuPopup> child = m_subMenuChild;
633
634 m_subMenuChild = NULL;
635
636 if (child)
637 {
638 IUnknown_SetSite(child, NULL);
639 child.Release();
640 }
641
642 if (!punkClient)
643 {
644 return S_OK;
645 }
646
647 return punkClient->QueryInterface(IID_PPV_ARG(IMenuPopup, &m_subMenuChild));
648 }
649
650 HRESULT STDMETHODCALLTYPE CMenuBand::GetClient(IUnknown **ppunkClient)
651 {
652 if (!ppunkClient)
653 return E_POINTER;
654 *ppunkClient = NULL;
655
656 if (m_subMenuChild)
657 {
658 m_subMenuChild->AddRef();
659 *ppunkClient = m_subMenuChild;
660 }
661
662 return S_OK;
663 }
664
665 HRESULT STDMETHODCALLTYPE CMenuBand::IsMenuMessage(MSG *pmsg)
666 {
667 return S_FALSE;
668 }
669
670 HRESULT STDMETHODCALLTYPE CMenuBand::TranslateMenuMessage(MSG *pmsg, LRESULT *plRet)
671 {
672 if (pmsg->message == WM_ACTIVATE && _IsPopup() == S_FALSE)
673 {
674 if (m_staticToolbar)
675 m_staticToolbar->Invalidate();
676 if (m_SFToolbar)
677 m_SFToolbar->Invalidate();
678 }
679
680 return S_FALSE;
681 }
682
683 HRESULT STDMETHODCALLTYPE CMenuBand::SetShellFolder(IShellFolder *psf, LPCITEMIDLIST pidlFolder, HKEY hKey, DWORD dwFlags)
684 {
685 if (!psf)
686 return E_INVALIDARG;
687
688 if (m_SFToolbar == NULL)
689 {
690 m_SFToolbar = new CMenuSFToolbar(this);
691 }
692
693 HRESULT hr = m_SFToolbar->SetShellFolder(psf, pidlFolder, hKey, dwFlags);
694 if (FAILED_UNEXPECTEDLY(hr))
695 return hr;
696
697 m_shellBottom = (dwFlags & SMSET_BOTTOM) != 0;
698
699 if (m_site)
700 {
701 HWND hwndParent;
702
703 hr = m_site->GetWindow(&hwndParent);
704 if (FAILED_UNEXPECTEDLY(hr))
705 return hr;
706
707 hr = m_SFToolbar->CreateToolbar(hwndParent, m_dwFlags);
708 if (FAILED_UNEXPECTEDLY(hr))
709 return hr;
710
711 hr = m_SFToolbar->FillToolbar();
712 }
713
714 return hr;
715 }
716
717 HRESULT STDMETHODCALLTYPE CMenuBand::GetShellFolder(DWORD *pdwFlags, LPITEMIDLIST *ppidl, REFIID riid, void **ppv)
718 {
719 if (m_SFToolbar)
720 return m_SFToolbar->GetShellFolder(pdwFlags, ppidl, riid, ppv);
721 return E_FAIL;
722 }
723
724 HRESULT STDMETHODCALLTYPE CMenuBand::OnWinEvent(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *theResult)
725 {
726 if (theResult)
727 *theResult = 0;
728
729 if (uMsg == WM_WININICHANGE && wParam == SPI_SETFLATMENU)
730 {
731 BOOL bFlatMenus;
732 SystemParametersInfo(SPI_GETFLATMENU, 0, &bFlatMenus, 0);
733 AdjustForTheme(bFlatMenus);
734
735 if (m_staticToolbar)
736 m_staticToolbar->OnWinEvent(hWnd, uMsg, wParam, lParam, theResult);
737
738 if (m_SFToolbar)
739 m_SFToolbar->OnWinEvent(hWnd, uMsg, wParam, lParam, theResult);
740
741 return S_OK;
742 }
743
744 if (m_staticToolbar && m_staticToolbar->IsWindowOwner(hWnd) == S_OK)
745 {
746 return m_staticToolbar->OnWinEvent(hWnd, uMsg, wParam, lParam, theResult);
747 }
748
749 if (m_SFToolbar && m_SFToolbar->IsWindowOwner(hWnd) == S_OK)
750 {
751 return m_SFToolbar->OnWinEvent(hWnd, uMsg, wParam, lParam, theResult);
752 }
753
754 return S_FALSE;
755 }
756
757 HRESULT STDMETHODCALLTYPE CMenuBand::IsWindowOwner(HWND hWnd)
758 {
759 if (m_staticToolbar && m_staticToolbar->IsWindowOwner(hWnd) == S_OK)
760 return S_OK;
761
762 if (m_SFToolbar && m_SFToolbar->IsWindowOwner(hWnd) == S_OK)
763 return S_OK;
764
765 return S_FALSE;
766 }
767
768 HRESULT CMenuBand::_CallCBWithItemId(UINT id, UINT uMsg, WPARAM wParam, LPARAM lParam)
769 {
770 return _CallCB(uMsg, wParam, lParam, id);
771 }
772
773 HRESULT CMenuBand::_CallCBWithItemPidl(LPITEMIDLIST pidl, UINT uMsg, WPARAM wParam, LPARAM lParam)
774 {
775 return _CallCB(uMsg, wParam, lParam, 0, pidl);
776 }
777
778 HRESULT CMenuBand::_CallCB(UINT uMsg, WPARAM wParam, LPARAM lParam, UINT id, LPITEMIDLIST pidl)
779 {
780 if (!m_psmc)
781 return S_FALSE;
782
783 SMDATA smData = { 0 };
784 smData.punk = static_cast<IShellMenu2*>(this);
785 smData.uId = id;
786 smData.uIdParent = m_uId;
787 smData.uIdAncestor = m_uIdAncestor;
788 smData.pidlItem = pidl;
789 smData.hwnd = m_menuOwner ? m_menuOwner : m_topLevelWindow;
790 smData.hmenu = m_hmenu;
791 if (m_SFToolbar)
792 m_SFToolbar->GetShellFolder(NULL, &smData.pidlFolder, IID_PPV_ARG(IShellFolder, &smData.psf));
793 HRESULT hr = m_psmc->CallbackSM(&smData, uMsg, wParam, lParam);
794 ILFree(smData.pidlFolder);
795 if (smData.psf)
796 smData.psf->Release();
797 return hr;
798 }
799
800 HRESULT CMenuBand::_TrackSubMenu(HMENU popup, INT x, INT y, RECT& rcExclude)
801 {
802 TPMPARAMS params = { sizeof(TPMPARAMS), rcExclude };
803 UINT flags = TPM_VERPOSANIMATION | TPM_VERTICAL | TPM_LEFTALIGN;
804 HWND hwnd = m_menuOwner ? m_menuOwner : m_topLevelWindow;
805
806 m_trackedPopup = popup;
807 m_trackedHwnd = hwnd;
808
809 m_focusManager->PushTrackedPopup(popup);
810 ::TrackPopupMenuEx(popup, flags, x, y, hwnd, &params);
811 m_focusManager->PopTrackedPopup(popup);
812
813 m_trackedPopup = NULL;
814 m_trackedHwnd = NULL;
815
816 _DisableMouseTrack(FALSE);
817
818 return S_OK;
819 }
820
821 HRESULT CMenuBand::_TrackContextMenu(IContextMenu * contextMenu, INT x, INT y)
822 {
823 HRESULT hr;
824 UINT uCommand;
825
826 // Ensure that the menu doesn't disappear on us
827 CComPtr<IContextMenu> ctxMenu = contextMenu;
828
829 HMENU popup = CreatePopupMenu();
830
831 if (popup == NULL)
832 return E_FAIL;
833
834 TRACE("Before Query\n");
835 hr = contextMenu->QueryContextMenu(popup, 0, 0, UINT_MAX, CMF_NORMAL);
836 if (FAILED_UNEXPECTEDLY(hr))
837 {
838 TRACE("Query failed\n");
839 DestroyMenu(popup);
840 return hr;
841 }
842
843 HWND hwnd = m_menuOwner ? m_menuOwner : m_topLevelWindow;
844
845 m_focusManager->PushTrackedPopup(popup);
846
847 TRACE("Before Tracking\n");
848 uCommand = ::TrackPopupMenuEx(popup, TPM_RETURNCMD, x, y, hwnd, NULL);
849
850 m_focusManager->PopTrackedPopup(popup);
851
852 if (uCommand != 0)
853 {
854 _MenuItemSelect(MPOS_FULLCANCEL);
855
856 TRACE("Before InvokeCommand\n");
857 CMINVOKECOMMANDINFO cmi = { 0 };
858 cmi.cbSize = sizeof(cmi);
859 cmi.lpVerb = MAKEINTRESOURCEA(uCommand);
860 cmi.hwnd = hwnd;
861 hr = contextMenu->InvokeCommand(&cmi);
862 TRACE("InvokeCommand returned hr=%08x\n", hr);
863 }
864 else
865 {
866 TRACE("TrackPopupMenu failed. Code=%d, LastError=%d\n", uCommand, GetLastError());
867 hr = S_FALSE;
868 }
869
870 DestroyMenu(popup);
871 return hr;
872 }
873
874 HRESULT CMenuBand::_GetTopLevelWindow(HWND*topLevel)
875 {
876 *topLevel = m_topLevelWindow;
877 return S_OK;
878 }
879
880 HRESULT CMenuBand::_ChangeHotItem(CMenuToolbarBase * tb, INT id, DWORD dwFlags)
881 {
882 if (m_hotBar == tb && m_hotItem == id)
883 return S_FALSE;
884
885 TRACE("Hot item changed from %p %p, to %p %p\n", m_hotBar, m_hotItem, tb, id);
886
887 _KillPopupTimers();
888
889 m_hotBar = tb;
890 m_hotItem = id;
891 if (m_staticToolbar) m_staticToolbar->ChangeHotItem(tb, id, dwFlags);
892 if (m_SFToolbar) m_SFToolbar->ChangeHotItem(tb, id, dwFlags);
893
894 _MenuItemSelect(MPOS_CHILDTRACKING);
895
896 return S_OK;
897 }
898
899 HRESULT CMenuBand::_ChangePopupItem(CMenuToolbarBase * tb, INT id)
900 {
901 TRACE("Popup item changed from %p %p, to %p %p\n", m_popupBar, m_popupItem, tb, id);
902
903 m_popupBar = tb;
904 m_popupItem = id;
905 if (m_staticToolbar) m_staticToolbar->ChangePopupItem(tb, id);
906 if (m_SFToolbar) m_SFToolbar->ChangePopupItem(tb, id);
907
908 return S_OK;
909 }
910
911 HRESULT CMenuBand::_KeyboardItemChange(DWORD change)
912 {
913 HRESULT hr;
914 CMenuToolbarBase *tb = m_hotBar;
915
916 if (!tb)
917 {
918 // If no hot item was selected choose the appropriate toolbar
919 if (change == VK_UP || change == VK_END)
920 {
921 if (m_staticToolbar)
922 tb = m_staticToolbar;
923 else
924 tb = m_SFToolbar;
925 }
926 else if (change == VK_DOWN || change == VK_HOME)
927 {
928 if (m_SFToolbar)
929 tb = m_SFToolbar;
930 else
931 tb = m_staticToolbar;
932 }
933 }
934
935 // Ask the first toolbar to change
936 hr = tb->KeyboardItemChange(change);
937
938 if (hr != S_FALSE)
939 return hr;
940
941 // Select the second toolbar based on the first
942 if (tb == m_SFToolbar && m_staticToolbar)
943 tb = m_staticToolbar;
944 else if (m_SFToolbar)
945 tb = m_SFToolbar;
946
947 if (!tb)
948 return hr;
949
950 // Ask the second toolbar to change
951 return tb->KeyboardItemChange(change == VK_DOWN ? VK_HOME : VK_END);
952 }
953
954 HRESULT CMenuBand::_MenuItemSelect(DWORD changeType)
955 {
956 // Needed to prevent the this point from vanishing mid-function
957 CComPtr<CMenuBand> safeThis = this;
958 HRESULT hr;
959
960 if (m_dwFlags & SMINIT_VERTICAL)
961 {
962 switch (changeType)
963 {
964 case VK_UP:
965 case VK_DOWN:
966 return _KeyboardItemChange(changeType);
967
968 // TODO: Left/Right across multi-column menus, if they ever work.
969 case VK_LEFT:
970 changeType = MPOS_SELECTLEFT;
971 break;
972 case VK_RIGHT:
973 changeType = MPOS_SELECTRIGHT;
974 break;
975 }
976 }
977 else
978 {
979 // In horizontal menubars, left/right are equivalent to vertical's up/down
980 switch (changeType)
981 {
982 case VK_LEFT:
983 hr = _KeyboardItemChange(VK_UP);
984 if (hr != S_FALSE)
985 return hr;
986 case VK_RIGHT:
987 hr = _KeyboardItemChange(VK_DOWN);
988 if (hr != S_FALSE)
989 return hr;
990 }
991 }
992
993 // In this context, the parent is the CMenuDeskBar, so when it bubbles upward,
994 // it is notifying the deskbar, and not the the higher-level menu.
995 // Same for the child: since it points to a CMenuDeskBar, it's not just recursing.
996 switch (changeType)
997 {
998 case MPOS_EXECUTE:
999 {
1000 CMenuToolbarBase * tb = m_hotBar;
1001 int item = m_hotItem;
1002 tb->PrepareExecuteItem(item);
1003 if (m_subMenuParent)
1004 {
1005 m_subMenuParent->OnSelect(changeType);
1006 }
1007 TRACE("Menu closed, executing item...\n");
1008 tb->ExecuteItem();
1009 break;
1010 }
1011 case MPOS_SELECTLEFT:
1012 if (m_parentBand && m_parentBand->_IsPopup()==S_FALSE)
1013 return m_parentBand->_MenuItemSelect(VK_LEFT);
1014 if (m_subMenuChild)
1015 return m_subMenuChild->OnSelect(MPOS_CANCELLEVEL);
1016 if (!m_subMenuParent)
1017 return S_OK;
1018 return m_subMenuParent->OnSelect(MPOS_CANCELLEVEL);
1019
1020 case MPOS_SELECTRIGHT:
1021 if (m_hotBar && m_hotItem >= 0 && m_hotBar->PopupItem(m_hotItem, TRUE) == S_OK)
1022 return S_FALSE;
1023 if (m_parentBand)
1024 return m_parentBand->_MenuItemSelect(VK_RIGHT);
1025 if (!m_subMenuParent)
1026 return S_OK;
1027 return m_subMenuParent->OnSelect(MPOS_SELECTRIGHT);
1028
1029 default:
1030 if (!m_subMenuParent)
1031 return S_OK;
1032 return m_subMenuParent->OnSelect(changeType);
1033 }
1034
1035 return S_OK;
1036 }
1037
1038 HRESULT CMenuBand::_CancelCurrentPopup()
1039 {
1040 if (m_subMenuChild)
1041 {
1042 HRESULT hr = m_subMenuChild->OnSelect(MPOS_CANCELLEVEL);
1043 return hr;
1044 }
1045
1046 if (m_trackedPopup)
1047 {
1048 ::SendMessage(m_trackedHwnd, WM_CANCELMODE, 0, 0);
1049 return S_OK;
1050 }
1051
1052 return S_FALSE;
1053 }
1054
1055 HRESULT CMenuBand::_OnPopupSubMenu(IShellMenu * childShellMenu, POINTL * pAt, RECTL * pExclude, BOOL keyInitiated)
1056 {
1057 HRESULT hr = 0;
1058 CComPtr<IBandSite> pBandSite;
1059 CComPtr<IDeskBar> pDeskBar;
1060
1061 // Create the necessary objects
1062 hr = CMenuSite_CreateInstance(IID_PPV_ARG(IBandSite, &pBandSite));
1063 if (FAILED_UNEXPECTEDLY(hr))
1064 return hr;
1065
1066 hr = CMenuDeskBar_CreateInstance(IID_PPV_ARG(IDeskBar, &pDeskBar));
1067 if (FAILED_UNEXPECTEDLY(hr))
1068 return hr;
1069
1070 hr = pDeskBar->SetClient(pBandSite);
1071 if (FAILED_UNEXPECTEDLY(hr))
1072 return hr;
1073
1074 hr = pBandSite->AddBand(childShellMenu);
1075 if (FAILED_UNEXPECTEDLY(hr))
1076 return hr;
1077
1078 //
1079 CComPtr<IMenuPopup> popup;
1080 hr = pDeskBar->QueryInterface(IID_PPV_ARG(IMenuPopup, &popup));
1081 if (FAILED_UNEXPECTEDLY(hr))
1082 return hr;
1083
1084 m_subMenuChild = popup;
1085
1086 if (m_subMenuParent)
1087 IUnknown_SetSite(popup, m_subMenuParent);
1088 else
1089 IUnknown_SetSite(popup, m_site);
1090
1091 DWORD flags = MPPF_RIGHT;
1092
1093 if (keyInitiated && m_dwFlags & SMINIT_VERTICAL)
1094 flags |= MPPF_INITIALSELECT;
1095
1096 popup->Popup(pAt, pExclude, flags);
1097
1098 return S_OK;
1099 }
1100
1101 HRESULT CMenuBand::_BeforeCancelPopup()
1102 {
1103 if (m_staticToolbar)
1104 m_staticToolbar->BeforeCancelPopup();
1105 if (m_SFToolbar)
1106 m_SFToolbar->BeforeCancelPopup();
1107 return S_OK;
1108 }
1109
1110 HRESULT CMenuBand::_DisableMouseTrack(BOOL bDisable)
1111 {
1112 if (m_staticToolbar)
1113 m_staticToolbar->DisableMouseTrack(bDisable);
1114 if (m_SFToolbar)
1115 m_SFToolbar->DisableMouseTrack(bDisable);
1116 return S_OK;
1117 }
1118
1119 HRESULT CMenuBand::_KillPopupTimers()
1120 {
1121 HRESULT hr = S_OK;
1122 if (m_staticToolbar)
1123 hr = m_staticToolbar->KillPopupTimer();
1124 if (FAILED(hr))
1125 return hr;
1126
1127 if (m_SFToolbar)
1128 hr = m_SFToolbar->KillPopupTimer();
1129
1130 return hr;
1131 }
1132
1133 HRESULT CMenuBand::_MenuBarMouseDown(HWND hwnd, INT item, BOOL isLButton)
1134 {
1135 if (m_staticToolbar && m_staticToolbar->IsWindowOwner(hwnd) == S_OK)
1136 m_staticToolbar->MenuBarMouseDown(item, isLButton);
1137 if (m_SFToolbar && m_SFToolbar->IsWindowOwner(hwnd) == S_OK)
1138 m_SFToolbar->MenuBarMouseDown(item, isLButton);
1139 return S_OK;
1140 }
1141
1142 HRESULT CMenuBand::_MenuBarMouseUp(HWND hwnd, INT item, BOOL isLButton)
1143 {
1144 if (m_staticToolbar && m_staticToolbar->IsWindowOwner(hwnd) == S_OK)
1145 m_staticToolbar->MenuBarMouseUp(item, isLButton);
1146 if (m_SFToolbar && m_SFToolbar->IsWindowOwner(hwnd) == S_OK)
1147 m_SFToolbar->MenuBarMouseUp(item, isLButton);
1148 return S_OK;
1149 }
1150
1151 HRESULT CMenuBand::_HasSubMenu()
1152 {
1153 return m_popupBar ? S_OK : S_FALSE;
1154 }
1155
1156 HRESULT CMenuBand::AdjustForTheme(BOOL bFlatStyle)
1157 {
1158 return IUnknown_QueryServiceExec(m_site, SID_SMenuPopup, &CGID_MenuDeskBar, 4, bFlatStyle, NULL, NULL);
1159 }
1160
1161 HRESULT STDMETHODCALLTYPE CMenuBand::InvalidateItem(LPSMDATA psmd, DWORD dwFlags)
1162 {
1163 UNIMPLEMENTED;
1164 return S_OK;
1165 }
1166
1167 HRESULT STDMETHODCALLTYPE CMenuBand::GetState(LPSMDATA psmd)
1168 {
1169 UNIMPLEMENTED;
1170 return S_OK;
1171 }
1172
1173 HRESULT STDMETHODCALLTYPE CMenuBand::SetMenuToolbar(IUnknown *punk, DWORD dwFlags)
1174 {
1175 UNIMPLEMENTED;
1176 return S_OK;
1177 }
1178
1179 HRESULT STDMETHODCALLTYPE CMenuBand::ResizeBorderDW(LPCRECT prcBorder, IUnknown *punkToolbarSite, BOOL fReserved)
1180 {
1181 UNIMPLEMENTED;
1182 return S_OK;
1183 }
1184
1185 HRESULT STDMETHODCALLTYPE CMenuBand::ContextSensitiveHelp(BOOL fEnterMode)
1186 {
1187 UNIMPLEMENTED;
1188 return S_OK;
1189 }
1190
1191 HRESULT STDMETHODCALLTYPE CMenuBand::GetSubMenu(THIS)
1192 {
1193 UNIMPLEMENTED;
1194 return S_OK;
1195 }
1196
1197 HRESULT STDMETHODCALLTYPE CMenuBand::SetToolbar(THIS)
1198 {
1199 UNIMPLEMENTED;
1200 return S_OK;
1201 }
1202
1203 HRESULT STDMETHODCALLTYPE CMenuBand::SetMinWidth(THIS)
1204 {
1205 UNIMPLEMENTED;
1206 return S_OK;
1207 }
1208
1209 HRESULT STDMETHODCALLTYPE CMenuBand::SetNoBorder(THIS)
1210 {
1211 UNIMPLEMENTED;
1212 return S_OK;
1213 }
1214
1215 HRESULT STDMETHODCALLTYPE CMenuBand::SetTheme(THIS)
1216 {
1217 UNIMPLEMENTED;
1218 return S_OK;
1219 }
1220
1221 HRESULT STDMETHODCALLTYPE CMenuBand::GetTop(THIS)
1222 {
1223 UNIMPLEMENTED;
1224 return S_OK;
1225 }
1226
1227 HRESULT STDMETHODCALLTYPE CMenuBand::GetBottom(THIS)
1228 {
1229 UNIMPLEMENTED;
1230 return S_OK;
1231 }
1232
1233 HRESULT STDMETHODCALLTYPE CMenuBand::GetTracked(THIS)
1234 {
1235 UNIMPLEMENTED;
1236 return S_OK;
1237 }
1238
1239 HRESULT STDMETHODCALLTYPE CMenuBand::GetParentSite(THIS)
1240 {
1241 UNIMPLEMENTED;
1242 return S_OK;
1243 }
1244
1245 HRESULT STDMETHODCALLTYPE CMenuBand::GetState(THIS)
1246 {
1247 UNIMPLEMENTED;
1248 return S_OK;
1249 }
1250
1251 HRESULT STDMETHODCALLTYPE CMenuBand::DoDefaultAction(THIS)
1252 {
1253 UNIMPLEMENTED;
1254 return S_OK;
1255 }
1256
1257 HRESULT STDMETHODCALLTYPE CMenuBand::IsEmpty(THIS)
1258 {
1259 UNIMPLEMENTED;
1260 return S_OK;
1261 }
1262
1263 HRESULT STDMETHODCALLTYPE CMenuBand::HasFocusIO()
1264 {
1265 if (m_popupBar)
1266 return S_OK;
1267 return S_FALSE;
1268 }
1269
1270 HRESULT STDMETHODCALLTYPE CMenuBand::TranslateAcceleratorIO(LPMSG lpMsg)
1271 {
1272 // TODO: Alt down -> toggle menu focus
1273 return S_FALSE;
1274 }
1275
1276 HRESULT STDMETHODCALLTYPE CMenuBand::IsDirty()
1277 {
1278 UNIMPLEMENTED;
1279 return S_OK;
1280 }
1281
1282 HRESULT STDMETHODCALLTYPE CMenuBand::Load(IStream *pStm)
1283 {
1284 UNIMPLEMENTED;
1285 return S_OK;
1286 }
1287
1288 HRESULT STDMETHODCALLTYPE CMenuBand::Save(IStream *pStm, BOOL fClearDirty)
1289 {
1290 UNIMPLEMENTED;
1291 return S_OK;
1292 }
1293
1294 HRESULT STDMETHODCALLTYPE CMenuBand::GetSizeMax(ULARGE_INTEGER *pcbSize)
1295 {
1296 UNIMPLEMENTED;
1297 return S_OK;
1298 }
1299
1300 HRESULT STDMETHODCALLTYPE CMenuBand::GetClassID(CLSID *pClassID)
1301 {
1302 UNIMPLEMENTED;
1303 return S_OK;
1304 }
1305
1306 HRESULT STDMETHODCALLTYPE CMenuBand::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds [], OLECMDTEXT *pCmdText)
1307 {
1308 UNIMPLEMENTED;
1309 return S_OK;
1310 }
1311
1312 extern "C"
1313 HRESULT WINAPI RSHELL_CMenuBand_CreateInstance(REFIID riid, LPVOID *ppv)
1314 {
1315 return ShellObjectCreator<CMenuBand>(riid, ppv);
1316 }