/*
-* Shell Menu Desk Bar
-*
-* Copyright 2014 David Quintana
-*
-* This library is free software; you can redistribute it and/or
-* modify it under the terms of the GNU Lesser General Public
-* License as published by the Free Software Foundation; either
-* version 2.1 of the License, or (at your option) any later version.
-*
-* This library is distributed in the hope that it will be useful,
-* but WITHOUT ANY WARRANTY; without even the implied warranty of
-* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
-* Lesser General Public License for more details.
-*
-* You should have received a copy of the GNU Lesser General Public
-* License along with this library; if not, write to the Free Software
-* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
-*/
+ * Shell Menu Desk Bar
+ *
+ * Copyright 2014 David Quintana
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
+ */
#include "precomp.h"
-#include "wraplog.h"
+#include <atlwin.h>
+#include <shlwapi_undoc.h>
+
+#include "CMenuDeskBar.h"
WINE_DEFAULT_DEBUG_CHANNEL(CMenuDeskBar);
-#define WRAP_LOG 1
-
-class CMenuDeskBar :
- public CComCoClass<CMenuDeskBar>,
- public CComObjectRootEx<CComMultiThreadModelNoCS>,
- public IOleCommandTarget,
- public IServiceProvider,
- public IInputObjectSite,
- public IInputObject,
- public IMenuPopup,
- public IObjectWithSite,
- public IBanneredBar,
- public IInitializeObject
-{
-public:
- CMenuDeskBar();
- ~CMenuDeskBar();
-
-private:
-#if WRAP_LOG
- IUnknown * m_IUnknown;
- IMenuPopup * m_IMenuPopup;
- IOleCommandTarget * m_IOleCommandTarget;
- IServiceProvider * m_IServiceProvider;
- IDeskBar * m_IDeskBar;
- IOleWindow * m_IOleWindow;
- IInputObjectSite * m_IInputObjectSite;
- IInputObject * m_IInputObject;
- IObjectWithSite * m_IObjectWithSite;
- IBanneredBar * m_IBanneredBar;
- IInitializeObject * m_IInitializeObject;
-#else
-#endif
-
-public:
- // *** IMenuPopup methods ***
- virtual HRESULT STDMETHODCALLTYPE Popup(POINTL *ppt, RECTL *prcExclude, MP_POPUPFLAGS dwFlags);
- virtual HRESULT STDMETHODCALLTYPE OnSelect(DWORD dwSelectType);
- virtual HRESULT STDMETHODCALLTYPE SetSubMenu(IMenuPopup *pmp, BOOL fSet);
-
- // *** IOleWindow methods ***
- virtual HRESULT STDMETHODCALLTYPE GetWindow(HWND *phwnd);
- virtual HRESULT STDMETHODCALLTYPE ContextSensitiveHelp(BOOL fEnterMode);
-
- // *** IObjectWithSite methods ***
- virtual HRESULT STDMETHODCALLTYPE SetSite(IUnknown *pUnkSite);
- virtual HRESULT STDMETHODCALLTYPE GetSite(REFIID riid, PVOID *ppvSite);
-
- // *** IBanneredBar methods ***
- virtual HRESULT STDMETHODCALLTYPE SetIconSize(DWORD iIcon);
- virtual HRESULT STDMETHODCALLTYPE GetIconSize(DWORD* piIcon);
- virtual HRESULT STDMETHODCALLTYPE SetBitmap(HBITMAP hBitmap);
- virtual HRESULT STDMETHODCALLTYPE GetBitmap(HBITMAP* phBitmap);
-
- // *** IInitializeObject methods ***
- virtual HRESULT STDMETHODCALLTYPE Initialize(THIS);
-
- // *** IOleCommandTarget methods ***
- virtual HRESULT STDMETHODCALLTYPE QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds [], OLECMDTEXT *pCmdText);
- virtual HRESULT STDMETHODCALLTYPE Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut);
-
- // *** IServiceProvider methods ***
- virtual HRESULT STDMETHODCALLTYPE QueryService(REFGUID guidService, REFIID riid, void **ppvObject);
-
- // *** IInputObjectSite methods ***
- virtual HRESULT STDMETHODCALLTYPE OnFocusChangeIS(LPUNKNOWN lpUnknown, BOOL bFocus);
-
- // *** IInputObject methods ***
- virtual HRESULT STDMETHODCALLTYPE UIActivateIO(BOOL bActivating, LPMSG lpMsg);
- virtual HRESULT STDMETHODCALLTYPE HasFocusIO(THIS);
- virtual HRESULT STDMETHODCALLTYPE TranslateAcceleratorIO(LPMSG lpMsg);
-
- // *** IDeskBar methods ***
- virtual HRESULT STDMETHODCALLTYPE SetClient(IUnknown *punkClient);
- virtual HRESULT STDMETHODCALLTYPE GetClient(IUnknown **ppunkClient);
- virtual HRESULT STDMETHODCALLTYPE OnPosRectChangeDB(LPRECT prc);
-
-
- DECLARE_NOT_AGGREGATABLE(CMenuDeskBar)
- DECLARE_PROTECT_FINAL_CONSTRUCT()
-
- BEGIN_COM_MAP(CMenuDeskBar)
- COM_INTERFACE_ENTRY_IID(IID_IMenuPopup, IMenuPopup)
- COM_INTERFACE_ENTRY_IID(IID_IOleCommandTarget, IOleCommandTarget)
- COM_INTERFACE_ENTRY_IID(IID_IServiceProvider, IServiceProvider)
- COM_INTERFACE_ENTRY_IID(IID_IInputObjectSite, IInputObjectSite)
- COM_INTERFACE_ENTRY_IID(IID_IInputObject, IInputObject)
- COM_INTERFACE_ENTRY_IID(IID_IDeskBar, IMenuPopup)
- COM_INTERFACE_ENTRY_IID(IID_IOleWindow, IMenuPopup)
- COM_INTERFACE_ENTRY_IID(IID_IObjectWithSite, IObjectWithSite)
- COM_INTERFACE_ENTRY_IID(IID_IBanneredBar, IBanneredBar)
- COM_INTERFACE_ENTRY_IID(IID_IInitializeObject, IInitializeObject)
- END_COM_MAP()
-};
+const static GUID CGID_MenuDeskBar = { 0x5C9F0A12, 0x959E, 0x11D0, { 0xA3, 0xA4, 0x00, 0xA0, 0xC9, 0x08, 0x26, 0x36 } };
extern "C"
-HRESULT CMenuDeskBar_Constructor(REFIID riid, LPVOID *ppv)
+HRESULT WINAPI CMenuDeskBar_Constructor(REFIID riid, LPVOID *ppv)
{
- *ppv = NULL;
+ return ShellObjectCreator<CMenuDeskBar>(riid, ppv);
+}
- CMenuDeskBar * deskbar = new CComObject<CMenuDeskBar>();
+CMenuDeskBar::CMenuDeskBar() :
+ m_Client(NULL),
+ m_ClientWindow(NULL),
+ m_IconSize(0),
+ m_Banner(NULL),
+ m_Shown(FALSE),
+ m_ShowFlags(0),
+ m_didAddRef(FALSE)
+{
+}
- if (!deskbar)
- return E_OUTOFMEMORY;
+CMenuDeskBar::~CMenuDeskBar()
+{
+}
- HRESULT hr = deskbar->QueryInterface(riid, ppv);
+LRESULT CMenuDeskBar::_OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
+{
+ if (!m_didAddRef)
+ {
+ this->AddRef();
+ m_didAddRef = TRUE;
+ }
- if (FAILED(hr))
- deskbar->Release();
+ bHandled = FALSE;
+ return 0;
+}
- return hr;
+void CMenuDeskBar::OnFinalMessage(HWND /* hWnd */)
+{
+ if (m_didAddRef)
+ {
+ this->Release();
+ m_didAddRef = FALSE;
+ }
}
-CMenuDeskBar::CMenuDeskBar()
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::Initialize(THIS)
{
- HRESULT hr;
- WrapLogOpen();
+ return S_OK;
+}
- hr = CoCreateInstance(CLSID_MenuDeskBar, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IMenuPopup, &m_IMenuPopup));
- hr = m_IMenuPopup->QueryInterface(IID_PPV_ARG(IUnknown, &m_IUnknown));
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetWindow(HWND *lphwnd)
+{
+ if (lphwnd == NULL)
+ return E_POINTER;
+ *lphwnd = m_hWnd;
+ return S_OK;
+}
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IOleCommandTarget, &m_IOleCommandTarget));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IServiceProvider, &m_IServiceProvider));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IDeskBar, &m_IDeskBar));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IOleWindow, &m_IOleWindow));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IInputObjectSite, &m_IInputObjectSite));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IInputObject, &m_IInputObject));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IObjectWithSite, &m_IObjectWithSite));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IBanneredBar, &m_IBanneredBar));
- hr = m_IUnknown->QueryInterface(IID_PPV_ARG(IInitializeObject, &m_IInitializeObject));
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::ContextSensitiveHelp(BOOL fEnterMode)
+{
+ return E_NOTIMPL;
}
-CMenuDeskBar::~CMenuDeskBar()
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnFocusChangeIS(IUnknown *punkObj, BOOL fSetFocus)
{
- m_IUnknown->Release();
- m_IMenuPopup->Release();
- m_IOleCommandTarget->Release();
- m_IServiceProvider->Release();
- m_IDeskBar->Release();
- m_IOleWindow->Release();
- m_IInputObjectSite->Release();
- m_IInputObject->Release();
- m_IObjectWithSite->Release();
- m_IBanneredBar->Release();
- m_IInitializeObject->Release();
+ CComPtr<IInputObjectSite> ios;
+
+ HRESULT hr = m_Client->QueryInterface(IID_PPV_ARG(IInputObjectSite, &ios));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
- WrapLogClose();
+ return ios->OnFocusChangeIS(punkObj, fSetFocus);
}
-// *** IMenuPopup methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::Popup(POINTL *ppt, RECTL *prcExclude, MP_POPUPFLAGS dwFlags)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds,
+ OLECMD prgCmds [], OLECMDTEXT *pCmdText)
{
- WrapLogEnter("CMenuDeskBar<%p>::Popup(POINTL *ppt=%p, RECTL *prcExclude=%p, MP_POPUPFLAGS dwFlags=%08x)\n", this, ppt, prcExclude, dwFlags);
- HRESULT hr = m_IMenuPopup->Popup(ppt, prcExclude, dwFlags);
- WrapLogExit("CMenuDeskBar::Popup() = %08x\n", hr);
- return hr;
+ return E_NOTIMPL;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnSelect(DWORD dwSelectType)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::Exec(const GUID *pguidCmdGroup, DWORD nCmdID,
+ DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
{
- WrapLogEnter("CMenuDeskBar<%p>::OnSelect(DWORD dwSelectType=%08x)\n", this, dwSelectType);
- HRESULT hr = m_IMenuPopup->OnSelect(dwSelectType);
- WrapLogExit("CMenuDeskBar::OnSelect() = %08x\n", hr);
- return hr;
+ if (IsEqualIID(*pguidCmdGroup, CGID_MenuDeskBar))
+ {
+ switch (nCmdID)
+ {
+ case 2: // refresh
+ return S_OK;
+ case 3: // load complete
+ return S_OK;
+ case 4: // set font metrics
+ return S_OK;
+ }
+ }
+ if (IsEqualIID(*pguidCmdGroup, CGID_Explorer))
+ {
+ }
+ else if (IsEqualIID(*pguidCmdGroup, IID_IDeskBarClient))
+ {
+ switch (nCmdID)
+ {
+ case 0:
+ // hide current band
+ break;
+ case 2:
+ break;
+ case 3:
+ break;
+ }
+ }
+ return E_NOTIMPL;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetSubMenu(IMenuPopup *pmp, BOOL fSet)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::QueryService(REFGUID guidService, REFIID riid, void **ppvObject)
{
- WrapLogEnter("CMenuDeskBar<%p>::SetSubMenu(IMenuPopup *pmp=%p, BOOL fSet=%d)\n", this, pmp, fSet);
- HRESULT hr = m_IMenuPopup->SetSubMenu(pmp, fSet);
- WrapLogExit("CMenuDeskBar::SetSubMenu() = %08x\n", hr);
- return hr;
+ HRESULT hr;
+
+ if (IsEqualGUID(guidService, SID_SMenuPopup) ||
+ IsEqualGUID(guidService, SID_SMenuBandParent) ||
+ IsEqualGUID(guidService, SID_STopLevelBrowser))
+ {
+ hr = this->QueryInterface(riid, ppvObject);
+ if (SUCCEEDED(hr))
+ return hr;
+ }
+
+ if (IsEqualGUID(guidService, SID_SMenuBandBottom) ||
+ IsEqualGUID(guidService, SID_SMenuBandBottomSelected) ||
+ IsEqualGUID(guidService, SID_SMenuBandChild))
+ {
+ if (m_Client == NULL)
+ return E_NOINTERFACE;
+
+ hr = IUnknown_QueryService(m_Client, guidService, riid, ppvObject);
+ if (SUCCEEDED(hr))
+ return hr;
+ }
+
+
+ if (m_Site == NULL)
+ return E_NOINTERFACE;
+
+ return IUnknown_QueryService(m_Site, guidService, riid, ppvObject);
}
-// *** IOleWindow methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetWindow(HWND *phwnd)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
{
- WrapLogEnter("CMenuDeskBar<%p>::GetWindow(HWND *phwnd=%p)\n", this, phwnd);
- HRESULT hr = m_IOleWindow->GetWindow(phwnd);
- if (phwnd) WrapLogMsg("*phwnd=%p\n", *phwnd);
- WrapLogExit("CMenuDeskBar::GetWindow() = %08x\n", hr);
- return hr;
+ return IUnknown_UIActivateIO(m_Client, fActivate, lpMsg);
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::ContextSensitiveHelp(BOOL fEnterMode)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::HasFocusIO()
{
- WrapLogEnter("CMenuDeskBar<%p>::ContextSensitiveHelp(BOOL fEnterMode=%d)\n", this, fEnterMode);
- HRESULT hr = m_IOleWindow->ContextSensitiveHelp(fEnterMode);
- WrapLogExit("CMenuDeskBar::ContextSensitiveHelp() = %08x\n", hr);
- return hr;
+ CComPtr<IInputObject> io;
+
+ HRESULT hr = m_Client->QueryInterface(IID_PPV_ARG(IInputObject, &io));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ return io->HasFocusIO();
}
-// *** IObjectWithSite methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetSite(IUnknown *pUnkSite)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::TranslateAcceleratorIO(LPMSG lpMsg)
{
- WrapLogEnter("CMenuDeskBar<%p>::SetSite(IUnknown *pUnkSite=%p)\n", this, pUnkSite);
- HRESULT hr = m_IObjectWithSite->SetSite(pUnkSite);
- WrapLogExit("CMenuDeskBar::SetSite() = %08x\n", hr);
- return hr;
+ CComPtr<IInputObject> io;
+
+ HRESULT hr = m_Client->QueryInterface(IID_PPV_ARG(IInputObject, &io));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ return io->TranslateAcceleratorIO(lpMsg);
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetSite(REFIID riid, PVOID *ppvSite)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetClient(IUnknown *punkClient)
{
- WrapLogEnter("CMenuDeskBar<%p>::GetSite(REFIID riid=%s, PVOID *ppvSite=%p)\n", this, Wrap(riid), ppvSite);
- HRESULT hr = m_IObjectWithSite->GetSite(riid, ppvSite);
- if (ppvSite) WrapLogMsg("*ppvSite=%p\n", *ppvSite);
- WrapLogExit("CMenuDeskBar::GetSite() = %08x\n", hr);
- return hr;
+ CComPtr<IDeskBarClient> pDeskBandClient;
+ HRESULT hr;
+
+ if (m_Client)
+ {
+ hr = m_Client->QueryInterface(IID_PPV_ARG(IDeskBarClient, &pDeskBandClient));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ pDeskBandClient->SetDeskBarSite(NULL);
+
+ pDeskBandClient = NULL;
+ m_Client = NULL;
+ }
+
+ if (punkClient == NULL)
+ return S_OK;
+
+ if (m_hWnd == NULL)
+ {
+ Create(NULL);
+ }
+
+ hr = punkClient->QueryInterface(IID_PPV_ARG(IUnknown, &m_Client));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ hr = m_Client->QueryInterface(IID_PPV_ARG(IDeskBarClient, &pDeskBandClient));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ hr = pDeskBandClient->SetDeskBarSite(static_cast<IDeskBar*>(this));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ return IUnknown_GetWindow(m_Client, &m_ClientWindow);
}
-// *** IBanneredBar methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetIconSize(DWORD iIcon)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetClient(IUnknown **ppunkClient)
{
- WrapLogEnter("CMenuDeskBar<%p>::SetIconSize(DWORD iIcon=%d)\n", this, iIcon);
- HRESULT hr = m_IBanneredBar->SetIconSize(iIcon);
- WrapLogExit("CMenuDeskBar::SetIconSize() = %08x\n", hr);
- return hr;
+ if (ppunkClient == NULL)
+ return E_POINTER;
+
+ if (!m_Client)
+ return E_FAIL;
+
+ return m_Client->QueryInterface(IID_PPV_ARG(IUnknown, ppunkClient));
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetIconSize(DWORD* piIcon)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnPosRectChangeDB(LPRECT prc)
{
- WrapLogEnter("CMenuDeskBar<%p>::GetIconSize(DWORD* piIcon=%p)\n", this, piIcon);
- HRESULT hr = m_IBanneredBar->GetIconSize(piIcon);
- if (piIcon) WrapLogMsg("*piIcon=%d\n", *piIcon);
- WrapLogExit("CMenuDeskBar::GetIconSize() = %08x\n", hr);
- return hr;
+ if (prc == NULL)
+ return E_POINTER;
+
+ return S_OK;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetBitmap(HBITMAP hBitmap)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetSite(IUnknown *pUnkSite)
{
- WrapLogEnter("CMenuDeskBar<%p>::SetBitmap(HBITMAP hBitmap=%p)\n", this, hBitmap);
- HRESULT hr = m_IBanneredBar->SetBitmap(hBitmap);
- WrapLogExit("CMenuDeskBar::SetBitmap() = %08x\n", hr);
- return hr;
+ // Windows closes the bar if this is called when the bar is shown
+
+ if (m_Shown)
+ _CloseBar();
+
+ m_SubMenuParent = NULL;
+
+ m_Site = pUnkSite;
+
+ if (m_Site)
+ {
+ IUnknown_QueryService(m_Site, SID_SMenuPopup, IID_PPV_ARG(IMenuPopup, &m_SubMenuParent));
+ }
+ else
+ {
+ SetClient(NULL);
+ DestroyWindow();
+ }
+
+ return S_OK;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetBitmap(HBITMAP* phBitmap)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetSite(REFIID riid, void **ppvSite)
{
- WrapLogEnter("CMenuDeskBar<%p>::GetBitmap(HBITMAP* phBitmap=%p)\n", this, phBitmap);
- HRESULT hr = m_IBanneredBar->GetBitmap(phBitmap);
- if (phBitmap) WrapLogMsg("*phBitmap=%p\n", *phBitmap);
- WrapLogExit("CMenuDeskBar::GetBitmap() = %08x\n", hr);
- return hr;
+ if (m_Site == NULL)
+ return E_FAIL;
+
+ return m_Site->QueryInterface(riid, ppvSite);
}
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::Popup(POINTL *ppt, RECTL *prcExclude, MP_POPUPFLAGS dwFlags)
+{
+ HRESULT hr;
+ CComPtr<IOleCommandTarget> oct;
+ CComPtr<IInputObject> io;
+ CComPtr<IDeskBand> band;
+ CComPtr<IDeskBarClient> dbc;
+
+ if (m_hWnd == NULL)
+ return E_FAIL;
-// *** IInitializeObject methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::Initialize(THIS)
+ hr = IUnknown_QueryService(m_Client, SID_SMenuBandChild, IID_PPV_ARG(IOleCommandTarget, &oct));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ hr = m_Client->QueryInterface(IID_PPV_ARG(IDeskBarClient, &dbc));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ // Windows calls this, but it appears to be unimplemented?
+ hr = dbc->SetModeDBC(1);
+ // Allow it to fail with E_NOTIMPL.
+
+ // No clue about the arg, using anything != 0
+ hr = dbc->UIActivateDBC(TRUE);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ RECT rc = { 0 };
+ hr = dbc->GetSize(0, &rc);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ // Unknown meaning
+ const int CMD = 19;
+ const int CMD_EXEC_OPT = 0;
+
+ hr = IUnknown_QueryServiceExec(m_Client, SID_SMenuBandChild, &CLSID_MenuBand, CMD, CMD_EXEC_OPT, NULL, NULL);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ ::AdjustWindowRect(&rc, ::GetWindowLong(m_hWnd, GWL_STYLE), FALSE);
+ ::OffsetRect(&rc, -rc.left, -rc.top);
+
+ if (m_Banner != NULL)
+ {
+ BITMAP bm;
+ ::GetObject(m_Banner, sizeof(bm), &bm);
+ rc.right += bm.bmWidth;
+ }
+
+ RECT rcWorkArea;
+ GetWindowRect(GetDesktopWindow(), &rcWorkArea);
+ int waHeight = rcWorkArea.bottom - rcWorkArea.top;
+
+ int x = ppt->x;
+ int y = ppt->y;
+ int cx = rc.right - rc.left;
+ int cy = rc.bottom - rc.top;
+
+ switch (dwFlags & 0xFF000000)
+ {
+ case MPPF_BOTTOM:
+ x = ppt->x;
+ y = ppt->y - rc.bottom;
+ break;
+ case MPPF_RIGHT:
+ x = ppt->x + rc.left;
+ y = ppt->y + rc.top;
+ break;
+ case MPPF_TOP | MPPF_ALIGN_LEFT:
+ x = ppt->x - rc.right;
+ y = ppt->y + rc.top;
+ break;
+ case MPPF_TOP | MPPF_ALIGN_RIGHT:
+ x = ppt->x;
+ y = ppt->y + rc.top;
+ break;
+ }
+
+ if (x + cx > rcWorkArea.right)
+ {
+ // FIXME: Works, but it's oversimplified.
+ x = prcExclude->left - cx;
+ dwFlags = (dwFlags & (~MPPF_TOP)) | MPPF_LEFT;
+ }
+
+ if (y < rcWorkArea.top)
+ {
+ y = rcWorkArea.top;
+ }
+
+ if (cy > waHeight)
+ {
+ cy = waHeight;
+ }
+
+ if (y + cy > rcWorkArea.bottom)
+ {
+ y = rcWorkArea.bottom - cy;
+ }
+
+ this->SetWindowPos(HWND_TOPMOST, x, y, cx, cy, SWP_SHOWWINDOW);
+
+ m_ShowFlags = dwFlags;
+ m_Shown = true;
+
+ // HACK: The bar needs to be notified of the size AFTER it is shown.
+ // Quick & dirty way of getting it done.
+ BOOL bHandled;
+ _OnSize(WM_SIZE, 0, 0, bHandled);
+
+ UIActivateIO(TRUE, NULL);
+
+ if (dwFlags & (MPPF_INITIALSELECT | MPPF_FINALSELECT))
+ {
+ const int CMD_SELECT = 5;
+ int CMD_SELECT_OPTS = dwFlags & MPPF_INITIALSELECT ? 0 : -2;
+ IUnknown_QueryServiceExec(m_Client, SID_SMenuBandChild, &CLSID_MenuBand, CMD_SELECT, CMD_SELECT_OPTS, NULL, NULL);
+ }
+
+ return S_OK;
+}
+
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetIconSize(THIS_ DWORD iIcon)
{
- WrapLogEnter("CMenuDeskBar<%p>::Initialize()\n", this);
- HRESULT hr = m_IInitializeObject->Initialize();
- WrapLogExit("CMenuDeskBar::Initialize() = %08x\n", hr);
+ HRESULT hr;
+ m_IconSize = iIcon;
+
+ // Unknown meaning (set flags? set icon size?)
+ const int CMD = 16;
+ const int CMD_EXEC_OPT = iIcon ? 0 : 2; // seems to work
+
+ hr = IUnknown_QueryServiceExec(m_Client, SID_SMenuBandChild, &CLSID_MenuBand, CMD, CMD_EXEC_OPT, NULL, NULL);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ BOOL bHandled;
+ _OnSize(WM_SIZE, 0, 0, bHandled);
+
return hr;
}
-// *** IOleCommandTarget methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds [], OLECMDTEXT *pCmdText)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetIconSize(THIS_ DWORD* piIcon)
{
- WrapLogEnter("CMenuDeskBar<%p>::QueryStatus(const GUID *pguidCmdGroup=%p, ULONG cCmds=%u, prgCmds=%p, pCmdText=%p)\n", this, pguidCmdGroup, cCmds, prgCmds, pCmdText);
- HRESULT hr = m_IOleCommandTarget->QueryStatus(pguidCmdGroup, cCmds, prgCmds, pCmdText);
- if (pguidCmdGroup) WrapLogMsg("*pguidCmdGroup=%s\n", Wrap(*pguidCmdGroup));
- WrapLogExit("CMenuDeskBar::QueryStatus() = %08x\n", hr);
- return hr;
+ if (piIcon)
+ *piIcon = m_IconSize;
+ return S_OK;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetBitmap(THIS_ HBITMAP hBitmap)
{
- //bool b;
+ m_Banner = hBitmap;
- WrapLogEnter("CMenuDeskBar<%p>::Exec(const GUID *pguidCmdGroup=%p, DWORD nCmdID=%d, DWORD nCmdexecopt=%d, VARIANT *pvaIn=%p, VARIANT *pvaOut=%p)\n", this, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut);
+ BOOL bHandled;
+ _OnSize(WM_SIZE, 0, 0, bHandled);
- //if (pguidCmdGroup && IsEqualGUID(*pguidCmdGroup, CLSID_MenuBand))
- //{
- // if (nCmdID == 19) // popup
- // {
- // b = true;
- // }
- //}
+ return S_OK;
+}
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetBitmap(THIS_ HBITMAP* phBitmap)
+{
+ if (phBitmap)
+ *phBitmap = m_Banner;
+ return S_OK;
+}
- if (pguidCmdGroup) WrapLogMsg("*pguidCmdGroup=%s\n", Wrap(*pguidCmdGroup));
- HRESULT hr = m_IOleCommandTarget->Exec(pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut);
- WrapLogExit("CMenuDeskBar::Exec() = %08x\n", hr);
- return hr;
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetSubMenu(IMenuPopup *pmp, BOOL fSet)
+{
+ // Called by the MenuBand to assign itself as the logical child of the DeskBar
+
+ if (fSet)
+ {
+ m_SubMenuChild = pmp;
+ }
+ else
+ {
+ if (m_SubMenuChild)
+ {
+ if (pmp == m_SubMenuChild)
+ {
+ m_SubMenuChild = NULL;
+ }
+ }
+ }
+ return S_OK;
}
-// *** IServiceProvider methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::QueryService(REFGUID guidService, REFIID riid, void **ppvObject)
+HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnSelect(DWORD dwSelectType)
{
- WrapLogEnter("CMenuDeskBar<%p>::QueryService(REFGUID guidService=%s, REFIID riid=%s, void **ppvObject=%p)\n", this, Wrap(guidService), Wrap(riid), ppvObject);
-
- //if (IsEqualIID(guidService, SID_SMenuBandChild))
- //{
- // WrapLogMsg("SID is SID_SMenuBandChild. Using QueryInterface of self instead of wrapped object.\n");
- // HRESULT hr = this->QueryInterface(riid, ppvObject);
- // if (ppvObject) WrapLogMsg("*ppvObject=%p\n", *ppvObject);
- // WrapLogExit("CMenuDeskBar::QueryService() = %08x\n", hr);
- // return hr;
- //}
- //else
+ CComPtr<IDeskBar> safeThis = this;
+
+ /* As far as I can tell, the submenu hierarchy looks like this:
+ *
+ * The DeskBar's Child is the Band it contains.
+ * The DeskBar's Parent is the SID_SMenuPopup of the Site.
+ *
+ * The Band's Child is the IMenuPopup of the child submenu.
+ * The Band's Parent is the SID_SMenuPopup of the Site (the DeskBar).
+ *
+ * When the DeskBar receives a selection event:
+ * If it requires closing the window, it will notify the Child (Band) using CancelLevel.
+ * If it has to spread upwards (everything but CancelLevel), it will notify the Parent.
+ *
+ * When the Band receives a selection event, this is where it gets fuzzy:
+ * In which cases does it call the Parent? Probably not CancelLevel.
+ * In which cases does it call the Child?
+ * How does it react to calls?
+ *
+ */
+
+ CComPtr<IMenuPopup> oldParent = m_SubMenuParent;
+
+ switch (dwSelectType)
{
- WrapLogMsg("SID not identified.\n");
+ case MPOS_EXECUTE:
+ case MPOS_FULLCANCEL:
+ case MPOS_CANCELLEVEL:
+
+ _CloseBar();
+
+ if (dwSelectType == MPOS_CANCELLEVEL)
+ return S_OK;
+
+ case MPOS_SELECTLEFT:
+ case MPOS_SELECTRIGHT:
+ case MPOS_CHILDTRACKING:
+ if (oldParent)
+ return oldParent->OnSelect(dwSelectType);
+ break;
}
- HRESULT hr = m_IServiceProvider->QueryService(guidService, riid, ppvObject);
- if (ppvObject) WrapLogMsg("*ppvObject=%p\n", *ppvObject);
- WrapLogExit("CMenuDeskBar::QueryService() = %08x\n", hr);
- return hr;
+
+ return S_OK;
}
-// *** IInputObjectSite methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnFocusChangeIS(LPUNKNOWN lpUnknown, BOOL bFocus)
+HRESULT CMenuDeskBar::_CloseBar()
{
- WrapLogEnter("CMenuDeskBar<%p>::OnFocusChangeIS(LPUNKNOWN lpUnknown=%p, BOOL bFocus=%d)\n", this, lpUnknown, bFocus);
- HRESULT hr = m_IInputObjectSite->OnFocusChangeIS(lpUnknown, bFocus);
- WrapLogExit("CMenuDeskBar::OnFocusChangeIS() = %08x\n", hr);
- return hr;
+ CComPtr<IDeskBar> safeThis = this;
+ CComPtr<IDeskBarClient> dbc;
+ HRESULT hr;
+
+ m_Shown = false;
+
+ if (m_SubMenuParent)
+ {
+ m_SubMenuParent->SetSubMenu(this, FALSE);
+ }
+
+ if (m_SubMenuChild)
+ {
+ hr = m_SubMenuChild->OnSelect(MPOS_CANCELLEVEL);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+ }
+
+ hr = m_Client->QueryInterface(IID_PPV_ARG(IDeskBarClient, &dbc));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ hr = dbc->UIActivateDBC(FALSE);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return hr;
+
+ SetWindowPos(NULL, 0, 0, 0, 0, SWP_HIDEWINDOW | SWP_NOACTIVATE | SWP_NOMOVE);
+
+ return UIActivateIO(FALSE, NULL);
}
-// *** IInputObject methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::UIActivateIO(BOOL bActivating, LPMSG lpMsg)
+BOOL CMenuDeskBar::_IsSubMenuParent(HWND hwnd)
{
- WrapLogEnter("CMenuDeskBar<%p>::UIActivateIO(BOOL bActivating=%d, LPMSG lpMsg=%p)\n", this, bActivating, lpMsg);
- HRESULT hr = m_IInputObject->UIActivateIO(bActivating, lpMsg);
- WrapLogExit("CMenuDeskBar::UIActivateIO() = %08x\n", hr);
- return hr;
+ CComPtr<IMenuPopup> popup = m_SubMenuParent;
+
+ while (popup)
+ {
+ HRESULT hr;
+ CComPtr<IOleWindow> window;
+
+ hr = popup->QueryInterface(IID_PPV_ARG(IOleWindow, &window));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return FALSE;
+
+ HWND parent;
+
+ hr = window->GetWindow(&parent);
+ if (SUCCEEDED(hr) && hwnd == parent)
+ return TRUE;
+
+ popup = NULL;
+ hr = IUnknown_GetSite(window, IID_PPV_ARG(IMenuPopup, &popup));
+ if (FAILED(hr))
+ return FALSE;
+ }
+
+ return FALSE;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::HasFocusIO(THIS)
+LRESULT CMenuDeskBar::_OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
- WrapLogEnter("CMenuDeskBar<%p>::HasFocusIO()\n", this);
- HRESULT hr = m_IInputObject->HasFocusIO();
- WrapLogExit("CMenuDeskBar::HasFocusIO() = %08x\n", hr);
- return hr;
+ if (m_Client)
+ {
+ RECT rc;
+
+ GetClientRect(&rc);
+
+ if (m_Banner != NULL)
+ {
+ BITMAP bm;
+ ::GetObject(m_Banner, sizeof(bm), &bm);
+ rc.left += bm.bmWidth;
+ }
+
+ ::SetWindowPos(m_ClientWindow, NULL, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, 0);
+ }
+
+ return 0;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::TranslateAcceleratorIO(LPMSG lpMsg)
+LRESULT CMenuDeskBar::_OnNotify(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
- WrapLogEnter("CMenuDeskBar<%p>::TranslateAcceleratorIO(LPMSG lpMsg=%p)\n", this, lpMsg);
- if (lpMsg) WrapLogMsg("*lpMsg=%s\n", Wrap(*lpMsg));
- HRESULT hr = m_IInputObject->TranslateAcceleratorIO(lpMsg);
- WrapLogExit("CMenuDeskBar::TranslateAcceleratorIO() = %08x\n", hr);
- return hr;
+ if (!m_Client)
+ return 0;
+
+ CComPtr<IWinEventHandler> winEventHandler;
+ HRESULT hr = m_Client->QueryInterface(IID_PPV_ARG(IWinEventHandler, &winEventHandler));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return 0;
+
+ if (winEventHandler)
+ {
+ LRESULT result;
+ hr = winEventHandler->OnWinEvent(NULL, uMsg, wParam, lParam, &result);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return 0;
+ return result;
+ }
+
+ return 0;
}
-// *** IDeskBar methods ***
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetClient(IUnknown *punkClient)
+LRESULT CMenuDeskBar::_OnPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
- WrapLogEnter("CMenuDeskBar<%p>::SetClient(IUnknown *punkClient=%p)\n", this, punkClient);
- HRESULT hr = m_IDeskBar->SetClient(punkClient);
- WrapLogExit("CMenuDeskBar::SetClient() = %08x\n", hr);
- return hr;
+ bHandled = FALSE;
+
+ if (m_Banner && !m_IconSize)
+ {
+ BITMAP bm;
+ PAINTSTRUCT ps;
+ HDC hdc = BeginPaint(&ps);
+
+ HDC hdcMem = ::CreateCompatibleDC(hdc);
+ HGDIOBJ hbmOld = ::SelectObject(hdcMem, m_Banner);
+
+ ::GetObject(m_Banner, sizeof(bm), &bm);
+
+ RECT rc;
+ if (!GetClientRect(&rc))
+ WARN("GetClientRect failed\n");
+
+ const int bx = bm.bmWidth;
+ const int by = bm.bmHeight;
+ const int cy = rc.bottom;
+
+ TRACE("Painting banner: %d by %d\n", bm.bmWidth, bm.bmHeight);
+
+ if (!::StretchBlt(hdc, 0, 0, bx, cy - by, hdcMem, 0, 0, bx, 1, SRCCOPY))
+ WARN("StretchBlt failed\n");
+
+ if (!::BitBlt(hdc, 0, cy - by, bx, by, hdcMem, 0, 0, SRCCOPY))
+ WARN("BitBlt failed\n");
+
+ ::SelectObject(hdcMem, hbmOld);
+ ::DeleteDC(hdcMem);
+
+ EndPaint(&ps);
+ }
+
+ return TRUE;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetClient(IUnknown **ppunkClient)
+LRESULT CMenuDeskBar::_OnActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
- WrapLogEnter("CMenuDeskBar<%p>::GetClient(IUnknown **ppunkClient=%p)\n", this, ppunkClient);
- HRESULT hr = m_IDeskBar->GetClient(ppunkClient);
- if (ppunkClient) WrapLogMsg("*ppunkClient=%p\n", *ppunkClient);
- WrapLogExit("CMenuDeskBar::GetClient() = %08x\n", hr);
- return hr;
+ // BUG in ReactOS: WM_ACTIVATE/WA_INACTIVE makes no sense with lParam==hWnd
+ if (LOWORD(wParam) != 0 || reinterpret_cast<HWND>(lParam) == m_hWnd)
+ {
+ return 0;
+ }
+
+ // HACK! I just want it to work !!!
+ CComPtr<IDeskBar> db;
+ HRESULT hr = IUnknown_QueryService(m_Client, SID_SMenuBandChild, IID_PPV_ARG(IDeskBar, &db));
+ if (FAILED_UNEXPECTEDLY(hr))
+ return 0;
+
+ CComPtr<IUnknown> punk;
+
+ hr = db->GetClient(&punk);
+ if (FAILED_UNEXPECTEDLY(hr))
+ return 0;
+
+ if (!punk && m_Shown)
+ {
+ if (!_IsSubMenuParent(reinterpret_cast<HWND>(lParam)))
+ {
+ OnSelect(MPOS_FULLCANCEL);
+ }
+ }
+
+ return 0;
}
-HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnPosRectChangeDB(LPRECT prc)
+LRESULT CMenuDeskBar::_OnMouseActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
{
- WrapLogEnter("CMenuDeskBar<%p>::OnPosRectChangeDB(RECT *prc=%p)\n", this, prc);
- HRESULT hr = m_IDeskBar->OnPosRectChangeDB(prc);
- if (prc) WrapLogMsg("*prc=%s\n", Wrap(*prc));
- WrapLogExit("CMenuDeskBar::OnPosRectChangeDB() = %08x\n", hr);
- return hr;
+ return MA_NOACTIVATE;
+}
+
+LRESULT CMenuDeskBar::_OnAppActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
+{
+ if (wParam == 0 && m_Shown)
+ {
+ OnSelect(MPOS_FULLCANCEL);
+ }
+ return 0;
}