455 lines
12 KiB
C++
455 lines
12 KiB
C++
|
//==============================================================;
|
||
|
//
|
||
|
// This source code is only intended as a supplement to existing Microsoft documentation.
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
|
||
|
// KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
||
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR
|
||
|
// PURPOSE.
|
||
|
//
|
||
|
// Copyright (C) 1999 Microsoft Corporation. All Rights Reserved.
|
||
|
//
|
||
|
//
|
||
|
//
|
||
|
//==============================================================;
|
||
|
|
||
|
#include "Comp.h"
|
||
|
#include "CompData.h"
|
||
|
#include "Space.h"
|
||
|
#include "DataObj.h"
|
||
|
#include <commctrl.h> // Needed for button styles...
|
||
|
#include <crtdbg.h>
|
||
|
#include <stdio.h> // needed for _stprintf
|
||
|
#include "globals.h"
|
||
|
#include "resource.h"
|
||
|
#include "DeleBase.h"
|
||
|
#include "CompData.h"
|
||
|
|
||
|
CComponent::CComponent(CComponentData *pParent)
|
||
|
: m_pParent(pParent), m_cref(0), m_ipConsole(NULL), m_pLastNode(NULL)
|
||
|
{
|
||
|
OBJECT_CREATED
|
||
|
}
|
||
|
|
||
|
CComponent::~CComponent()
|
||
|
{
|
||
|
OBJECT_DESTROYED
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::QueryInterface(REFIID riid, LPVOID *ppv)
|
||
|
{
|
||
|
if (!ppv)
|
||
|
return E_FAIL;
|
||
|
|
||
|
*ppv = NULL;
|
||
|
|
||
|
if (IsEqualIID(riid, IID_IUnknown))
|
||
|
*ppv = static_cast<IComponent *>(this);
|
||
|
else if (IsEqualIID(riid, IID_IComponent))
|
||
|
*ppv = static_cast<IComponent *>(this);
|
||
|
else if (IsEqualIID(riid, IID_IResultOwnerData))
|
||
|
*ppv = static_cast<IResultOwnerData *>(this);
|
||
|
|
||
|
if (*ppv)
|
||
|
{
|
||
|
reinterpret_cast<IUnknown *>(*ppv)->AddRef();
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
return E_NOINTERFACE;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CComponent::AddRef()
|
||
|
{
|
||
|
return InterlockedIncrement((LONG *)&m_cref);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CComponent::Release()
|
||
|
{
|
||
|
if (InterlockedDecrement((LONG *)&m_cref) == 0)
|
||
|
{
|
||
|
delete this;
|
||
|
return 0;
|
||
|
}
|
||
|
return m_cref;
|
||
|
|
||
|
}
|
||
|
|
||
|
///////////////////////////////
|
||
|
// Interface IComponent
|
||
|
///////////////////////////////
|
||
|
STDMETHODIMP CComponent::Initialize(
|
||
|
/* [in] */ LPCONSOLE lpConsole)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
// Save away all the interfaces we'll need.
|
||
|
// Fail if we can't QI the required interfaces.
|
||
|
|
||
|
m_ipConsole = lpConsole;
|
||
|
m_ipConsole->AddRef();
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::Notify(
|
||
|
/* [in] */ LPDATAOBJECT lpDataObject,
|
||
|
/* [in] */ MMC_NOTIFY_TYPE event,
|
||
|
/* [in] */ LPARAM arg,
|
||
|
/* [in] */ LPARAM param)
|
||
|
{
|
||
|
MMCN_Crack(FALSE, lpDataObject, NULL, this, event, arg, param);
|
||
|
|
||
|
HRESULT hr = S_FALSE;
|
||
|
|
||
|
// MMCN_VIEW_CHANGE
|
||
|
|
||
|
if (MMCN_VIEW_CHANGE == event) {
|
||
|
switch (param) {//arg holds the data. For a scope item, this is the
|
||
|
//item's HSCOPEITEM. For a result item, this is
|
||
|
//the item's nId value, but we don't use it
|
||
|
|
||
|
//param holds the hint passed to IConsole::UpdateAllViews.
|
||
|
//hint is a value of the UPDATE_VIEWS_HINT enumeration
|
||
|
|
||
|
case UPDATE_SCOPEITEM:
|
||
|
hr = m_ipConsole->SelectScopeItem( (HSCOPEITEM)arg );
|
||
|
_ASSERT( S_OK == hr);
|
||
|
break;
|
||
|
case UPDATE_RESULTITEM:
|
||
|
CDelegationBase *base = GetOurDataObject(lpDataObject)->GetBaseNodeObject();
|
||
|
hr = base->OnUpdateItem(m_ipConsole, (long)arg, RESULT);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
//Remaining notifications
|
||
|
|
||
|
CDelegationBase *base = GetOurDataObject(lpDataObject)->GetBaseNodeObject();
|
||
|
|
||
|
switch (event)
|
||
|
{
|
||
|
case MMCN_SHOW:
|
||
|
hr = base->OnShow(m_ipConsole, (BOOL)arg, (HSCOPEITEM)param);
|
||
|
break;
|
||
|
|
||
|
case MMCN_ADD_IMAGES:
|
||
|
hr = base->OnAddImages((IImageList *)arg, (HSCOPEITEM)param);
|
||
|
break;
|
||
|
|
||
|
case MMCN_SELECT:
|
||
|
|
||
|
//check for multiselection
|
||
|
if ( MMC_MULTI_SELECT_COOKIE == GetOurDataObject(lpDataObject)->GetCookie() )
|
||
|
{
|
||
|
if ( (BOOL)LOWORD(arg) == 0 && (BOOL)HIWORD(arg) == 1 )
|
||
|
{
|
||
|
//We need the cookie of any of the multiselection items
|
||
|
//to enable the delete verb for all the items.
|
||
|
MMC_COOKIE ourCookie = GetOurDataObject(lpDataObject)->GetMultiSelectCookie(0);
|
||
|
|
||
|
base = reinterpret_cast<CDelegationBase *>(ourCookie);
|
||
|
hr = base->OnSelect(m_ipConsole, (BOOL)LOWORD(arg), (BOOL)HIWORD(arg));
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
else
|
||
|
hr = base->OnSelect(m_ipConsole, (BOOL)LOWORD(arg), (BOOL)HIWORD(arg));
|
||
|
break;
|
||
|
|
||
|
case MMCN_REFRESH:
|
||
|
hr = base->OnRefresh(m_pParent->m_ipConsole);
|
||
|
break;
|
||
|
|
||
|
case MMCN_DELETE:
|
||
|
|
||
|
//check for multiselection. if true, delete each item
|
||
|
if ( MMC_MULTI_SELECT_COOKIE == GetOurDataObject(lpDataObject)->GetCookie() )
|
||
|
{
|
||
|
|
||
|
int n = 0;
|
||
|
MMC_COOKIE ourCookie;
|
||
|
|
||
|
while ( ourCookie = GetOurDataObject(lpDataObject)->GetMultiSelectCookie(n) )
|
||
|
{
|
||
|
base = reinterpret_cast<CDelegationBase *>(ourCookie);
|
||
|
hr = base->OnDelete(m_ipConsole);
|
||
|
n++;
|
||
|
//Uncomment the following line to display a message box
|
||
|
//for each item deletion.
|
||
|
//DisplayMessageBox(base);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
else
|
||
|
{
|
||
|
//select item deletion
|
||
|
hr = base->OnDelete(m_ipConsole);
|
||
|
}
|
||
|
|
||
|
//Now call IConsole::UpdateAllViews to redraw all views
|
||
|
//owned by the parent scope item. OnRefresh already does
|
||
|
//this for us, so use it.
|
||
|
//Do this for both multiselection and single selection
|
||
|
hr = base->OnRefresh(m_pParent->m_ipConsole);
|
||
|
|
||
|
break;
|
||
|
|
||
|
case MMCN_RENAME:
|
||
|
hr = base->OnRename((LPOLESTR)param);
|
||
|
|
||
|
//Now call IConsole::UpdateAllViews to redraw the item in all views.
|
||
|
hr = m_pParent->m_ipConsole->UpdateAllViews(lpDataObject, 0, UPDATE_RESULTITEM);
|
||
|
_ASSERT( S_OK == hr);
|
||
|
|
||
|
break;
|
||
|
|
||
|
}//end switch
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::Destroy(
|
||
|
/* [in] */ MMC_COOKIE cookie)
|
||
|
{
|
||
|
if (m_ipConsole) {
|
||
|
m_ipConsole->Release();
|
||
|
m_ipConsole = NULL;
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
STDMETHODIMP CComponent::QueryDataObject(
|
||
|
/* [in] */ MMC_COOKIE cookie,
|
||
|
/* [in] */ DATA_OBJECT_TYPES type,
|
||
|
/* [out] */ LPDATAOBJECT __RPC_FAR *ppDataObject)
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
|
||
|
CDataObject *pObj = NULL;
|
||
|
CDelegationBase *pBase = NULL;
|
||
|
|
||
|
//Use The IS_SPECIAL_COOKIE macro to see if cookie is a special cookie
|
||
|
if ( IS_SPECIAL_COOKIE (cookie) ) {
|
||
|
if ( MMC_MULTI_SELECT_COOKIE == cookie) {
|
||
|
|
||
|
pObj = new CDataObject(cookie, type);
|
||
|
|
||
|
if (!pObj)
|
||
|
return E_OUTOFMEMORY;
|
||
|
|
||
|
//create the multiselection data object
|
||
|
hr = GetCurrentSelections(pObj);
|
||
|
_ASSERT( SUCCEEDED(hr) );
|
||
|
|
||
|
hr = pObj->QueryInterface(IID_IDataObject, (void **)ppDataObject);
|
||
|
_ASSERT( SUCCEEDED(hr) );
|
||
|
|
||
|
return hr;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Remaining code for "regular" cookies, and for the next item
|
||
|
//during a multiselection
|
||
|
|
||
|
if (IsBadReadPtr((void *)cookie, sizeof(CDelegationBase))) {
|
||
|
if (NULL == m_pLastNode)
|
||
|
return E_FAIL;
|
||
|
|
||
|
pBase = m_pLastNode->GetChildPtr((int)cookie);
|
||
|
} else {
|
||
|
pBase = (cookie == 0) ? m_pParent->m_pStaticNode : (CDelegationBase *)cookie;
|
||
|
}
|
||
|
|
||
|
if (pBase == NULL)
|
||
|
return E_FAIL;
|
||
|
|
||
|
pObj = new CDataObject((MMC_COOKIE)pBase, type);
|
||
|
|
||
|
if (!pObj)
|
||
|
return E_OUTOFMEMORY;
|
||
|
|
||
|
pObj->QueryInterface(IID_IDataObject, (void **)ppDataObject);
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::GetResultViewType(
|
||
|
/* [in] */ MMC_COOKIE cookie,
|
||
|
/* [out] */ LPOLESTR __RPC_FAR *ppViewType,
|
||
|
/* [out] */ long __RPC_FAR *pViewOptions)
|
||
|
{
|
||
|
CDelegationBase *base = m_pLastNode = (CDelegationBase *)cookie;
|
||
|
|
||
|
//
|
||
|
// Ask for default listview.
|
||
|
//
|
||
|
if (base == NULL)
|
||
|
{
|
||
|
*pViewOptions = MMC_VIEW_OPTIONS_NONE;
|
||
|
*ppViewType = NULL;
|
||
|
}
|
||
|
else
|
||
|
return base->GetResultViewType(ppViewType, pViewOptions);
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::GetDisplayInfo(
|
||
|
/* [out][in] */ RESULTDATAITEM __RPC_FAR *pResultDataItem)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
CDelegationBase *base = NULL;
|
||
|
|
||
|
// if they are asking for the RDI_STR we have one of those to give
|
||
|
|
||
|
if (pResultDataItem->lParam) {
|
||
|
base = (CDelegationBase *)pResultDataItem->lParam;
|
||
|
if (pResultDataItem->mask & RDI_STR) {
|
||
|
LPCTSTR pszT = base->GetDisplayName(pResultDataItem->nCol);
|
||
|
MAKE_WIDEPTR_FROMTSTR_ALLOC(pszW, pszT);
|
||
|
pResultDataItem->str = pszW;
|
||
|
}
|
||
|
|
||
|
if (pResultDataItem->mask & RDI_IMAGE) {
|
||
|
pResultDataItem->nImage = base->GetBitmapIndex();
|
||
|
}
|
||
|
} else {
|
||
|
m_pLastNode->GetChildColumnInfo(pResultDataItem);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
STDMETHODIMP CComponent::CompareObjects(
|
||
|
/* [in] */ LPDATAOBJECT lpDataObjectA,
|
||
|
/* [in] */ LPDATAOBJECT lpDataObjectB)
|
||
|
{
|
||
|
CDelegationBase *baseA = GetOurDataObject(lpDataObjectA)->GetBaseNodeObject();
|
||
|
CDelegationBase *baseB = GetOurDataObject(lpDataObjectB)->GetBaseNodeObject();
|
||
|
|
||
|
// compare the object pointers
|
||
|
if (baseA->GetCookie() == baseB->GetCookie())
|
||
|
return S_OK;
|
||
|
|
||
|
return S_FALSE;
|
||
|
}
|
||
|
|
||
|
///////////////////////////////
|
||
|
// Interface IComponent
|
||
|
///////////////////////////////
|
||
|
STDMETHODIMP CComponent::FindItem(
|
||
|
/* [in] */ LPRESULTFINDINFO pFindInfo,
|
||
|
/* [out] */ int __RPC_FAR *pnFoundIndex)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::CacheHint(
|
||
|
/* [in] */ int nStartIndex,
|
||
|
/* [in] */ int nEndIndex)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CComponent::SortItems(
|
||
|
/* [in] */ int nColumn,
|
||
|
/* [in] */ DWORD dwSortOptions,
|
||
|
/* [in] */ LPARAM lUserParam)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
|
||
|
///////////////////////////////////////////
|
||
|
// GetCurrentSelections() finds the currently
|
||
|
// selected result items and the data object
|
||
|
// passed to it with their cookie values.
|
||
|
// The multi-select cookie is cached so that
|
||
|
// we don't have to calculate multiselection
|
||
|
// dataobject for other notifications.
|
||
|
// It is destroyed at appropriate time.
|
||
|
///////////////////////////////////////////
|
||
|
|
||
|
HRESULT CComponent::GetCurrentSelections(CDataObject *pMultiSelectDataObject)
|
||
|
{
|
||
|
|
||
|
HRESULT hr = S_FALSE;
|
||
|
|
||
|
//GetCurrentSelections only works for multiselection data objects
|
||
|
if ( !( MMC_MULTI_SELECT_COOKIE == GetOurDataObject(pMultiSelectDataObject)->GetCookie() ) )
|
||
|
return hr = E_INVALIDARG;
|
||
|
|
||
|
IResultData *pResultData = NULL;
|
||
|
|
||
|
hr = m_ipConsole->QueryInterface(IID_IResultData, (void **)&pResultData);
|
||
|
_ASSERT( SUCCEEDED(hr) );
|
||
|
|
||
|
RESULTDATAITEM rdi;
|
||
|
|
||
|
BOOL isLastSelected = FALSE;
|
||
|
int nIndex = -1;
|
||
|
int nIndexCookies = 0;
|
||
|
|
||
|
while (!isLastSelected)
|
||
|
{
|
||
|
ZeroMemory(&rdi, sizeof(RESULTDATAITEM) );
|
||
|
rdi.mask = RDI_STATE; // nState is valid
|
||
|
rdi.nCol = 0;
|
||
|
rdi.nIndex = nIndex; // nIndex == -1 to start at first item
|
||
|
rdi.nState = LVIS_SELECTED; // only interested in selected items
|
||
|
|
||
|
|
||
|
hr = pResultData->GetNextItem(&rdi);
|
||
|
_ASSERT( SUCCEEDED(hr) );
|
||
|
|
||
|
if (rdi.nIndex != -1) {
|
||
|
|
||
|
//rdi is the RESULTDATAITEM of a selected item. add its
|
||
|
//lParam to the pCookies array of the pMultiSelectDataObject data object
|
||
|
|
||
|
_ASSERT( nIndexCookies < 20 ); // MAX_COOKIES == 20
|
||
|
pMultiSelectDataObject->AddMultiSelectCookie(nIndexCookies, rdi.lParam);
|
||
|
nIndexCookies++;
|
||
|
nIndex = rdi.nIndex;
|
||
|
}
|
||
|
|
||
|
else
|
||
|
isLastSelected = TRUE;
|
||
|
|
||
|
}
|
||
|
|
||
|
pResultData->Release();
|
||
|
|
||
|
return hr;
|
||
|
|
||
|
}
|
||
|
|
||
|
void CComponent::DisplayMessageBox(CDelegationBase* base)
|
||
|
{
|
||
|
|
||
|
_TCHAR szVehicle[128];
|
||
|
static _TCHAR buf[128];
|
||
|
|
||
|
_stprintf(buf, _T("%s"), base->GetDisplayName() );
|
||
|
|
||
|
wsprintf(szVehicle, _T("%s deleted"), buf);
|
||
|
|
||
|
int ret = 0;
|
||
|
MAKE_WIDEPTR_FROMTSTR_ALLOC(wszVehicle, szVehicle);
|
||
|
m_ipConsole->MessageBox(wszVehicle,
|
||
|
L"Vehicle command", MB_OK | MB_ICONINFORMATION, &ret);
|
||
|
|
||
|
return;
|
||
|
}
|