/* * 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 "shellmenu.h" #include #include #include "CMenuDeskBar.h" /* 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? * */ WINE_DEFAULT_DEBUG_CHANNEL(CMenuDeskBar); CMenuDeskBar::CMenuDeskBar() : m_Client(NULL), m_ClientWindow(NULL), m_IconSize(0), m_Banner(NULL), m_Shown(FALSE), m_ShowFlags(0), m_didAddRef(FALSE) { } CMenuDeskBar::~CMenuDeskBar() { } LRESULT CMenuDeskBar::_OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { if (!m_didAddRef) { this->AddRef(); m_didAddRef = TRUE; } bHandled = FALSE; return 0; } void CMenuDeskBar::OnFinalMessage(HWND /* hWnd */) { if (m_didAddRef) { this->Release(); m_didAddRef = FALSE; } } HRESULT STDMETHODCALLTYPE CMenuDeskBar::Initialize(THIS) { return S_OK; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetWindow(HWND *lphwnd) { if (lphwnd == NULL) return E_POINTER; *lphwnd = m_hWnd; return S_OK; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::ContextSensitiveHelp(BOOL fEnterMode) { return E_NOTIMPL; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnFocusChangeIS(IUnknown *punkObj, BOOL fSetFocus) { return IUnknown_OnFocusChangeIS(m_Client, punkObj, fSetFocus); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds [], OLECMDTEXT *pCmdText) { return E_NOTIMPL; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut) { 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 _AdjustForTheme(nCmdexecopt); } } 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::QueryService(REFGUID guidService, REFIID riid, void **ppvObject) { 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); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::UIActivateIO(BOOL fActivate, LPMSG lpMsg) { return IUnknown_UIActivateIO(m_Client, fActivate, lpMsg); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::HasFocusIO() { return IUnknown_HasFocusIO(m_Client); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::TranslateAcceleratorIO(LPMSG lpMsg) { return IUnknown_TranslateAcceleratorIO(m_Client, lpMsg); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetClient(IUnknown *punkClient) { CComPtr 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(this)); if (FAILED_UNEXPECTEDLY(hr)) return hr; return IUnknown_GetWindow(m_Client, &m_ClientWindow); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetClient(IUnknown **ppunkClient) { if (ppunkClient == NULL) return E_POINTER; if (!m_Client) return E_FAIL; return m_Client->QueryInterface(IID_PPV_ARG(IUnknown, ppunkClient)); } HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnPosRectChangeDB(LPRECT prc) { if (prc == NULL) return E_POINTER; return S_OK; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetSite(IUnknown *pUnkSite) { // 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::GetSite(REFIID riid, void **ppvSite) { if (m_Site == NULL) return E_FAIL; return m_Site->QueryInterface(riid, ppvSite); } static void AdjustForExcludeArea(BOOL alignLeft, BOOL alignTop, BOOL preferVertical, PINT px, PINT py, INT cx, INT cy, RECTL rcExclude) { RECT rcWindow = { *px, *py, *px + cx, *py + cy }; if (rcWindow.right > rcExclude.left && rcWindow.left < rcExclude.right && rcWindow.bottom > rcExclude.top && rcWindow.top < rcExclude.bottom) { if (preferVertical) { if (alignTop && rcWindow.bottom > rcExclude.top) *py = rcExclude.top - cy; else if (!alignTop && rcWindow.top < rcExclude.bottom) *py = rcExclude.bottom; else if (alignLeft && rcWindow.right > rcExclude.left) *px = rcExclude.left - cx; else if (!alignLeft && rcWindow.left < rcExclude.right) *px = rcExclude.right; } else { if (alignLeft && rcWindow.right > rcExclude.left) *px = rcExclude.left - cx; else if (!alignLeft && rcWindow.left < rcExclude.right) *px = rcExclude.right; else if (alignTop && rcWindow.bottom > rcExclude.top) *py = rcExclude.top - cy; else if (!alignTop && rcWindow.top < rcExclude.bottom) *py = rcExclude.bottom; } } } HRESULT STDMETHODCALLTYPE CMenuDeskBar::Popup(POINTL *ppt, RECTL *prcExclude, MP_POPUPFLAGS dwFlags) { HRESULT hr; CComPtr oct; CComPtr io; CComPtr band; CComPtr dbc; if (m_hWnd == NULL) return E_FAIL; 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 cxWorkArea = rcWorkArea.right - rcWorkArea.left; int cyWorkArea = rcWorkArea.bottom - rcWorkArea.top; int x = ppt->x; int y = ppt->y; int cx = rc.right - rc.left; int cy = rc.bottom - rc.top; // TODO: Make alignLeft default to TRUE in LTR systems or whenever necessary. BOOL alignLeft = FALSE; BOOL alignTop = FALSE; BOOL preferVertical = FALSE; switch (dwFlags & MPPF_POS_MASK) { case MPPF_TOP: alignTop = TRUE; preferVertical = TRUE; break; case MPPF_LEFT: alignLeft = TRUE; break; case MPPF_BOTTOM: alignTop = FALSE; preferVertical = TRUE; break; case MPPF_RIGHT: alignLeft = FALSE; break; } // Try the selected alignment and verify that it doesn't escape the work area. if (alignLeft) { x = ppt->x - cx; } else { x = ppt->x; } if (alignTop) { y = ppt->y - cy; } else { y = ppt->y; } if (prcExclude) AdjustForExcludeArea(alignLeft, alignTop, preferVertical, &x, &y, cx, cy, *prcExclude); // Verify that it doesn't escape the work area, and flip. if (alignLeft) { if (x < rcWorkArea.left && (ppt->x+cx) <= rcWorkArea.right) { alignLeft = FALSE; if (prcExclude) x = prcExclude->right - ((x + cx) - prcExclude->left); else x = ppt->x; } } else { if ((ppt->x + cx) > rcWorkArea.right && x >= rcWorkArea.left) { alignLeft = TRUE; if (prcExclude) x = prcExclude->left - cx + (prcExclude->right - x); else x = ppt->x - cx; } } BOOL flipV = FALSE; if (alignTop) { if (y < rcWorkArea.top && (ppt->y + cy) <= rcWorkArea.bottom) { alignTop = FALSE; if (prcExclude) y = prcExclude->bottom - ((y + cy) - prcExclude->top); else y = ppt->y; flipV = true; } } else { if ((ppt->y + cy) > rcWorkArea.bottom && y >= rcWorkArea.top) { alignTop = TRUE; if (prcExclude) y = prcExclude->top - cy + (prcExclude->bottom - y); else y = ppt->y - cy; flipV = true; } } if (prcExclude) AdjustForExcludeArea(alignLeft, alignTop, preferVertical, &x, &y, cx, cy, *prcExclude); if (x < rcWorkArea.left) x = rcWorkArea.left; if (cx > cxWorkArea) cx = cxWorkArea; if (x + cx > rcWorkArea.right) x = rcWorkArea.right - cx; if (y < rcWorkArea.top) y = rcWorkArea.top; if (cy > cyWorkArea) cy = cyWorkArea; if (y + cy > rcWorkArea.bottom) y = rcWorkArea.bottom - cy; int flags = SWP_SHOWWINDOW | SWP_NOACTIVATE; this->SetWindowPos(HWND_TOPMOST, x, y, cx, cy, flags); if (flipV) { if (dwFlags & MPPF_INITIALSELECT) dwFlags = (dwFlags ^ MPPF_INITIALSELECT) | MPPF_FINALSELECT; else if (dwFlags & MPPF_FINALSELECT) dwFlags = (dwFlags ^ MPPF_FINALSELECT) | MPPF_INITIALSELECT; } 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) { 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; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetIconSize(THIS_ DWORD* piIcon) { if (piIcon) *piIcon = m_IconSize; return S_OK; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::SetBitmap(THIS_ HBITMAP hBitmap) { if (m_Banner && m_Banner != hBitmap) ::DeleteObject(m_Banner); m_Banner = hBitmap; BOOL bHandled; _OnSize(WM_SIZE, 0, 0, bHandled); return S_OK; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::GetBitmap(THIS_ HBITMAP* phBitmap) { if (phBitmap) *phBitmap = m_Banner; return S_OK; } 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; } HRESULT STDMETHODCALLTYPE CMenuDeskBar::OnSelect(DWORD dwSelectType) { CComPtr safeThis = this; CComPtr oldParent = m_SubMenuParent; TRACE("OnSelect dwSelectType=%d\n", this, dwSelectType); switch (dwSelectType) { 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; } return S_OK; } HRESULT CMenuDeskBar::_CloseBar() { CComPtr dbc; HRESULT hr; // Ensure that our data isn't destroyed while we are working CComPtr safeThis = this; 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; if (m_hWnd) SetWindowPos(NULL, 0, 0, 0, 0, SWP_HIDEWINDOW | SWP_NOACTIVATE | SWP_NOMOVE); return UIActivateIO(FALSE, NULL); } BOOL CMenuDeskBar::_IsSubMenuParent(HWND hwnd) { CComPtr popup = m_SubMenuParent; while (popup) { HRESULT hr; HWND parent; hr = IUnknown_GetWindow(popup, &parent); if (FAILED_UNEXPECTEDLY(hr)) return FALSE; if (hwnd == parent) return TRUE; hr = IUnknown_GetSite(popup, IID_PPV_ARG(IMenuPopup, &popup)); if (FAILED(hr)) return FALSE; } return FALSE; } LRESULT CMenuDeskBar::_OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { if (m_Client) { RECT rc; GetClientRect(&rc); if (m_Banner && m_IconSize != BMICON_SMALL) { 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; } LRESULT CMenuDeskBar::_OnNotify(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { if (!m_Client) return 0; CComPtr 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; } LRESULT CMenuDeskBar::_OnPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { 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; } LRESULT CMenuDeskBar::_OnActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { // BUG in ReactOS: WM_ACTIVATE/WA_INACTIVE makes no sense with lParam==hWnd if (LOWORD(wParam) != 0 || reinterpret_cast(lParam) == m_hWnd) { return 0; } // HACK! I just want it to work !!! CComPtr db; HRESULT hr = IUnknown_QueryService(m_Client, SID_SMenuBandChild, IID_PPV_ARG(IDeskBar, &db)); if (FAILED_UNEXPECTEDLY(hr)) return 0; CComPtr punk; hr = db->GetClient(&punk); if (FAILED_UNEXPECTEDLY(hr)) return 0; if (!punk && m_Shown) { if (!_IsSubMenuParent(reinterpret_cast(lParam))) { OnSelect(MPOS_FULLCANCEL); } } return 0; } LRESULT CMenuDeskBar::_OnMouseActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { return MA_NOACTIVATE; } LRESULT CMenuDeskBar::_OnAppActivate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { #if 0 if (wParam == 0 && m_Shown) { OnSelect(MPOS_FULLCANCEL); } #endif return 0; } LRESULT CMenuDeskBar::_OnWinIniChange(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { if (wParam == SPI_SETFLATMENU) return _OnNotify(uMsg, wParam, lParam, bHandled); return 0; } LRESULT CMenuDeskBar::_OnNcPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { /* If it is a flat style menu we need to handle WM_NCPAINT * and paint the border with the right colour */ if ((GetStyle() & WS_BORDER) == 0) { /* This isn't a flat style menu. */ bHandled = FALSE; return 0; } HDC hdc; RECT rcWindow; hdc = GetWindowDC(); GetWindowRect(&rcWindow); OffsetRect(&rcWindow, -rcWindow.left, -rcWindow.top); FrameRect(hdc, &rcWindow, GetSysColorBrush(COLOR_BTNSHADOW)); ReleaseDC(hdc); return 0; } LRESULT CMenuDeskBar::_OnClose(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled) { /* Prevent the CMenuDeskBar from destroying on being sent a WM_CLOSE */ return 0; } HRESULT CMenuDeskBar::_AdjustForTheme(BOOL bFlatStyle) { DWORD style = bFlatStyle ? WS_BORDER : WS_CLIPCHILDREN|WS_DLGFRAME; DWORD mask = WS_BORDER|WS_CLIPCHILDREN|WS_DLGFRAME; SHSetWindowBits(m_hWnd, GWL_STYLE, mask, style); return S_OK; } extern "C" HRESULT WINAPI RSHELL_CMenuDeskBar_CreateInstance(REFIID riid, LPVOID *ppv) { return ShellObjectCreator(riid, ppv); }