2016-11-29 15:38:55 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: shell32
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
|
|
* FILE: dll/win32/shell32/CDefViewBckgrndMenu.cpp
|
|
|
|
* PURPOSE: background context menu of the CDefView
|
|
|
|
* PROGRAMMERS: Giannis Adamopoulos
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <precomp.h>
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shell);
|
|
|
|
|
|
|
|
class CDefViewBckgrndMenu :
|
|
|
|
public CComObjectRootEx<CComMultiThreadModelNoCS>,
|
|
|
|
public IContextMenu3,
|
|
|
|
public IObjectWithSite
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
CComPtr<IUnknown> m_site;
|
|
|
|
CComPtr<IShellFolder> m_psf;
|
|
|
|
CComPtr<IContextMenu> m_folderCM;
|
|
|
|
|
2017-08-14 15:25:58 +00:00
|
|
|
UINT m_idCmdFirst;
|
|
|
|
UINT m_LastFolderCMId;
|
|
|
|
|
2016-11-29 15:38:55 +00:00
|
|
|
BOOL _bIsDesktopBrowserMenu();
|
|
|
|
BOOL _bCanPaste();
|
|
|
|
public:
|
|
|
|
CDefViewBckgrndMenu();
|
|
|
|
~CDefViewBckgrndMenu();
|
|
|
|
HRESULT Initialize(IShellFolder* psf);
|
|
|
|
|
|
|
|
// IContextMenu
|
2024-03-04 23:43:39 +00:00
|
|
|
STDMETHOD(QueryContextMenu)(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags) override;
|
|
|
|
STDMETHOD(InvokeCommand)(LPCMINVOKECOMMANDINFO lpcmi) override;
|
|
|
|
STDMETHOD(GetCommandString)(UINT_PTR idCommand, UINT uFlags, UINT *lpReserved, LPSTR lpszName, UINT uMaxNameLen) override;
|
2016-11-29 15:38:55 +00:00
|
|
|
|
|
|
|
// IContextMenu2
|
2024-03-04 23:43:39 +00:00
|
|
|
STDMETHOD(HandleMenuMsg)(UINT uMsg, WPARAM wParam, LPARAM lParam) override;
|
2016-11-29 15:38:55 +00:00
|
|
|
|
|
|
|
// IContextMenu3
|
2024-03-04 23:43:39 +00:00
|
|
|
STDMETHOD(HandleMenuMsg2)(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *plResult) override;
|
2016-11-29 15:38:55 +00:00
|
|
|
|
|
|
|
// IObjectWithSite
|
2024-03-04 23:43:39 +00:00
|
|
|
STDMETHOD(SetSite)(IUnknown *pUnkSite) override;
|
|
|
|
STDMETHOD(GetSite)(REFIID riid, void **ppvSite) override;
|
2016-11-29 15:38:55 +00:00
|
|
|
|
|
|
|
BEGIN_COM_MAP(CDefViewBckgrndMenu)
|
|
|
|
COM_INTERFACE_ENTRY_IID(IID_IContextMenu, IContextMenu)
|
|
|
|
COM_INTERFACE_ENTRY_IID(IID_IContextMenu2, IContextMenu2)
|
|
|
|
COM_INTERFACE_ENTRY_IID(IID_IContextMenu3, IContextMenu3)
|
|
|
|
COM_INTERFACE_ENTRY_IID(IID_IObjectWithSite, IObjectWithSite)
|
|
|
|
END_COM_MAP()
|
|
|
|
};
|
|
|
|
|
|
|
|
CDefViewBckgrndMenu::CDefViewBckgrndMenu()
|
|
|
|
{
|
2017-08-14 15:25:58 +00:00
|
|
|
m_idCmdFirst = 0;
|
|
|
|
m_LastFolderCMId = 0;
|
2016-11-29 15:38:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CDefViewBckgrndMenu::~CDefViewBckgrndMenu()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL CDefViewBckgrndMenu::_bIsDesktopBrowserMenu()
|
|
|
|
{
|
|
|
|
if (!m_site)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Get a pointer to the shell browser */
|
|
|
|
CComPtr<IShellView> psv;
|
|
|
|
HRESULT hr = IUnknown_QueryService(m_site, SID_IFolderView, IID_PPV_ARG(IShellView, &psv));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
FOLDERSETTINGS FolderSettings;
|
|
|
|
hr = psv->GetCurrentInfo(&FolderSettings);
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return ((FolderSettings.fFlags & FWF_DESKTOP) == FWF_DESKTOP);
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL CDefViewBckgrndMenu::_bCanPaste()
|
|
|
|
{
|
|
|
|
/* If the folder doesn't have a drop target we can't paste */
|
|
|
|
CComPtr<IDropTarget> pdt;
|
|
|
|
HRESULT hr = m_psf->CreateViewObject(NULL, IID_PPV_ARG(IDropTarget, &pdt));
|
|
|
|
if (FAILED(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* We can only paste if CFSTR_SHELLIDLIST is present in the clipboard */
|
|
|
|
CComPtr<IDataObject> pDataObj;
|
|
|
|
hr = OleGetClipboard(&pDataObj);
|
|
|
|
if (FAILED(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
STGMEDIUM medium;
|
|
|
|
FORMATETC formatetc;
|
|
|
|
|
|
|
|
/* Set the FORMATETC structure*/
|
|
|
|
InitFormatEtc(formatetc, RegisterClipboardFormatW(CFSTR_SHELLIDLIST), TYMED_HGLOBAL);
|
|
|
|
hr = pDataObj->GetData(&formatetc, &medium);
|
|
|
|
if (FAILED(hr))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
ReleaseStgMedium(&medium);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
CDefViewBckgrndMenu::Initialize(IShellFolder* psf)
|
|
|
|
{
|
|
|
|
m_psf = psf;
|
2018-02-13 22:03:35 +00:00
|
|
|
|
|
|
|
/* Get the context menu of the folder. Do it here because someone may call
|
|
|
|
InvokeCommand without calling QueryContextMenu. It is fine if this fails */
|
|
|
|
m_psf->CreateViewObject(NULL, IID_PPV_ARG(IContextMenu, &m_folderCM));
|
|
|
|
|
2016-11-29 15:38:55 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::SetSite(IUnknown *pUnkSite)
|
|
|
|
{
|
|
|
|
m_site = pUnkSite;
|
2018-02-14 22:23:18 +00:00
|
|
|
|
|
|
|
if(m_folderCM)
|
|
|
|
IUnknown_SetSite(m_folderCM, pUnkSite);
|
|
|
|
|
2016-11-29 15:38:55 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2021-09-13 01:33:14 +00:00
|
|
|
HRESULT
|
|
|
|
WINAPI
|
2016-11-29 15:38:55 +00:00
|
|
|
CDefViewBckgrndMenu::GetSite(REFIID riid, void **ppvSite)
|
|
|
|
{
|
|
|
|
if (!m_site)
|
|
|
|
return E_FAIL;
|
|
|
|
|
|
|
|
return m_site->QueryInterface(riid, ppvSite);
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
HMENU hMenuPart;
|
2017-08-14 15:25:58 +00:00
|
|
|
UINT cIds = 0;
|
|
|
|
|
|
|
|
/* This is something the implementations of IContextMenu should never really do.
|
2021-09-13 01:33:14 +00:00
|
|
|
However CDefViewBckgrndMenu is more or less an overengineering result, its code could really be part of the
|
2017-08-14 15:25:58 +00:00
|
|
|
CDefView. Given this, I think that abusing the interface here is not that bad since only CDefView is the ony
|
2021-09-13 01:33:14 +00:00
|
|
|
user of this class. Here we need to do two things to keep things as simple as possible.
|
2017-08-14 15:25:58 +00:00
|
|
|
First we want the menu part added by the shell folder to be the first to add so as to make as few id translations
|
2021-09-13 01:33:14 +00:00
|
|
|
as possible. Second, we want to add the default part of the background menu without shifted ids, so as
|
|
|
|
to let the CDefView fill some parts like filling the arrange modes or checking the view mode. In order
|
2017-08-14 15:25:58 +00:00
|
|
|
for that to work we need to save idCmdFirst because our caller will pass id offsets to InvokeCommand.
|
|
|
|
This makes it impossible to concatenate the CDefViewBckgrndMenu with other menus since it abuses IContextMenu
|
|
|
|
but as stated above, its sole user is CDefView and should really be that way. */
|
|
|
|
m_idCmdFirst = idCmdFirst;
|
|
|
|
|
2018-02-13 22:03:35 +00:00
|
|
|
/* Let the shell folder add any items it wants to add in the background context menu */
|
|
|
|
if (m_folderCM)
|
2017-08-14 15:25:58 +00:00
|
|
|
{
|
|
|
|
hr = m_folderCM->QueryContextMenu(hMenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
m_LastFolderCMId = LOWORD(hr);
|
|
|
|
cIds = m_LastFolderCMId;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WARN("QueryContextMenu failed!\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WARN("GetUIObjectOf didn't give any context menu!\n");
|
|
|
|
}
|
2016-11-29 15:38:55 +00:00
|
|
|
|
|
|
|
/* Load the default part of the background context menu */
|
|
|
|
hMenuPart = LoadMenuW(shell32_hInstance, L"MENU_002");
|
|
|
|
if (hMenuPart)
|
|
|
|
{
|
|
|
|
/* Don't show the view submenu for the desktop */
|
|
|
|
if (_bIsDesktopBrowserMenu())
|
|
|
|
{
|
|
|
|
DeleteMenu(hMenuPart, FCIDM_SHVIEW_VIEW, MF_BYCOMMAND);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable the paste options if it is not possible */
|
|
|
|
if (!_bCanPaste())
|
|
|
|
{
|
|
|
|
EnableMenuItem(hMenuPart, FCIDM_SHVIEW_INSERT, MF_BYCOMMAND | MF_GRAYED);
|
|
|
|
EnableMenuItem(hMenuPart, FCIDM_SHVIEW_INSERTLINK, MF_BYCOMMAND | MF_GRAYED);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* merge general background context menu in */
|
2017-08-14 15:25:58 +00:00
|
|
|
Shell_MergeMenus(hMenu, GetSubMenu(hMenuPart, 0), indexMenu, 0, idCmdLast, MM_DONTREMOVESEPS | MM_SUBMENUSHAVEIDS | MM_ADDSEPARATOR);
|
2016-11-29 15:38:55 +00:00
|
|
|
DestroyMenu(hMenuPart);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERR("Failed to load menu from resource!\n");
|
|
|
|
}
|
|
|
|
|
2017-08-14 15:25:58 +00:00
|
|
|
return MAKE_HRESULT(SEVERITY_SUCCESS, 0, cIds);
|
2016-11-29 15:38:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::InvokeCommand(LPCMINVOKECOMMANDINFO lpcmi)
|
|
|
|
{
|
2017-08-14 15:25:58 +00:00
|
|
|
UINT idCmd = LOWORD(lpcmi->lpVerb);
|
2018-02-14 22:23:18 +00:00
|
|
|
|
|
|
|
if (HIWORD(lpcmi->lpVerb) && !strcmp(lpcmi->lpVerb, CMDSTR_VIEWLISTA))
|
|
|
|
{
|
|
|
|
idCmd = FCIDM_SHVIEW_LISTVIEW;
|
|
|
|
}
|
|
|
|
else if (HIWORD(lpcmi->lpVerb) && !strcmp(lpcmi->lpVerb, CMDSTR_VIEWDETAILSA))
|
|
|
|
{
|
|
|
|
idCmd = FCIDM_SHVIEW_REPORTVIEW;
|
|
|
|
}
|
|
|
|
else if(HIWORD(lpcmi->lpVerb) != 0 || idCmd < m_LastFolderCMId)
|
2017-08-14 15:25:58 +00:00
|
|
|
{
|
2018-02-05 16:12:00 +00:00
|
|
|
if (m_folderCM)
|
|
|
|
{
|
|
|
|
return m_folderCM->InvokeCommand(lpcmi);
|
|
|
|
}
|
|
|
|
WARN("m_folderCM is NULL!\n");
|
|
|
|
return E_NOTIMPL;
|
2017-08-14 15:25:58 +00:00
|
|
|
}
|
2018-02-14 22:23:18 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* The default part of the background menu doesn't have shifted ids so we need to convert the id offset to the real id */
|
|
|
|
idCmd += m_idCmdFirst;
|
|
|
|
}
|
2017-08-14 15:25:58 +00:00
|
|
|
|
2016-11-29 15:38:55 +00:00
|
|
|
/* The commands that are handled by the def view are forwarded to it */
|
2017-08-14 15:25:58 +00:00
|
|
|
switch (idCmd)
|
2016-11-29 15:38:55 +00:00
|
|
|
{
|
2017-08-18 17:32:20 +00:00
|
|
|
case FCIDM_SHVIEW_INSERT:
|
|
|
|
case FCIDM_SHVIEW_INSERTLINK:
|
2018-02-05 16:12:00 +00:00
|
|
|
if (m_folderCM)
|
|
|
|
{
|
|
|
|
lpcmi->lpVerb = MAKEINTRESOURCEA(idCmd);
|
|
|
|
return m_folderCM->InvokeCommand(lpcmi);
|
|
|
|
}
|
|
|
|
WARN("m_folderCM is NULL!\n");
|
|
|
|
return E_NOTIMPL;
|
2016-11-29 15:38:55 +00:00
|
|
|
case FCIDM_SHVIEW_BIGICON:
|
|
|
|
case FCIDM_SHVIEW_SMALLICON:
|
|
|
|
case FCIDM_SHVIEW_LISTVIEW:
|
|
|
|
case FCIDM_SHVIEW_REPORTVIEW:
|
|
|
|
case FCIDM_SHVIEW_AUTOARRANGE:
|
|
|
|
case FCIDM_SHVIEW_SNAPTOGRID:
|
|
|
|
case FCIDM_SHVIEW_REFRESH:
|
2020-04-20 02:49:38 +00:00
|
|
|
case FCIDM_SHVIEW_ALIGNTOGRID:
|
2016-11-29 15:38:55 +00:00
|
|
|
if (!m_site)
|
|
|
|
return E_FAIL;
|
|
|
|
|
|
|
|
/* Get a pointer to the shell browser */
|
|
|
|
CComPtr<IShellView> psv;
|
|
|
|
HRESULT hr = IUnknown_QueryService(m_site, SID_IFolderView, IID_PPV_ARG(IShellView, &psv));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
HWND hwndSV = NULL;
|
|
|
|
if (SUCCEEDED(psv->GetWindow(&hwndSV)))
|
2017-08-14 15:25:58 +00:00
|
|
|
SendMessageW(hwndSV, WM_COMMAND, MAKEWPARAM(idCmd, 0), 0);
|
2016-11-29 15:38:55 +00:00
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2017-08-14 15:25:58 +00:00
|
|
|
ERR("Got unknown command id %ul\n", LOWORD(lpcmi->lpVerb));
|
|
|
|
return E_FAIL;
|
2016-11-29 15:38:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::GetCommandString(UINT_PTR idCommand, UINT uFlags, UINT *lpReserved, LPSTR lpszName, UINT uMaxNameLen)
|
|
|
|
{
|
|
|
|
if (m_folderCM)
|
|
|
|
{
|
|
|
|
return m_folderCM->GetCommandString(idCommand, uFlags, lpReserved, lpszName, uMaxNameLen);
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::HandleMenuMsg(UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
if(m_folderCM)
|
|
|
|
{
|
|
|
|
CComPtr<IContextMenu2> pfolderCM2;
|
|
|
|
HRESULT hr = m_folderCM->QueryInterface(IID_PPV_ARG(IContextMenu2, &pfolderCM2));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
return pfolderCM2->HandleMenuMsg(uMsg, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
WINAPI
|
|
|
|
CDefViewBckgrndMenu::HandleMenuMsg2(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *plResult)
|
|
|
|
{
|
|
|
|
if(m_folderCM)
|
|
|
|
{
|
|
|
|
CComPtr<IContextMenu3> pfolderCM3;
|
|
|
|
HRESULT hr = m_folderCM->QueryInterface(IID_PPV_ARG(IContextMenu3, &pfolderCM3));
|
|
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
return pfolderCM3->HandleMenuMsg2(uMsg, wParam, lParam, plResult);
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT
|
|
|
|
CDefViewBckgrndMenu_CreateInstance(IShellFolder* psf, REFIID riid, void **ppv)
|
|
|
|
{
|
|
|
|
return ShellObjectCreatorInit<CDefViewBckgrndMenu>(psf, riid, ppv);
|
|
|
|
}
|