mirror of https://github.com/tongzx/nt5src
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.
2512 lines
76 KiB
2512 lines
76 KiB
// This is a part of the Microsoft Management Console.
|
|
// Copyright (C) 1995-1996 Microsoft Corporation
|
|
// All rights reserved.
|
|
//
|
|
// This source code is only intended as a supplement to the
|
|
// Microsoft Management Console and related
|
|
// electronic documentation provided with the interfaces.
|
|
|
|
|
|
#include "precomp.hxx"
|
|
#include "process.h"
|
|
|
|
#include <atlimpl.cpp>
|
|
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
long CSnapin::lDataObjectRefCount = 0;
|
|
|
|
#if 1 // BUGBUG - until this gets put in the regular build environment
|
|
extern const IID IID_IClassAdmin; /* = {0x00000191,0x0000,0x0000,{0xC0,0x00,0x00,0x00,0x00,0x00,0x00,0x46}}; */
|
|
#endif
|
|
|
|
extern const CLSID CLSID_Snapin = {0xbdc67e00,0x8ea5,0x11d0,{0x8d,0x3c,0x00,0xa0,0xc9,0x0d,0xca,0xe7}};
|
|
extern const wchar_t * szCLSID_Snapin = L"{bdc67e00-8ea5-11d0-8d3c-00a0c90dcae7}";
|
|
//const CLSID CLSID_Snapin = {0x18731372,0x1D79,0x11D0,{0xA2,0x9B,0x00,0xC0,0x4F,0xD9,0x09,0xDD}};
|
|
|
|
// Main NodeType GUID on numeric format
|
|
extern const GUID cNodeType = {0xf8b3a900,0x8ea5,0x11d0,{0x8d,0x3c,0x00,0xa0,0xc9,0x0d,0xca,0xe7}};
|
|
//const GUID cNodeType = {0x44092d22,0x1d7e,0x11D0,{0xA2,0x9B,0x00,0xC0,0x4F,0xD9,0x09,0xDD}};
|
|
|
|
// Main NodeType GUID on string format
|
|
extern const wchar_t* cszNodeType = L"{f8b3a900-8ea5-11d0-8d3c-00a0c90dcae7}";
|
|
//const wchar_t* cszNodeType = L"{44092d22-1d7e-11d0-a29b-00c04fd909dd}";
|
|
|
|
// Internal private format
|
|
const wchar_t* SNAPIN_INTERNAL = L"APPMGR_INTERNAL";
|
|
|
|
#define ARRAYLEN(x) (sizeof(x) / sizeof((x)[0]))
|
|
|
|
static MMCBUTTON SnapinButtons[] =
|
|
{
|
|
{ 0, 1, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Folder"), _T("New Folder") },
|
|
{ 1, 2, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Inbox"), _T("Mail Inbox")},
|
|
{ 2, 3, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Outbox"), _T("Mail Outbox") },
|
|
{ 3, 4, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Send"), _T("Send Message") },
|
|
{ 0, 0, TBSTATE_ENABLED, TBSTYLE_SEP, _T(" "), _T("") },
|
|
{ 4, 5, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Trash"), _T("Trash") },
|
|
{ 5, 6, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Open"), _T("Open Folder")},
|
|
{ 6, 7, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("News"), _T("Today's News") },
|
|
{ 7, 8, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("INews"), _T("Internet News") },
|
|
|
|
};
|
|
|
|
static MMCBUTTON SnapinButtons2[] =
|
|
{
|
|
{ 0, 10, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Compose"), _T("Compose Message") },
|
|
{ 1, 20, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Print"), _T("Print Message") },
|
|
{ 2, 30, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Find"), _T("Find Message") },
|
|
{ 0, 0, TBSTATE_ENABLED, TBSTYLE_SEP, _T(" "), _T("") },
|
|
{ 3, 40, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Inbox"), _T("Inbox") },
|
|
{ 4, 50, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Smile"), _T("Smile :-)") },
|
|
{ 5, 60, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Reply"), _T("Reply") },
|
|
{ 0, 0, TBSTATE_ENABLED, TBSTYLE_SEP , _T(" "), _T("") },
|
|
{ 6, 70, TBSTATE_ENABLED, TBSTYLE_BUTTON, _T("Reply All"), _T("Reply All") },
|
|
|
|
};
|
|
|
|
// Utility function to delete an registry key and all of it's children
|
|
LONG RegDeleteTree(HKEY hKey, LPCTSTR lpSubKey)
|
|
{
|
|
HKEY hKeyNew;
|
|
LONG lResult = RegOpenKey(hKey, lpSubKey, &hKeyNew);
|
|
if (lResult != ERROR_SUCCESS)
|
|
{
|
|
return lResult;
|
|
}
|
|
TCHAR szName[256];
|
|
while (ERROR_SUCCESS == RegEnumKey(hKeyNew, 0, szName, 256))
|
|
{
|
|
RegDeleteTree(hKeyNew, szName);
|
|
}
|
|
RegCloseKey(hKeyNew);
|
|
return RegDeleteKey(hKey, lpSubKey);
|
|
}
|
|
|
|
|
|
INTERNAL* ExtractInternalFormat(LPDATAOBJECT lpDataObject)
|
|
{
|
|
INTERNAL* internal = NULL;
|
|
|
|
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
|
|
FORMATETC formatetc = { CDataObject::m_cfInternal, NULL,
|
|
DVASPECT_CONTENT, -1, TYMED_HGLOBAL
|
|
};
|
|
|
|
if (!lpDataObject)
|
|
return NULL;
|
|
|
|
|
|
// Allocate memory for the stream
|
|
stgmedium.hGlobal = GlobalAlloc(GMEM_SHARE, sizeof(INTERNAL));
|
|
|
|
// Attempt to get data from the object
|
|
do
|
|
{
|
|
if (stgmedium.hGlobal == NULL)
|
|
break;
|
|
|
|
if (FAILED(lpDataObject->GetDataHere(&formatetc, &stgmedium)))
|
|
break;
|
|
|
|
internal = reinterpret_cast<INTERNAL*>(stgmedium.hGlobal);
|
|
|
|
if (internal == NULL)
|
|
break;
|
|
|
|
} while (FALSE);
|
|
|
|
return internal;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// Return TRUE if we are enumerating our main folder
|
|
|
|
BOOL CSnapin::IsEnumerating(LPDATAOBJECT lpDataObject)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
|
|
STGMEDIUM stgmedium = { TYMED_HGLOBAL, NULL };
|
|
FORMATETC formatetc = { CDataObject::m_cfNodeType, NULL,
|
|
DVASPECT_CONTENT, -1, TYMED_HGLOBAL
|
|
};
|
|
|
|
// Allocate memory for the stream
|
|
stgmedium.hGlobal = GlobalAlloc(GMEM_SHARE, sizeof(GUID));
|
|
|
|
// Attempt to get data from the object
|
|
do
|
|
{
|
|
if (stgmedium.hGlobal == NULL)
|
|
break;
|
|
|
|
if (FAILED(lpDataObject->GetDataHere(&formatetc, &stgmedium)))
|
|
break;
|
|
|
|
GUID* nodeType = reinterpret_cast<GUID*>(stgmedium.hGlobal);
|
|
|
|
if (nodeType == NULL)
|
|
break;
|
|
|
|
// Is this my main node (static folder node type)
|
|
if (*nodeType == cNodeType)
|
|
bResult = TRUE;
|
|
|
|
} while (FALSE);
|
|
|
|
|
|
// Free resources
|
|
if (stgmedium.hGlobal != NULL)
|
|
GlobalFree(stgmedium.hGlobal);
|
|
|
|
return bResult;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CSnapin's IComponent implementation
|
|
|
|
STDMETHODIMP CSnapin::GetResultViewType(long cookie, BSTR* ppViewType, LONG * pViewOptions)
|
|
{
|
|
// Use default view
|
|
return S_FALSE;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::Initialize(LPCONSOLE lpConsole)
|
|
{
|
|
ASSERT(lpConsole != NULL);
|
|
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
// Save the IConsole pointer
|
|
m_pConsole = lpConsole;
|
|
m_pConsole->AddRef();
|
|
|
|
// Load resource strings
|
|
LoadResources();
|
|
|
|
// QI for a IHeaderCtrl
|
|
HRESULT hr = m_pConsole->QueryInterface(IID_IHeaderCtrl,
|
|
reinterpret_cast<void**>(&m_pHeader));
|
|
|
|
hr = m_pConsole->QueryInterface(IID_IPropertySheetProvider,
|
|
(void **)&m_pIPropertySheetProvider);
|
|
|
|
// Give the console the header control interface pointer
|
|
if (SUCCEEDED(hr))
|
|
m_pConsole->SetHeader(m_pHeader);
|
|
|
|
m_pConsole->QueryInterface(IID_IResultData,
|
|
reinterpret_cast<void**>(&m_pResult));
|
|
|
|
hr = m_pConsole->QueryResultImageList(&m_pImageResult);
|
|
ASSERT(hr == S_OK);
|
|
|
|
hr = m_pConsole->QueryConsoleVerb(&m_pConsoleVerb);
|
|
ASSERT(hr == S_OK);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::Notify(LPDATAOBJECT lpDataObject, MMC_NOTIFY_TYPE event, long arg, long param)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
long cookie;
|
|
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
|
|
if (event == MMCN_PROPERTY_CHANGE)
|
|
{
|
|
hr = OnPropertyChange(param);
|
|
}
|
|
else if (event == MMCN_VIEW_CHANGE)
|
|
{
|
|
hr = OnUpdateView(lpDataObject);
|
|
}
|
|
else
|
|
{
|
|
INTERNAL* pInternal = ExtractInternalFormat(lpDataObject);
|
|
|
|
if (pInternal == NULL)
|
|
{
|
|
cookie = 0;
|
|
}
|
|
else
|
|
{
|
|
cookie = pInternal->m_cookie;
|
|
}
|
|
|
|
|
|
switch(event)
|
|
{
|
|
case MMCN_ACTIVATE:
|
|
hr = OnActivate(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_CLICK:
|
|
hr = OnResultItemClkOrDblClk(cookie, FALSE);
|
|
break;
|
|
|
|
case MMCN_DBLCLICK:
|
|
if (pInternal->m_type == CCT_RESULT)
|
|
hr = OnResultItemClkOrDblClk(cookie, TRUE);
|
|
else
|
|
hr = S_FALSE;
|
|
break;
|
|
|
|
case MMCN_ADD_IMAGES:
|
|
hr = OnAddImages(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_SHOW:
|
|
hr = OnShow(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_MINIMIZED:
|
|
hr = OnMinimize(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_SELECT:
|
|
hr = OnSelect(pInternal->m_type, cookie, arg, param);
|
|
break;
|
|
|
|
// Note - Future expansion of notify types possible
|
|
default:
|
|
ASSERT(FALSE); // Handle new messages
|
|
hr = E_UNEXPECTED;
|
|
break;
|
|
}
|
|
|
|
FREE_INTERNAL(pInternal);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::Destroy(long cookie)
|
|
{
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
// Release the interfaces that we QI'ed
|
|
if (m_pConsole != NULL)
|
|
{
|
|
// Tell the console to release the header control interface
|
|
m_pConsole->SetHeader(NULL);
|
|
SAFE_RELEASE(m_pHeader);
|
|
|
|
SAFE_RELEASE(m_pResult);
|
|
SAFE_RELEASE(m_pImageResult);
|
|
SAFE_RELEASE(m_pConsoleVerb);
|
|
|
|
// Release the IConsole interface last
|
|
SAFE_RELEASE(m_pConsole);
|
|
if (m_pComponentData)
|
|
{
|
|
((IComponentData*)m_pComponentData)->Release(); // QI'ed in IComponentDataImpl::CreateComponent
|
|
}
|
|
SAFE_RELEASE(m_pIAppManagerActions); // ditto
|
|
|
|
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::QueryDataObject(long cookie, DATA_OBJECT_TYPES type,
|
|
LPDATAOBJECT* ppDataObject)
|
|
{
|
|
// Delegate it to the IComponentData
|
|
ASSERT(m_pComponentData != NULL);
|
|
return m_pComponentData->QueryDataObject(cookie, type, ppDataObject);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CSnapin's implementation specific members
|
|
|
|
DEBUG_DECLARE_INSTANCE_COUNTER(CSnapin);
|
|
|
|
CSnapin::CSnapin()
|
|
{
|
|
DEBUG_INCREMENT_INSTANCE_COUNTER(CSnapin);
|
|
CSnapin::lDataObjectRefCount = 0;
|
|
m_lViewMode = LVS_REPORT;
|
|
Construct();
|
|
}
|
|
|
|
CSnapin::~CSnapin()
|
|
{
|
|
#if DBG==1
|
|
ASSERT(dbg_cRef == 0);
|
|
#endif
|
|
|
|
DEBUG_DECREMENT_INSTANCE_COUNTER(CSnapin);
|
|
|
|
// SAFE_RELEASE(m_pToolbar1);
|
|
// SAFE_RELEASE(m_pToolbar2);
|
|
|
|
// SAFE_RELEASE(m_pControlbar);
|
|
|
|
// Make sure the interfaces have been released
|
|
ASSERT(m_pConsole == NULL);
|
|
ASSERT(m_pHeader == NULL);
|
|
// ASSERT(m_pToolbar1 == NULL);
|
|
// ASSERT(m_pToolbar2 == NULL);
|
|
|
|
|
|
// delete m_pbmpToolbar1;
|
|
// delete m_pbmpToolbar2;
|
|
|
|
Construct();
|
|
|
|
ASSERT(CSnapin::lDataObjectRefCount == 0);
|
|
|
|
}
|
|
|
|
void CSnapin::Construct()
|
|
{
|
|
#if DBG==1
|
|
dbg_cRef = 0;
|
|
#endif
|
|
|
|
m_pConsole = NULL;
|
|
m_pHeader = NULL;
|
|
|
|
m_pResult = NULL;
|
|
m_pImageResult = NULL;
|
|
m_pComponentData = NULL;
|
|
m_pIAppManagerActions = NULL;
|
|
// m_pToolbar1 = NULL;
|
|
// m_pToolbar2 = NULL;
|
|
// m_pControlbar = NULL;
|
|
|
|
// m_pbmpToolbar1 = NULL;
|
|
// m_pbmpToolbar2 = NULL;
|
|
}
|
|
|
|
// Array of menu item commands to be inserted into the context menu.
|
|
// Note - the first item is the menu text,
|
|
// the second item is the status string
|
|
|
|
CONTEXTMENUITEM menuItems[] =
|
|
{
|
|
{
|
|
L"", L"",
|
|
0, CCM_INSERTIONPOINTID_PRIMARY_TOP, MFT_SEPARATOR, CCM_SPECIAL_SEPARATOR
|
|
},
|
|
{
|
|
L"", L"",
|
|
IDM_ADD_APP, CCM_INSERTIONPOINTID_PRIMARY_TOP, 0, 0
|
|
},
|
|
{
|
|
L"", L"",
|
|
IDM_UPDATE_APP, CCM_INSERTIONPOINTID_PRIMARY_TOP, 0, 0
|
|
},
|
|
{
|
|
L"", L"",
|
|
IDM_DEL_APP, CCM_INSERTIONPOINTID_PRIMARY_TOP, 0, 0
|
|
},
|
|
{
|
|
L"", L"",
|
|
IDM_ADD_FROM_IE, CCM_INSERTIONPOINTID_PRIMARY_TOP, 0, 0
|
|
},
|
|
{
|
|
L"", L"",
|
|
IDM_REFRESH, CCM_INSERTIONPOINTID_PRIMARY_TOP, 0, 0
|
|
}
|
|
};
|
|
|
|
|
|
CString szExtension;
|
|
CString szFilter;
|
|
|
|
void CSnapin::LoadResources()
|
|
{
|
|
// Load strings from resources
|
|
|
|
m_column1.LoadString(IDS_NAME);
|
|
m_column2.LoadString(IDS_TYPE);
|
|
m_column3.LoadString(IDS_SIZE);
|
|
m_column4.LoadString(IDS_LOC);
|
|
m_column5.LoadString(IDS_MACH);
|
|
m_column6.LoadString(IDS_DESC);
|
|
m_column7.LoadString(IDS_PATH);
|
|
m_szAddApp.LoadString(IDM_ADD_APP);
|
|
m_szAddAppDesc.LoadString(IDS_ADD_APP_DESC);
|
|
m_szDelApp.LoadString(IDM_DEL_APP);
|
|
m_szDelAppDesc.LoadString(IDS_DEL_APP_DESC);
|
|
m_szUpdateApp.LoadString(IDM_UPDATE_APP);
|
|
m_szUpdateAppDesc.LoadString(IDS_UPDATE_APP_DESC);
|
|
m_szRefresh.LoadString(IDM_REFRESH);
|
|
m_szRefreshDesc.LoadString(IDS_REFRESH_DESC);
|
|
m_szAddFromIe.LoadString(IDM_ADD_FROM_IE);
|
|
m_szAddFromIeDesc.LoadString(IDS_ADD_FROM_IE_DESC);
|
|
menuItems[1].strName = (LPWSTR)((LPCOLESTR)m_szAddApp);
|
|
menuItems[1].strStatusBarText = (LPWSTR)((LPCOLESTR)m_szAddAppDesc);
|
|
menuItems[2].strName = (LPWSTR)((LPCOLESTR)m_szUpdateApp);
|
|
menuItems[2].strStatusBarText = (LPWSTR)((LPCOLESTR)m_szUpdateAppDesc);
|
|
menuItems[3].strName = (LPWSTR)((LPCOLESTR)m_szDelApp);
|
|
menuItems[3].strStatusBarText = (LPWSTR)((LPCOLESTR)m_szDelAppDesc);
|
|
menuItems[4].strName = (LPWSTR)((LPCOLESTR)m_szAddFromIe);
|
|
menuItems[4].strStatusBarText = (LPWSTR)((LPCOLESTR)m_szAddFromIeDesc);
|
|
menuItems[5].strName = (LPWSTR)((LPCOLESTR)m_szRefresh);
|
|
menuItems[5].strStatusBarText = (LPWSTR)((LPCOLESTR)m_szRefreshDesc);
|
|
szExtension.LoadString(IDS_DEF_EXTENSION);
|
|
szFilter.LoadString(IDS_EXTENSION_FILTER);
|
|
m_szFolderTitle.LoadString(IDS_FOLDER_TITLE);
|
|
}
|
|
|
|
HRESULT CSnapin::InitializeHeaders(long cookie)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
ASSERT(m_pHeader);
|
|
|
|
// Put the correct headers depending on the cookie
|
|
// Note - cookie ignored for this sample
|
|
m_pHeader->InsertColumn(0, m_column1, LVCFMT_LEFT, 100); // Name
|
|
m_pHeader->InsertColumn(1, m_column2, LVCFMT_LEFT, 75); // Type
|
|
// m_pHeader->InsertColumn(2, m_column3, LVCFMT_RIGHT, 50); // Size
|
|
m_pHeader->InsertColumn(2, m_column4, LVCFMT_RIGHT, 100); // localle
|
|
m_pHeader->InsertColumn(3, m_column5, LVCFMT_LEFT, 75); // machine
|
|
m_pHeader->InsertColumn(4, m_column6, LVCFMT_LEFT, 75); // description
|
|
m_pHeader->InsertColumn(5, m_column7, LVCFMT_LEFT, 150); // path
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CSnapin::InitializeBitmaps(long cookie)
|
|
{
|
|
ASSERT(m_pImageResult != NULL);
|
|
|
|
CBitmap bmp16x16;
|
|
CBitmap bmp32x32;
|
|
|
|
// Load the bitmaps from the dll
|
|
bmp16x16.LoadBitmap(IDB_16x16);
|
|
bmp32x32.LoadBitmap(IDB_32x32);
|
|
|
|
// Set the images
|
|
m_pImageResult->ImageListSetStrip(reinterpret_cast<long*>(static_cast<HBITMAP>(bmp16x16)),
|
|
reinterpret_cast<long*>(static_cast<HBITMAP>(bmp32x32)),
|
|
0, RGB(255,0,255));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::GetDisplayInfo(LPRESULTDATAITEM pResult)
|
|
{
|
|
ASSERT(pResult != NULL);
|
|
if (pResult)
|
|
{
|
|
if (pResult->lParam == -1)
|
|
{
|
|
switch (pResult->nCol)
|
|
{
|
|
case 0:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)m_szFolderTitle);
|
|
break;
|
|
default:
|
|
pResult->str = (BSTR)_T("");
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
std::map<long, APP_DATA>::iterator i = m_pComponentData->m_AppData.find(pResult->lParam);
|
|
if (i != m_pComponentData->m_AppData.end())
|
|
{
|
|
APP_DATA & data = i->second;
|
|
switch (pResult->nCol)
|
|
{
|
|
case 0:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szName);
|
|
break;
|
|
case 1:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szType);
|
|
break;
|
|
case 2:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szLoc);
|
|
break;
|
|
case 3:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szMach);
|
|
break;
|
|
case 4:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szDesc);
|
|
break;
|
|
case 5:
|
|
pResult->str = (unsigned short *)((LPCOLESTR)data.szIconPath);
|
|
break;
|
|
}
|
|
if (pResult->str == NULL)
|
|
pResult->str = (BSTR)_T("");
|
|
}
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// IExtendContextMenu Implementation
|
|
|
|
STDMETHODIMP CSnapin::AddMenuItems(LPDATAOBJECT pDataObject,
|
|
LPCONTEXTMENUCALLBACK pContextMenuCallback, LONG * pInsertionAllowed)
|
|
{
|
|
return m_pComponentData->
|
|
AddMenuItems(pDataObject, pContextMenuCallback, pInsertionAllowed);
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::Command(long nCommandID, LPDATAOBJECT pDataObject)
|
|
{
|
|
return m_pComponentData->
|
|
Command(nCommandID, pDataObject);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// IComponentData implementation
|
|
|
|
DEBUG_DECLARE_INSTANCE_COUNTER(CComponentDataImpl);
|
|
|
|
CComponentDataImpl::CComponentDataImpl()
|
|
: m_bIsDirty(TRUE)
|
|
{
|
|
HKEY hKey;
|
|
DWORD dwDisp;
|
|
|
|
DEBUG_INCREMENT_INSTANCE_COUNTER(CComponentDataImpl);
|
|
|
|
m_pScope = NULL;
|
|
m_pConsole = NULL;
|
|
m_pIClassAdmin = NULL;
|
|
m_fLoaded = FALSE;
|
|
m_fExtension = FALSE;
|
|
m_pIGPTInformation = NULL;
|
|
m_lLastAllocated = 0;
|
|
|
|
//
|
|
// This creates the magic "GPTSupport" key in HKCR so that Darwin
|
|
// generates full link files.
|
|
//
|
|
|
|
if (RegCreateKeyEx (HKEY_CLASSES_ROOT, TEXT("GPTSupport"), 0, NULL,
|
|
REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey,
|
|
&dwDisp) == ERROR_SUCCESS)
|
|
{
|
|
RegCloseKey (hKey);
|
|
}
|
|
}
|
|
|
|
CComponentDataImpl::~CComponentDataImpl()
|
|
{
|
|
|
|
DEBUG_DECREMENT_INSTANCE_COUNTER(CComponentDataImpl);
|
|
|
|
ASSERT(m_pScope == NULL);
|
|
ASSERT(CSnapin::lDataObjectRefCount == 0);
|
|
}
|
|
#include <msi.h>
|
|
|
|
STDMETHODIMP CComponentDataImpl::Initialize(LPUNKNOWN pUnknown)
|
|
{
|
|
ASSERT(pUnknown != NULL);
|
|
HRESULT hr;
|
|
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
// MMC should only call ::Initialize once!
|
|
ASSERT(m_pScope == NULL);
|
|
pUnknown->QueryInterface(IID_IConsoleNameSpace,
|
|
reinterpret_cast<void**>(&m_pScope));
|
|
ASSERT(hr == S_OK);
|
|
|
|
hr = pUnknown->QueryInterface(IID_IConsole, reinterpret_cast<void**>(&m_pConsole));
|
|
ASSERT(hr == S_OK);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CSnapin::OnAddImages(long cookie, long arg, long param)
|
|
{
|
|
if (arg == 0)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// add the images for the scope tree
|
|
CBitmap bmp16x16;
|
|
CBitmap bmp32x32;
|
|
LPIMAGELIST lpScopeImage = (LPIMAGELIST)arg;
|
|
|
|
// Load the bitmaps from the dll
|
|
bmp16x16.LoadBitmap(IDB_16x16);
|
|
bmp32x32.LoadBitmap(IDB_32x32);
|
|
|
|
// Set the images
|
|
lpScopeImage->ImageListSetStrip(reinterpret_cast<long*>(static_cast<HBITMAP>(bmp16x16)),
|
|
reinterpret_cast<long*>(static_cast<HBITMAP>(bmp32x32)),
|
|
0, RGB(255,0,255));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::CreateComponent(LPCOMPONENT* ppComponent)
|
|
{
|
|
ASSERT(ppComponent != NULL);
|
|
|
|
CComObject<CSnapin>* pObject;
|
|
CComObject<CSnapin>::CreateInstance(&pObject);
|
|
ASSERT(pObject != NULL);
|
|
|
|
m_pSnapin = pObject;
|
|
|
|
|
|
// Store IComponentData
|
|
pObject->SetIComponentData(this);
|
|
|
|
return pObject->QueryInterface(IID_IComponent,
|
|
reinterpret_cast<void**>(ppComponent));
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::Notify(LPDATAOBJECT lpDataObject, MMC_NOTIFY_TYPE event, long arg, long param)
|
|
{
|
|
ASSERT(m_pScope != NULL);
|
|
HRESULT hr;
|
|
|
|
// Since it's my folder it has an internal format.
|
|
// Design Note: for extension. I can use the fact, that the data object doesn't have
|
|
// my internal format and I should look at the node type and see how to extend it.
|
|
if (event == MMCN_PROPERTY_CHANGE)
|
|
{
|
|
hr = OnProperties(param);
|
|
}
|
|
else
|
|
{
|
|
INTERNAL* pInternal = ExtractInternalFormat(lpDataObject);
|
|
long cookie = 0;
|
|
if (pInternal != NULL)
|
|
{
|
|
cookie = pInternal->m_cookie;
|
|
FREE_INTERNAL(pInternal);
|
|
}
|
|
else
|
|
{
|
|
// only way we could not be able to extract our own format is if we're operating as an extension
|
|
m_fExtension = TRUE;
|
|
}
|
|
|
|
if (m_pIGPTInformation == NULL)
|
|
{
|
|
hr = lpDataObject->QueryInterface(IID_IGPTInformation,
|
|
reinterpret_cast<void**>(&m_pIGPTInformation));
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
WCHAR szBuffer[MAX_PATH];
|
|
do
|
|
{
|
|
hr = m_pIGPTInformation->GetCSPath(szBuffer, MAX_PATH);
|
|
if (FAILED(hr)) break;
|
|
m_szLDAP_Path = "ADCS:";
|
|
m_szLDAP_Path += szBuffer;
|
|
|
|
hr = m_pIGPTInformation->GetGPTPath(GPT_SECTION_USER, szBuffer, MAX_PATH);
|
|
if (FAILED(hr)) break;
|
|
m_szGPT_Path = szBuffer;
|
|
m_szGPT_Path += L"\\Applications";
|
|
|
|
if (SUCCEEDED(CreateApplicationDirectories()))
|
|
{
|
|
m_fLoaded = TRUE;
|
|
}
|
|
} while (0);
|
|
}
|
|
}
|
|
|
|
|
|
switch(event)
|
|
{
|
|
// case MMCN_ADD:
|
|
// hr = OnAdd(cookie, arg, param);
|
|
// break;
|
|
|
|
case MMCN_DELETE:
|
|
hr = OnDelete(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_RENAME:
|
|
hr = OnRename(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_EXPAND:
|
|
{
|
|
hr = OnExpand(cookie, arg, param);
|
|
}
|
|
break;
|
|
|
|
case MMCN_SELECT:
|
|
hr = OnSelect(cookie, arg, param);
|
|
break;
|
|
|
|
case MMCN_CONTEXTMENU:
|
|
hr = OnContextMenu(cookie, arg, param);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::Destroy()
|
|
{
|
|
// Delete enumerated scope items
|
|
DeleteList();
|
|
|
|
SAFE_RELEASE(m_pScope);
|
|
SAFE_RELEASE(m_pConsole);
|
|
SAFE_RELEASE(m_pIClassAdmin);
|
|
SAFE_RELEASE(m_pIGPTInformation);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::QueryDataObject(long cookie, DATA_OBJECT_TYPES type, LPDATAOBJECT* ppDataObject)
|
|
{
|
|
ASSERT(ppDataObject != NULL);
|
|
|
|
CComObject<CDataObject>* pObject;
|
|
|
|
CComObject<CDataObject>::CreateInstance(&pObject);
|
|
ASSERT(pObject != NULL);
|
|
|
|
// Save cookie and type for delayed rendering
|
|
pObject->SetType(type);
|
|
pObject->SetCookie(cookie);
|
|
|
|
return pObject->QueryInterface(IID_IDataObject,
|
|
reinterpret_cast<void**>(ppDataObject));
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//// IPersistStreamInit interface members
|
|
|
|
STDMETHODIMP CComponentDataImpl::GetClassID(CLSID *pClassID)
|
|
{
|
|
ASSERT(pClassID != NULL);
|
|
|
|
// Copy the CLSID for this snapin
|
|
*pClassID = CLSID_Snapin;
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::IsDirty()
|
|
{
|
|
// Always save / Always dirty.
|
|
return ThisIsDirty() ? S_OK : S_FALSE;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::Load(IStream *pStm)
|
|
{
|
|
ASSERT(pStm);
|
|
|
|
// Read the string
|
|
TCHAR psz[MAX_PATH]; // BUGBUG - really should be WCHAR to avoid problems in case
|
|
// it's ever compiled for MBCS
|
|
ULONG nBytesRead;
|
|
ULONG cb;
|
|
HRESULT hr = pStm->Read(&cb, sizeof(ULONG), &nBytesRead);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pStm->Read(psz, cb, &nBytesRead);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (cb > MAX_PATH * sizeof(TCHAR))
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
m_szLDAP_Path = psz;
|
|
|
|
hr = pStm->Read(&cb, sizeof(ULONG), &nBytesRead);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (cb > MAX_PATH * sizeof(TCHAR))
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
hr = pStm->Read(psz, cb, &nBytesRead);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
m_szGPT_Path = psz;
|
|
m_fLoaded = TRUE;
|
|
ClearDirty();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return SUCCEEDED(hr) ? S_OK : E_FAIL;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::Save(IStream *pStm, BOOL fClearDirty)
|
|
{
|
|
ASSERT(pStm);
|
|
|
|
// Write the string
|
|
ULONG nBytesWritten;
|
|
ULONG cb = (m_szLDAP_Path.GetLength() + 1) * sizeof(TCHAR);
|
|
HRESULT hr = pStm->Write(&cb, sizeof(ULONG), &nBytesWritten);
|
|
if (FAILED(hr))
|
|
return STG_E_CANTSAVE;
|
|
hr = pStm->Write(m_szLDAP_Path, cb, &nBytesWritten);
|
|
if (FAILED(hr))
|
|
return STG_E_CANTSAVE;
|
|
|
|
cb = (m_szGPT_Path.GetLength() + 1) * sizeof(TCHAR);
|
|
hr = pStm->Write(&cb, sizeof(ULONG), &nBytesWritten);
|
|
if (FAILED(hr))
|
|
return STG_E_CANTSAVE;
|
|
hr = pStm->Write(m_szGPT_Path, cb, &nBytesWritten);
|
|
|
|
if (FAILED(hr))
|
|
return STG_E_CANTSAVE;
|
|
|
|
if (fClearDirty)
|
|
ClearDirty();
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::GetSizeMax(ULARGE_INTEGER *pcbSize)
|
|
{
|
|
ASSERT(pcbSize);
|
|
|
|
ULONG cb = (m_szLDAP_Path.GetLength() + m_szGPT_Path.GetLength() + 2) * sizeof(TCHAR) + 2 * sizeof(ULONG);
|
|
// Set the size of the string to be saved
|
|
ULISet32(*pcbSize, cb);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::InitNew(void)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
// IAppManagerActions methods
|
|
|
|
STDMETHODIMP CComponentDataImpl::CanPackageBeAssigned(ULONG cookie)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
std::map<long, APP_DATA>::iterator i = m_AppData.find(cookie);
|
|
if (i != m_AppData.end())
|
|
{
|
|
APP_DATA & data = i->second;
|
|
// If it is already assigned or if the path points to the GPT then it can
|
|
// be assigned.
|
|
if (data.pDetails->dwActFlags & ACTFLG_Assigned)
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
CString szTemp = data.szPath;
|
|
szTemp.MakeLower();
|
|
int i = szTemp.Find(_T("\\published\\"));
|
|
if (i < 0)
|
|
{
|
|
i = szTemp.Find(_T("\\assigned\\")); // cover all the bases
|
|
}
|
|
if (i >= 0)
|
|
{
|
|
// finally make sure it's got an .aas extension
|
|
if (szTemp.Right(4) == _T(".aas"))
|
|
{
|
|
DWORD dwAttributes = GetFileAttributes(data.szPath);
|
|
if ((dwAttributes != 0xffffffff) && ((dwAttributes & FILE_ATTRIBUTE_DIRECTORY) == 0))
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::NotifyClients(BOOL f)
|
|
{
|
|
// Notify clients of change
|
|
if (m_pIGPTInformation)
|
|
{
|
|
m_pIGPTInformation->NotifyClients(f);
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::MovePackageToAssigned(ULONG cookie)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
// first validate that we've got a script file that can be moved
|
|
if (SUCCEEDED(CanPackageBeAssigned(cookie)))
|
|
{
|
|
APP_DATA & data = m_AppData[cookie];
|
|
// don't need to validate the cookie because CanPackageBeAssigned does it.
|
|
|
|
// need to build the destination path
|
|
CString szTemp = data.szPath;
|
|
szTemp.MakeLower();
|
|
int iSplitpoint = szTemp.Find(_T("published"));
|
|
if (iSplitpoint >= 0)
|
|
{
|
|
CString szDestination = data.szPath.Left(iSplitpoint);
|
|
szDestination += _T("assigned");
|
|
szDestination += data.szPath.Mid(iSplitpoint + strlen("published"));
|
|
|
|
// move the script file
|
|
if (!MoveFileEx(data.szPath, szDestination, MOVEFILE_COPY_ALLOWED|MOVEFILE_WRITE_THROUGH))
|
|
return (hr);
|
|
|
|
// update the path in the data packet
|
|
data.szPath = szDestination;
|
|
data.pDetails->pszPath = (LPOLESTR)(LPCOLESTR) data.szPath;
|
|
|
|
}
|
|
else
|
|
{
|
|
if (szTemp.Find(_T("assigned")) >= 0)
|
|
{
|
|
hr = S_OK; // already in the assigned directory
|
|
}
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::MovePackageToPublished(ULONG cookie)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
// first validate that we've got a script file that can be moved
|
|
if (SUCCEEDED(CanPackageBeAssigned(cookie)))
|
|
{
|
|
APP_DATA & data = m_AppData[cookie];
|
|
// don't need to validate pData because CanPackageBeAssigned does it.
|
|
|
|
// need to build the destination path
|
|
CString szTemp = data.szPath;
|
|
szTemp.MakeLower();
|
|
int iSplitpoint = szTemp.Find(_T("assigned"));
|
|
if (iSplitpoint >= 0)
|
|
{
|
|
CString szDestination = data.szPath.Left(iSplitpoint);
|
|
szDestination += _T("published");
|
|
szDestination += data.szPath.Mid(iSplitpoint + strlen("assigned"));
|
|
|
|
// move the script file
|
|
if (!MoveFileEx(data.szPath, szDestination, MOVEFILE_COPY_ALLOWED|MOVEFILE_WRITE_THROUGH))
|
|
return(hr);
|
|
|
|
// update the path in the data packet
|
|
data.szPath = szDestination;
|
|
data.pDetails->pszPath = (LPOLESTR)(LPCOLESTR) data.szPath;
|
|
|
|
// Notify clients of change
|
|
if (m_pIGPTInformation)
|
|
{
|
|
m_pIGPTInformation->NotifyClients(FALSE);
|
|
}
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
if (szTemp.Find(_T("published")) >= 0)
|
|
{
|
|
hr = S_OK; // already in the published directory
|
|
}
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::ReloadPackageData(ULONG cookie)
|
|
{
|
|
// put up an hourglass (this could take a while)
|
|
CHourglass hourglass;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::CreateApplicationDirectories(VOID)
|
|
{
|
|
TCHAR szDir[MAX_PATH];
|
|
LPTSTR lpEnd;
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
lstrcpy (szDir, m_szGPT_Path);
|
|
lpEnd = szDir + lstrlen(szDir);
|
|
|
|
do
|
|
{
|
|
#if 0
|
|
lstrcpy (lpEnd, TEXT("\\Assigned\\x86\\WinNT"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
|
|
lstrcpy (lpEnd, TEXT("\\Assigned\\x86\\Win95"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
#else
|
|
lstrcpy (lpEnd, TEXT("\\Assigned\\x86"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
lstrcpy (lpEnd, TEXT("\\Assigned\\Alpha"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
#if 0
|
|
lstrcpy (lpEnd, TEXT("\\Published\\x86\\WinNT"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
|
|
lstrcpy (lpEnd, TEXT("\\Published\\x86\\Win95"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
#else
|
|
lstrcpy (lpEnd, TEXT("\\Published\\x86"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
#endif
|
|
lstrcpy (lpEnd, TEXT("\\Published\\Alpha"));
|
|
if (!CreateNestedDirectory (szDir, NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
break;
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
UINT CComponentDataImpl::CreateNestedDirectory (LPTSTR lpDirectory, LPSECURITY_ATTRIBUTES lpSecurityAttributes)
|
|
{
|
|
TCHAR szDirectory[MAX_PATH];
|
|
LPTSTR lpEnd;
|
|
|
|
|
|
//
|
|
// Check for NULL pointer
|
|
//
|
|
|
|
if (!lpDirectory || !(*lpDirectory)) {
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
return 0;
|
|
}
|
|
|
|
|
|
//
|
|
// First, see if we can create the directory without having
|
|
// to build parent directories.
|
|
//
|
|
|
|
if (CreateDirectory (lpDirectory, lpSecurityAttributes)) {
|
|
return 1;
|
|
}
|
|
|
|
//
|
|
// If this directory exists already, this is OK too.
|
|
//
|
|
|
|
if (GetLastError() == ERROR_ALREADY_EXISTS) {
|
|
return ERROR_ALREADY_EXISTS;
|
|
}
|
|
|
|
|
|
//
|
|
// No luck, copy the string to a buffer we can munge
|
|
//
|
|
|
|
lstrcpy (szDirectory, lpDirectory);
|
|
|
|
|
|
//
|
|
// Find the first subdirectory name
|
|
//
|
|
|
|
lpEnd = szDirectory;
|
|
|
|
if (szDirectory[1] == TEXT(':')) {
|
|
lpEnd += 3;
|
|
} else if (szDirectory[1] == TEXT('\\')) {
|
|
|
|
//
|
|
// Skip the first two slashes
|
|
//
|
|
|
|
lpEnd += 2;
|
|
|
|
//
|
|
// Find the slash between the server name and
|
|
// the share name.
|
|
//
|
|
|
|
while (*lpEnd && *lpEnd != TEXT('\\')) {
|
|
lpEnd++;
|
|
}
|
|
|
|
if (!(*lpEnd)) {
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// Skip the slash, and find the slash between
|
|
// the share name and the directory name.
|
|
//
|
|
|
|
lpEnd++;
|
|
|
|
while (*lpEnd && *lpEnd != TEXT('\\')) {
|
|
lpEnd++;
|
|
}
|
|
|
|
if (!(*lpEnd)) {
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// Leave pointer at the beginning of the directory.
|
|
//
|
|
|
|
lpEnd++;
|
|
|
|
|
|
} else if (szDirectory[0] == TEXT('\\')) {
|
|
lpEnd++;
|
|
}
|
|
|
|
while (*lpEnd) {
|
|
|
|
while (*lpEnd && *lpEnd != TEXT('\\')) {
|
|
lpEnd++;
|
|
}
|
|
|
|
if (*lpEnd == TEXT('\\')) {
|
|
*lpEnd = TEXT('\0');
|
|
|
|
if (!CreateDirectory (szDirectory, NULL)) {
|
|
|
|
if (GetLastError() != ERROR_ALREADY_EXISTS) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
*lpEnd = TEXT('\\');
|
|
lpEnd++;
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// Create the final directory
|
|
//
|
|
|
|
if (CreateDirectory (szDirectory, lpSecurityAttributes)) {
|
|
return 1;
|
|
}
|
|
|
|
if (GetLastError() == ERROR_ALREADY_EXISTS) {
|
|
return ERROR_ALREADY_EXISTS;
|
|
}
|
|
|
|
|
|
//
|
|
// Failed
|
|
//
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//// Notify handlers for IComponentData
|
|
|
|
HRESULT CComponentDataImpl::OnAdd(long cookie, long arg, long param)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnDelete(long cookie, long arg, long param)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnRename(long cookie, long arg, long param)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnExpand(long cookie, long arg, long param)
|
|
{
|
|
if (arg == TRUE)
|
|
{
|
|
// Did Initialize get called?
|
|
ASSERT(m_pScope != NULL);
|
|
|
|
EnumerateScopePane(cookie,
|
|
param);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnSelect(long cookie, long arg, long param)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnContextMenu(long cookie, long arg, long param)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::OnProperties(long param)
|
|
{
|
|
if (param == NULL)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
ASSERT(param != NULL);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
void CComponentDataImpl::EnumerateScopePane(long cookie, HSCOPEITEM pParent)
|
|
{
|
|
// We only have one folder so this is really easy.
|
|
if (cookie != NULL)
|
|
return ;
|
|
|
|
if (m_fExtension)
|
|
{
|
|
// if we're an extension then add a root folder to hang everything off of
|
|
SCOPEDATAITEM * m_pScopeItem = new SCOPEDATAITEM;
|
|
memset(m_pScopeItem, 0, sizeof(SCOPEDATAITEM));
|
|
m_pScopeItem->mask = SDI_STR | SDI_PARAM;
|
|
m_pScopeItem->relativeID = pParent;
|
|
m_pScopeItem->displayname = (unsigned short *)-1;
|
|
m_pScopeItem->lParam = -1; // made up cookie for my main folder
|
|
m_pScope->InsertItem(m_pScopeItem);
|
|
}
|
|
}
|
|
|
|
void CComponentDataImpl::DeleteList()
|
|
{
|
|
return;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::GetDisplayInfo(SCOPEDATAITEM* pScopeDataItem)
|
|
{
|
|
ASSERT(pScopeDataItem != NULL);
|
|
if (pScopeDataItem == NULL)
|
|
return E_POINTER;
|
|
|
|
if (pScopeDataItem->lParam == -1)
|
|
{
|
|
TCHAR szBuffer[256];
|
|
::LoadString(ghInstance, IDS_FOLDER_TITLE, szBuffer, 256);
|
|
m_szFolderTitle = szBuffer;
|
|
pScopeDataItem->displayname = (unsigned short *)((LPCOLESTR)m_szFolderTitle);
|
|
}
|
|
else
|
|
{
|
|
ASSERT(pScopeDataItem->mask == TVIF_TEXT);
|
|
pScopeDataItem->displayname = (unsigned short *)((LPCOLESTR)m_AppData[pScopeDataItem->lParam].szName);
|
|
}
|
|
|
|
ASSERT(pScopeDataItem->displayname != NULL);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CComponentDataImpl::CompareObjects(LPDATAOBJECT lpDataObjectA, LPDATAOBJECT lpDataObjectB)
|
|
{
|
|
if (lpDataObjectA == NULL || lpDataObjectB == NULL)
|
|
return E_POINTER;
|
|
|
|
// Make sure both data object are mine
|
|
INTERNAL* pA;
|
|
INTERNAL* pB;
|
|
HRESULT hr = S_FALSE;
|
|
|
|
pA = ExtractInternalFormat(lpDataObjectA);
|
|
pB = ExtractInternalFormat(lpDataObjectB);
|
|
|
|
if (pA != NULL && pB != NULL)
|
|
hr = (*pA == *pB) ? S_OK : S_FALSE;
|
|
|
|
FREE_INTERNAL(pA);
|
|
FREE_INTERNAL(pB);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// IExtendPropertySheet Implementation
|
|
|
|
STDMETHODIMP CSnapin::CreatePropertyPages(LPPROPERTYSHEETCALLBACK lpProvider,
|
|
long handle,
|
|
LPDATAOBJECT lpIDataObject)
|
|
{
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
INTERNAL* pInternal = ExtractInternalFormat(lpIDataObject);
|
|
if (m_pIClassAdmin && (pInternal->m_type == CCT_RESULT))
|
|
{
|
|
HRESULT hr;
|
|
|
|
APP_DATA & data = m_pComponentData->m_AppData[pInternal->m_cookie];
|
|
|
|
// Create the property page
|
|
CGeneralPage* pPage = new CGeneralPage();
|
|
CPackageDetails * pDetails = new CPackageDetails();
|
|
pPage->m_hConsoleHandle = handle;
|
|
pPage->m_pData = &data;
|
|
pPage->m_cookie = pInternal->m_cookie;
|
|
FREE_INTERNAL(pInternal);
|
|
|
|
pPage->m_szName = data.szName;
|
|
|
|
pDetails->m_hConsoleHandle = handle;
|
|
pDetails->m_pData = &data;
|
|
|
|
// marshal the IClassAdmin interface to the details page
|
|
hr = CoMarshalInterThreadInterfaceInStream(IID_IClassAdmin, m_pIClassAdmin, &(pDetails->m_pIStream));
|
|
|
|
// marshal the IClassAdmin interface to the general page
|
|
hr = CoMarshalInterThreadInterfaceInStream(IID_IClassAdmin, m_pIClassAdmin, &(pPage->m_pIStream));
|
|
|
|
// marshal the IAppManagerActions interface to the general page
|
|
|
|
hr = CoMarshalInterThreadInterfaceInStream(IID_IAppManagerActions, m_pIAppManagerActions, & (pPage->m_pIStreamAM));
|
|
|
|
// Object gets deleted when the page is destroyed
|
|
ASSERT(lpProvider != NULL);
|
|
|
|
hr = MMCPropPageCallback(&pPage->m_psp);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = MMCPropPageCallback(&pDetails->m_psp);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
HPROPSHEETPAGE hPage = CreatePropertySheetPage(&pPage->m_psp);
|
|
HPROPSHEETPAGE hDetails = CreatePropertySheetPage(&pDetails->m_psp);
|
|
|
|
if (hPage == NULL || hDetails == NULL)
|
|
return E_UNEXPECTED;
|
|
|
|
lpProvider->AddPage(hPage);
|
|
#if DBG==1
|
|
lpProvider->AddPage(hDetails);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::QueryPagesFor(LPDATAOBJECT lpDataObject)
|
|
{
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
// Look at the data object and see if it an item that we want to have a property sheet
|
|
INTERNAL* pInternal = ExtractInternalFormat(lpDataObject);
|
|
if (CCT_RESULT == pInternal->m_type)
|
|
{
|
|
FREE_INTERNAL(pInternal);
|
|
return S_OK;
|
|
}
|
|
|
|
FREE_INTERNAL(pInternal);
|
|
return S_FALSE;
|
|
}
|
|
|
|
BOOL CComponentDataImpl::IsScopePaneNode(LPDATAOBJECT lpDataObject)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
INTERNAL* pInternal = ExtractInternalFormat(lpDataObject);
|
|
|
|
if (pInternal->m_type == CCT_SCOPE)
|
|
bResult = TRUE;
|
|
|
|
FREE_INTERNAL(pInternal);
|
|
|
|
return bResult;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// IExtendContextMenu implementation
|
|
//
|
|
STDMETHODIMP CComponentDataImpl::AddMenuItems(LPDATAOBJECT pDataObject,
|
|
LPCONTEXTMENUCALLBACK pContextMenuCallback, LONG * pInsertionAllowed)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
INTERNAL* pInternal = ExtractInternalFormat(pDataObject);
|
|
|
|
|
|
do {
|
|
|
|
//
|
|
// Add Application menu item
|
|
//
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[1]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
|
|
//
|
|
// Update & Remove application if this is a result pane item
|
|
//
|
|
|
|
if (pInternal->m_type == CCT_RESULT)
|
|
{
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[2]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[3]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
//
|
|
// Separator
|
|
//
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[0]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
|
|
//
|
|
// Import Application menu item
|
|
//
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[4]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
|
|
//
|
|
// Separator
|
|
//
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[0]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
|
|
//
|
|
// Refresh menu item
|
|
//
|
|
|
|
hr = pContextMenuCallback->AddItem(&menuItems[5]);
|
|
|
|
if (FAILED(hr))
|
|
break;
|
|
|
|
|
|
} while (FALSE);
|
|
|
|
|
|
FREE_INTERNAL(pInternal);
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CComponentDataImpl::InitializeClassAdmin()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fCancel = FALSE;
|
|
#if 0
|
|
if (!m_fLoaded)
|
|
{
|
|
// initialize dialog with a default path
|
|
// MH Remember last URL
|
|
GetProfileString(L"Appmgr", L"DefCS", L"ADCS:LDAP:", m_szLDAP_Path.GetBuffer(_MAX_PATH), _MAX_PATH);
|
|
m_szLDAP_Path.ReleaseBuffer();
|
|
|
|
m_szGPT_Path = "C:\\GPT\\User\\Applications";
|
|
}
|
|
#endif
|
|
do
|
|
{
|
|
#if 0
|
|
if (!m_fLoaded)
|
|
{
|
|
// If I have to ask for a path then I must not have recieved one
|
|
// from the GPT so the GPT snapin is probably in an invalid
|
|
// state. Therefore I should forget about the GPT snapin.
|
|
// If I don't do this then the GPT snapin is very likely to
|
|
// throw exceptions when I try to send it notifications.
|
|
if (m_pIGPTInformation)
|
|
{
|
|
SAFE_RELEASE(m_pIGPTInformation);
|
|
m_pIGPTInformation = NULL;
|
|
}
|
|
|
|
// ask for a path
|
|
|
|
CInitDlg dlgInit;
|
|
dlgInit.m_szLDAP_Path = m_szLDAP_Path;
|
|
dlgInit.m_szGPT_Path = m_szGPT_Path;
|
|
int iReturn = dlgInit.DoModal();
|
|
if (iReturn = IDOK)
|
|
{
|
|
m_szLDAP_Path= dlgInit.m_szLDAP_Path;
|
|
WriteProfileString(L"Appmgr", L"DefCS", m_szLDAP_Path); // MH Remember last URL
|
|
m_szGPT_Path = dlgInit.m_szGPT_Path;
|
|
}
|
|
m_fLoaded = TRUE;
|
|
}
|
|
#endif
|
|
if (!m_pIClassAdmin)
|
|
{
|
|
// get the IClassAdmin
|
|
LPBC pbc = NULL;
|
|
hr = CreateBindCtx(0, &pbc);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ULONG chEaten = 0;
|
|
IMoniker * pmk = NULL;
|
|
|
|
hr = MkParseDisplayName(pbc, (LPCOLESTR) m_szLDAP_Path, &chEaten, &pmk);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pmk->BindToObject(pbc, NULL, IID_IClassAdmin, (void **) & m_pIClassAdmin);
|
|
// make sure directories are created:
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = CreateApplicationDirectories();
|
|
}
|
|
SAFE_RELEASE(pmk);
|
|
}
|
|
|
|
SAFE_RELEASE(pbc);
|
|
}
|
|
if (FAILED(hr))
|
|
{
|
|
m_fLoaded = FALSE;
|
|
TCHAR szBuffer[256];
|
|
if (!m_pIClassAdmin)
|
|
{
|
|
::LoadString(ghInstance, IDS_CSADMINFAILED, szBuffer, 256);
|
|
}
|
|
else
|
|
{
|
|
m_pIClassAdmin->Release();
|
|
m_pIClassAdmin = NULL;
|
|
::LoadString(ghInstance, IDS_GPTFAILED, szBuffer, 256);
|
|
}
|
|
int iReturn = ::MessageBox(NULL, m_szLDAP_Path,
|
|
szBuffer,
|
|
MB_RETRYCANCEL);
|
|
if (iReturn == IDCANCEL)
|
|
{
|
|
fCancel = TRUE;
|
|
}
|
|
}
|
|
}
|
|
} while ((!fCancel) && (!m_pIClassAdmin));
|
|
return hr;
|
|
}
|
|
|
|
#include <list>
|
|
|
|
typedef struct tagCABLIST
|
|
{
|
|
FILETIME ft;
|
|
CString szPath;
|
|
bool operator<(const struct tagCABLIST& st)
|
|
{
|
|
return CompareFileTime(&ft, &st.ft) < 0;
|
|
}
|
|
} CABLIST;
|
|
|
|
|
|
STDMETHODIMP CComponentDataImpl::Command(long nCommandID, LPDATAOBJECT pDataObject)
|
|
{
|
|
// Note - snap-ins need to look at the data object and determine
|
|
// in what context the command is being called.
|
|
|
|
// Handle each of the commands.
|
|
switch (nCommandID)
|
|
{
|
|
case IDM_ADD_APP:
|
|
case IDM_UPDATE_APP:
|
|
{
|
|
// put up an hourglass (this could take a while)
|
|
CHourglass hourglass;
|
|
CString szFileName;
|
|
|
|
|
|
if (nCommandID == IDM_UPDATE_APP)
|
|
{
|
|
INTERNAL* pInternal = ExtractInternalFormat(pDataObject);
|
|
if (pInternal)
|
|
{
|
|
APP_DATA & data = m_AppData[pInternal->m_cookie];
|
|
|
|
szFileName = data.szIconPath;
|
|
|
|
FREE_INTERNAL(pInternal);
|
|
}
|
|
}
|
|
|
|
|
|
CFileDialog cfd(TRUE,
|
|
szExtension,
|
|
szFileName,
|
|
OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT | OFN_FILEMUSTEXIST,
|
|
szFilter);
|
|
if (IDOK == cfd.DoModal())
|
|
{
|
|
|
|
if (nCommandID == IDM_ADD_APP)
|
|
{
|
|
LONG index;
|
|
BOOL bFound = FALSE;
|
|
|
|
// make sure the user isn't adding something that
|
|
// already exists
|
|
|
|
std::map<long, APP_DATA>::iterator i = m_AppData.begin();
|
|
while (i != m_AppData.end())
|
|
{
|
|
APP_DATA &Data = i->second;
|
|
|
|
TCHAR Drive [_MAX_DRIVE];
|
|
TCHAR Dir [_MAX_DIR];
|
|
TCHAR Name [_MAX_FNAME];
|
|
TCHAR Ext [_MAX_EXT];
|
|
TCHAR szFile[_MAX_PATH];
|
|
|
|
_tsplitpath( Data.szIconPath, Drive, Dir, Name, Ext );
|
|
lstrcpy( szFile, Name );
|
|
lstrcat( szFile, Ext );
|
|
|
|
if (lstrcmpi (szFile, cfd.m_ofn.lpstrFileTitle) == 0)
|
|
{
|
|
bFound = TRUE;
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
if (bFound)
|
|
{
|
|
TCHAR szBuffer[256];
|
|
TCHAR szTitle[100];
|
|
|
|
::LoadString(ghInstance, IDS_ADDEXISTSALREADY, szBuffer, 256);
|
|
::LoadString(ghInstance, IDS_SNAPIN_DESC, szTitle, 100);
|
|
m_pConsole->MessageBox(szBuffer,
|
|
szTitle,
|
|
MB_OK, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// user selected an application
|
|
UNIVERSAL_NAME_INFO * pUni = new UNIVERSAL_NAME_INFO;
|
|
ULONG cbSize = sizeof(UNIVERSAL_NAME_INFO);
|
|
BOOL bAssigned = FALSE;
|
|
HRESULT hr = WNetGetUniversalName(cfd.m_ofn.lpstrFile,
|
|
UNIVERSAL_NAME_INFO_LEVEL,
|
|
pUni,
|
|
&cbSize);
|
|
if (ERROR_MORE_DATA == hr) // we expect this to be true
|
|
{
|
|
delete [] pUni;
|
|
pUni = (UNIVERSAL_NAME_INFO *) new BYTE [cbSize];
|
|
hr = WNetGetUniversalName(cfd.m_ofn.lpstrFile,
|
|
UNIVERSAL_NAME_INFO_LEVEL,
|
|
pUni,
|
|
&cbSize);
|
|
}
|
|
|
|
int i;
|
|
char * szPackagePath = NULL;
|
|
|
|
if (S_OK == hr)
|
|
{
|
|
i = WTOALEN(pUni->lpUniversalName);
|
|
szPackagePath = new char [i+1];
|
|
WTOA(szPackagePath, pUni->lpUniversalName, i);
|
|
}
|
|
else
|
|
{
|
|
i = WTOALEN(cfd.m_ofn.lpstrFile);
|
|
szPackagePath = new char [i+1];
|
|
WTOA(szPackagePath, cfd.m_ofn.lpstrFile, i);
|
|
}
|
|
delete[] pUni;
|
|
|
|
i = WTOALEN((LPCOLESTR)m_szGPT_Path);
|
|
char * szFilePath = new char [i+1];
|
|
WTOA(szFilePath, m_szGPT_Path, i);
|
|
|
|
|
|
if (nCommandID == IDM_UPDATE_APP)
|
|
{
|
|
if (SUCCEEDED(RemovePackage(pDataObject, &bAssigned)))
|
|
{
|
|
hr = AddMSIPackage(pDataObject, szPackagePath, szFilePath, cfd.m_ofn.lpstrFileTitle, &bAssigned);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = AddMSIPackage(pDataObject, szPackagePath, szFilePath, cfd.m_ofn.lpstrFileTitle, &bAssigned);
|
|
}
|
|
|
|
delete [] szPackagePath;
|
|
delete [] szFilePath;
|
|
|
|
|
|
// Notify clients of change
|
|
if (SUCCEEDED(hr) && m_pIGPTInformation && bAssigned)
|
|
{
|
|
m_pIGPTInformation->NotifyClients(FALSE);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case IDM_ADD_FROM_IE:
|
|
if (m_pIClassAdmin)
|
|
{
|
|
// Locate IE4
|
|
HKEY hkey;
|
|
LONG r;
|
|
TCHAR szPath[MAX_PATH];
|
|
TCHAR szFullPath[MAX_PATH];
|
|
r = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
|
|
TEXT("SOFTWARE\\Microsoft\\IE4\\Setup"),
|
|
0,
|
|
KEY_READ,
|
|
&hkey);
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
DWORD cbData = MAX_PATH * sizeof(TCHAR);
|
|
r = RegQueryValueEx(hkey,
|
|
TEXT("Path"),
|
|
0,
|
|
NULL,
|
|
(LPBYTE)szPath,
|
|
&cbData);
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
ExpandEnvironmentStrings(szPath, szFullPath, MAX_PATH);
|
|
lstrcat(szFullPath, TEXT("\\iexplore.exe"));
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
// Put up dialog informing user to close IE4 when he's ready
|
|
// to continue.
|
|
TCHAR szBuffer[1024];
|
|
TCHAR szCaption[256];
|
|
::LoadString(ghInstance, IDS_SPAWNMSG, szBuffer, 1024);
|
|
::LoadString(ghInstance, IDS_SPAWNCAPTION, szCaption, 256);
|
|
|
|
int iReturn = ::MessageBox(NULL,
|
|
szBuffer,
|
|
szCaption,
|
|
MB_YESNO);
|
|
if (IDYES == iReturn)
|
|
{
|
|
// Take the starting time stamp
|
|
FILETIME ftStart;
|
|
GetSystemTimeAsFileTime(&ftStart);
|
|
|
|
// Start IE4 and wait for it to be closed
|
|
BOOL f;
|
|
STARTUPINFO startupinfo;
|
|
memset (&startupinfo, 0, sizeof(startupinfo));
|
|
PROCESS_INFORMATION processinfo;
|
|
f = CreateProcess(NULL,
|
|
szFullPath,
|
|
NULL, // process attributes
|
|
NULL, // thread attributes
|
|
FALSE, // bInheritHandles
|
|
CREATE_DEFAULT_ERROR_MODE | CREATE_NEW_CONSOLE | NORMAL_PRIORITY_CLASS,
|
|
NULL, // lpEnvironment
|
|
NULL, // lpCurrentDirectory
|
|
&startupinfo,
|
|
&processinfo);
|
|
if (f)
|
|
{
|
|
DWORD dw;
|
|
MSG msg;
|
|
do
|
|
{
|
|
dw = MsgWaitForMultipleObjects(1, &processinfo.hProcess, FALSE, INFINITE, QS_ALLINPUT);
|
|
// if we don't process Windows messages
|
|
// here, we run the risk of causing a
|
|
// deadlock
|
|
if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
|
|
{
|
|
DispatchMessage(&msg);
|
|
}
|
|
} while (dw != WAIT_OBJECT_0 );
|
|
}
|
|
|
|
// Take the ending time stamp
|
|
FILETIME ftEnd;
|
|
GetSystemTimeAsFileTime(&ftEnd);
|
|
|
|
r = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
|
|
TEXT("SOFTWARE\\Microsoft\\Code Store Database\\Distribution Units"),
|
|
0,
|
|
KEY_READ,
|
|
&hkey);
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
DWORD cSubKeys,
|
|
cbMaxSubKeyLen;
|
|
|
|
r = RegQueryInfoKey(hkey,
|
|
NULL, // lpClass
|
|
NULL, // lpcbClass
|
|
0, // reserved
|
|
&cSubKeys,
|
|
&cbMaxSubKeyLen,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
// Build a list of cab files installed after the
|
|
// time stamp and order them by time.
|
|
std::list<CABLIST> cablist;
|
|
LPTSTR lpName = new TCHAR[cbMaxSubKeyLen + 1];
|
|
DWORD iSubKey;
|
|
DWORD cbName;
|
|
FILETIME ftSubKey;
|
|
for (iSubKey = 0; iSubKey < cSubKeys; iSubKey++)
|
|
{
|
|
HKEY hkSubKey;
|
|
cbName = cbMaxSubKeyLen + 1;
|
|
r = RegEnumKeyEx(hkey,
|
|
iSubKey,
|
|
lpName,
|
|
&cbName,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&ftSubKey);
|
|
if ((ERROR_SUCCESS == r) &&
|
|
(CompareFileTime(&ftStart, &ftSubKey) <= 0) &&
|
|
(CompareFileTime(&ftSubKey, &ftEnd) <= 0))
|
|
{
|
|
CString szSubKey = lpName;
|
|
szSubKey += "\\DownloadInformation";
|
|
HKEY hkeyInfo;
|
|
r = RegOpenKeyEx(hkey,
|
|
szSubKey,
|
|
0,
|
|
KEY_READ,
|
|
&hkeyInfo);
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
TCHAR szPath[MAX_PATH];
|
|
DWORD cbData = MAX_PATH * sizeof(TCHAR);
|
|
r = RegQueryValueEx(hkeyInfo,
|
|
TEXT("CODEBASE"),
|
|
0,
|
|
NULL,
|
|
(LPBYTE)szPath,
|
|
&cbData);
|
|
if (ERROR_SUCCESS == r)
|
|
{
|
|
// add this one to the list
|
|
CABLIST cl;
|
|
cl.szPath = szPath;
|
|
cl.ft = ftSubKey;
|
|
cablist.push_back(cl);
|
|
}
|
|
RegCloseKey(hkeyInfo);
|
|
}
|
|
}
|
|
}
|
|
RegCloseKey(hkey);
|
|
delete [] lpName;
|
|
|
|
// sort the list by file time stamps
|
|
cablist.sort();
|
|
|
|
// for each cab file in the list
|
|
std::list<CABLIST>::iterator i;
|
|
for (i=cablist.begin(); i != cablist.end(); i++)
|
|
{
|
|
int x;
|
|
char * szPackagePath = NULL;
|
|
x = WTOALEN(i->szPath);
|
|
szPackagePath = new char [x+1];
|
|
WTOA(szPackagePath, i->szPath, x);
|
|
|
|
x = WTOALEN((LPCOLESTR)m_szGPT_Path);
|
|
char * szFilePath = new char [x+1];
|
|
WTOA(szFilePath, m_szGPT_Path, x);
|
|
HWND hwnd;
|
|
m_pConsole->GetMainWindow(&hwnd);
|
|
|
|
// install the cab file
|
|
HRESULT hr = UpdateClassStoreFromIE(m_pIClassAdmin, szPackagePath, szFilePath, 1, ftStart, i->ft, hwnd);
|
|
|
|
ftStart = i->ft;
|
|
|
|
delete [] szPackagePath;
|
|
delete [] szFilePath;
|
|
|
|
if (S_OK != hr)
|
|
{
|
|
TCHAR szBuffer[256];
|
|
::LoadString(ghInstance, IDS_ADDFAILED, szBuffer, 256);
|
|
m_pConsole->MessageBox(szBuffer,
|
|
i->szPath,
|
|
MB_OK, NULL);
|
|
}
|
|
else
|
|
{
|
|
// add an entry to the result pane
|
|
PACKAGEDETAIL * pd = new PACKAGEDETAIL;
|
|
int n = i->szPath.ReverseFind('/');
|
|
CString szName = i->szPath.Mid(n+1);
|
|
hr = m_pIClassAdmin->GetPackageDetails((LPOLESTR)((LPCOLESTR) szName), pd);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
APP_DATA data;
|
|
data.szName = pd->pszPackageName;
|
|
if (pd->dwActFlags & ACTFLG_Assigned)
|
|
{
|
|
data.type = DT_ASSIGNED;
|
|
}
|
|
else
|
|
{
|
|
data.type = DT_PUBLISHED;
|
|
}
|
|
data.szPath = pd->pszPath;
|
|
data.szIconPath = pd->pszIconPath;
|
|
data.szDesc = pd->pszProductName;
|
|
data.pDetails = pd;
|
|
|
|
RESULTDATAITEM resultItem;
|
|
|
|
resultItem.mask = RDI_STR | RDI_IMAGE | RDI_PARAM;
|
|
resultItem.str = MMC_CALLBACK;
|
|
resultItem.nImage = 1;
|
|
data.fBlockDeletion = FALSE;
|
|
SetStringData(&data);
|
|
m_lLastAllocated++;
|
|
m_AppData[m_lLastAllocated] = data;
|
|
// BUGBUG - need to make sure that m_lLastAllocated
|
|
// hasn't already been used!
|
|
resultItem.lParam = m_lLastAllocated;
|
|
hr = m_pSnapin->m_pResult->InsertItem(&resultItem);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
m_AppData[m_lLastAllocated].itemID = resultItem.itemID;
|
|
m_pSnapin->m_pResult->Sort(0, 0, -1);
|
|
}
|
|
|
|
// Notify clients of change
|
|
if (m_pIGPTInformation && (data.type == DT_ASSIGNED))
|
|
{
|
|
m_pIGPTInformation->NotifyClients(FALSE);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// put up dialog informing user that IE4 isn't installed
|
|
// on this machine
|
|
// UNDONE
|
|
}
|
|
|
|
}
|
|
break;
|
|
case IDM_REFRESH:
|
|
if (m_pIClassAdmin)
|
|
{
|
|
|
|
std::map <long, APP_DATA>::iterator i;
|
|
for (i=m_AppData.begin(); i != m_AppData.end(); i++)
|
|
{
|
|
m_pSnapin->m_pResult->DeleteItem(i->second.itemID, 0);
|
|
m_AppData.erase(i);
|
|
}
|
|
m_lLastAllocated = 0;
|
|
m_pSnapin->EnumerateResultPane(0);
|
|
}
|
|
break;
|
|
case IDM_DEL_APP:
|
|
{
|
|
BOOL bAssigned = FALSE;
|
|
|
|
if (SUCCEEDED(RemovePackage(pDataObject, &bAssigned)))
|
|
{
|
|
// Notify clients of change
|
|
if (m_pIGPTInformation && bAssigned)
|
|
{
|
|
m_pIGPTInformation->NotifyClients(FALSE);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CComponentDataImpl::AddMSIPackage(LPDATAOBJECT pDataObject, LPSTR szPackagePath, LPSTR szFilePath, LPOLESTR lpFileTitle, BOOL *bAssigned)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if (m_pIClassAdmin)
|
|
{
|
|
ASSERT(m_pConsole);
|
|
{
|
|
char szPackageName[MAX_PATH];
|
|
DWORD cchPackageName = MAX_PATH;
|
|
HWND hwnd;
|
|
m_pConsole->GetMainWindow(&hwnd);
|
|
|
|
hr = UpdateClassStore(m_pIClassAdmin, szPackagePath, szFilePath, szPackageName, cchPackageName, !(*bAssigned), hwnd);
|
|
|
|
if (S_OK != hr)
|
|
{
|
|
TCHAR szBuffer[256];
|
|
// check to see if the reason is because there
|
|
// were no COM Serverss
|
|
if (hr == MAKE_HRESULT( SEVERITY_SUCCESS, 0, 1 ))
|
|
{
|
|
::LoadString(ghInstance, IDS_NOCOMSVR, szBuffer, 256);
|
|
}
|
|
else
|
|
{
|
|
::LoadString(ghInstance, IDS_ADDFAILED, szBuffer, 256);
|
|
//#if DBG == 1
|
|
#if 0
|
|
TCHAR szDebugBuffer[256];
|
|
DWORD dw = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
hr,
|
|
0,
|
|
szDebugBuffer,
|
|
sizeof(szDebugBuffer) / sizeof(szDebugBuffer[0]),
|
|
NULL);
|
|
if (0 == dw)
|
|
{
|
|
wsprintf(szDebugBuffer, TEXT("(HRESULT: 0x%lX)"), hr);
|
|
}
|
|
wcscat(szBuffer, szDebugBuffer);
|
|
#endif
|
|
}
|
|
m_pConsole->MessageBox(szBuffer,
|
|
lpFileTitle,
|
|
MB_OK, NULL);
|
|
}
|
|
else
|
|
{
|
|
PACKAGEDETAIL * pd = new PACKAGEDETAIL;
|
|
WCHAR wszPackageName[MAX_PATH];
|
|
ATOW(wszPackageName, szPackageName, MAX_PATH);
|
|
hr = m_pIClassAdmin->GetPackageDetails(wszPackageName, pd);
|
|
//hr = m_pIClassAdmin->GetPackageDetails(lpFileTitle, pd);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
APP_DATA data;
|
|
data.szName = pd->pszPackageName;
|
|
if (pd->dwActFlags & ACTFLG_Assigned)
|
|
{
|
|
data.type = DT_ASSIGNED;
|
|
}
|
|
else
|
|
{
|
|
data.type = DT_PUBLISHED;
|
|
}
|
|
data.szPath = pd->pszPath;
|
|
data.szIconPath = pd->pszIconPath;
|
|
data.szDesc = pd->pszProductName;
|
|
data.pDetails = pd;
|
|
data.fBlockDeletion = FALSE;
|
|
SetStringData(&data);
|
|
m_lLastAllocated++;
|
|
m_AppData[m_lLastAllocated] = data;
|
|
|
|
RESULTDATAITEM resultItem;
|
|
|
|
resultItem.mask = RDI_STR | RDI_IMAGE | RDI_PARAM;
|
|
resultItem.str = MMC_CALLBACK;
|
|
resultItem.nImage = 1;
|
|
// BUGBUG - need to make sure that m_lLastAllocated
|
|
// hasn't already been used!
|
|
resultItem.lParam = m_lLastAllocated;
|
|
hr = m_pSnapin->m_pResult->InsertItem(&resultItem);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
m_AppData[m_lLastAllocated].itemID = resultItem.itemID;
|
|
m_pSnapin->m_pResult->Sort(0, 0, -1);
|
|
}
|
|
|
|
// Notify clients of change
|
|
if (data.type == DT_ASSIGNED)
|
|
{
|
|
*bAssigned = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CComponentDataImpl::RemovePackage(LPDATAOBJECT pDataObject, BOOL *bAssigned)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if (m_pIClassAdmin)
|
|
{
|
|
ASSERT(m_pConsole);
|
|
INTERNAL* pInternal = ExtractInternalFormat(pDataObject);
|
|
if (pInternal->m_type == CCT_RESULT)
|
|
{
|
|
// put up an hourglass (this could take a while)
|
|
CHourglass hourglass;
|
|
APP_DATA & data = m_AppData[pInternal->m_cookie];
|
|
|
|
if (!data.fBlockDeletion) // make sure it's not being held open by a property page
|
|
{
|
|
// We need to make sure it gets removed from
|
|
// the GPT before we delete it from the class store.
|
|
|
|
if (data.pDetails->PathType==DrwFilePath) // MH: don't touch the GPT for anything but Darwin files!
|
|
DeleteFile(data.szPath);
|
|
|
|
if (data.type == DT_ASSIGNED)
|
|
*bAssigned = TRUE;
|
|
|
|
|
|
hr = DeletePackageAndDependants(m_pIClassAdmin, (LPOLESTR)((LPCOLESTR) data.szName), data.pDetails);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = m_pSnapin->m_pResult->DeleteItem(data.itemID, 0);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
m_AppData.erase(pInternal->m_cookie);
|
|
m_pSnapin->m_pResult->Sort(0, 0, -1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
FREE_INTERNAL(pInternal);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// IExtendControlbar implementation
|
|
//
|
|
|
|
#if 0
|
|
|
|
STDMETHODIMP CSnapin::SetControlbar(LPCONTROLBAR pControlbar)
|
|
{
|
|
|
|
TRACE(_T("CSnapin::SetControlbar(%ld)\n"),pControlbar);
|
|
// Please don't delete this. Required to make sure we pick up the bitmap
|
|
AFX_MANAGE_STATE(AfxGetStaticModuleState());
|
|
|
|
if (pControlbar != NULL)
|
|
{
|
|
// Hold on to the controlbar interface.
|
|
if (m_pControlbar)
|
|
{
|
|
m_pControlbar->Release();
|
|
}
|
|
|
|
m_pControlbar = pControlbar;
|
|
m_pControlbar->AddRef();
|
|
|
|
HRESULT hr=S_FALSE;
|
|
|
|
// Create the Toolbar 1
|
|
if (!m_pToolbar1)
|
|
{
|
|
hr = m_pControlbar->Create(TOOLBAR, this, reinterpret_cast<LPUNKNOWN*>(&m_pToolbar1));
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
|
|
// Add the bitmap
|
|
m_pbmpToolbar1 = new CBitmap;
|
|
m_pbmpToolbar1->LoadBitmap(IDB_TOOLBAR1);
|
|
hr = m_pToolbar1->AddBitmap(11, *m_pbmpToolbar1, 16, 16, RGB(255, 0, 255));
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
// Add the buttons to the toolbar
|
|
hr = m_pToolbar1->AddButtons(ARRAYLEN(SnapinButtons), SnapinButtons);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
|
|
|
|
// TOOLBAR 2
|
|
// Create the Toolbar 2
|
|
if (!m_pToolbar2)
|
|
{
|
|
hr = m_pControlbar->Create(TOOLBAR, this, reinterpret_cast<LPUNKNOWN*>(&m_pToolbar2));
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
// Add the bitmap
|
|
m_pbmpToolbar2 = new CBitmap;
|
|
m_pbmpToolbar2->LoadBitmap(IDB_TOOLBAR2);
|
|
hr = m_pToolbar2->AddBitmap(36, *m_pbmpToolbar2, 16, 16, RGB(192,192,192));
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
// Add the buttons to the toolbar
|
|
hr = m_pToolbar2->AddButtons(ARRAYLEN(SnapinButtons2), SnapinButtons2);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SAFE_RELEASE(m_pControlbar);
|
|
}
|
|
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::ControlbarNotify(MMC_NOTIFY_TYPE event, long arg, long param)
|
|
{
|
|
HRESULT hr=S_FALSE;
|
|
// Temp temp
|
|
static BOOL bSwap=FALSE;
|
|
|
|
switch (event)
|
|
{
|
|
#if 0
|
|
case MMCN_ACTIVATE:
|
|
TRACE(_T("CSnapin::ControlbarNotify - MMCN_ACTIVATE\n"));
|
|
// Need to handle this.
|
|
// Verify that we can enable and disable buttons based on selection
|
|
|
|
if (arg == TRUE)
|
|
{
|
|
m_pToolbar1->SetButtonState(3, BUTTONPRESSED, TRUE);
|
|
}
|
|
else
|
|
{
|
|
BOOL bState=FALSE;
|
|
hr = m_pToolbar1->GetButtonState(3, BUTTONPRESSED, &bState);
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (bState)
|
|
m_pToolbar1->SetButtonState(3, BUTTONPRESSED, FALSE);
|
|
}
|
|
|
|
break;
|
|
#endif // 0
|
|
|
|
case MMCN_BTN_CLICK:
|
|
TRACE(_T("CSnapin::ControlbarNotify - MMCN_BTN_CLICK\n"));
|
|
// Temp code
|
|
TCHAR szMessage[MAX_PATH];
|
|
wsprintf(szMessage,_T("CommandID %ld was not handled by the snapin!!!"),param);
|
|
AfxMessageBox(szMessage);
|
|
|
|
break;
|
|
case MMCN_SELECT:
|
|
TRACE(_T("CSnapin::ControlbarNotify - MMCN_SEL_CHANGE\n"));
|
|
{
|
|
LPDATAOBJECT* ppDataObject = reinterpret_cast<LPDATAOBJECT*>(param);
|
|
|
|
// Attach the toolbars to the window
|
|
hr = m_pControlbar->Attach(TOOLBAR, (LPUNKNOWN) m_pToolbar1);
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
hr = m_pControlbar->Attach(TOOLBAR, (LPUNKNOWN) m_pToolbar2);
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
}
|
|
|
|
bSwap = !bSwap ;
|
|
|
|
if (bSwap == TRUE)
|
|
{
|
|
m_pToolbar1->SetButtonState(1, ENABLED, FALSE); // 1 = CMD ID
|
|
m_pToolbar1->SetButtonState(2, CHECKED, TRUE); // 2 = CMD ID
|
|
m_pToolbar1->SetButtonState(3, HIDDEN, TRUE); // 3 = CMD ID
|
|
m_pToolbar1->SetButtonState(4, INDETERMINATE, TRUE); // 4 = CMD ID
|
|
m_pToolbar1->SetButtonState(5, BUTTONPRESSED, TRUE); // 5 = CMD ID
|
|
|
|
// Just for fun let's add another style
|
|
m_pToolbar1->SetButtonState(2, BUTTONPRESSED, TRUE); // 4 = CMD ID
|
|
}
|
|
else
|
|
{
|
|
BOOL bState=FALSE;
|
|
hr = m_pToolbar1->GetButtonState(1, ENABLED, &bState);
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
if (bState == FALSE)
|
|
m_pToolbar1->SetButtonState(1, ENABLED, TRUE);
|
|
|
|
|
|
// Above is the correct way
|
|
m_pToolbar1->SetButtonState(2, CHECKED, FALSE);
|
|
m_pToolbar1->SetButtonState(3, HIDDEN, FALSE);
|
|
m_pToolbar1->SetButtonState(4, INDETERMINATE, FALSE);
|
|
m_pToolbar1->SetButtonState(5, BUTTONPRESSED, FALSE);
|
|
|
|
// Better remove the additional style
|
|
m_pToolbar1->SetButtonState(2, BUTTONPRESSED, FALSE); // 4 = CMD ID
|
|
|
|
}
|
|
|
|
|
|
break;
|
|
|
|
|
|
default:
|
|
ASSERT(FALSE); // Unhandle event
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
#endif
|
|
|
|
STDMETHODIMP CSnapin::CompareObjects(LPDATAOBJECT lpDataObjectA, LPDATAOBJECT lpDataObjectB)
|
|
{
|
|
if (lpDataObjectA == NULL || lpDataObjectB == NULL)
|
|
return E_POINTER;
|
|
|
|
// Make sure both data object are mine
|
|
INTERNAL* pA;
|
|
INTERNAL* pB;
|
|
HRESULT hr = S_FALSE;
|
|
|
|
pA = ExtractInternalFormat(lpDataObjectA);
|
|
pB = ExtractInternalFormat(lpDataObjectB);
|
|
|
|
if (pA != NULL && pB != NULL)
|
|
hr = (*pA == *pB) ? S_OK : S_FALSE;
|
|
|
|
FREE_INTERNAL(pA);
|
|
FREE_INTERNAL(pB);
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSnapin::Compare(long lUserParam, long cookieA, long cookieB, int* pnResult)
|
|
{
|
|
if (pnResult == NULL)
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_POINTER;
|
|
}
|
|
|
|
// check col range
|
|
int nCol = *pnResult;
|
|
|
|
*pnResult = 0;
|
|
|
|
APP_DATA & dataA = m_pComponentData->m_AppData[cookieA];
|
|
APP_DATA & dataB = m_pComponentData->m_AppData[cookieB];
|
|
// compare the two based on column and the cookies
|
|
|
|
switch (nCol)
|
|
{
|
|
case 0:
|
|
*pnResult = dataA.szName.CompareNoCase(dataB.szName);
|
|
break;
|
|
case 1:
|
|
*pnResult = dataA.szType.CompareNoCase(dataB.szType);
|
|
break;
|
|
case 2:
|
|
*pnResult = dataA.pDetails->Locale - dataB.pDetails->Locale;
|
|
break;
|
|
case 3:
|
|
*pnResult = dataA.szMach.CompareNoCase(dataB.szMach);
|
|
break;
|
|
case 4:
|
|
*pnResult = dataA.szDesc.CompareNoCase(dataB.szDesc);
|
|
break;
|
|
case 5:
|
|
*pnResult = dataA.szIconPath.CompareNoCase(dataB.szIconPath);
|
|
break;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|