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.
3728 lines
104 KiB
3728 lines
104 KiB
//=--------------------------------------------------------------------------=
|
|
// proxy.cpp
|
|
//=--------------------------------------------------------------------------=
|
|
// Copyright (c) 1999, Microsoft Corp.
|
|
// All Rights Reserved
|
|
// Information Contained Herein Is Proprietary and Confidential.
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// MMC Interfcace Proxy and Stub Functions
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
// This file contains proxy and stub functions for MMC methods that are not
|
|
// remotable using a MIDL generated proxy and stub. Non-remotable methods have
|
|
// parameters that are ambiguous i.e. can be casted to different data types.
|
|
// For example, IComponentData::Notify() is passed an event and two additional
|
|
// LPARAM arguments that are interpreted according to the event. Sometimes an
|
|
// LPARAM contains a simple value such as a long or two BOOLs and sometimes it
|
|
// contains an IDataObject *. MIDL doesn't know the difference between
|
|
// MMCN_SELECT and MMCN_PRINT so we need to write some code to help out.
|
|
//
|
|
// The version of MMC.IDL in the designer directory has added a [local]
|
|
// attribute to all non-remotable methods. In addition, an extra method has
|
|
// been added to the same interface that is a remotable version of the method.
|
|
// The remotable version has more parameters and represents the union of all
|
|
// possible interpretations of the ambiguous parameter. For example,
|
|
// IExtendControlbar::ControlbarNotify() is defined as:
|
|
//
|
|
// [helpstring("User actions"), local]
|
|
// HRESULT ControlbarNotify([in] MMC_NOTIFY_TYPE event,
|
|
// [in] LPARAM arg, [in] LPARAM param);
|
|
//
|
|
// This method can receive MMCN_SELECT, MMCN_BTN_CLICK, and MMCN_MENU_BTNCLICK.
|
|
// The union of all possible parameter types is used in the following method
|
|
// added to that interface:
|
|
//
|
|
//
|
|
// HRESULT RemControlbarNotify([in] MMC_NOTIFY_TYPE event,
|
|
// [in] LPARAM lparam,
|
|
// [in] IDataObject *piDataObject,
|
|
// [in] MENUBUTTONDATA *MenuButtonData);
|
|
//
|
|
// Note, that normal, in-proc, non-remoted versions of IExtendControlbar do not
|
|
// have this extra method in their vtable because no one is going to call it.
|
|
// It is only used in the MILD generated proxy object.
|
|
//
|
|
// In order to tell MIDL which method remotes ControlbarNotify() an attribute
|
|
// control file (ACF) is used. The entry in the ACF for IExtendControlbar is:
|
|
//
|
|
// interface IExtendControlbar
|
|
// {
|
|
// [call_as (ControlbarNotify)]
|
|
// RemControlbarNotify();
|
|
//
|
|
// }
|
|
//
|
|
// This says that RemControlbarNotify should be called when remoting
|
|
// ControlbarNotify(). MIDL generates the proxy/stub code as usual but it only
|
|
// generates the prototypes for ControlbarNotify proxy and stub. We have to
|
|
// write these routines.
|
|
//
|
|
// When a remote client has an IExtendControlbarNotify pointer it actually
|
|
// points into the proxy vtable. MIDL sets the ControlbarNotify entry pointing to
|
|
// our IExtendControlbar_ControlbarNotify_Proxy() function below. That function
|
|
// interprets the parameters and then calls the MIDL generated
|
|
// IExtendControlbar_RemControlbarNotify_Proxy() that packs up the parameters
|
|
// and sends them off to the server. If a parameters is not applicable, (e.g.
|
|
// MMCN_SELECT does not receive a pointer to a MENUBUTTONDATA stuct), then a
|
|
// pointer to an empty struct or zeroes are sent.
|
|
//
|
|
// When the packet reaches the server side the MIDL generated
|
|
// IExtendControlbar_RemControlbarNotify_Stub() unpacks them and then calls our
|
|
// IExtendControlbar_ControlbarNotify_Stub() passing it the parameters and the
|
|
// IExtendControlbar pointer into the server. This function interprets the
|
|
// parameters and then calls ControlbarNotify in the server.
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
#include "mmc.h"
|
|
|
|
extern HRESULT GetClipboardFormat
|
|
(
|
|
WCHAR *pwszFormatName,
|
|
CLIPFORMAT *pcfFormat
|
|
);
|
|
|
|
extern HRESULT CreateMultiSelDataObject
|
|
(
|
|
IDataObject **ppiDataObjects,
|
|
long cDataObjects,
|
|
IDataObject **ppiMultiSelDataObject
|
|
);
|
|
|
|
|
|
static HRESULT MenuButtonClickProxy
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
IDataObject *piDataObject,
|
|
MENUBUTTONDATA *pMenuButtonData
|
|
);
|
|
|
|
static HRESULT IsMultiSelect(IDataObject *piDataObject, BOOL *pfMultiSelect)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD *pdwMultiSelect = NULL;
|
|
BOOL fGotData = FALSE;
|
|
FORMATETC FmtEtc;
|
|
STGMEDIUM StgMed;
|
|
|
|
ZeroMemory(&FmtEtc, sizeof(FmtEtc));
|
|
ZeroMemory(&StgMed, sizeof(StgMed));
|
|
|
|
*pfMultiSelect = FALSE;
|
|
|
|
if (NULL == piDataObject)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (IS_SPECIAL_DATAOBJECT(piDataObject))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = GetClipboardFormat(CCF_MMC_MULTISELECT_DATAOBJECT, &FmtEtc.cfFormat);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
FmtEtc.dwAspect = DVASPECT_CONTENT;
|
|
FmtEtc.lindex = -1L;
|
|
FmtEtc.tymed = TYMED_HGLOBAL;
|
|
StgMed.tymed = TYMED_HGLOBAL;
|
|
|
|
hr = piDataObject->lpVtbl->GetData(piDataObject, &FmtEtc, &StgMed);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
fGotData = TRUE;
|
|
}
|
|
else
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
|
|
// Ignore any failures and assume that it is not multi-select. Snap-ins
|
|
// should return DV_E_FORMATETC or DV_E_CLIPFORMAT but in practice that
|
|
// is not the case. For example, the IIS snap-in returns E_NOTIMPL.
|
|
// It would be impossible to cover the range of reasonable return codes so
|
|
// we treat any error as format not supported.
|
|
|
|
if (fGotData)
|
|
{
|
|
pdwMultiSelect = (DWORD *)GlobalLock(StgMed.hGlobal);
|
|
|
|
if ((DWORD)1 == *pdwMultiSelect)
|
|
{
|
|
*pfMultiSelect = TRUE;
|
|
}
|
|
}
|
|
|
|
Cleanup:
|
|
if (NULL != pdwMultiSelect)
|
|
{
|
|
(void)GlobalUnlock(StgMed.hGlobal);
|
|
}
|
|
|
|
if (fGotData)
|
|
{
|
|
ReleaseStgMedium(&StgMed);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
static HRESULT InterpretMultiSelect
|
|
(
|
|
IDataObject *piDataObject,
|
|
long *pcDataObjects,
|
|
IDataObject ***pppiDataObjects
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
SMMCDataObjects *pMMCDataObjects = NULL;
|
|
BOOL fGotData = FALSE;
|
|
size_t cbObjectTypes = 0;
|
|
long i = 0;
|
|
FORMATETC FmtEtc;
|
|
STGMEDIUM StgMed;
|
|
|
|
ZeroMemory(&FmtEtc, sizeof(FmtEtc));
|
|
ZeroMemory(&StgMed, sizeof(StgMed));
|
|
|
|
*pcDataObjects = 0;
|
|
*pppiDataObjects = NULL;
|
|
|
|
// Get the SMMCDataObjects structure from MMC
|
|
|
|
hr = GetClipboardFormat(CCF_MULTI_SELECT_SNAPINS, &FmtEtc.cfFormat);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
FmtEtc.dwAspect = DVASPECT_CONTENT;
|
|
FmtEtc.lindex = -1L;
|
|
FmtEtc.tymed = TYMED_HGLOBAL;
|
|
StgMed.tymed = TYMED_HGLOBAL;
|
|
|
|
hr = piDataObject->lpVtbl->GetData(piDataObject, &FmtEtc, &StgMed);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
fGotData = TRUE;
|
|
|
|
pMMCDataObjects = (SMMCDataObjects *)GlobalLock(StgMed.hGlobal);
|
|
if (NULL == pMMCDataObjects)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Allocate an array of IDataObject and copy the IDataObjects to it
|
|
|
|
*pcDataObjects = pMMCDataObjects->count;
|
|
*pppiDataObjects = (IDataObject **)GlobalAlloc(GPTR,
|
|
pMMCDataObjects->count * sizeof(IDataObject *));
|
|
|
|
if (NULL == *pppiDataObjects)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
for (i = 0; i < *pcDataObjects; i++)
|
|
{
|
|
(*pppiDataObjects)[i] = pMMCDataObjects->lpDataObject[i];
|
|
}
|
|
|
|
Cleanup:
|
|
if (NULL != pMMCDataObjects)
|
|
{
|
|
(void)GlobalUnlock(StgMed.hGlobal);
|
|
}
|
|
|
|
if (fGotData)
|
|
{
|
|
ReleaseStgMedium(&StgMed);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
void CheckForSpecialDataObjects
|
|
(
|
|
IDataObject **ppiDataObject,
|
|
BOOL *pfSpecialDataObject,
|
|
long *plSpecialDataObject
|
|
)
|
|
{
|
|
long lSpecialDataObject = (long)(*ppiDataObject);
|
|
|
|
if (IS_SPECIAL_DATAOBJECT(lSpecialDataObject))
|
|
{
|
|
*plSpecialDataObject = lSpecialDataObject;
|
|
*ppiDataObject = NULL;
|
|
*pfSpecialDataObject = TRUE;
|
|
}
|
|
else
|
|
{
|
|
*pfSpecialDataObject = FALSE;
|
|
}
|
|
}
|
|
|
|
static HRESULT SetRemote(IUnknown *This)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IMMCRemote *piMMCRemote = NULL;
|
|
DWORD cbFileName = 0;
|
|
char szModuleFileName[MAX_PATH] = "";
|
|
|
|
// Call IMMCRemote methods: ObjectIsRemote and SetMMCExePath so that the
|
|
// snap-in will know it is remote and so that it will have MMC.EXE's full
|
|
// path in order to build taskpad display strings.
|
|
|
|
hr = This->lpVtbl->QueryInterface(This, &IID_IMMCRemote,
|
|
(void **)&piMMCRemote);
|
|
if (FAILED(hr))
|
|
{
|
|
// If the object doesn't support IMMCRemote that is not an error.
|
|
// The designer runtime will get this QI on both its main object and
|
|
// its IComponent object but only the main object needs to support the
|
|
// interface.
|
|
hr = S_OK;
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = piMMCRemote->lpVtbl->ObjectIsRemote(piMMCRemote);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
cbFileName = GetModuleFileName(NULL, // get executable that loaded us (MMC)
|
|
szModuleFileName,
|
|
sizeof(szModuleFileName));
|
|
|
|
if (0 == cbFileName)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = piMMCRemote->lpVtbl->SetMMCExePath(piMMCRemote, szModuleFileName);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = piMMCRemote->lpVtbl->SetMMCCommandLine(piMMCRemote, GetCommandLine());
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
Cleanup:
|
|
if (NULL != piMMCRemote)
|
|
{
|
|
piMMCRemote->lpVtbl->Release(piMMCRemote);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbar_SetControlbar_Proxy
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
LPCONTROLBAR pControlbar
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// Make sure the snap-in knows we are remoted. We do this here because
|
|
// this is the first opportunity for the proxy to inform a toolbar
|
|
// extension that it is remote.
|
|
|
|
hr = SetRemote((IUnknown *)This);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = IExtendControlbar_RemSetControlbar_Proxy(This, pControlbar);
|
|
|
|
Cleanup:
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbar_SetControlbar_Stub
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
LPCONTROLBAR pControlbar
|
|
)
|
|
{
|
|
return This->lpVtbl->SetControlbar(This, pControlbar);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbar_ControlbarNotify_Proxy
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
MMC_NOTIFY_TYPE event,
|
|
LPARAM arg,
|
|
LPARAM param
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// If this is not a menu button click then we can use the generated remoting
|
|
// code with the arg and param unions
|
|
|
|
if (MMCN_MENU_BTNCLICK == event)
|
|
{
|
|
hr = MenuButtonClickProxy(This,
|
|
(IDataObject *)arg,
|
|
(MENUBUTTONDATA *)param);
|
|
goto Cleanup;
|
|
}
|
|
// Get any IDataObject associated with the event
|
|
|
|
switch (event)
|
|
{
|
|
case MMCN_SELECT:
|
|
piDataObject = (IDataObject *)param;
|
|
break;
|
|
|
|
case MMCN_BTN_CLICK:
|
|
piDataObject = (IDataObject *)arg;
|
|
break;
|
|
|
|
default:
|
|
piDataObject = NULL;
|
|
break;
|
|
}
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects,
|
|
&ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendControlbar_RemControlbarNotify_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
event, arg, param);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbar_ControlbarNotify_Stub
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject **ppiDataObjects,
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
MMC_NOTIFY_TYPE event,
|
|
LPARAM arg,
|
|
LPARAM param
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Put the IDataObject into the corresponding parameter for the event
|
|
|
|
switch (event)
|
|
{
|
|
case MMCN_SELECT:
|
|
param = (LPARAM)piDataObject;
|
|
break;
|
|
|
|
case MMCN_BTN_CLICK:
|
|
arg = (LPARAM)piDataObject;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// Call into the snap-in with all parameters appearing as they would
|
|
// when in-proc.
|
|
|
|
hr = This->lpVtbl->ControlbarNotify(This, event, arg, param);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
static HRESULT MenuButtonClickProxy
|
|
(
|
|
IExtendControlbar __RPC_FAR *This,
|
|
IDataObject *piDataObject,
|
|
MENUBUTTONDATA *pMenuButtonData
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
POPUP_MENUDEF *pPopupMenuDef = NULL;
|
|
HMENU hMenu = NULL;
|
|
UINT uiSelectedItemID = 0;
|
|
IExtendControlbarRemote *piECRemote = NULL;
|
|
long i = 0;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
|
|
|
|
// The generated remoting cannot easily handle what we need to do so we get
|
|
// IExtendControlbarRemote on the snap-in. This interface has methods that
|
|
// allow us to ask the snap-in for its popup menu items, display the menu
|
|
// here on the MMC side, and then tell the snap-in which item was selected.
|
|
|
|
hr = This->lpVtbl->QueryInterface(This, &IID_IExtendControlbarRemote,
|
|
(void **)&piECRemote);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Tell the snap-in about the menu button click and get back its list of
|
|
// popup menu items.
|
|
|
|
hr = piECRemote->lpVtbl->MenuButtonClick(piECRemote,
|
|
piDataObject,
|
|
pMenuButtonData->idCommand,
|
|
&pPopupMenuDef);
|
|
|
|
if ( FAILED(hr) || (NULL == pPopupMenuDef) )
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Create an empty Win32 menu
|
|
|
|
hMenu = CreatePopupMenu();
|
|
if (NULL == hMenu)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Iterate through each of the items and add them to the menu
|
|
|
|
for (i = 0; i < pPopupMenuDef->cMenuItems; i++)
|
|
{
|
|
if (!AppendMenu(hMenu,
|
|
pPopupMenuDef->MenuItems[i].uiFlags,
|
|
pPopupMenuDef->MenuItems[i].uiItemID,
|
|
pPopupMenuDef->MenuItems[i].pszItemText))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
// If the owner HWND is NULL then this is an extension and it does not have
|
|
// access to IConsole2 on MMC to get the main frame HWND. In this case just
|
|
// use the active window on this thread.
|
|
|
|
if (NULL == pPopupMenuDef->hwndMenuOwner)
|
|
{
|
|
pPopupMenuDef->hwndMenuOwner = GetActiveWindow();
|
|
}
|
|
|
|
// Display the popup menu and wait for a selection.
|
|
|
|
uiSelectedItemID = (UINT)TrackPopupMenu(
|
|
hMenu, // menu to display
|
|
TPM_LEFTALIGN | // align left side of menu with x
|
|
TPM_TOPALIGN | // align top of menu with y
|
|
TPM_NONOTIFY | // don't send any messages during selection
|
|
TPM_RETURNCMD | // make the ret val the selected item
|
|
TPM_LEFTBUTTON, // allow selection with left button only
|
|
pMenuButtonData->x, // left side coordinate
|
|
pMenuButtonData->y, // top coordinate
|
|
0, // reserved,
|
|
pPopupMenuDef->hwndMenuOwner, // owner window, this comes from snap-in
|
|
// as it can call IConsole2->GetMainWindow
|
|
NULL); // not used
|
|
|
|
// A zero return could indicate either an error or that the user hit
|
|
// Escape or clicked off of the menu to cancel the operation. GetLastError()
|
|
// determines whether there was an error. Either way we're done but set the
|
|
// hr first.
|
|
|
|
if (0 == uiSelectedItemID)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If i is non-zero then it contains the ID of the selected item.
|
|
// Tell the snap-in what was selected and pass it the extra IUnknown it
|
|
// included in its menu definition (this is snap-in defined and it allows
|
|
// the snap-in to include some more identifying information to handle the
|
|
// event).
|
|
|
|
if (0 != uiSelectedItemID)
|
|
{
|
|
hr = piECRemote->lpVtbl->PopupMenuClick(
|
|
piECRemote,
|
|
piDataObject,
|
|
uiSelectedItemID,
|
|
pPopupMenuDef->punkSnapInDefined);
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
if (NULL != piECRemote)
|
|
{
|
|
piECRemote->lpVtbl->Release(piECRemote);
|
|
}
|
|
if (NULL != hMenu)
|
|
{
|
|
(void)DestroyMenu(hMenu);
|
|
}
|
|
|
|
if (NULL != pPopupMenuDef)
|
|
{
|
|
for (i = 0; i < pPopupMenuDef->cMenuItems; i++)
|
|
{
|
|
if (NULL != pPopupMenuDef->MenuItems[i].pszItemText)
|
|
{
|
|
CoTaskMemFree(pPopupMenuDef->MenuItems[i].pszItemText);
|
|
}
|
|
}
|
|
if (NULL != pPopupMenuDef->punkSnapInDefined)
|
|
{
|
|
pPopupMenuDef->punkSnapInDefined->lpVtbl->Release(pPopupMenuDef->punkSnapInDefined);
|
|
}
|
|
CoTaskMemFree(pPopupMenuDef);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbarRemote_MenuButtonClick_Proxy
|
|
(
|
|
IExtendControlbarRemote __RPC_FAR *This,
|
|
IDataObject __RPC_FAR *piDataObject,
|
|
int idCommand,
|
|
POPUP_MENUDEF __RPC_FAR *__RPC_FAR *ppPopupMenuDef
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendControlbarRemote_RemMenuButtonClick_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
idCommand,
|
|
ppPopupMenuDef);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbarRemote_MenuButtonClick_Stub
|
|
(
|
|
IExtendControlbarRemote __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
int idCommand,
|
|
POPUP_MENUDEF __RPC_FAR *__RPC_FAR *ppPopupMenuDef
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->MenuButtonClick(This, piDataObject,
|
|
idCommand, ppPopupMenuDef);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbarRemote_PopupMenuClick_Proxy
|
|
(
|
|
IExtendControlbarRemote __RPC_FAR *This,
|
|
IDataObject __RPC_FAR *piDataObject,
|
|
UINT uIDItem,
|
|
IUnknown __RPC_FAR *punkParam
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendControlbarRemote_RemPopupMenuClick_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
uIDItem,
|
|
punkParam);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendControlbarRemote_PopupMenuClick_Stub
|
|
(
|
|
IExtendControlbarRemote __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
UINT uIDItem,
|
|
IUnknown __RPC_FAR *punkParam
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->PopupMenuClick(This, piDataObject, uIDItem, punkParam);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_Initialize_Proxy
|
|
(
|
|
IComponentData *This,
|
|
LPUNKNOWN pUnknown
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// Tell the object it is remote and give the path to mmc.exe
|
|
|
|
hr = SetRemote((IUnknown *)This);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Now pass on the Initiaize call normally. Using this order allows a snap-in
|
|
// to know it is remote prior to its IComponentData::Initialize in case it
|
|
// needs that information up front.
|
|
|
|
hr = IComponentData_RemInitialize_Proxy(This, pUnknown);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_Initialize_Stub
|
|
(
|
|
IComponentData *This,
|
|
LPUNKNOWN pUnknown
|
|
)
|
|
{
|
|
return This->lpVtbl->Initialize(This, pUnknown);
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_CreateComponent_Proxy
|
|
(
|
|
IComponentData *This,
|
|
LPCOMPONENT *ppComponent
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// Tell the object it is remote and give the path to mmc.exe
|
|
|
|
hr = SetRemote((IUnknown *)This);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Now pass on the CreateComponent call normally. Using this order allows a
|
|
// snap-in to know it is remote prior to its
|
|
// IComponentData::CreateComponent in case it needs that information up
|
|
// front.
|
|
|
|
// We do this in IComponentData::Initialize and
|
|
// IComponentData::CreateComponent. Most cases will use Initialize but in
|
|
// MMC 1.1 a taskpad extension does not receive IComponentData::Initialize.
|
|
// MMC only calls IComponentData::CreateComponent. As a taskpad extension
|
|
// may need to resolve a res:// URL to use the mmc.exe path we need to do
|
|
// it here as well.
|
|
|
|
hr = IComponentData_RemCreateComponent_Proxy(This, ppComponent);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_CreateComponent_Stub
|
|
(
|
|
IComponentData *This,
|
|
LPCOMPONENT *ppComponent
|
|
)
|
|
{
|
|
return This->lpVtbl->CreateComponent(This, ppComponent);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_Notify_Proxy
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
LPDATAOBJECT piDataObject,
|
|
MMC_NOTIFY_TYPE event,
|
|
LPARAM arg,
|
|
LPARAM param
|
|
)
|
|
{
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
ICDNotifyParam ParamUnion;
|
|
ZeroMemory(&ParamUnion, sizeof(ParamUnion));
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
ParamUnion.value = param;
|
|
return IComponentData_RemNotify_Proxy(This, piDataObject,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
event, arg, &ParamUnion);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_Notify_Stub
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
LPDATAOBJECT piDataObject,
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
MMC_NOTIFY_TYPE event,
|
|
LPARAM arg,
|
|
ICDNotifyParam *pParamUnion
|
|
)
|
|
{
|
|
if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
return This->lpVtbl->Notify(This, piDataObject,
|
|
event, arg, pParamUnion->value);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_CompareObjects_Proxy
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
IDataObject *piDataObjectA,
|
|
IDataObject *piDataObjectB
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
BOOL fIsMultiSelectA = FALSE;
|
|
long cDataObjectsA = 1L;
|
|
IDataObject **ppiDataObjectsA = NULL;
|
|
BOOL fSpecialDataObjectA = FALSE;
|
|
long lSpecialDataObjectA = 0;
|
|
|
|
BOOL fIsMultiSelectB = FALSE;
|
|
long cDataObjectsB = 1L;
|
|
IDataObject **ppiDataObjectsB = NULL;
|
|
BOOL fSpecialDataObjectB = FALSE;
|
|
long lSpecialDataObjectB = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObjectA, &fSpecialDataObjectA, &lSpecialDataObjectA);
|
|
|
|
CheckForSpecialDataObjects(&piDataObjectB, &fSpecialDataObjectB, &lSpecialDataObjectB);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObjectA)
|
|
{
|
|
hr = IsMultiSelect(piDataObjectA, &fIsMultiSelectA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (!fSpecialDataObjectB)
|
|
{
|
|
hr = IsMultiSelect(piDataObjectB, &fIsMultiSelectB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelectA)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObjectA, &cDataObjectsA, &ppiDataObjectsA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjectsA = &piDataObjectA;
|
|
}
|
|
|
|
if (fIsMultiSelectB)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObjectB, &cDataObjectsB, &ppiDataObjectsB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjectsB = &piDataObjectB;
|
|
}
|
|
|
|
hr = IComponentData_RemCompareObjects_Proxy(This,
|
|
cDataObjectsA,
|
|
ppiDataObjectsA,
|
|
fSpecialDataObjectA,
|
|
lSpecialDataObjectA,
|
|
|
|
cDataObjectsB,
|
|
ppiDataObjectsB,
|
|
fSpecialDataObjectB,
|
|
lSpecialDataObjectB);
|
|
Cleanup:
|
|
if ( fIsMultiSelectA && (NULL != ppiDataObjectsA) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjectsA);
|
|
}
|
|
if ( fIsMultiSelectB && (NULL != ppiDataObjectsB) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjectsB);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_CompareObjects_Stub
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
|
|
long cDataObjectsA,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjectsA[ ],
|
|
BOOL fSpecialDataObjectA,
|
|
long lSpecialDataObjectA,
|
|
|
|
long cDataObjectsB,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjectsB[ ],
|
|
BOOL fSpecialDataObjectB,
|
|
long lSpecialDataObjectB
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObjectA = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObjectA = NULL;
|
|
IDataObject *piDataObjectB = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObjectB = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjectsA > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjectsA, cDataObjectsA,
|
|
&piMultiSelDataObjectA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObjectA = piMultiSelDataObjectA;
|
|
}
|
|
else if (fSpecialDataObjectA)
|
|
{
|
|
piDataObjectA = (IDataObject *)lSpecialDataObjectA;
|
|
}
|
|
else
|
|
{
|
|
piDataObjectA = ppiDataObjectsA[0];
|
|
}
|
|
|
|
if (cDataObjectsB > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjectsB, cDataObjectsB,
|
|
&piMultiSelDataObjectB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObjectB = piMultiSelDataObjectB;
|
|
}
|
|
else if (fSpecialDataObjectB)
|
|
{
|
|
piDataObjectB = (IDataObject *)lSpecialDataObjectB;
|
|
}
|
|
else
|
|
{
|
|
piDataObjectB = ppiDataObjectsB[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->CompareObjects(This, piDataObjectA, piDataObjectB);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObjectA)
|
|
{
|
|
piMultiSelDataObjectA->lpVtbl->Release(piMultiSelDataObjectA);
|
|
}
|
|
if (NULL != piMultiSelDataObjectB)
|
|
{
|
|
piMultiSelDataObjectB->lpVtbl->Release(piMultiSelDataObjectB);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_Notify_Proxy
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
LPDATAOBJECT piDataObject,
|
|
MMC_NOTIFY_TYPE event,
|
|
LPARAM arg,
|
|
LPARAM param
|
|
)
|
|
{
|
|
ICNotifyArg ArgUnion;
|
|
ICNotifyParam ParamUnion;
|
|
ICOutParam *pOutParam = NULL;
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
ZeroMemory(&ArgUnion, sizeof(ArgUnion));
|
|
ZeroMemory(&ParamUnion, sizeof(ParamUnion));
|
|
|
|
// Switch any potential multiselect data objects with arg/param so that
|
|
// piDataObject always contains the potential multiselect.
|
|
|
|
switch (event)
|
|
{
|
|
case MMCN_QUERY_PASTE:
|
|
ArgUnion.pidoQueryPasteTarget = piDataObject;
|
|
piDataObject = (IDataObject *)arg;
|
|
ParamUnion.value = param;
|
|
break;
|
|
|
|
case MMCN_PASTE:
|
|
ArgUnion.pidoPasteTarget = piDataObject;
|
|
piDataObject = (IDataObject *)arg;
|
|
// Pass through param as an LPARAM rather than the IDataObject **
|
|
// it really is. This is just to let the stub know whether it is
|
|
// a copy or a move. If it is a move the CUTORMOVE IDataObject will
|
|
// be in the ICOutParam returned from the stub.
|
|
ParamUnion.value = param;
|
|
break;
|
|
|
|
case MMCN_RESTORE_VIEW:
|
|
ArgUnion.value = arg;
|
|
// Don't pass param because it is a BOOL * that will not be
|
|
// marshaled. The BOOL will be received in the ICOutParam returned
|
|
// from the stub.
|
|
break;
|
|
|
|
default:
|
|
ArgUnion.value = arg;
|
|
ParamUnion.value = param;
|
|
}
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IComponent_RemNotify_Proxy(This,
|
|
cDataObjects, ppiDataObjects,
|
|
fSpecialDataObject, lSpecialDataObject,
|
|
event, &ArgUnion, &ParamUnion, &pOutParam);
|
|
Cleanup:
|
|
if (NULL != pOutParam)
|
|
{
|
|
if (MMCN_PASTE == event)
|
|
{
|
|
*((IDataObject **)param) = pOutParam->pidoCutOrMove;
|
|
}
|
|
else if (MMCN_RESTORE_VIEW == event)
|
|
{
|
|
*((BOOL *)param) = pOutParam->fRestoreHandled;
|
|
}
|
|
CoTaskMemFree(pOutParam);
|
|
}
|
|
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_Notify_Stub
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject **ppiDataObjects,
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
MMC_NOTIFY_TYPE event,
|
|
ICNotifyArg *pArgUnion,
|
|
ICNotifyParam *pParamUnion,
|
|
ICOutParam **ppOutParam
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
LPARAM Arg = 0;
|
|
LPARAM Param = 0;
|
|
|
|
*ppOutParam = (ICOutParam *)CoTaskMemAlloc(sizeof(ICOutParam));
|
|
if (NULL == *ppOutParam)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
ZeroMemory(*ppOutParam, sizeof(ICOutParam));
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// If the event required swaping Arg and IDataObject then swap it back
|
|
// before calling into the object. For MMCN_QUERY_PASTE and
|
|
// MMCN_RESTORE_VIEW we need to make Param contain the out pointer.
|
|
|
|
switch (event)
|
|
{
|
|
case MMCN_PASTE:
|
|
Arg = (LPARAM)piDataObject;
|
|
piDataObject = pArgUnion->pidoPasteTarget;
|
|
if (0 == pParamUnion->value)
|
|
{
|
|
// This is a copy, pass zero in Param so snap-in will know that
|
|
Param = 0;
|
|
}
|
|
else
|
|
{
|
|
// This is a move. Pass the address of the IDataObject in
|
|
// the ICOutParam that we will return to the proxy.
|
|
Param = (LPARAM)&((*ppOutParam)->pidoCutOrMove);
|
|
}
|
|
break;
|
|
|
|
case MMCN_QUERY_PASTE:
|
|
Arg = (LPARAM)piDataObject;
|
|
piDataObject = pArgUnion->pidoQueryPasteTarget;
|
|
Param = pParamUnion->value;
|
|
break;
|
|
|
|
case MMCN_RESTORE_VIEW:
|
|
Arg = pArgUnion->value;
|
|
Param = (LPARAM)&((*ppOutParam)->fRestoreHandled);
|
|
break;
|
|
|
|
default:
|
|
Arg = pArgUnion->value;
|
|
Param = pParamUnion->value;
|
|
break;
|
|
}
|
|
|
|
hr = This->lpVtbl->Notify(This, piDataObject, event, Arg, Param);
|
|
|
|
Cleanup:
|
|
if (FAILED(hr))
|
|
{
|
|
if (NULL != *ppOutParam)
|
|
{
|
|
CoTaskMemFree(*ppOutParam);
|
|
*ppOutParam = NULL;
|
|
}
|
|
}
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_CompareObjects_Proxy
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
IDataObject *piDataObjectA,
|
|
IDataObject *piDataObjectB
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
BOOL fIsMultiSelectA = FALSE;
|
|
long cDataObjectsA = 1L;
|
|
IDataObject **ppiDataObjectsA = NULL;
|
|
BOOL fSpecialDataObjectA = FALSE;
|
|
long lSpecialDataObjectA = 0;
|
|
|
|
BOOL fIsMultiSelectB = FALSE;
|
|
long cDataObjectsB = 1L;
|
|
IDataObject **ppiDataObjectsB = NULL;
|
|
BOOL fSpecialDataObjectB = FALSE;
|
|
long lSpecialDataObjectB = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObjectA, &fSpecialDataObjectA, &lSpecialDataObjectA);
|
|
|
|
CheckForSpecialDataObjects(&piDataObjectB, &fSpecialDataObjectB, &lSpecialDataObjectB);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObjectA)
|
|
{
|
|
hr = IsMultiSelect(piDataObjectA, &fIsMultiSelectA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (!fSpecialDataObjectB)
|
|
{
|
|
hr = IsMultiSelect(piDataObjectB, &fIsMultiSelectB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelectA)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObjectA, &cDataObjectsA, &ppiDataObjectsA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjectsA = &piDataObjectA;
|
|
}
|
|
|
|
if (fIsMultiSelectB)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObjectB, &cDataObjectsB, &ppiDataObjectsB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjectsB = &piDataObjectB;
|
|
}
|
|
|
|
hr = IComponent_RemCompareObjects_Proxy(This,
|
|
cDataObjectsA,
|
|
ppiDataObjectsA,
|
|
fSpecialDataObjectA,
|
|
lSpecialDataObjectA,
|
|
|
|
cDataObjectsB,
|
|
ppiDataObjectsB,
|
|
fSpecialDataObjectB,
|
|
lSpecialDataObjectB);
|
|
Cleanup:
|
|
if ( fIsMultiSelectA && (NULL != ppiDataObjectsA) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjectsA);
|
|
}
|
|
if ( fIsMultiSelectB && (NULL != ppiDataObjectsB) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjectsB);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_CompareObjects_Stub
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
|
|
long cDataObjectsA,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjectsA[ ],
|
|
BOOL fSpecialDataObjectA,
|
|
long lSpecialDataObjectA,
|
|
|
|
long cDataObjectsB,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjectsB[ ],
|
|
BOOL fSpecialDataObjectB,
|
|
long lSpecialDataObjectB
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObjectA = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObjectA = NULL;
|
|
IDataObject *piDataObjectB = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObjectB = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjectsA > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjectsA, cDataObjectsA,
|
|
&piMultiSelDataObjectA);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObjectA = piMultiSelDataObjectA;
|
|
}
|
|
else if (fSpecialDataObjectA)
|
|
{
|
|
piDataObjectA = (IDataObject *)lSpecialDataObjectA;
|
|
}
|
|
else
|
|
{
|
|
piDataObjectA = ppiDataObjectsA[0];
|
|
}
|
|
|
|
if (cDataObjectsB > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjectsB, cDataObjectsB,
|
|
&piMultiSelDataObjectB);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObjectB = piMultiSelDataObjectB;
|
|
}
|
|
else if (fSpecialDataObjectB)
|
|
{
|
|
piDataObjectB = (IDataObject *)lSpecialDataObjectB;
|
|
}
|
|
else
|
|
{
|
|
piDataObjectB = ppiDataObjectsB[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->CompareObjects(This, piDataObjectA, piDataObjectB);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObjectA)
|
|
{
|
|
piMultiSelDataObjectA->lpVtbl->Release(piMultiSelDataObjectA);
|
|
}
|
|
if (NULL != piMultiSelDataObjectB)
|
|
{
|
|
piMultiSelDataObjectB->lpVtbl->Release(piMultiSelDataObjectB);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// SCOPEDATAITEM Marshaling
|
|
//
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
// Caveat: When returning a string in SCOPEDATAITEM MMC does not use a
|
|
// callee allocate/caller free strategy. When in-proc, the owner of the memory
|
|
// must insure that it stays alive for as long as the caller is expected to
|
|
// use it (e.g. scope item display name must remain valid for the life of the
|
|
// scope item). When out-of-proc, that returned string will be allocated by
|
|
// the proxy using CoTaskMemAlloc() and it will never be freed so there will
|
|
// be some leaks.
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
static void SCOPEDATAITEM_TO_WIRE
|
|
(
|
|
SCOPEDATAITEM *psdi,
|
|
WIRE_SCOPEDATAITEM *pwsdi
|
|
)
|
|
{
|
|
pwsdi->mask = psdi->mask;
|
|
pwsdi->nImage = psdi->nImage;
|
|
pwsdi->nOpenImage = psdi->nOpenImage;
|
|
pwsdi->nState = psdi->nState;
|
|
pwsdi->cChildren = psdi->cChildren;
|
|
pwsdi->lParam = psdi->lParam;
|
|
pwsdi->relativeID = psdi->relativeID;
|
|
pwsdi->ID = psdi->ID;
|
|
|
|
if ( SDI_STR != (psdi->mask & SDI_STR) )
|
|
{
|
|
pwsdi->pwszDisplayName = NULL;
|
|
pwsdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
else if (MMC_CALLBACK == psdi->displayname)
|
|
{
|
|
pwsdi->pwszDisplayName = NULL;
|
|
pwsdi->fUsingCallbackForString = TRUE;
|
|
}
|
|
else if (NULL == psdi->displayname)
|
|
{
|
|
pwsdi->pwszDisplayName = NULL;
|
|
pwsdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
else
|
|
{
|
|
// A string is being passed. Need to CoTaskMemAlloc() it so that
|
|
// the MIDL generated stub can free it after transmission
|
|
|
|
int cbString = (lstrlenW(psdi->displayname) + 1) * sizeof(psdi->displayname[0]);
|
|
|
|
pwsdi->pwszDisplayName = (LPOLESTR)CoTaskMemAlloc(cbString);
|
|
if (NULL == pwsdi->pwszDisplayName)
|
|
{
|
|
RpcRaiseException( E_OUTOFMEMORY );
|
|
}
|
|
else
|
|
{
|
|
memcpy(pwsdi->pwszDisplayName, psdi->displayname, cbString);
|
|
}
|
|
pwsdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
static void WIRE_TO_SCOPEDATAITEM
|
|
(
|
|
WIRE_SCOPEDATAITEM *pwsdi,
|
|
SCOPEDATAITEM *psdi
|
|
)
|
|
{
|
|
psdi->mask = pwsdi->mask;
|
|
psdi->nImage = pwsdi->nImage;
|
|
psdi->nOpenImage = pwsdi->nOpenImage;
|
|
psdi->nState = pwsdi->nState;
|
|
psdi->cChildren = pwsdi->cChildren;
|
|
psdi->lParam = pwsdi->lParam;
|
|
psdi->relativeID = pwsdi->relativeID;
|
|
psdi->ID = pwsdi->ID;
|
|
|
|
if ( SDI_STR != (psdi->mask & SDI_STR) )
|
|
{
|
|
psdi->displayname = NULL;
|
|
}
|
|
else if (pwsdi->fUsingCallbackForString)
|
|
{
|
|
psdi->displayname = MMC_CALLBACK;
|
|
}
|
|
else
|
|
{
|
|
psdi->displayname = pwsdi->pwszDisplayName;
|
|
}
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_GetDisplayInfo_Proxy
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
SCOPEDATAITEM __RPC_FAR *pScopeDataItem
|
|
)
|
|
{
|
|
WIRE_SCOPEDATAITEM wsdi;
|
|
HRESULT hr;
|
|
|
|
// Make sure the string pointer is NULL so that it is not marshaled as it
|
|
// is never passed from MMC to the snap-in. (MMC might not have initialized
|
|
// the pointer).
|
|
|
|
pScopeDataItem->displayname = NULL;
|
|
|
|
SCOPEDATAITEM_TO_WIRE(pScopeDataItem, &wsdi);
|
|
|
|
hr = IComponentData_RemGetDisplayInfo_Proxy(This, &wsdi);
|
|
WIRE_TO_SCOPEDATAITEM(&wsdi, pScopeDataItem);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponentData_GetDisplayInfo_Stub
|
|
(
|
|
IComponentData __RPC_FAR *This,
|
|
WIRE_SCOPEDATAITEM __RPC_FAR *pwsdi
|
|
)
|
|
{
|
|
SCOPEDATAITEM sdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_SCOPEDATAITEM(pwsdi, &sdi);
|
|
hr = This->lpVtbl->GetDisplayInfo(This, &sdi);
|
|
SCOPEDATAITEM_TO_WIRE(&sdi, pwsdi);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_InsertItem_Proxy
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
LPSCOPEDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_SCOPEDATAITEM wsdi;
|
|
HRESULT hr;
|
|
HSCOPEITEM ItemID;
|
|
|
|
SCOPEDATAITEM_TO_WIRE(pItem, &wsdi);
|
|
hr = IConsoleNameSpace_RemInsertItem_Proxy(This, &wsdi, &ItemID);
|
|
|
|
// The only returned field is the item ID so copy it from the wire
|
|
// structure to the client structure
|
|
|
|
pItem->ID = ItemID;
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_InsertItem_Stub
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
WIRE_SCOPEDATAITEM __RPC_FAR *pwsdi,
|
|
HSCOPEITEM __RPC_FAR *pItemID
|
|
)
|
|
{
|
|
SCOPEDATAITEM sdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_SCOPEDATAITEM(pwsdi, &sdi);
|
|
hr = This->lpVtbl->InsertItem(This, &sdi);
|
|
|
|
// The only returned field is the itemID.
|
|
|
|
*pItemID = sdi.ID;
|
|
return hr;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_SetItem_Proxy
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
LPSCOPEDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_SCOPEDATAITEM wsdi;
|
|
|
|
SCOPEDATAITEM_TO_WIRE(pItem, &wsdi);
|
|
return IConsoleNameSpace_RemSetItem_Proxy(This, &wsdi);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_SetItem_Stub
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
WIRE_SCOPEDATAITEM __RPC_FAR *pwsdi
|
|
)
|
|
{
|
|
SCOPEDATAITEM sdi;
|
|
|
|
WIRE_TO_SCOPEDATAITEM(pwsdi, &sdi);
|
|
return This->lpVtbl->SetItem(This, &sdi);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_GetItem_Proxy
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
LPSCOPEDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_SCOPEDATAITEM wsdi;
|
|
HRESULT hr;
|
|
|
|
// Make sure the string pointer is NULL so that it is not marshaled as it
|
|
// is never passed from the snap-in to MMC. (It might not have be
|
|
// initialized).
|
|
|
|
pItem->displayname = NULL;
|
|
|
|
SCOPEDATAITEM_TO_WIRE(pItem, &wsdi);
|
|
hr = IConsoleNameSpace_RemGetItem_Proxy(This, &wsdi);
|
|
WIRE_TO_SCOPEDATAITEM(&wsdi, pItem);
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IConsoleNameSpace_GetItem_Stub
|
|
(
|
|
IConsoleNameSpace __RPC_FAR *This,
|
|
WIRE_SCOPEDATAITEM __RPC_FAR *pwsdi
|
|
)
|
|
{
|
|
SCOPEDATAITEM sdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_SCOPEDATAITEM(pwsdi, &sdi);
|
|
hr = This->lpVtbl->GetItem(This, &sdi);
|
|
SCOPEDATAITEM_TO_WIRE(&sdi, pwsdi);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// RESULTDATAITEM Marshaling
|
|
//
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
// Caveat: When returning a string in RESULTDATAITEM MMC does not use a
|
|
// callee allocate/caller free strategy. When in-proc, the owner of the memory
|
|
// must insure that it stays alive for as long as the caller is expected to
|
|
// use it (e.g. list item column data must remain valid for the life of the
|
|
// list item). When out-of-proc, that returned string will be allocated by
|
|
// the proxy using SysAllocString() and it will never be freed so there will
|
|
// be some leaks.
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
static void RESULTDATAITEM_TO_WIRE
|
|
(
|
|
RESULTDATAITEM *prdi,
|
|
WIRE_RESULTDATAITEM *pwrdi
|
|
)
|
|
{
|
|
pwrdi->mask = prdi->mask;
|
|
pwrdi->bScopeItem = prdi->bScopeItem;
|
|
pwrdi->itemID = prdi->itemID;
|
|
pwrdi->nIndex = prdi->nIndex;
|
|
pwrdi->nCol = prdi->nCol;
|
|
pwrdi->nImage = prdi->nImage;
|
|
pwrdi->nState = prdi->nState;
|
|
pwrdi->lParam = prdi->lParam;
|
|
pwrdi->iIndent = prdi->iIndent;
|
|
|
|
if ( RDI_STR != (prdi->mask & RDI_STR) )
|
|
{
|
|
pwrdi->str = NULL;
|
|
pwrdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
else if (MMC_CALLBACK == prdi->str)
|
|
{
|
|
pwrdi->str = NULL;
|
|
pwrdi->fUsingCallbackForString = TRUE;
|
|
}
|
|
else if (NULL == prdi->str)
|
|
{
|
|
pwrdi->str = NULL;
|
|
pwrdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
else
|
|
{
|
|
// A string is being passed. Need to CoTaskMemAlloc() it so that
|
|
// the MIDL generated stub can free it after transmission
|
|
|
|
int cbString = (lstrlenW(prdi->str) + 1) * sizeof(prdi->str[0]);
|
|
|
|
pwrdi->str = (LPOLESTR)CoTaskMemAlloc(cbString);
|
|
if (NULL == pwrdi->str)
|
|
{
|
|
RpcRaiseException( E_OUTOFMEMORY );
|
|
}
|
|
else
|
|
{
|
|
memcpy(pwrdi->str, prdi->str, cbString);
|
|
}
|
|
pwrdi->fUsingCallbackForString = FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static void WIRE_TO_RESULTDATAITEM
|
|
(
|
|
WIRE_RESULTDATAITEM *pwrdi,
|
|
RESULTDATAITEM *prdi
|
|
)
|
|
{
|
|
prdi->mask = pwrdi->mask;
|
|
prdi->bScopeItem = pwrdi->bScopeItem;
|
|
prdi->itemID = pwrdi->itemID;
|
|
prdi->nIndex = pwrdi->nIndex;
|
|
prdi->nCol = pwrdi->nCol;
|
|
prdi->nImage = pwrdi->nImage;
|
|
prdi->nState = pwrdi->nState;
|
|
prdi->lParam = pwrdi->lParam;
|
|
prdi->iIndent = pwrdi->iIndent;
|
|
|
|
if ( RDI_STR != (prdi->mask & RDI_STR) )
|
|
{
|
|
prdi->str = NULL;
|
|
}
|
|
else if (pwrdi->fUsingCallbackForString)
|
|
{
|
|
prdi->str = MMC_CALLBACK;
|
|
}
|
|
else
|
|
{
|
|
prdi->str = pwrdi->str;
|
|
}
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_GetDisplayInfo_Proxy
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
RESULTDATAITEM __RPC_FAR *pResultDataItem
|
|
)
|
|
{
|
|
WIRE_RESULTDATAITEM wrdi;
|
|
HRESULT hr;
|
|
|
|
// Make sure the string pointer is NULL so that it is not marshaled as it
|
|
// is never passed from MMC to the snap-in. (MMC might not have initialized
|
|
// the pointer).
|
|
|
|
pResultDataItem->str = NULL;
|
|
|
|
RESULTDATAITEM_TO_WIRE(pResultDataItem, &wrdi);
|
|
|
|
hr = IComponent_RemGetDisplayInfo_Proxy(This, &wrdi);
|
|
WIRE_TO_RESULTDATAITEM(&wrdi, pResultDataItem);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IComponent_GetDisplayInfo_Stub
|
|
(
|
|
IComponent __RPC_FAR *This,
|
|
WIRE_RESULTDATAITEM __RPC_FAR *pwrdi
|
|
)
|
|
{
|
|
RESULTDATAITEM rdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_RESULTDATAITEM(pwrdi, &rdi);
|
|
hr = This->lpVtbl->GetDisplayInfo(This, &rdi);
|
|
RESULTDATAITEM_TO_WIRE(&rdi, pwrdi);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_InsertItem_Proxy
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
LPRESULTDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_RESULTDATAITEM wrdi;
|
|
HRESULT hr;
|
|
HRESULTITEM ItemID;
|
|
|
|
RESULTDATAITEM_TO_WIRE(pItem, &wrdi);
|
|
hr = IResultData_RemInsertItem_Proxy(This, &wrdi, &ItemID);
|
|
|
|
// The only returned field is the itemID so copy it from the wire
|
|
// structure to the client structure
|
|
|
|
pItem->itemID = ItemID;
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_InsertItem_Stub
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
WIRE_RESULTDATAITEM __RPC_FAR *pwrdi,
|
|
HRESULTITEM __RPC_FAR *pItemID
|
|
)
|
|
{
|
|
RESULTDATAITEM rdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_RESULTDATAITEM(pwrdi, &rdi);
|
|
hr = This->lpVtbl->InsertItem(This, &rdi);
|
|
|
|
// The only returned field is the itemID.
|
|
|
|
*pItemID = rdi.itemID;
|
|
return hr;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_SetItem_Proxy
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
LPRESULTDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_RESULTDATAITEM wrdi;
|
|
|
|
RESULTDATAITEM_TO_WIRE(pItem, &wrdi);
|
|
return IResultData_RemSetItem_Proxy(This, &wrdi);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_SetItem_Stub
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
WIRE_RESULTDATAITEM __RPC_FAR *pwrdi
|
|
)
|
|
{
|
|
RESULTDATAITEM rdi;
|
|
|
|
WIRE_TO_RESULTDATAITEM(pwrdi, &rdi);
|
|
return This->lpVtbl->SetItem(This, &rdi);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_GetItem_Proxy
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
LPRESULTDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_RESULTDATAITEM wrdi;
|
|
HRESULT hr;
|
|
|
|
// Make sure the string pointer is NULL so that it is not marshaled as it
|
|
// is never passed from the snap-in to MMC. (It might not have be
|
|
// initialized).
|
|
|
|
pItem->str = NULL;
|
|
|
|
RESULTDATAITEM_TO_WIRE(pItem, &wrdi);
|
|
hr = IResultData_RemGetItem_Proxy(This, &wrdi);
|
|
WIRE_TO_RESULTDATAITEM(&wrdi, pItem);
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_GetItem_Stub
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
WIRE_RESULTDATAITEM __RPC_FAR *pwrdi
|
|
)
|
|
{
|
|
RESULTDATAITEM rdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_RESULTDATAITEM(pwrdi, &rdi);
|
|
hr = This->lpVtbl->GetItem(This, &rdi);
|
|
RESULTDATAITEM_TO_WIRE(&rdi, pwrdi);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_GetNextItem_Proxy
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
LPRESULTDATAITEM pItem
|
|
)
|
|
{
|
|
WIRE_RESULTDATAITEM wrdi;
|
|
HRESULT hr;
|
|
|
|
// Make sure the string pointer is NULL so that it is not marshaled as it
|
|
// is never passed from the snap-in to MMC. (It might not have be
|
|
// initialized).
|
|
|
|
pItem->str = NULL;
|
|
|
|
RESULTDATAITEM_TO_WIRE(pItem, &wrdi);
|
|
hr = IResultData_RemGetNextItem_Proxy(This, &wrdi);
|
|
WIRE_TO_RESULTDATAITEM(&wrdi, pItem);
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IResultData_GetNextItem_Stub
|
|
(
|
|
IResultData __RPC_FAR *This,
|
|
WIRE_RESULTDATAITEM __RPC_FAR *pwrdi
|
|
)
|
|
{
|
|
RESULTDATAITEM rdi;
|
|
HRESULT hr;
|
|
|
|
WIRE_TO_RESULTDATAITEM(pwrdi, &rdi);
|
|
hr = This->lpVtbl->GetNextItem(This, &rdi);
|
|
RESULTDATAITEM_TO_WIRE(&rdi, pwrdi);
|
|
return hr;
|
|
}
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// HICON Marshaling
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
// In wtypes.idl HICON is defined with the wire_marshal attribute with its
|
|
// 'on-the-wire' type as a pointer to a RemotableHandle. RemotableHandle is
|
|
// defined in wtypes.idl as
|
|
//
|
|
// typedef union _RemotableHandle switch( long fContext ) u
|
|
// {
|
|
// case WDT_INPROC_CALL: long hInproc;
|
|
// case WDT_REMOTE_CALL: long hRemote;
|
|
// } RemotableHandle;
|
|
//
|
|
// A wire_marshal type must supply routines to size, marhsal, unmarshal, and
|
|
// free marshaling data. Those routines are in ole32.dll but someone forgot to
|
|
// export them. (ole32 also has routines to marshal bitmaps, hwnds, etc. that
|
|
// are all exported). The code has been plagiarized here from ole32. The source
|
|
// is in \\savik\cairo\src\ole32\oleprx32\proxy\transmit.cxx with some macros in
|
|
// transmit.h in that same directory.
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
|
|
//
|
|
// The following defines and macros are from transmit.h. Note that
|
|
// USER_CALL_CTXT_MASK is from rpcndr.h and WDT_REMOTE_CALL is from wtypes.idl.
|
|
//
|
|
|
|
#define ALIGN( pStuff, cAlign ) \
|
|
pStuff = (unsigned char *)((ULONG_PTR)((pStuff) + (cAlign)) & ~ (cAlign))
|
|
|
|
#define LENGTH_ALIGN( Length, cAlign ) \
|
|
Length = (((Length) + (cAlign)) & ~ (cAlign))
|
|
|
|
#define PULONG_LV_CAST *(unsigned long __RPC_FAR * __RPC_FAR *)&
|
|
|
|
|
|
#define DIFFERENT_MACHINE_CALL( Flags) \
|
|
(USER_CALL_CTXT_MASK(Flags) == MSHCTX_DIFFERENTMACHINE)
|
|
|
|
#define WDT_HANDLE_MARKER WDT_INPROC_CALL
|
|
|
|
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
// HICON_UserSize
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// Parameters:
|
|
// unsigned long __RPC_FAR *pFlags [in] data format & context (see below)
|
|
// unsigned long StartingSize [in] current buffer size
|
|
// HICON __RPC_FAR *hIcon [in] HICON to be marshaled
|
|
//
|
|
//
|
|
// Flags Layout
|
|
// ============
|
|
//
|
|
//----------------------------------------------------------------------
|
|
// Bits Flag Value
|
|
//----------------------------------------------------------------------
|
|
// 31-24 Floating-point representation 0 = IEEE
|
|
// 1 = VAX
|
|
// 2 = Cray
|
|
// 3 = IBM
|
|
//----------------------------------------------------------------------
|
|
// 23-20 Integer and floating-point byte
|
|
// order 0 = Big-endian
|
|
// 1 = Little-endian
|
|
//----------------------------------------------------------------------
|
|
// 19-16 Character representation 0 = ASCII
|
|
// 1 = EBCDIC
|
|
//----------------------------------------------------------------------
|
|
// 15-0 Marshaling context flag 0 = MSHCTX_LOCAL
|
|
// 1 = MSHCTX_NOSHAREDMEM
|
|
// 2 = MSHCTX_DIFFERENTMSCHINE
|
|
// 3 = MSHCTX_INPROC
|
|
//----------------------------------------------------------------------
|
|
//
|
|
// Output:
|
|
// New buffer size after adding amount needed for HICON marshaled data
|
|
//
|
|
// Notes:
|
|
//
|
|
// Called from MIDL generated proxy to determine buffer size needed for
|
|
// marhaled data.
|
|
//
|
|
|
|
unsigned long __RPC_USER HICON_UserSize
|
|
(
|
|
unsigned long __RPC_FAR *pFlags,
|
|
unsigned long StartingSize,
|
|
HICON __RPC_FAR *hIcon
|
|
)
|
|
{
|
|
if (NULL == hIcon)
|
|
{
|
|
return StartingSize;
|
|
}
|
|
|
|
// If marshaling context is to a different machine then we don't support
|
|
// that.
|
|
|
|
if ( DIFFERENT_MACHINE_CALL(*pFlags) )
|
|
{
|
|
RpcRaiseException( RPC_S_INVALID_TAG );
|
|
}
|
|
|
|
// Make sure that our data will fall at a long boundary
|
|
|
|
LENGTH_ALIGN( StartingSize, 3 );
|
|
|
|
//Add the length
|
|
|
|
return StartingSize + 8;
|
|
}
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
// HICON_UserMarhsal
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// Parameters:
|
|
// unsigned long __RPC_FAR *pFlags [in] data format & context (see above)
|
|
// unsigned char __RPC_FAR *pBuffer [in] current buffer size
|
|
// HICON __RPC_FAR *hIcon [in] HICON to be marshaled
|
|
//
|
|
//
|
|
// Output:
|
|
// Pointer to buffer location following HICON's marshaling data
|
|
//
|
|
// Notes:
|
|
//
|
|
// Called from MIDL generated proxy to marshal an HICON
|
|
//
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER HICON_UserMarshal
|
|
(
|
|
unsigned long __RPC_FAR *pFlags,
|
|
unsigned char __RPC_FAR *pBuffer,
|
|
HICON __RPC_FAR *hIcon
|
|
)
|
|
{
|
|
if (NULL == hIcon)
|
|
{
|
|
return pBuffer;
|
|
}
|
|
|
|
if ( DIFFERENT_MACHINE_CALL(*pFlags) )
|
|
{
|
|
RpcRaiseException( RPC_S_INVALID_TAG );
|
|
}
|
|
|
|
// Make sure that our data will fall at a long boundary
|
|
|
|
ALIGN( pBuffer, 3 );
|
|
|
|
*( PULONG_LV_CAST pBuffer)++ = WDT_HANDLE_MARKER;
|
|
*( PULONG_LV_CAST pBuffer)++ = *((long *)hIcon);
|
|
|
|
return pBuffer;
|
|
}
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
// HICON_UserUnmarhsal
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// Parameters:
|
|
// unsigned long __RPC_FAR *pFlags [in] data format & context (see above)
|
|
// unsigned char __RPC_FAR *pBuffer [in] current buffer size
|
|
// HICON __RPC_FAR *hIcon [in] HICON to be marshaled
|
|
//
|
|
//
|
|
// Output:
|
|
// Pointer to buffer location following HICON's marshaling data
|
|
//
|
|
// Notes:
|
|
//
|
|
// Called from MIDL generated stub to unmarshal an HICON
|
|
//
|
|
|
|
unsigned char __RPC_FAR *__RPC_USER HICON_UserUnmarshal
|
|
(
|
|
unsigned long __RPC_FAR *pFlags,
|
|
unsigned char __RPC_FAR *pBuffer,
|
|
HICON __RPC_FAR *hIcon
|
|
)
|
|
{
|
|
unsigned long HandleMarker;
|
|
|
|
ALIGN( pBuffer, 3 );
|
|
|
|
HandleMarker = *( PULONG_LV_CAST pBuffer)++;
|
|
|
|
if ( HandleMarker == WDT_HANDLE_MARKER )
|
|
*((long *)hIcon) = *( PULONG_LV_CAST pBuffer)++;
|
|
else
|
|
RpcRaiseException( RPC_S_INVALID_TAG );
|
|
|
|
return pBuffer;
|
|
}
|
|
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
// HICON_UserUnmarhsal
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// Parameters:
|
|
// unsigned long __RPC_FAR *pFlags [in] data format & context (see above)
|
|
// HICON __RPC_FAR *hIcon [in] HICON that was unmarshaled
|
|
//
|
|
//
|
|
// Output:
|
|
// None
|
|
//
|
|
// Notes:
|
|
//
|
|
// Called from MIDL generated stub to free any associated marshaling data
|
|
// allocated during unmarshaling for embedded pointers. Not used for HICON.
|
|
//
|
|
|
|
void __RPC_USER HICON_UserFree
|
|
(
|
|
unsigned long __RPC_FAR *pFlags,
|
|
HICON __RPC_FAR *hIcon
|
|
)
|
|
{
|
|
}
|
|
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// IImageList Marshaling
|
|
//
|
|
//
|
|
//=--------------------------------------------------------------------------=
|
|
// These methods needed call_as because the HICON and HBITMAP parameters are
|
|
// specified as long pointers in the original IDL.
|
|
//=--------------------------------------------------------------------------=
|
|
|
|
HRESULT STDMETHODCALLTYPE IImageList_ImageListSetIcon_Proxy
|
|
(
|
|
IImageList __RPC_FAR *This,
|
|
LONG_PTR __RPC_FAR *pIcon,
|
|
long nLoc
|
|
)
|
|
{
|
|
return IImageList_RemImageListSetIcon_Proxy(This, (HICON)pIcon, nLoc);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IImageList_ImageListSetIcon_Stub
|
|
(
|
|
IImageList __RPC_FAR *This,
|
|
HICON hIcon,
|
|
long nLoc
|
|
)
|
|
{
|
|
return This->lpVtbl->ImageListSetIcon(This, (LONG_PTR __RPC_FAR*)hIcon, nLoc);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IImageList_ImageListSetStrip_Proxy
|
|
(
|
|
IImageList __RPC_FAR *This,
|
|
LONG_PTR __RPC_FAR *pBMapSm,
|
|
LONG_PTR __RPC_FAR *pBMapLg,
|
|
long nStartLoc,
|
|
COLORREF cMask
|
|
)
|
|
{
|
|
return IImageList_RemImageListSetStrip_Proxy(This,
|
|
(HBITMAP)pBMapSm,
|
|
(HBITMAP)pBMapLg,
|
|
nStartLoc,
|
|
cMask);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IImageList_ImageListSetStrip_Stub
|
|
(
|
|
IImageList __RPC_FAR *This,
|
|
HBITMAP hbmSmall,
|
|
HBITMAP hbmLarge,
|
|
long nStartLoc,
|
|
COLORREF cMask
|
|
)
|
|
{
|
|
return This->lpVtbl->ImageListSetStrip(This,
|
|
(LONG_PTR __RPC_FAR*)hbmSmall,
|
|
(LONG_PTR __RPC_FAR*)hbmLarge,
|
|
nStartLoc,
|
|
cMask);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet_CreatePropertyPages_Proxy
|
|
(
|
|
IExtendPropertySheet __RPC_FAR *This,
|
|
LPPROPERTYSHEETCALLBACK lpProvider,
|
|
LONG_PTR handle,
|
|
LPDATAOBJECT lpIDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
WIRE_PROPERTYPAGES *pPages = NULL;
|
|
WIRE_PROPERTYPAGE *pPage = NULL;
|
|
ULONG i = 0;
|
|
ULONG j = 0;
|
|
IExtendPropertySheetRemote *piExtendPropertySheetRemote = NULL;
|
|
IRemotePropertySheetManager *piRemotePropertySheetManager = NULL;
|
|
|
|
// Make sure the snap-in knows we are remoted. We do this here because
|
|
// this is the first opportunity for the proxy to inform a property page
|
|
// extension that it is remote and pass it data such as the MMC.exe path
|
|
// and the MMC command line.
|
|
|
|
hr = SetRemote((IUnknown *)This);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Call the IExtendPropertySheetRemote method which will return a filled in
|
|
// WIRE_PROPERTYPAGES from the remoted snap-in.
|
|
|
|
hr = This->lpVtbl->QueryInterface(This, &IID_IExtendPropertySheetRemote,
|
|
(void **)&piExtendPropertySheetRemote);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = piExtendPropertySheetRemote->lpVtbl->CreatePropertyPageDefs(
|
|
piExtendPropertySheetRemote, lpIDataObject, &pPages);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If there are pages (snap-in might not have added any) then
|
|
// CoCreateInstance the remote property sheet manager using the clsid
|
|
// returned in WIRE_PROPERTYPAGES. This object will be created in-proc here
|
|
// on the MMC side.
|
|
|
|
if (0 == pPages->cPages)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hr = CoCreateInstance(&pPages->clsidRemotePropertySheetManager,
|
|
NULL, // no aggregation,
|
|
CLSCTX_INPROC_SERVER,
|
|
&IID_IRemotePropertySheetManager,
|
|
(void **)&piRemotePropertySheetManager);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Pass the remote property sheet manager the WIRE_PROPERTYPAGES and let it
|
|
// actually create the property pages and add them to the sheet here on the
|
|
// MMC side.
|
|
|
|
hr = piRemotePropertySheetManager->lpVtbl->CreateRemotePages(
|
|
piRemotePropertySheetManager,
|
|
lpProvider,
|
|
handle,
|
|
lpIDataObject,
|
|
pPages);
|
|
Cleanup:
|
|
if (NULL != piRemotePropertySheetManager)
|
|
{
|
|
piRemotePropertySheetManager->lpVtbl->Release(piRemotePropertySheetManager);
|
|
}
|
|
|
|
if (NULL != piExtendPropertySheetRemote)
|
|
{
|
|
piExtendPropertySheetRemote->lpVtbl->Release(piExtendPropertySheetRemote);
|
|
}
|
|
|
|
// Free the WIRE_PROPERTYPAGES and all of its contents.
|
|
|
|
if (NULL != pPages)
|
|
{
|
|
// Release the object and free the title for each individual page
|
|
|
|
for (i = 0, pPage = &pPages->aPages[0]; i < pPages->cPages; i++, pPage++)
|
|
{
|
|
if (NULL != pPage->apunkObjects)
|
|
{
|
|
for (j = 0; j < pPage->cObjects; j++)
|
|
{
|
|
if (NULL != pPage->apunkObjects[j])
|
|
{
|
|
pPage->apunkObjects[j]->lpVtbl->Release(pPage->apunkObjects[j]);
|
|
}
|
|
}
|
|
CoTaskMemFree(pPage->apunkObjects);
|
|
}
|
|
if (NULL != pPage->pwszTitle)
|
|
{
|
|
CoTaskMemFree(pPage->pwszTitle);
|
|
}
|
|
}
|
|
|
|
// Free the ProgID prefix
|
|
|
|
if (NULL != pPages->pwszProgIDStart)
|
|
{
|
|
CoTaskMemFree(pPages->pwszProgIDStart);
|
|
}
|
|
|
|
// Free all of the snap-in's property page info
|
|
|
|
if (NULL != pPages->pPageInfos)
|
|
{
|
|
for (i = 0; i < pPages->pPageInfos->cPages; i++)
|
|
{
|
|
if (NULL != pPages->pPageInfos->aPageInfo[i].pwszTitle)
|
|
{
|
|
CoTaskMemFree(pPages->pPageInfos->aPageInfo[i].pwszTitle);
|
|
}
|
|
if (NULL != pPages->pPageInfos->aPageInfo[i].pwszProgID)
|
|
{
|
|
CoTaskMemFree(pPages->pPageInfos->aPageInfo[i].pwszProgID);
|
|
}
|
|
}
|
|
CoTaskMemFree(pPages->pPageInfos);
|
|
}
|
|
|
|
// Free all of the objects associated with the sheet
|
|
|
|
if (NULL != pPages->apunkObjects)
|
|
{
|
|
for (i = 0; i < pPages->cObjects; i++)
|
|
{
|
|
if (NULL != pPages->apunkObjects[i])
|
|
{
|
|
pPages->apunkObjects[i]->lpVtbl->Release(pPages->apunkObjects[i]);
|
|
}
|
|
}
|
|
CoTaskMemFree(pPages->apunkObjects);
|
|
}
|
|
|
|
|
|
// Release the extra object and the WIRE_PROPERTYPAGES struct itself
|
|
|
|
if (NULL != pPages->punkExtra)
|
|
{
|
|
pPages->punkExtra->lpVtbl->Release(pPages->punkExtra);
|
|
}
|
|
CoTaskMemFree(pPages);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//=--------------------------------------------------------------------------=
|
|
// IExtendPropertySheet_CreatePropertyPages_Stub
|
|
//=--------------------------------------------------------------------------=
|
|
//
|
|
// Parameters:
|
|
// IExtendPropertySheet __RPC_FAR *This [in] this pointer
|
|
//
|
|
// Output:
|
|
//
|
|
// Notes:
|
|
//
|
|
// This stub is never called because
|
|
// IExtendPropertySheet_CreatePropertyPages_Proxy() (see above) reroutes the
|
|
// call to IExtendPropertySheetRemote::CreatePropertyPageDefs().
|
|
//
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet_CreatePropertyPages_Stub
|
|
(
|
|
IExtendPropertySheet __RPC_FAR *This
|
|
)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet_QueryPagesFor_Proxy
|
|
(
|
|
IExtendPropertySheet __RPC_FAR *This,
|
|
IDataObject *piDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendPropertySheet_RemQueryPagesFor_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet_QueryPagesFor_Stub
|
|
(
|
|
IExtendPropertySheet __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->QueryPagesFor(This, piDataObject);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet2_GetWatermarks_Proxy
|
|
(
|
|
IExtendPropertySheet2 __RPC_FAR *This,
|
|
IDataObject *piDataObject,
|
|
HBITMAP *lphWatermark,
|
|
HBITMAP *lphHeader,
|
|
HPALETTE *lphPalette,
|
|
BOOL *bStretch
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendPropertySheet2_RemGetWatermarks_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
lphWatermark,
|
|
lphHeader,
|
|
lphPalette,
|
|
bStretch);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheet2_GetWatermarks_Stub
|
|
(
|
|
IExtendPropertySheet2 __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
HBITMAP *lphWatermark,
|
|
HBITMAP *lphHeader,
|
|
HPALETTE *lphPalette,
|
|
BOOL *bStretch
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->GetWatermarks(This,
|
|
piDataObject,
|
|
lphWatermark,
|
|
lphHeader,
|
|
lphPalette,
|
|
bStretch);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheetRemote_CreatePropertyPageDefs_Proxy
|
|
(
|
|
IExtendPropertySheetRemote __RPC_FAR *This,
|
|
IDataObject *piDataObject,
|
|
WIRE_PROPERTYPAGES **ppPages
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendPropertySheetRemote_RemCreatePropertyPageDefs_Proxy(
|
|
This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
ppPages);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendPropertySheetRemote_CreatePropertyPageDefs_Stub
|
|
(
|
|
IExtendPropertySheetRemote __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
WIRE_PROPERTYPAGES **ppPages
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->CreatePropertyPageDefs(This, piDataObject, ppPages);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IPropertySheetProvider_CreatePropertySheet_Proxy
|
|
(
|
|
IPropertySheetProvider __RPC_FAR *This,
|
|
LPCWSTR title,
|
|
boolean type,
|
|
MMC_COOKIE cookie,
|
|
IDataObject *piDataObject,
|
|
DWORD dwOptions
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IPropertySheetProvider_RemCreatePropertySheet_Proxy(This,
|
|
title,
|
|
type,
|
|
cookie,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
dwOptions);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IPropertySheetProvider_CreatePropertySheet_Stub
|
|
(
|
|
IPropertySheetProvider __RPC_FAR *This,
|
|
LPCWSTR title,
|
|
boolean type,
|
|
MMC_COOKIE cookie,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
DWORD dwOptions
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->CreatePropertySheet(This, title, type, cookie, piDataObject, dwOptions);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IPropertySheetProvider_FindPropertySheet_Proxy
|
|
(
|
|
IPropertySheetProvider __RPC_FAR *This,
|
|
MMC_COOKIE cookie,
|
|
IComponent *piComponent,
|
|
IDataObject *piDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IPropertySheetProvider_RemFindPropertySheet_Proxy(This,
|
|
cookie,
|
|
piComponent,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IPropertySheetProvider_FindPropertySheet_Stub
|
|
(
|
|
IPropertySheetProvider __RPC_FAR *This,
|
|
MMC_COOKIE cookie,
|
|
IComponent *piComponent,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->FindPropertySheet(This, cookie, piComponent, piDataObject);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendContextMenu_AddMenuItems_Proxy
|
|
(
|
|
IExtendContextMenu __RPC_FAR *This,
|
|
LPDATAOBJECT piDataObject,
|
|
LPCONTEXTMENUCALLBACK piCallback,
|
|
long __RPC_FAR *pInsertionAllowed
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Make sure the snap-in knows we are remoted. We do this here because
|
|
// this is the first opportunity for the proxy to inform a context menu
|
|
// extension that it is remote.
|
|
|
|
hr = SetRemote((IUnknown *)This);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendContextMenu_RemAddMenuItems_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
piCallback,
|
|
pInsertionAllowed);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendContextMenu_AddMenuItems_Stub
|
|
(
|
|
IExtendContextMenu __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
LPCONTEXTMENUCALLBACK piCallback,
|
|
long __RPC_FAR *pInsertionAllowed
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->AddMenuItems(This, piDataObject,
|
|
piCallback, pInsertionAllowed);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendContextMenu_Command_Proxy
|
|
(
|
|
IExtendContextMenu __RPC_FAR *This,
|
|
long lCommandID,
|
|
LPDATAOBJECT piDataObject
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL fIsMultiSelect = FALSE;
|
|
long cDataObjects = 1L;
|
|
IDataObject **ppiDataObjects = NULL;
|
|
BOOL fSpecialDataObject = FALSE;
|
|
long lSpecialDataObject = 0;
|
|
|
|
// Check for special data objects such DOBJ_CUSTOMWEB etc.
|
|
|
|
CheckForSpecialDataObjects(&piDataObject, &fSpecialDataObject, &lSpecialDataObject);
|
|
|
|
// If this is a mutliple selection then we need to extract the data
|
|
// objects in the HGLOBAL
|
|
|
|
if (!fSpecialDataObject)
|
|
{
|
|
hr = IsMultiSelect(piDataObject, &fIsMultiSelect);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
if (fIsMultiSelect)
|
|
{
|
|
hr = InterpretMultiSelect(piDataObject, &cDataObjects, &ppiDataObjects);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ppiDataObjects = &piDataObject;
|
|
}
|
|
|
|
hr = IExtendContextMenu_RemCommand_Proxy(This,
|
|
cDataObjects,
|
|
ppiDataObjects,
|
|
fSpecialDataObject,
|
|
lSpecialDataObject,
|
|
lCommandID);
|
|
Cleanup:
|
|
if ( fIsMultiSelect && (NULL != ppiDataObjects) )
|
|
{
|
|
(void)GlobalFree(ppiDataObjects);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IExtendContextMenu_Command_Stub
|
|
(
|
|
IExtendContextMenu __RPC_FAR *This,
|
|
long cDataObjects,
|
|
IDataObject __RPC_FAR *__RPC_FAR ppiDataObjects[ ],
|
|
BOOL fSpecialDataObject,
|
|
long lSpecialDataObject,
|
|
long lCommandID
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IDataObject *piDataObject = NULL; // Not AddRef()ed
|
|
IDataObject *piMultiSelDataObject = NULL;
|
|
|
|
// If there is more than one data object then we need to pack them into a
|
|
// a separate data object that appears as a multi-select data object.
|
|
|
|
if (cDataObjects > 1L)
|
|
{
|
|
hr = CreateMultiSelDataObject(ppiDataObjects, cDataObjects,
|
|
&piMultiSelDataObject);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
piDataObject = piMultiSelDataObject;
|
|
}
|
|
else if (fSpecialDataObject)
|
|
{
|
|
piDataObject = (IDataObject *)lSpecialDataObject;
|
|
}
|
|
else
|
|
{
|
|
piDataObject = ppiDataObjects[0];
|
|
}
|
|
|
|
// Call the snap-in
|
|
|
|
hr = This->lpVtbl->Command(This, lCommandID, piDataObject);
|
|
|
|
Cleanup:
|
|
if (NULL != piMultiSelDataObject)
|
|
{
|
|
piMultiSelDataObject->lpVtbl->Release(piMultiSelDataObject);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IColumnData_GetColumnConfigData_Proxy
|
|
(
|
|
IColumnData __RPC_FAR *This,
|
|
SColumnSetID __RPC_FAR *pColID,
|
|
MMC_COLUMN_SET_DATA __RPC_FAR *__RPC_FAR *ppColSetData
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
size_t cbColData = 0;
|
|
MMC_COLUMN_SET_DATA *pColSetData = NULL;
|
|
|
|
// Call the proxy and get the returned data from MMC
|
|
|
|
hr = IColumnData_RemGetColumnConfigData_Proxy(This, pColID, ppColSetData);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If no data was returned then there is nothing to do
|
|
|
|
if (NULL == *ppColSetData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (NULL == (*ppColSetData)->pColData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// At this point the MIDL-generated proxy has returned an MMC_COLUMN_SET_DATA
|
|
// in which the embedded pointer pColData points to a separate block of
|
|
// memory which must be freed independently. The snap-in thinks that
|
|
// pColData points into the same block of memory so it will only call
|
|
// CoTaskMemFree for the MMC_COLUMN_SET_DATA. We need to allocate a new single
|
|
// block in the format the snap-in is expecting and free the memory returned
|
|
// from the proxy.
|
|
|
|
cbColData = sizeof(MMC_COLUMN_DATA) * (*ppColSetData)->nNumCols;
|
|
|
|
pColSetData = (MMC_COLUMN_SET_DATA *)CoTaskMemAlloc(
|
|
sizeof(MMC_COLUMN_SET_DATA) + cbColData);
|
|
|
|
if (NULL == pColSetData)
|
|
{
|
|
CoTaskMemFree((*ppColSetData)->pColData);
|
|
CoTaskMemFree(*ppColSetData);
|
|
*ppColSetData = NULL;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
// Copy the column set data
|
|
|
|
memcpy(pColSetData, (*ppColSetData), sizeof(MMC_COLUMN_SET_DATA));
|
|
|
|
// Set the embedded pointer to point immediately following the
|
|
// MMC_COLUMN_SET_DATA
|
|
|
|
pColSetData->pColData = (MMC_COLUMN_DATA *)(pColSetData + 1);
|
|
|
|
// Copy the column data
|
|
|
|
memcpy(pColSetData->pColData, (*ppColSetData)->pColData, cbColData);
|
|
|
|
// Free the data returned from the proxy
|
|
|
|
CoTaskMemFree((*ppColSetData)->pColData);
|
|
CoTaskMemFree(*ppColSetData);
|
|
|
|
// Return the new single block pointer to the snap-in
|
|
|
|
*ppColSetData = pColSetData;
|
|
}
|
|
|
|
Cleanup:
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IColumnData_GetColumnConfigData_Stub
|
|
(
|
|
IColumnData __RPC_FAR *This,
|
|
SColumnSetID __RPC_FAR *pColID,
|
|
MMC_COLUMN_SET_DATA __RPC_FAR *__RPC_FAR *ppColSetData
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
size_t cbColData = 0;
|
|
MMC_COLUMN_DATA *pColData = NULL;
|
|
|
|
// Call into MMC and get the returned data
|
|
|
|
hr = This->lpVtbl->GetColumnConfigData(This, pColID, ppColSetData);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If no data was returned then there is nothing to do
|
|
|
|
if (NULL == *ppColSetData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (NULL == (*ppColSetData)->pColData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// At this point MMC has returned a pointer to an MMC_COLUMN_SET_DATA
|
|
// that contains an embedded pointer into the same block of memory (pColData).
|
|
// MMC expects that the caller will make a single call to CoTaskMemFree().
|
|
// The MIDL-generated stub thinks the embedded pointer needs to be freed
|
|
// separately so we need to reconstruct the output to use two separate
|
|
// blocks.
|
|
|
|
// Allocate a new MMC_COLUMN_DATA array
|
|
|
|
cbColData = sizeof(MMC_COLUMN_DATA) * (*ppColSetData)->nNumCols;
|
|
|
|
pColData = (MMC_COLUMN_DATA *)CoTaskMemAlloc(cbColData);
|
|
if (NULL == pColData)
|
|
{
|
|
CoTaskMemFree(*ppColSetData);
|
|
*ppColSetData = NULL;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
// Copy the column data
|
|
memcpy(pColData, (*ppColSetData)->pColData, cbColData);
|
|
|
|
// Overwrite the existing embedded pointer. There will be no memory leak
|
|
// because that pointer pointed into the same block as the
|
|
// MMC_COLUMN_SET_DATA and the stub will free the MMC_COLUMN_SET_DATA pointer.
|
|
// Now both pointers can be freed independently as the stub expects.
|
|
|
|
(*ppColSetData)->pColData = pColData;
|
|
}
|
|
|
|
Cleanup:
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IColumnData_GetColumnSortData_Proxy
|
|
(
|
|
IColumnData __RPC_FAR *This,
|
|
SColumnSetID __RPC_FAR *pColID,
|
|
MMC_SORT_SET_DATA __RPC_FAR *__RPC_FAR *ppColSortData
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
size_t cbSortData = 0;
|
|
MMC_SORT_SET_DATA *pColSortData = NULL;
|
|
|
|
// Call the proxy and get the returned data from MMC
|
|
|
|
hr = IColumnData_RemGetColumnSortData_Proxy(This, pColID, ppColSortData);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If no data was returned then there is nothing to do
|
|
|
|
if (NULL == *ppColSortData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (NULL == (*ppColSortData)->pSortData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// At this point the MIDL-generated proxy has returned an MMC_SORT_SET_DATA
|
|
// in which the embedded pointer pSortData points to a separate block of
|
|
// memory which must be freed independently. The snap-in thinks that
|
|
// pSortData points into the same block of memory so it will only call
|
|
// CoTaskMemFree for the MMC_SORT_SET_DATA. We need to allocate a new single
|
|
// block in the format the snap-in is expecting and free the memory returned
|
|
// from the proxy.
|
|
|
|
cbSortData = sizeof(MMC_SORT_DATA) * (*ppColSortData)->nNumItems;
|
|
|
|
pColSortData = (MMC_SORT_SET_DATA *)CoTaskMemAlloc(
|
|
sizeof(MMC_SORT_SET_DATA) + cbSortData);
|
|
|
|
if (NULL == pColSortData)
|
|
{
|
|
CoTaskMemFree((*ppColSortData)->pSortData);
|
|
CoTaskMemFree(*ppColSortData);
|
|
*ppColSortData = NULL;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
// Copy the sort set data
|
|
|
|
memcpy(pColSortData, (*ppColSortData), sizeof(MMC_SORT_SET_DATA));
|
|
|
|
// Set the embedded pointer to point immediately following the
|
|
// MMC_SORT_SET_DATA
|
|
|
|
pColSortData->pSortData = (MMC_SORT_DATA *)(pColSortData + 1);
|
|
|
|
// Copy the sort data
|
|
|
|
memcpy(pColSortData->pSortData, (*ppColSortData)->pSortData, cbSortData);
|
|
|
|
// Free the data returned from the proxy
|
|
|
|
CoTaskMemFree((*ppColSortData)->pSortData);
|
|
CoTaskMemFree(*ppColSortData);
|
|
|
|
// Return the new single block pointer to the snap-in
|
|
|
|
*ppColSortData = pColSortData;
|
|
}
|
|
|
|
Cleanup:
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE IColumnData_GetColumnSortData_Stub
|
|
(
|
|
IColumnData __RPC_FAR *This,
|
|
SColumnSetID __RPC_FAR *pColID,
|
|
MMC_SORT_SET_DATA __RPC_FAR *__RPC_FAR *ppColSortData
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
size_t cbSortData = 0;
|
|
MMC_SORT_DATA *pSortData = NULL;
|
|
|
|
// Call into MMC and get the returned data
|
|
|
|
hr = This->lpVtbl->GetColumnSortData(This, pColID, ppColSortData);
|
|
if (FAILED(hr))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// If no data was returned then there is nothing to do
|
|
|
|
if (NULL == *ppColSortData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (NULL == (*ppColSortData)->pSortData)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
// At this point MMC has returned a pointer to an MMC_SORT_SET_DATA
|
|
// that contains an embedded pointer into the same block of memory (pSortData).
|
|
// MMC expects that the caller will make a single call to CoTaskMemFree().
|
|
// The MIDL-generated stub thinks the embedded pointer needs to be freed
|
|
// separately so we need to reconstruct the output to use two separate
|
|
// blocks.
|
|
|
|
// Allocate a new MMC_SORT_DATA array
|
|
|
|
cbSortData = sizeof(MMC_SORT_DATA) * (*ppColSortData)->nNumItems;
|
|
|
|
pSortData = (MMC_SORT_DATA *)CoTaskMemAlloc(cbSortData);
|
|
if (NULL == pSortData)
|
|
{
|
|
CoTaskMemFree(*ppColSortData);
|
|
*ppColSortData = NULL;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
// Copy the column data
|
|
memcpy(pSortData, (*ppColSortData)->pSortData, cbSortData);
|
|
|
|
// Overwrite the existing embedded pointer. There will be no memory leak
|
|
// because that pointer pointed into the same block as the
|
|
// MMC_SORT_SET_DATA and the stub will free the MMC_SORT_SET_DATA pointer.
|
|
// Now both pointers can be freed independently as the stub expects.
|
|
|
|
(*ppColSortData)->pSortData = pSortData;
|
|
}
|
|
|
|
Cleanup:
|
|
return hr;
|
|
}
|