Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1525 lines
25 KiB

/*++
Copyright (c) 1994-1999 Microsoft Corporation
Module Name :
inetmgr.cpp
Abstract:
Main MMC snap-in code
Author:
Ronald Meijer (ronaldm)
Project:
Internet Services Manager
Revision History:
--*/
#include "stdafx.h"
#include "common.h"
#include "InetMgrApp.h"
#include "iisobj.h"
#include "guids.h"
#ifdef _DEBUG
#undef THIS_FILE
static char BASED_CODE THIS_FILE[] = __FILE__;
#endif
#define new DEBUG_NEW
//const GUID * CCompMgrExtData::m_NODETYPE = &cCompMgmtService;
//const OLECHAR * CCompMgrExtData::m_SZNODETYPE = OLESTR("476e6446-aaff-11d0-b944-00c04fd8d5b0");
//const OLECHAR * CCompMgrExtData::m_SZDISPLAY_NAME = OLESTR("CMSnapin");
//const CLSID * CCompMgrExtData::m_SNAPIN_CLASSID = &CLSID_InetMgr;
extern CInetmgrApp theApp;
static HRESULT
GetHelpTopic(LPOLESTR *lpCompiledHelpFile)
{
if (lpCompiledHelpFile == NULL)
return E_INVALIDARG;
CString strFilePath, strWindowsPath, strBuffer;
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// Use system API to get windows directory.
UINT uiResult = GetWindowsDirectory(strWindowsPath.GetBuffer(MAX_PATH), MAX_PATH);
strWindowsPath.ReleaseBuffer();
if (uiResult <= 0 || uiResult > MAX_PATH)
{
return E_FAIL;
}
if (!strFilePath.LoadString(IDS_HELPFILE))
{
return E_FAIL;
}
strBuffer = strWindowsPath;
strBuffer += _T('\\');
strBuffer += strFilePath;
*lpCompiledHelpFile
= reinterpret_cast<LPOLESTR>(CoTaskMemAlloc((strBuffer.GetLength() + 1)
* sizeof(_TCHAR)));
if (*lpCompiledHelpFile == NULL)
return E_OUTOFMEMORY;
USES_CONVERSION;
_tcscpy(*lpCompiledHelpFile, T2OLE((LPTSTR)(LPCTSTR)strBuffer));
return S_OK;
}
//
// CInetMgrComponent Implementation
//
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
CInetMgrComponent::CInetMgrComponent()
/*++
Routine Description:
Constructor
Arguments:
None
Return Value:
N/A
--*/
{
}
HRESULT
CInetMgrComponent::Notify(
IN LPDATAOBJECT lpDataObject,
IN MMC_NOTIFY_TYPE event,
IN LPARAM arg,
IN LPARAM param
)
/*++
Routine Description:
Notification handler.
Arguments:
LPDATAOBJECT lpDataObject : Data object
MMC_NOTIFY_TYPE event : Notification event
long arg : Event specific argument
long param : Event specific parameter
Return Value:
HRESULT
--*/
{
if (lpDataObject != NULL)
{
//
// Pass it on to IComponentImpl
//
return IComponentImpl<CInetMgrComponent>::Notify(lpDataObject, event, arg, param);
}
return E_NOTIMPL;
}
HRESULT
CInetMgrComponent::GetClassID(
OUT CLSID * pClassID
)
/*++
Routine Description:
Get class ID for storage stream.
Arguments:
CLSID * pClassID : Returns class ID information
Return Value:
HRESULT
--*/
{
*pClassID = CLSID_InetMgr;
return S_OK;
}
STDMETHODIMP
CInetMgrComponent::GetHelpTopic(LPOLESTR *lpCompiledHelpFile)
{
return ::GetHelpTopic(lpCompiledHelpFile);
}
STDMETHODIMP
CInetMgrComponent::GetLinkedTopics(LPOLESTR *lpCompiledHelpFile)
{
return S_FALSE;
}
HRESULT
CInetMgrComponent::IsDirty()
/*++
Routine Description:
Check to see if we need to write to the cache.
Arguments:
None
Return Value:
S_OK if dirty, S_FALSE if not
--*/
{
TRACEEOLID("CInetMgrComponent::IsDirty");
return S_FALSE;
}
HRESULT
CInetMgrComponent::InitNew(
IN OUT IStorage * pStg
)
/*++
Routine Description:
Initialize storage stream.
Arguments:
IStorage * pStg : Storage stream
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::InitNew");
return S_OK;
}
HRESULT
CInetMgrComponent::Load(
IN OUT IStorage * pStg
)
/*++
Routine Description:
Load from the storage stream
Arguments:
IStorage * pStg : Storage stream
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::Load");
return S_OK;
}
/* virtual */
HRESULT
STDMETHODCALLTYPE
CInetMgrComponent::Save(
IN OUT IStorage * pStgSave,
IN BOOL fSameAsLoad
)
/*++
Routine Description:
Save to to the storage stream.
Arguments:
IStorage * pStgSave : Storage stream
BOOL fSameAsLoad : TRUE if same as load
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::Save");
return S_OK;
}
/* virtual */
HRESULT
STDMETHODCALLTYPE
CInetMgrComponent::SaveCompleted(IStorage * pStgNew)
/*++
Routine Description:
Save completed.
Arguments:
IStorage * pStgNew : Storage stream
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::SaveCompleted");
return S_OK;
}
/* virtual */
HRESULT
STDMETHODCALLTYPE
CInetMgrComponent::HandsOffStorage()
/*++
Routine Description:
Hands off storage.
Arguments:
None
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::HandsOffStorage");
return S_OK;
}
/* virtual */
HRESULT
CInetMgrComponent::SetControlbar(
IN LPCONTROLBAR lpControlBar
)
/*++
Routine Description:
Set/Reset the control bar
Arguments:
LPCONTROLBAR lpControlBar : Control bar pointer or NULL
Return Value:
HRESULT
--*/
{
return CIISObject::__SetControlbar(lpControlBar, this);
}
/* virtual */
HRESULT
CInetMgrComponent::ControlbarNotify(
IN MMC_NOTIFY_TYPE event,
IN LPARAM arg,
IN LPARAM param
)
/*++
Routine Description:
Handle control bar notification message. Figure out the CIISObject
selected, and pass the notification message off to it.
Arguments:
MMC_NOTIFY_TYPE event : Notification message
long arg : Message specific argument
long param : Message specific parameter
Return Value:
HRESULT
--*/
{
HRESULT hr = S_OK;
CSnapInItem * pItem = NULL;
DATA_OBJECT_TYPES type;
//
// Special casing this is annoying...
//
// CODEWORK: Handle MMCN_HELP and others
//
if (event == MMCN_BTN_CLICK)
{
hr = m_pComponentData->GetDataClass((IDataObject *)arg, &pItem, &type);
}
else if (event == MMCN_SELECT)
{
hr = m_pComponentData->GetDataClass((IDataObject *)param, &pItem, &type);
}
//
// Find out CIISObject this belongs to and pass on
// the message
//
CIISObject * pObject = (CIISObject *)pItem;
if (SUCCEEDED(hr) && pObject != NULL)
{
hr = pObject->ControlbarNotify(event, arg, param);
}
return hr;
}
/* virtual */
HRESULT
CInetMgrComponent::Compare(
IN RDCOMPARE * prdc,
OUT int * pnResult
)
/*++
Routine Description:
Compare method used for sorting the result and scope panes.
Arguments:
RDCOMPARE * prdc : Compare structure
int * pnResult : Returns result
Return Value:
HRESULT
--*/
{
if (!pnResult || !prdc || !prdc->prdch1->cookie || !prdc->prdch2->cookie)
{
ASSERT_MSG("Invalid parameter(s)");
return E_POINTER;
}
CIISObject * pObjectA = (CIISObject *)prdc->prdch1->cookie;
CIISObject * pObjectB = (CIISObject *)prdc->prdch2->cookie;
*pnResult = pObjectA->CompareResultPaneItem(pObjectB, prdc->nColumn);
return S_OK;
}
/* virtual */
HRESULT
CInetMgrComponent::CompareObjects(
IN LPDATAOBJECT lpDataObjectA,
IN LPDATAOBJECT lpDataObjectB
)
/*++
Routine Description:
Compare two data objects. This method is used to see if a property
sheet for the given data object is already open
Arguments:
LPDATAOBJECT lpDataObjectA : A data object
LPDATAOBJECT lpDataObjectB : B data object
Return Value:
S_OK if they match, S_FALSE otherwise
--*/
{
//
// Pass it on to IComponentImpl
//
return IComponentImpl<CInetMgrComponent>::CompareObjects(lpDataObjectA, lpDataObjectB);
}
//
// CInetMgr Implementation
//
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
/* static */ DWORD CInetMgr::_dwSignature = 0x3517;
/* static */ LPCTSTR CInetMgr::_szStream = _T("CInetMgr");
/* static */
void
WINAPI
CInetMgr::ObjectMain(
IN bool bStarting
)
/*++
Routine Description:
CInetMgr main entry point
Arguments:
bool bStarting : TRUE if starting
Return Value:
None
--*/
{
if (bStarting)
{
//
// Register clipboard formats
//
CSnapInItem::Init();
CIISObject::Init();
}
}
CInetMgr::CInetMgr()
{
//
// Initialize the root node
//
CIISObject::Initialize();
m_pNode = new CIISRoot;
ASSERT_PTR(m_pNode);
m_pComponentData = this;
}
CInetMgr::~CInetMgr()
/*++
Routine Description:
Destructor
Arguments:
N/A
Return Value:
N/A
--*/
{
//
// Clean up the root node
//
delete m_pNode;
m_pNode = NULL;
CIISObject::Destroy();
}
HRESULT
CInetMgr::Initialize(
IN LPUNKNOWN lpUnknown
)
/*++
Routine Description:
Initialize the snap-in
Arguments:
LPUNKNOWN lpUnknown : IUnknown
Return Value:
HRESULT
--*/
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
HRESULT hr =
IComponentDataImpl<CInetMgr, CInetMgrComponent>::Initialize(lpUnknown);
if (FAILED(hr))
{
return hr;
}
hr = CIISObject::AttachScopeView(lpUnknown);
if (FAILED(hr))
{
TRACEEOLID("failed to query console name space interface");
return hr;
}
CComPtr<IImageList> lpImageList;
hr = m_spConsole->QueryScopeImageList(&lpImageList);
if (FAILED(hr) || lpImageList == NULL)
{
TRACEEOLID("IConsole::QueryScopeImageList failed");
return E_UNEXPECTED;
}
return CIISObject::SetImageList(lpImageList);
}
HRESULT
CInetMgr::OnPropertyChange(LPARAM arg, LPARAM param)
{
HRESULT hr = S_OK;
if (param != 0)
{
CSnapInItem * pNode = (CSnapInItem *)param;
LPDATAOBJECT pDataObject = NULL;
pNode->GetDataObject(&pDataObject, CCT_SCOPE);
hr = m_spConsole->UpdateAllViews(pDataObject, param, 0);
}
return hr;
}
HRESULT
CInetMgr::Notify(
LPDATAOBJECT lpDataObject,
MMC_NOTIFY_TYPE event,
LPARAM arg,
LPARAM param
)
{
HRESULT hr = S_OK;
if (lpDataObject == NULL)
{
switch (event)
{
case MMCN_PROPERTY_CHANGE:
hr = OnPropertyChange(arg, param);
break;
case MMCN_SNAPINHELP:
break;
default:
break;
}
}
else
{
hr = IComponentDataImpl<CInetMgr, CInetMgrComponent>::Notify(
lpDataObject, event, arg, param);
}
return hr;
}
HRESULT
CInetMgr::GetClassID(CLSID * pClassID)
/*++
Routine Description:
Get class ID for storage stream
Arguments:
CLSID * pClassID : Returns class ID information
Return Value:
HRESULT
--*/
{
*pClassID = CLSID_InetMgr;
return S_OK;
}
STDMETHODIMP
CInetMgr::GetHelpTopic(LPOLESTR *lpCompiledHelpFile)
{
return ::GetHelpTopic(lpCompiledHelpFile);
}
STDMETHODIMP
CInetMgr::GetLinkedTopics(LPOLESTR *lpCompiledHelpFile)
{
return S_FALSE;
}
HRESULT
CInetMgr::IsDirty()
/*++
Routine Description:
Check to see if we need to write to the cache.
Arguments:
None
Return Value:
S_OK if dirty, S_FALSE if not
--*/
{
TRACEEOLID("CInetMgrComponent::IsDirty");
ASSERT_PTR(m_pNode);
if (IsExtension())
{
return FALSE;
}
else
{
return ((CIISRoot *)m_pNode)->m_scServers.IsDirty() ? S_OK : S_FALSE;
}
}
HRESULT
CInetMgr::InitNew(IStorage * pStg)
/*++
Routine Description:
Initialize new storage stream (newly created console file)
Arguments:
IStorage * pStg : Storage stream
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::InitNew");
//
// We could create the stream here, but it's just as easy to
// create it inside Save().
//
return S_OK;
}
HRESULT
CInetMgr::Load(IStorage * pStg)
/*++
Routine Description:
Load machine cache from the storage stream.
Arguments:
IStorage * pStg : Storage stream
Return Value:
HRESULT
--*/
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
TRACEEOLID("CInetMgrComponent::Load");
if (IsExtension())
{
return S_OK;
}
ASSERT_READ_WRITE_PTR(pStg);
DWORD cBytesRead;
DWORD dw;
HRESULT hr = S_OK;
CIISServerCache & cache = ((CIISRoot *)m_pNode)->m_scServers;
IStream * pStream = NULL;
ASSERT(cache.IsEmpty());
do
{
hr = pStg->OpenStream(
_szStream,
NULL,
STGM_READ | STGM_SHARE_EXCLUSIVE,
0L,
&pStream
);
if (FAILED(hr))
{
break;
}
//
// Read and verify the signature
//
hr = pStream->Read(&dw, sizeof(dw), &cBytesRead);
ASSERT(SUCCEEDED(hr) && cBytesRead == sizeof(dw));
if (FAILED(hr))
{
break;
}
if (dw != _dwSignature)
{
hr = HRESULT_FROM_WIN32(ERROR_FILE_CORRUPT);
break;
}
//
// Read number of machines in the cache
//
DWORD cMachines;
hr = pStream->Read(&cMachines, sizeof(cMachines), &cBytesRead);
ASSERT(SUCCEEDED(hr) && cBytesRead == sizeof(cMachines));
if (FAILED(hr))
{
break;
}
TRACEEOLID("Reading " << cMachines << " machines from cache");
CIISMachine * pMachine;
//
// Read each machine from the cache
//
for (dw = 0; dw < cMachines; ++dw)
{
hr = CIISMachine::ReadFromStream(pStream, &pMachine);
if (FAILED(hr))
{
break;
}
if (!cache.Add(pMachine))
{
delete pMachine;
}
}
}
while(FALSE);
if (pStream)
{
pStream->Release();
}
if (hr == STG_E_FILENOTFOUND)
{
//
// Stream was not initialized. This is acceptable.
//
hr = S_OK;
}
//
// Mark cache as clean
//
cache.SetDirty(FALSE);
return hr;
}
/* virtual */
HRESULT STDMETHODCALLTYPE
CInetMgr::Save(IStorage * pStgSave, BOOL fSameAsLoad)
/*++
Routine Description:
Save computer cache to to the storage stream.
Arguments:
IStorage * pStgSave : Storage stream
BOOL fSameAsLoad : TRUE if same as load
Return Value:
HRESULT
--*/
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
TRACEEOLID("CInetMgrComponent::Save");
if (IsExtension())
{
return S_OK;
}
//
// Write the computer names to the cache
//
ASSERT_READ_WRITE_PTR(pStgSave);
DWORD cBytesWritten;
HRESULT hr = STG_E_CANTSAVE;
IStream * pStream = NULL;
CIISServerCache & cache = ((CIISRoot *)m_pNode)->m_scServers;
do
{
hr = pStgSave->CreateStream(
_szStream,
STGM_CREATE | STGM_WRITE | STGM_SHARE_EXCLUSIVE,
0L,
0L,
&pStream
);
if (FAILED(hr))
{
break;
}
//
// Write the signature
//
hr = pStream->Write(&_dwSignature, sizeof(_dwSignature), &cBytesWritten);
ASSERT(SUCCEEDED(hr) && cBytesWritten == sizeof(_dwSignature));
if (FAILED(hr))
{
break;
}
//
// Write number of entries.
//
INT_PTR dw = cache.GetCount();
hr = pStream->Write(&dw, sizeof(dw), &cBytesWritten);
ASSERT(SUCCEEDED(hr) && cBytesWritten == sizeof(dw));
if (FAILED(hr))
{
break;
}
//
// Write each string -- but write them in reverse
// order to improve our sort performance when we load
// the cache.
//
CIISMachine * pMachine = cache.GetLast();
while(pMachine)
{
hr = pMachine->WriteToStream(pStream);
if (FAILED(hr))
{
break;
}
pMachine = cache.GetPrev();
}
}
while(FALSE);
if (pStream)
{
pStream->Release();
}
if (SUCCEEDED(hr))
{
//
// Mark cache as clean
//
cache.SetDirty(FALSE);
}
return hr;
}
/* virtual */
HRESULT
STDMETHODCALLTYPE
CInetMgr::SaveCompleted(IStorage * pStgNew)
/*++
Routine Description:
Save completed notification.
Arguments:
IStorage * pStgNew : Storage stream
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::SaveCompleted");
//
// Nothing to do
//
return S_OK;
}
/* virtual */
HRESULT
STDMETHODCALLTYPE
CInetMgr::HandsOffStorage()
/*++
Routine Description:
Hands off storage.
Arguments:
None
Return Value:
HRESULT
--*/
{
TRACEEOLID("CInetMgrComponent::HandsOffStorage");
//
// Nothing to do
//
return S_OK;
}
/* virtual */
HRESULT
CInetMgr::CompareObjects(LPDATAOBJECT lpDataObjectA, LPDATAOBJECT lpDataObjectB)
/*++
Routine Description:
Compare two data objects. This method is used by MMC to see if a property
sheet for the given data object is already open.
Arguments:
LPDATAOBJECT lpDataObjectA : A data object
LPDATAOBJECT lpDataObjectB : B data object
Return Value:
S_OK if they match, S_FALSE otherwise
--*/
{
HRESULT hr = E_POINTER;
do
{
if (!lpDataObjectA || !lpDataObjectB)
{
TRACEEOLID("IComponentData::CompareObjects called with NULL ptr");
break;
}
CSnapInItem * pItemA;
CSnapInItem * pItemB;
DATA_OBJECT_TYPES type;
hr = m_pComponentData->GetDataClass(lpDataObjectA, &pItemA, &type);
if (SUCCEEDED(hr))
{
hr = m_pComponentData->GetDataClass(lpDataObjectB, &pItemB, &type);
}
if (FAILED(hr))
{
break;
}
if (!pItemA || !pItemB)
{
hr = E_POINTER;
break;
}
if (pItemA == pItemB)
{
//
// Literally the same object
//
hr = S_OK;
break;
}
CIISObject * pObjectA = (CIISObject *)pItemA;
CIISObject * pObjectB = (CIISObject *)pItemB;
hr = !pObjectA->CompareScopeItem(pObjectB) ? S_OK : S_FALSE;
}
while(FALSE);
return hr;
}
HRESULT
CInetMgr::GetDataClass(
IDataObject * pDataObject,
CSnapInItem ** ppItem,
DATA_OBJECT_TYPES * pType)
{
if (ppItem == NULL)
return E_POINTER;
if (pType == NULL)
return E_POINTER;
*ppItem = NULL;
*pType = CCT_UNINITIALIZED;
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
FORMATETC formatetc = { CSnapInItem::m_CCF_NODETYPE,
NULL,
DVASPECT_CONTENT,
-1,
TYMED_HGLOBAL
};
stgmedium.hGlobal = GlobalAlloc(0, sizeof(GUID));
if (stgmedium.hGlobal == NULL)
return E_OUTOFMEMORY;
HRESULT hr = pDataObject->GetDataHere(&formatetc, &stgmedium);
if (FAILED(hr))
{
GlobalFree(stgmedium.hGlobal);
return hr;
}
GUID guid;
memcpy(&guid, stgmedium.hGlobal, sizeof(GUID));
GlobalFree(stgmedium.hGlobal);
hr = S_OK;
if (IsEqualGUID(guid, cCompMgmtService))
{
if (!IsExtension())
{
CIISRoot * pRootExt = new CIISRoot;
if (pRootExt == NULL)
{
return E_OUTOFMEMORY;
}
hr = pRootExt->InitAsExtension(pDataObject);
if (FAILED(hr))
{
return hr;
}
if (m_pNode != NULL)
{
delete m_pNode;
}
m_pNode = pRootExt;
}
*ppItem = m_pNode;
return hr;
}
return CSnapInItem::GetDataClass(pDataObject, ppItem, pType);
};
BOOL
CInetMgr::IsExtension()
{
ASSERT(m_pNode != NULL);
CIISRoot * pRoot = (CIISRoot *)m_pNode;
return pRoot->IsExtension();
}
//
// CInetMrgAbout Class
//
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
HRESULT
CInetMgrAbout::GetStringHelper(UINT nStringID, LPOLESTR * lpString)
/*++
Routine Description:
Get resource string helper function. Called from inline string fetcher
methods.
Arguments:
UINT nStringID : String ID from local resource segment
LPOLESTR * lpString : Returns the string
Return Value:
HRESULT
--*/
{
USES_CONVERSION;
TCHAR szBuf[256];
if (::LoadString(
_Module.GetResourceInstance(),
nStringID,
szBuf,
256) == 0)
{
return E_FAIL;
}
*lpString = (LPOLESTR)::CoTaskMemAlloc(
(lstrlen(szBuf) + 1) * sizeof(OLECHAR)
);
if (*lpString == NULL)
{
return E_OUTOFMEMORY;
}
::ocscpy(*lpString, T2OLE(szBuf));
return S_OK;
}
HRESULT
CInetMgrAbout::GetSnapinImage(HICON * hAppIcon)
/*++
Routine Description:
Get the icon for this snapin.
Arguments:
HICON * hAppIcon : Return handle to the icon
Return Value:
HRESULT
--*/
{
if (hAppIcon == NULL)
{
return E_POINTER;
}
m_hSnapinIcon = ::LoadIcon(
_Module.GetModuleInstance(),
MAKEINTRESOURCE(IDI_INETMGR)
);
*hAppIcon = m_hSnapinIcon;
ASSERT(*hAppIcon != NULL);
return (*hAppIcon != NULL) ? S_OK : E_FAIL;
}
HRESULT
CInetMgrAbout::GetStaticFolderImage(
HBITMAP * phSmallImage,
HBITMAP * phSmallImageOpen,
HBITMAP * phLargeImage,
COLORREF * prgbMask
)
/*++
Routine Description:
Get the static folder images.
Arguments:
HBITMAP * phSmallImage : Small folder
HBITMAP * phSmallImageOpen : Small open folder
HBITMAP * phLargeImage : Large image
COLORREF * prgbMask : Mask
Return Value:
HRESULT
--*/
{
if (!phSmallImage || !phSmallImageOpen || !phLargeImage || !prgbMask)
{
return HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER);
}
m_hSmallImage = (HBITMAP)::LoadImage(
_Module.GetModuleInstance(),
MAKEINTRESOURCE(IDB_SMALL_ROOT),
IMAGE_BITMAP,
0,
0,
LR_DEFAULTCOLOR
);
m_hLargeImage = (HBITMAP)::LoadImage(
_Module.GetModuleInstance(),
MAKEINTRESOURCE(IDB_LARGE_ROOT),
IMAGE_BITMAP,
0,
0,
LR_DEFAULTCOLOR
);
*phSmallImage = m_hSmallImage;
*phSmallImageOpen = m_hSmallImage;
*phLargeImage = m_hLargeImage;
*prgbMask = RGB_BK_IMAGES;
return *phSmallImage && *phLargeImage ? S_OK : E_FAIL;
}
CInetMgrAbout::~CInetMgrAbout()
{
if (m_hSmallImage != NULL)
{
::DeleteObject(m_hSmallImage);
}
if (m_hLargeImage != NULL)
{
::DeleteObject(m_hLargeImage);
}
if (m_hSnapinIcon != NULL)
{
::DestroyIcon(m_hSnapinIcon);
}
}
#if 0
HRESULT
ExtractComputerNameExt(IDataObject * pDataObject, CString& strComputer)
{
//
// Find the computer name from the ComputerManagement snapin
//
CLIPFORMAT CCF_MyComputMachineName = (CLIPFORMAT)RegisterClipboardFormat(MYCOMPUT_MACHINE_NAME);
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
FORMATETC formatetc = {
CCF_MyComputMachineName, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL
};
//
// Allocate memory for the stream
//
int len = MAX_PATH;
stgmedium.hGlobal = GlobalAlloc(GMEM_SHARE, len);
if(stgmedium.hGlobal == NULL)
return ERROR_NOT_ENOUGH_MEMORY;
HRESULT hr = pDataObject->GetDataHere(&formatetc, &stgmedium);
ASSERT(SUCCEEDED(hr));
//
// Get the computer name
//
strComputer = (LPTSTR)stgmedium.hGlobal;
GlobalFree(stgmedium.hGlobal);
return hr;
}
HRESULT
CCompMgrExtData::Init(IDataObject * pDataObject)
{
return ExtractComputerNameExt(pDataObject, m_ExtMachineName);
}
HRESULT
STDMETHODCALLTYPE
CCompMgrExtData::Notify(
MMC_NOTIFY_TYPE event,
LPARAM arg,
LPARAM param,
IComponentData* pComponentData,
IComponent* pComponent,
DATA_OBJECT_TYPES type)
{
CError err;
CComPtr<IConsole> pConsole;
CComQIPtr<IHeaderCtrl, &IID_IHeaderCtrl> pHeader;
CComQIPtr<IResultData, &IID_IResultData> pResultData;
if (pComponentData != NULL)
{
pConsole = ((CInetMgr *)pComponentData)->m_spConsole;
}
else
{
pConsole = ((CInetMgrComponent *)pComponent)->m_spConsole;
}
CComQIPtr<IConsoleNameSpace2, &IID_IConsoleNameSpace2> pScope = pConsole;
switch (event)
{
case MMCN_EXPAND:
err = EnumerateScopePane((HSCOPEITEM)param, pScope);
break;
default:
err = CSnapInItemImpl<CCompMgrExtData, TRUE>::Notify(event, arg, param, pComponentData, pComponent, type);
break;
}
return err;
}
HRESULT
CCompMgrExtData::EnumerateScopePane(HSCOPEITEM hParent, IConsoleNameSpace2 * pScope)
{
CError err;
ASSERT_PTR(pScope);
DWORD dwMask = SDI_PARENT;
SCOPEDATAITEM scopeDataItem;
::ZeroMemory(&scopeDataItem, sizeof(SCOPEDATAITEM));
scopeDataItem.mask =
SDI_STR | SDI_IMAGE | SDI_CHILDREN | SDI_OPENIMAGE | SDI_PARAM | dwMask;
scopeDataItem.displayname = MMC_CALLBACK;
scopeDataItem.nImage = scopeDataItem.nOpenImage = MMC_IMAGECALLBACK;//QueryImage();
scopeDataItem.lParam = (LPARAM)this;
scopeDataItem.relativeID = hParent;
scopeDataItem.cChildren = 1;
err = pScope->InsertItem(&scopeDataItem);
if (err.Succeeded())
{
//
// Cache the scope item handle
//
ASSERT(m_hScopeItem == NULL);
m_hScopeItem = scopeDataItem.ID;
// MMC_IMAGECALLBACK doesn't work in InsertItem. Update it here.
scopeDataItem.mask = SDI_IMAGE | SDI_OPENIMAGE;
err = pScope->SetItem(&scopeDataItem);
}
return err;
}
#endif