2020-09-30 16:53:55 +02:00

501 lines
16 KiB
C++

// qryitem.cpp - CQueryItem class
#include "stdafx.h"
#include "scopenode.h"
#include "namemap.h"
#include "qryitem.h"
#include <algorithm>
extern HWND g_hwndMain;
UINT CQueryItem::m_cfDisplayName = RegisterClipboardFormat(TEXT("CCF_DISPLAY_NAME"));
UINT CQueryItem::m_cfSnapInClsid = RegisterClipboardFormat(TEXT("CCF_SNAPIN_CLSID"));
UINT CQueryItem::m_cfNodeType = RegisterClipboardFormat(TEXT("CCF_NODETYPE"));
UINT CQueryItem::m_cfszNodeType = RegisterClipboardFormat(TEXT("CCF_SZNODETYPE"));
// {68D2DFD9-86A7-4964-8263-BA025C358992}
static const GUID GUID_QueryItem =
{ 0x68d2dfd9, 0x86a7, 0x4964, { 0x82, 0x63, 0xba, 0x2, 0x5c, 0x35, 0x89, 0x92 } };
/////////////////////////////////////////////////////////////////////////////////////////////
// CQueryItem
HRESULT CQueryItem::Initialize(CQueryableNode* pQueryNode, CRowItem* pRowItem)
{
VALIDATE_POINTER( pQueryNode );
VALIDATE_POINTER( pRowItem );
m_spQueryNode = pQueryNode;
m_pRowItem = new CRowItem(*pRowItem);
if (m_pRowItem == NULL) return E_OUTOFMEMORY;
return S_OK;
}
/////////////////////////////////////////////////////////////////////////////////////////////////
// Notification handlers
BEGIN_NOTIFY_MAP(CQueryItem)
ON_SELECT()
ON_DBLCLICK()
ON_NOTIFY(MMCN_CONTEXTHELP, OnHelp)
END_NOTIFY_MAP()
HRESULT CQueryItem::OnHelp(LPCONSOLE2 pConsole, LPARAM /*arg*/, LPARAM /*param*/)
{
VALIDATE_POINTER( pConsole );
tstring strHelpFile = _T("");
tstring strHelpTopic = _T("");
tstring strHelpFull = _T("");
strHelpFile = StrLoadString(IDS_HELPFILE);
if( strHelpFile.empty() ) return E_FAIL;
// Special Hack to get a different help topic for the first two nodes.
int nNodeID = m_spQueryNode->GetNodeID();
switch( nNodeID )
{
case 2:
{
// Users Node
strHelpTopic = StrLoadString(IDS_USERSHELPTOPIC);
break;
}
case 3:
{
// Printers Node
strHelpTopic = StrLoadString(IDS_PRINTERSHELPTOPIC);
break;
}
default:
{
strHelpTopic = StrLoadString(IDS_DEFAULTHELPTOPIC);
break;
}
}
if( strHelpTopic.empty() ) return E_FAIL;
// Build path to %systemroot%\help
TCHAR szWindowsDir[MAX_PATH+1] = {0};
UINT nSize = GetSystemWindowsDirectory( szWindowsDir, MAX_PATH );
if( nSize == 0 || nSize > MAX_PATH )
{
return E_FAIL;
}
strHelpFull = szWindowsDir;
strHelpFull += _T("\\Help\\");
strHelpFull += strHelpFile;
strHelpFull += _T("::/");
strHelpFull += strHelpTopic;
// Show the Help topic
CComQIPtr<IDisplayHelp> spHelp = pConsole;
if( !spHelp ) return E_NOINTERFACE;
return spHelp->ShowTopic( (LPTSTR)strHelpFull.c_str() );
}
HRESULT CQueryItem::OnSelect(LPCONSOLE2 pConsole, BOOL bSelect, BOOL bScope)
{
VALIDATE_POINTER( pConsole );
ASSERT(!bScope);
if( bSelect )
{
CComPtr<IConsoleVerb> pConsVerb;
pConsole->QueryConsoleVerb(&pConsVerb);
ASSERT(pConsVerb != NULL);
if (pConsVerb != NULL)
{
// Row item has class display name, so get internal name from class map
DisplayNameMap* pNameMap = DisplayNames::GetClassMap();
if (pNameMap == NULL)
return E_FAIL;
ASSERT(m_pRowItem != NULL && m_pRowItem->size() >= ROWITEM_USER_INDEX);
ASSERT(m_spQueryNode != NULL);
LPCWSTR pszClass = pNameMap->GetInternalName((*m_pRowItem)[ROWITEM_CLASS_INDEX]);
// Get menu items for this class from the owning query node
int iDefault;
BOOL bPropertyMenu;
reinterpret_cast<CQueryNode*>(m_pRowItem->GetOwnerParam())->GetClassMenuItems(pszClass, m_vMenus, &iDefault, &bPropertyMenu);
// if property menu enabled
if (bPropertyMenu)
{
// Enable property button and menu item
pConsVerb->SetVerbState(MMC_VERB_PROPERTIES, ENABLED, TRUE);
pConsVerb->SetVerbState(MMC_VERB_PROPERTIES, HIDDEN, FALSE);
// if no default menu item defined, make properties verb the default
pConsVerb->SetDefaultVerb( (iDefault >= 0) ? MMC_VERB_NONE : MMC_VERB_PROPERTIES);
}
}
}
return S_OK;
}
HRESULT CQueryItem::OnDblClick(LPCONSOLE2 pConsole)
{
VALIDATE_POINTER(pConsole);
// Row item has class display name, so get internal name from class map
DisplayNameMap* pNameMap = DisplayNames::GetClassMap();
if (pNameMap == NULL)
return E_FAIL;
ASSERT(m_pRowItem != NULL && m_pRowItem->size() >= ROWITEM_USER_INDEX);
ASSERT(m_spQueryNode != NULL);
LPCWSTR pszClass = pNameMap->GetInternalName((*m_pRowItem)[ROWITEM_CLASS_INDEX]);
// Get menu items for this class from the owning query node
int iDefault;
BOOL bPropMenu;
CQueryNode* pQueryNode = reinterpret_cast<CQueryNode*>(m_pRowItem->GetOwnerParam());
if( !pQueryNode ) return E_FAIL;
pQueryNode->GetClassMenuItems(pszClass, m_vMenus, &iDefault, &bPropMenu);
// if no default menu item, return
if (iDefault < 0)
return S_FALSE;
// if Active directory command, create AD menu extension
if (m_vMenus[iDefault]->MenuType() == MENUTYPE_ACTDIR)
{
// Create a directory extension object and use it to get the actual menu cmds for the selected object
// (we might have one already if AddMenuItems was called before)
if (m_pADExt == NULL)
m_pADExt = new CActDirExt();
if( !m_pADExt ) return E_OUTOFMEMORY;
HRESULT hr = m_pADExt->Initialize(pszClass, m_pRowItem->GetObjPath());
RETURN_ON_FAILURE(hr);
menu_vector vADMenus;
hr = m_pADExt->GetMenuItems(vADMenus);
RETURN_ON_FAILURE(hr);
if( m_vMenus.size() <= iDefault )
{
return E_FAIL;
}
LPCWSTR pszName = static_cast<CActDirMenuCmd*>((CMenuCmd*)m_vMenus[iDefault])->ADName();
LPCWSTR pszNoLocName = static_cast<CActDirMenuCmd*>((CMenuCmd*)m_vMenus[iDefault])->ADNoLocName();
if( !pszName || !pszNoLocName ) return E_FAIL;
// if the default command is not provided by the extension, return
menu_vector::iterator iter;
for( iter = vADMenus.begin(); iter != vADMenus.end(); iter++ )
{
if( _tcslen(pszNoLocName) )
{
if( _tcscmp(iter->strNoLoc.c_str(),pszNoLocName) == 0 )
break;
}
else if( _tcscmp(iter->strPlain.c_str(), pszName) == 0 )
{
break;
}
}
if( iter == vADMenus.end() )
{
return S_FALSE;
}
}
// Execute the command as though it had been selected
return MenuCommand(pConsole, iDefault);
}
HRESULT CQueryItem::AddMenuItems(LPCONTEXTMENUCALLBACK pCallback, long* plAllowed)
{
VALIDATE_POINTER( pCallback );
VALIDATE_POINTER( plAllowed );
if( !m_spQueryNode || !m_pRowItem ) return E_FAIL;
HRESULT hr = S_OK;
if (!(*plAllowed & CCM_INSERTIONALLOWED_TOP))
return S_OK;
CComQIPtr<IContextMenuCallback2> spContext2 = pCallback;
if( !spContext2 ) return E_NOINTERFACE;
ASSERT( m_pRowItem->size() >= ROWITEM_USER_INDEX );
//--------------------------- *** HACK ALERT *** -----------------------------------------
// One or more AD menu extensions allow window message processing while initializing
// and getting menu items. This causes reentrancy problems because MMC message handlers
// can execute before this method returns. Specifically, the following can happen:
//
// 1. The user right clicks in a taskpad list while the focus is elsewhere.
// 2. The right button down event causes MMC to call this method to update task buttons.
// 3. An AD menu extn processes messages allowing the button up event to go to MMC.
// 4. MMC sees this as a context menu event and calls this method recursively.
// 5. An AV occurs in nodemgr because a deleted COnContextMenu object is referenced.
//
// This can be prevented by not processing menu item requests when the right button is down.
// The only time this occurs is during the above scenario. The only ill effect is that the
// task buttons are not enabled until the mouse button up occurs when MMC gets the menu
// items again.
//-----------------------------------------------------------------------------------------
if (GetKeyState(VK_RBUTTON) < 0)
return S_OK;
DisplayNameMap* pNameMap = DisplayNames::GetClassMap();
if (pNameMap == NULL)
return E_FAIL;
LPCWSTR pszClass = pNameMap->GetInternalName((*m_pRowItem)[ROWITEM_CLASS_INDEX]);
// Get menu items for this class from the owning query node
int iDefault = 0;
BOOL bPropertyMenu;
reinterpret_cast<CQueryNode*>(m_pRowItem->GetOwnerParam())->GetClassMenuItems(pszClass, m_vMenus, &iDefault, &bPropertyMenu);
// Create a directory extension object and use it to get the actual menu cmds for the selected object
// (we might have one already if AddMenuItems was called before)
if (m_pADExt == NULL)
m_pADExt = new CActDirExt();
if( !m_pADExt ) return E_OUTOFMEMORY;
hr = m_pADExt->Initialize(pszClass, m_pRowItem->GetObjPath());
RETURN_ON_FAILURE(hr);
menu_vector vADMenus;
hr = m_pADExt->GetMenuItems(vADMenus);
RETURN_ON_FAILURE(hr);
ASSERT(vADMenus.size() > 0);
ASSERT(vADMenus.begin() != vADMenus.end());
menucmd_vector::iterator itMenu;
long lCmdID = 0;
for (itMenu = m_vMenus.begin(); itMenu != m_vMenus.end(); ++itMenu, ++lCmdID)
{
// if AD menu cmd and not enabled by the selected object, skip it
if ( (*itMenu)->MenuType() == MENUTYPE_ACTDIR )
{
BOOL bFound = FALSE;
menu_vector::iterator iter = vADMenus.begin();
while(iter != vADMenus.end())
{
LPCWSTR pszName = static_cast<CActDirMenuCmd*>((CMenuCmd*)(*itMenu))->ADName();
LPCWSTR pszNoLocName = static_cast<CActDirMenuCmd*>((CMenuCmd*)(*itMenu))->ADNoLocName();
if( pszNoLocName && wcslen(pszNoLocName) )
{
if( _tcscmp(iter->strNoLoc.c_str(), pszNoLocName) == 0 )
{
bFound = TRUE;
break;
}
}
else if( pszName && _tcscmp(iter->strPlain.c_str(), pszName) == 0 )
{
bFound = TRUE;
break;
}
iter++;
}
if (!bFound)
{
continue;
}
}
CONTEXTMENUITEM2 item;
OLECHAR szGuid[50] = {0};
::StringFromGUID2((*itMenu)->NoLocID(), szGuid, 50);
item.strName = const_cast<LPWSTR>((*itMenu)->Name());
item.strStatusBarText = L"";
item.lCommandID = lCmdID;
item.lInsertionPointID = CCM_INSERTIONPOINTID_PRIMARY_TOP;
item.fFlags = 0;
item.fSpecialFlags = (lCmdID == iDefault) ? CCM_SPECIAL_DEFAULT_ITEM : 0;
item.strLanguageIndependentName = szGuid;
hr = spContext2->AddItem(&item);
ASSERT(SUCCEEDED(hr));
}
return hr;
}
HRESULT
CQueryItem::QueryPagesFor()
{
ASSERT(m_pRowItem != NULL && m_pRowItem->size() >= ROWITEM_USER_INDEX);
ASSERT(m_spQueryNode != NULL);
// Row item has class display name, so get internal name from class map
DisplayNameMap* pNameMap = DisplayNames::GetClassMap();
if (pNameMap == NULL)
return E_FAIL;
LPCWSTR pszClass = pNameMap->GetInternalName((*m_pRowItem)[ROWITEM_CLASS_INDEX]);
// Create a directory extension object
// (we might have one already if AddMenuItems was called before)
if (m_pADExt == NULL)
m_pADExt = new CActDirExt();
ASSERT(m_pADExt != NULL);
if (m_pADExt == NULL)
return E_OUTOFMEMORY;
HRESULT hr = m_pADExt->Initialize(pszClass, m_pRowItem->GetObjPath());
RETURN_ON_FAILURE(hr);
hpage_vector vhPages;
hr = m_pADExt->GetPropertyPages(vhPages);
if (SUCCEEDED(hr) && vhPages.size() > 0)
{
CPropertySheet sheet;
// Set title to name of item
// Can't use SetTitle becuase if wrongly asserts (pszText == NULL)
sheet.m_psh.pszCaption = (*m_pRowItem)[ROWITEM_NAME_INDEX];
sheet.m_psh.dwFlags |= PSH_PROPTITLE;
hpage_vector::iterator itPage;
for (itPage = vhPages.begin(); itPage != vhPages.end(); ++itPage)
{
BOOL bStat = sheet.AddPage(*itPage);
ASSERT(bStat);
}
sheet.DoModal(g_hwndMain);
}
return S_FALSE;
}
class CRefreshCallback : public CEventCallback
{
public:
CRefreshCallback(HANDLE hProcess, CQueryableNode* pQueryNode)
: m_hProcess(hProcess), m_spQueryNode(pQueryNode) {}
virtual void Execute()
{
if( m_spQueryNode )
{
m_spQueryNode->OnRefresh(NULL);
}
CloseHandle(m_hProcess);
}
HANDLE m_hProcess;
CQueryableNodePtr m_spQueryNode;
};
HRESULT
CQueryItem::MenuCommand(LPCONSOLE2 pConsole, long lCommand)
{
VALIDATE_POINTER(pConsole);
ASSERT( lCommand < m_vMenus.size() && lCommand >= 0 );
if( lCommand >= m_vMenus.size() || lCommand < 0 )
return E_INVALIDARG;
HRESULT hr = E_FAIL;
switch (m_vMenus[lCommand]->MenuType())
{
case MENUTYPE_SHELL:
{
// Create a query Lookup object to translate the command parameters
CQueryLookup lookup(m_spQueryNode, m_pRowItem);
HANDLE hProcess = NULL;
hr = static_cast<CShellMenuCmd*>((CMenuCmd*)m_vMenus[lCommand])->Execute(&lookup, &hProcess);
// if process started and auto-refresh wanted, setup event-triggered callback
if (SUCCEEDED(hr) && hProcess != NULL && m_vMenus[lCommand]->IsAutoRefresh())
{
CallbackOnEvent(hProcess, new CRefreshCallback(hProcess, m_spQueryNode));
}
break;
}
case MENUTYPE_ACTDIR:
{
ASSERT(m_pADExt != NULL);
BOMMENU bmMenu;
bmMenu.strPlain = static_cast<CActDirMenuCmd*>((CMenuCmd*)m_vMenus[lCommand])->ADName();
bmMenu.strNoLoc = static_cast<CActDirMenuCmd*>((CMenuCmd*)m_vMenus[lCommand])->ADNoLocName();
hr = m_pADExt->Execute(&bmMenu);
// if commans should auto-refresh, do it now
if (SUCCEEDED(hr) && m_vMenus[lCommand]->IsAutoRefresh())
{
ASSERT(m_spQueryNode != NULL);
m_spQueryNode->OnRefresh(NULL);
}
break;
}
default:
ASSERT(0 && L"Unhandled menu command type");
}
return hr;
}
HRESULT CQueryItem::GetDataImpl(UINT cf, HGLOBAL* phGlobal)
{
VALIDATE_POINTER( phGlobal );
HRESULT hr = DV_E_FORMATETC;
if (cf == m_cfDisplayName)
{
hr = DataToGlobal(phGlobal, (*m_pRowItem)[0], wcslen((*m_pRowItem)[0]) * sizeof(WCHAR));
}
else if (cf == m_cfSnapInClsid)
{
hr = DataToGlobal(phGlobal, &CLSID_BOMSnapIn, sizeof(GUID));
}
else if (cf == m_cfNodeType)
{
hr = DataToGlobal(phGlobal, &GUID_QueryItem, sizeof(GUID));
}
else if (cf == m_cfszNodeType)
{
WCHAR szGuid[GUID_STRING_LEN+1];
StringFromGUID2(GUID_QueryItem, szGuid, GUID_STRING_LEN+1);
hr = DataToGlobal(phGlobal, szGuid, GUID_STRING_SIZE);
}
return hr;
}