422 lines
12 KiB
C++
422 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 "DataObj.h"
|
|
#include <commctrl.h> // Needed for button styles...
|
|
#include <crtdbg.h>
|
|
#include "resource.h"
|
|
#include "DeleBase.h"
|
|
#include "CompData.h"
|
|
#include "globals.h"
|
|
|
|
|
|
#define ARRAYLEN(x) (sizeof(x) / sizeof((x)[0]))
|
|
|
|
CComponent::CComponent(CComponentData *parent)
|
|
: m_pComponentData(parent), m_cref(0), m_ipConsole(NULL)
|
|
{
|
|
OBJECT_CREATED
|
|
m_ipControlBar = NULL;
|
|
m_ipToolbar = NULL;
|
|
}
|
|
|
|
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);
|
|
|
|
// first things first, make sure that when MMC
|
|
// asks if we do property sheets, that we actually
|
|
// say "yes"
|
|
else if (IsEqualIID(riid, IID_IExtendPropertySheet))
|
|
*ppv = static_cast<IExtendPropertySheet2 *>(this);
|
|
else if (IsEqualIID(riid, IID_IExtendPropertySheet2))
|
|
*ppv = static_cast<IExtendPropertySheet2 *>(this);
|
|
else if (IsEqualIID(riid, IID_IExtendControlbar))
|
|
*ppv = static_cast<IExtendControlbar *>(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;
|
|
CDelegationBase *base = NULL;
|
|
|
|
// we need to watch for property change and delegate it
|
|
// a little differently, we're actually going to send
|
|
// the CDelegationBase object pointer in the property page
|
|
// PSN_APPLY handler via MMCPropPageNotify()
|
|
if (MMCN_PROPERTY_CHANGE != event && MMCN_VIEW_CHANGE != event) {
|
|
if (NULL == lpDataObject)
|
|
return S_FALSE;
|
|
|
|
base = GetOurDataObject(lpDataObject)->GetBaseNodeObject();
|
|
|
|
} else if (MMCN_PROPERTY_CHANGE == event) {
|
|
base = (CDelegationBase *)param;
|
|
}
|
|
|
|
|
|
// MMCN_VIEW_CHANGE
|
|
|
|
static CDelegationBase *pLastPasteQuery = NULL;
|
|
|
|
if (MMCN_VIEW_CHANGE == event) {
|
|
|
|
switch (param) {//arg holds the data. For a scope item, this is the
|
|
//item's myhscopeitem. 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:
|
|
base = GetOurDataObject(lpDataObject)->GetBaseNodeObject();
|
|
hr = base->OnUpdateItem(m_ipConsole, (long)arg, RESULT);
|
|
break;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
//The remaining notifications
|
|
|
|
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:
|
|
hr = base->OnSelect(this, m_ipConsole, (BOOL)LOWORD(arg), (BOOL)HIWORD(arg));
|
|
break;
|
|
|
|
case MMCN_RENAME:
|
|
hr = base->OnRename((LPOLESTR)param);
|
|
|
|
//Now call IConsole::UpdateAllViews to redraw the item in all views.
|
|
hr = m_pComponentData->m_ipConsole->UpdateAllViews(lpDataObject, 0, UPDATE_RESULTITEM);
|
|
_ASSERT( S_OK == hr);
|
|
break;
|
|
|
|
case MMCN_REFRESH:
|
|
//we pass CComponentData's stored IConsole pointer here,
|
|
//so that the IConsole::UpdateAllViews can be called in OnRefresh
|
|
hr = base->OnRefresh(m_pComponentData->m_ipConsole);
|
|
break;
|
|
|
|
case MMCN_DELETE: {
|
|
//first delete the selected result item
|
|
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.
|
|
hr = base->OnRefresh(m_pComponentData->m_ipConsole);
|
|
break;
|
|
}
|
|
|
|
// handle the property change notification if we need to do anything
|
|
// special with it
|
|
case MMCN_PROPERTY_CHANGE:
|
|
//we pass CComponentData's stored IConsole pointer here,
|
|
//so that the IConsole::UpdateAllViews can be called in OnPropertyChange
|
|
hr = base->OnPropertyChange(m_pComponentData->m_ipConsole, this);
|
|
break;
|
|
}
|
|
|
|
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)
|
|
{
|
|
CDataObject *pObj = NULL;
|
|
|
|
if (cookie == 0)
|
|
pObj = new CDataObject((MMC_COOKIE)m_pComponentData->m_pStaticNode, type);
|
|
else
|
|
pObj = new CDataObject(cookie, 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 = (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();
|
|
}
|
|
}
|
|
|
|
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 IExtendPropertySheet2
|
|
///////////////////////////////////
|
|
HRESULT CComponent::CreatePropertyPages(
|
|
/* [in] */ LPPROPERTYSHEETCALLBACK lpProvider,
|
|
/* [in] */ LONG_PTR handle,
|
|
/* [in] */ LPDATAOBJECT lpIDataObject)
|
|
{
|
|
CDelegationBase *base = GetOurDataObject(lpIDataObject)->GetBaseNodeObject();
|
|
|
|
return base->CreatePropertyPages(lpProvider, handle);
|
|
}
|
|
|
|
HRESULT CComponent::QueryPagesFor(
|
|
/* [in] */ LPDATAOBJECT lpDataObject)
|
|
{
|
|
CDelegationBase *base = GetOurDataObject(lpDataObject)->GetBaseNodeObject();
|
|
|
|
return base->HasPropertySheets();
|
|
}
|
|
|
|
HRESULT CComponent::GetWatermarks(
|
|
/* [in] */ LPDATAOBJECT lpIDataObject,
|
|
/* [out] */ HBITMAP __RPC_FAR *lphWatermark,
|
|
/* [out] */ HBITMAP __RPC_FAR *lphHeader,
|
|
/* [out] */ HPALETTE __RPC_FAR *lphPalette,
|
|
/* [out] */ BOOL __RPC_FAR *bStretch)
|
|
{
|
|
CDelegationBase *base = GetOurDataObject(lpIDataObject)->GetBaseNodeObject();
|
|
|
|
return base->GetWatermarks(lphWatermark, lphHeader, lphPalette, bStretch);
|
|
}
|
|
|
|
///////////////////////////////
|
|
// Interface IExtendControlBar
|
|
///////////////////////////////
|
|
static MMCBUTTON SnapinButtons1[] =
|
|
{
|
|
{ 0, ID_BUTTONSTART, TBSTATE_ENABLED, TBSTYLE_GROUP, L"Start Vehicle", L"Start Vehicle" },
|
|
{ 1, ID_BUTTONPAUSE, TBSTATE_ENABLED, TBSTYLE_GROUP, L"Pause Vehicle", L"Pause Vehicle"},
|
|
{ 2, ID_BUTTONSTOP, TBSTATE_ENABLED, TBSTYLE_GROUP, L"Stop Vehicle", L"Stop Vehicle" },
|
|
};
|
|
|
|
HRESULT CComponent::SetControlbar(
|
|
/* [in] */ LPCONTROLBAR pControlbar)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
//
|
|
// Clean up
|
|
//
|
|
|
|
// if we've got a cached toolbar, release it
|
|
if (m_ipToolbar) {
|
|
m_ipToolbar->Release();
|
|
m_ipToolbar = NULL;
|
|
}
|
|
|
|
// if we've got a cached control bar, release it
|
|
if (m_ipControlBar) {
|
|
m_ipControlBar->Release();
|
|
m_ipControlBar = NULL;
|
|
}
|
|
|
|
|
|
//
|
|
// Install new pieces if necessary
|
|
//
|
|
|
|
// if a new one came in, cache and AddRef
|
|
if (pControlbar) {
|
|
m_ipControlBar = pControlbar;
|
|
m_ipControlBar->AddRef();
|
|
|
|
hr = m_ipControlBar->Create(TOOLBAR, // type of control to be created
|
|
dynamic_cast<IExtendControlbar *>(this),
|
|
reinterpret_cast<IUnknown **>(&m_ipToolbar));
|
|
_ASSERT(SUCCEEDED(hr));
|
|
|
|
// The IControlbar::Create AddRefs the toolbar object it created
|
|
// so no need to do any addref on the interface.
|
|
|
|
// add the bitmap to the toolbar
|
|
HBITMAP hbmp = LoadBitmap(g_hinst, MAKEINTRESOURCE(IDR_TOOLBAR1));
|
|
hr = m_ipToolbar->AddBitmap(3, hbmp, 16, 16, RGB(0, 128, 128)); // NOTE, hardcoded value 3
|
|
_ASSERT(SUCCEEDED(hr));
|
|
|
|
// Add the buttons to the toolbar
|
|
hr = m_ipToolbar->AddButtons(ARRAYLEN(SnapinButtons1), SnapinButtons1);
|
|
_ASSERT(SUCCEEDED(hr));
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CComponent::ControlbarNotify(
|
|
/* [in] */ MMC_NOTIFY_TYPE event,
|
|
/* [in] */ LPARAM arg,
|
|
/* [in] */ LPARAM param)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (event == MMCN_SELECT) {
|
|
BOOL bScope = (BOOL) LOWORD(arg);
|
|
BOOL bSelect = (BOOL) HIWORD(arg);
|
|
|
|
CDelegationBase *base = GetOurDataObject(reinterpret_cast<IDataObject *>(param))->GetBaseNodeObject();
|
|
hr = base->OnSetToolbar(m_ipControlBar, m_ipToolbar, bScope, bSelect);
|
|
}
|
|
|
|
else if (event == MMCN_BTN_CLICK) {
|
|
CDelegationBase *base = GetOurDataObject(reinterpret_cast<IDataObject *>(arg))->GetBaseNodeObject();
|
|
hr = base->OnToolbarCommand(m_pComponentData->m_ipConsole, (MMC_CONSOLE_VERB)param, reinterpret_cast<IDataObject *>(arg));
|
|
}
|
|
|
|
return hr;
|
|
}
|