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.
1998 lines
59 KiB
1998 lines
59 KiB
//+---------------------------------------------------------------------
|
|
//
|
|
// File: sdv.cxx
|
|
//
|
|
//------------------------------------------------------------------------
|
|
|
|
//[ srvrdv_overview
|
|
/*
|
|
SrvrDV Overview
|
|
|
|
The SrvrDV base class implements the persistent data and view aspects common
|
|
to most OLE Compound Document objects. It implements the IDataObject,
|
|
IViewObject, and IPersist family of interfaces.
|
|
|
|
The set of formats supported in IDataObject::GetData and SetData methods is
|
|
very object dependent. SrvrDV uses a table approach to allow each derived
|
|
class to specify precisely exactly the set of formats supported. For each
|
|
of GetData and SetData there is a pair of tables. One is a table of FORMATETC
|
|
structures that specify information about the format supported. A parallel
|
|
table contains a pointer to a function that implements each format supported.
|
|
SrvrDV has a set of static methods that implement the standard OLE clipboard
|
|
formats. The derived class can include these methods in its Get/Set tables.
|
|
|
|
*/
|
|
//]
|
|
|
|
#include "headers.hxx"
|
|
#pragma hdrstop
|
|
|
|
OLECHAR szContents[] = OLETEXT("contents");
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SrvrDV, protected
|
|
//
|
|
// Synopsis: Constructor for SrvrCtrl object
|
|
//
|
|
// Notes: To create a properly initialized object you must
|
|
// call the Init method immediately after construction.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
SrvrDV::SrvrDV(void)
|
|
{
|
|
DOUT(TEXT("SrvrDV: Constructing\r\n"));
|
|
|
|
_pmk = NULL;
|
|
_lpstrDisplayName = NULL;
|
|
_sizel.cx = 0; _sizel.cy = 0;
|
|
_pDataAdviseHolder = NULL;
|
|
|
|
_fFrozen = FALSE;
|
|
_pViewAdviseHolder = NULL;
|
|
|
|
_fDirty = FALSE;
|
|
_fNoScribble = FALSE;
|
|
_pStg = NULL;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Init, protected
|
|
//
|
|
// Synopsis: Fully initializes a SrvrCtrl object as part of a compound
|
|
// document server aggregate
|
|
//
|
|
// Arguments: [pClass] -- The initialized class descriptor for the server
|
|
// [pCtrl] -- The control subobject of the server we are a part of.
|
|
//
|
|
// Returns: NOERROR if successful
|
|
//
|
|
// Notes: The Init method of the control subobject creates the data/view and
|
|
// inplace subobjects of the object and calls the respective Init methods
|
|
// on each, including this one.
|
|
// The class descriptor pointer is saved in the protected _pClass
|
|
// member variable where it is accessible during the lifetime
|
|
// of the object.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::Init(LPCLASSDESCRIPTOR pClass, LPSRVRCTRL pCtrl)
|
|
{
|
|
_pClass = pClass;
|
|
_pCtrl = pCtrl;
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Init, protected
|
|
//
|
|
// Synopsis: Fully initializes a SrvrCtrl object from an existing object
|
|
// to be used as a data transfer object
|
|
//
|
|
// Arguments: [pClass] -- The initialized class descriptor for the server
|
|
// [pDV] -- The data/view subobject of the object that is the source
|
|
// of this transfer data object
|
|
//
|
|
// Returns: NOERROR if successful
|
|
//
|
|
// Notes: This method is used in the process of the GetClipboardCopy method
|
|
// for obtaining a transfer data object from an existing object.
|
|
// The class descriptor pointer is saved in the protected _pClass
|
|
// member variable where it is accessible during the lifetime
|
|
// of the object.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::Init(LPCLASSDESCRIPTOR pClass, LPSRVRDV pDV)
|
|
{
|
|
_pClass = pClass; // stash the class descriptor pointer
|
|
_pCtrl = NULL; // we are a free-floating transfer data object
|
|
|
|
// copy over member variables as appropriate
|
|
_pmk = pDV->_pmk;
|
|
if (_pmk != NULL)
|
|
_pmk->AddRef();
|
|
|
|
TaskAllocString(pDV->_lpstrDisplayName, &_lpstrDisplayName);
|
|
_sizel = pDV->_sizel;
|
|
|
|
// we create a temporary IStorage and take a snapshot of the storage.
|
|
// The temporary IStorage will be automatically deleted when this
|
|
// data object is released
|
|
// What happens when we run out of memory?
|
|
// Should re-try with disk-based docfile?
|
|
//
|
|
HRESULT hr;
|
|
LPSTORAGE pStg;
|
|
if (OK(hr = CreateStorageOnHGlobal(NULL, &pStg)))
|
|
{
|
|
// do a "Save Copy As" into our storage instance.
|
|
if (OK(hr = OleSave((LPPERSISTSTORAGE)pDV, pStg, FALSE)))
|
|
{
|
|
if (OK(hr = pStg->Commit(STGC_DEFAULT)))
|
|
{
|
|
if (OK(hr = ((LPPERSISTSTORAGE)pDV)->SaveCompleted(NULL)))
|
|
(_pStg = pStg)->AddRef();
|
|
}
|
|
}
|
|
pStg->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::~SrvrDV, protected
|
|
//
|
|
// Synopsis: Destructor for the SrvrDV object
|
|
//
|
|
// Notes: The destructor is called as a result of the servers
|
|
// reference count going to 0. It releases any held resources
|
|
// including advise holders and storages.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
SrvrDV::~SrvrDV(void)
|
|
{
|
|
if (_pmk != NULL)
|
|
_pmk->Release();
|
|
|
|
if (_lpstrDisplayName != NULL)
|
|
TaskFreeString(_lpstrDisplayName);
|
|
|
|
if (_pDataAdviseHolder != NULL)
|
|
_pDataAdviseHolder->Release();
|
|
|
|
if (_pViewAdviseHolder != NULL)
|
|
_pViewAdviseHolder->Release();
|
|
|
|
if (_pStg != NULL)
|
|
_pStg->Release();
|
|
|
|
DOUT(TEXT("SrvrDV: Destructed\r\n"));
|
|
}
|
|
|
|
#ifdef DOCGEN // documentation for pure virtual function
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetClipboardCopy, public
|
|
//
|
|
// Synopsis: Produces a data-transfer object representing a snapshot
|
|
// of this data/view object
|
|
//
|
|
// Arguments: [ppDV] -- The place where the data-transfer object is returned
|
|
//
|
|
// Returns: Success if the data-transfer object was created.
|
|
//
|
|
// Notes: This method is called
|
|
// as a result of an IOleObject::GetClipboardData call on the
|
|
// control subobject. The overridden method should create a new
|
|
// instance of itself and call the appropriate Init method on
|
|
// the new instance passing the `this' pointer. This way the
|
|
// new instance can initialize itself as a snapshot of this object.
|
|
// All servers must override this pure virtual method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetClipboardCopy(LPSRVRDV FAR* ppDV)
|
|
{}
|
|
#endif // DOCGEN
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SetExtent, public
|
|
//
|
|
// Synopsis: Informs the object that it has a new expected size
|
|
//
|
|
// Notes: IOleObject::SetExtent and GetExtent methods are passed
|
|
// directly from the control subobject to the data/view subobject
|
|
// via these methods. See OLE documentation for the
|
|
// arguments and return values.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::SetExtent(DWORD dwAspect, SIZEL& sizel)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
switch(dwAspect)
|
|
{
|
|
case DVASPECT_CONTENT:
|
|
case DVASPECT_DOCPRINT:
|
|
#if 1
|
|
// NTBug #20692: Fail Metafile setextent, because
|
|
// Powerpoint tries to resize things in weird ways.
|
|
hr = E_FAIL; // icon aspect is fixed size
|
|
#else
|
|
_sizel = sizel;
|
|
#endif
|
|
break;
|
|
case DVASPECT_THUMBNAIL: //REVIEW: what size is a thumbnail?
|
|
case DVASPECT_ICON:
|
|
DOUT(TEXT("SrvrDV::SetExtent E_FAIL\r\n"));
|
|
hr = E_FAIL; // icon aspect is fixed size
|
|
break;
|
|
default:
|
|
DOUT(TEXT("SrvrDV::SetExtent E_INVALIDARG\r\n"));
|
|
hr = E_INVALIDARG;
|
|
break;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetExtent, public
|
|
//
|
|
// Synopsis: Requests the current size for some draw aspect of the object
|
|
//
|
|
// Notes: IOleObject::SetExtent and GetExtent methods are passed
|
|
// directly from the control subobject to the data/view subobject
|
|
// via these methods. See OLE documentation for the
|
|
// arguments and return values.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetExtent(DWORD dwAspect, LPSIZEL lpsizel)
|
|
{
|
|
#if DBG
|
|
TCHAR achTemp[256];
|
|
wsprintf(achTemp,TEXT("SrvrDV::GetExtent (dwApsect = %ld, cx = %ld, cy = %ld)\r\n"),
|
|
dwAspect, _sizel.cx, _sizel.cy);
|
|
DOUT(achTemp);
|
|
#endif
|
|
|
|
HRESULT hr = NOERROR;
|
|
switch(dwAspect) {
|
|
default:
|
|
DOUT(TEXT("SrvrDV::GetExtent INVALIDARG\r\n"));
|
|
case DVASPECT_CONTENT:
|
|
case DVASPECT_DOCPRINT:
|
|
*lpsizel = _sizel;
|
|
break;
|
|
|
|
case DVASPECT_THUMBNAIL:
|
|
case DVASPECT_ICON:
|
|
// The iconic view is actually a metafile of the
|
|
// icon with a text-string underneath.
|
|
// This isn't the right calculation...
|
|
lpsizel->cx = HimetricFromHPix(32);
|
|
lpsizel->cy = HimetricFromVPix(32);
|
|
break;
|
|
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::OnDataChange, public
|
|
//
|
|
// Synopsis: Sets the dirty flag and raises data and view changed
|
|
// to all registered advises
|
|
//
|
|
// Arguments: [dwAdvf] -- from the ADVF Data Advise flags.
|
|
// Usually this is 0. ADVF_DATAONSTOP is used
|
|
// when the object is closing down.
|
|
// Notes: This function should be called whenever the native
|
|
// data of the object is modified.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
void
|
|
SrvrDV::OnDataChange(DWORD dwAdvf)
|
|
{
|
|
_fDirty = TRUE;
|
|
if (_pViewAdviseHolder != NULL)
|
|
_pViewAdviseHolder->SendOnViewChange(DVASPECT_CONTENT);
|
|
if (_pDataAdviseHolder != NULL)
|
|
_pDataAdviseHolder->SendOnDataChange((LPDATAOBJECT)this, 0, dwAdvf);
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SetMoniker, public
|
|
//
|
|
// Synopsis: Informs the object of its new, full moniker
|
|
//
|
|
// Arguments: [pmk] -- full moniker to the object
|
|
//
|
|
// Notes: The data/view subobject records the full moniker to the
|
|
// object so it can properly dispense the standard OLE
|
|
// Object Descriptor and Link Source clipboard formats.
|
|
// This method is called whenever the IOleObject::SetMoniker
|
|
// method is called on the control subobject.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
void
|
|
SrvrDV::SetMoniker(LPMONIKER pmk)
|
|
{
|
|
if (_pmk != NULL)
|
|
{
|
|
_pmk->Release();
|
|
|
|
if (_lpstrDisplayName != NULL) // flush our cached display name
|
|
{
|
|
TaskFreeString(_lpstrDisplayName);
|
|
_lpstrDisplayName = NULL;
|
|
}
|
|
}
|
|
|
|
_pmk = pmk;
|
|
|
|
if (_pmk != NULL)
|
|
{
|
|
_pmk->AddRef();
|
|
}
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetMoniker, public
|
|
//
|
|
// Synopsis: Returns the full moniker to the object
|
|
//
|
|
// Arguments: [dwAssign] -- See IOleObject::GetMoniker
|
|
// [ppmk] -- The place where the moniker is returned
|
|
//
|
|
// Returns: Success if the moniker is available.
|
|
//
|
|
// Notes: This returns the moniker that this data/view subobject has
|
|
// previously recorded. If no moniker is yet assigned
|
|
// then the moniker is requested from the client site
|
|
// via the IOleObject::GetMoniker method on the control subobject.
|
|
// This method is used by the GetOBJECTDESCRIPTOR and GetLINKSOURCE
|
|
// methods.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetMoniker(DWORD dwAssign, LPMONIKER FAR* ppmk)
|
|
{
|
|
*ppmk = NULL; // set out params to NULL
|
|
|
|
HRESULT hr = NOERROR;
|
|
if (_pmk == NULL)
|
|
{
|
|
if (_pCtrl == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetMoniker E_INVALIDARG\r\n"));
|
|
hr = MK_E_UNAVAILABLE;
|
|
}
|
|
else
|
|
{
|
|
hr = _pCtrl->GetMoniker(dwAssign, OLEWHICHMK_OBJFULL, &_pmk);
|
|
}
|
|
}
|
|
|
|
if (OK(hr))
|
|
(*ppmk = _pmk)->AddRef();
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetMonikerDisplayName, public
|
|
//
|
|
// Synopsis: Returns the display name from the object's moniker
|
|
//
|
|
// Notes: The display name of the object is used in for
|
|
// dispensing the Object Descriptor clipboard format.
|
|
// The caller must not free the string returned.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
LPOLESTR
|
|
SrvrDV::GetMonikerDisplayName(DWORD dwAssign)
|
|
{
|
|
//
|
|
// NOTE: default dwAssign is OLEGETMONIKER_ONLYIFTHERE
|
|
//
|
|
// we maintain a moniker display name cache in the form of member
|
|
// variable _lpstrDisplayName.
|
|
//
|
|
// If we don't have a display name cached then take our moniker and
|
|
// squeeze a display name out of it
|
|
//
|
|
if (!_lpstrDisplayName)
|
|
{
|
|
LPMONIKER pmk;
|
|
if (OK(GetMoniker(dwAssign, &pmk)))
|
|
{
|
|
::GetMonikerDisplayName(pmk, &_lpstrDisplayName);
|
|
pmk->Release();
|
|
}
|
|
}
|
|
return _lpstrDisplayName;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetEMBEDDEDOBJECT, public
|
|
//
|
|
// Synopsis: Implementation of IDataObject::GetData and GetDataHere
|
|
// for the standard Embedded Object clipboard format
|
|
//
|
|
// Arguments: [pv] -- pointer to a SrvrDV object
|
|
// [pformatetc] -- as in IDataObject::GetData, GetDataHere
|
|
// [pmedium] -- as in IDataObject::GetData, GetDataHere
|
|
// [fHere] -- TRUE for GetDataHere, FALSE for GetData
|
|
//
|
|
// Returns: Success if the clipboard format could be dispensed
|
|
//
|
|
// Notes: This and the other static GetXXX methods are for use
|
|
// in the server's Get format tables.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetEMBEDDEDOBJECT( LPSRVRDV pDV,
|
|
LPFORMATETC pformatetc,
|
|
LPSTGMEDIUM pmedium,
|
|
BOOL fHere)
|
|
{
|
|
LPPERSISTSTORAGE pPStg = (LPPERSISTSTORAGE)(LPSRVRDV)pDV;
|
|
HRESULT hr = NOERROR;
|
|
if (!fHere && pmedium)
|
|
{
|
|
// fill in the pmedium structure
|
|
pmedium->tymed = TYMED_ISTORAGE;
|
|
hr = StgCreateDocfile(NULL,
|
|
STGM_DFRALL | STGM_CREATE | STGM_DELETEONRELEASE,
|
|
0L,
|
|
&pmedium->pstg);
|
|
pmedium->pUnkForRelease = NULL;
|
|
}
|
|
|
|
if (OK(hr) && pmedium && pPStg)
|
|
{
|
|
if (OK(hr = OleSave(pPStg, pmedium->pstg, FALSE)))
|
|
hr = pPStg->SaveCompleted(NULL);
|
|
|
|
// if we failed somehow and yet created a docfile, then we will
|
|
// release the docfile to delete it
|
|
//
|
|
if (!OK(hr) && !fHere)
|
|
pmedium->pstg->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetMETAFILEPICT, public
|
|
//
|
|
// Synopsis: Implementation of IDataObject::GetData and GetDataHere
|
|
// for the standard Metafilepict clipboard format
|
|
//
|
|
// Arguments: [pv] -- pointer to a SrvrDV object
|
|
// [pformatetc] -- as in IDataObject::GetData, GetDataHere
|
|
// [pmedium] -- as in IDataObject::GetData, GetDataHere
|
|
// [fHere] -- TRUE for GetDataHere, FALSE for GetData
|
|
//
|
|
// Returns: Success if the clipboard format could be dispensed
|
|
//
|
|
// Notes: This member function uses IViewObject::Draw to construct
|
|
// the metafile pict.
|
|
// This and the other static GetXXX methods are for use
|
|
// in the server's Get format tables.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetMETAFILEPICT( LPSRVRDV pDV,
|
|
LPFORMATETC pformatetc,
|
|
LPSTGMEDIUM pmedium,
|
|
BOOL fHere)
|
|
{
|
|
DOUT(TEXT("o2base/SrvrDV::GetMETAFILEPICT\r\n"));
|
|
|
|
LPVIEWOBJECT pView = (LPVIEWOBJECT)pDV;
|
|
SIZEL sizel;
|
|
pDV->GetExtent(pformatetc->dwAspect, &sizel);
|
|
|
|
RECT rc = { 0, 0, sizel.cx, sizel.cy };
|
|
HRESULT hr = NOERROR;
|
|
if (!fHere)
|
|
{
|
|
// fill in the pmedium structure
|
|
pmedium->tymed = TYMED_MFPICT;
|
|
pmedium->hGlobal = GlobalAlloc(GMEM_SHARE, sizeof(METAFILEPICT));
|
|
if (pmedium->hGlobal == NULL)
|
|
{
|
|
DOUT(TEXT("o2base/SrvrDV::GetMETAFILEPICT failed\r\n"));
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
pmedium->pUnkForRelease = NULL;
|
|
}
|
|
|
|
if (OK(hr))
|
|
{
|
|
HMETAFILE hmf;
|
|
if (OK(hr = DrawMetafile(pView, rc, pformatetc->dwAspect, &hmf)))
|
|
{
|
|
LPMETAFILEPICT pPict = (LPMETAFILEPICT)GlobalLock(pmedium->hGlobal);
|
|
if (pPict == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetMETAFILEPICT E_INVALIDARG\r\n"));
|
|
|
|
DeleteMetaFile(hmf);
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
// fill in the object descriptor
|
|
pPict->mm = MM_ANISOTROPIC;
|
|
pPict->hMF = hmf;
|
|
pPict->xExt = rc.right;
|
|
pPict->yExt = rc.bottom;
|
|
|
|
GlobalUnlock(pmedium->hGlobal);
|
|
}
|
|
}
|
|
|
|
// if we failed somehow and yet allocated memory,
|
|
// then we will release it here...
|
|
//
|
|
if (!OK(hr) && !fHere)
|
|
{
|
|
GlobalFree(pmedium->hGlobal);
|
|
pmedium->hGlobal = NULL;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetOBJECTDESCRIPTOR, public
|
|
//
|
|
// Synopsis: Implementation of IDataObject::GetData and GetDataHere
|
|
// for the standard Object Descriptor clipboard format
|
|
//
|
|
// Arguments: [pv] -- pointer to a SrvrDV object
|
|
// [pformatetc] -- as in IDataObject::GetData, GetDataHere
|
|
// [pmedium] -- as in IDataObject::GetData, GetDataHere
|
|
// [fHere] -- TRUE for GetDataHere, FALSE for GetData
|
|
//
|
|
// Returns: Success if the clipboard format could be dispensed
|
|
//
|
|
// Notes: This and the other static GetXXX methods are for use
|
|
// in the server's Get format tables.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetOBJECTDESCRIPTOR( LPSRVRDV pDV,
|
|
LPFORMATETC pformatetc,
|
|
LPSTGMEDIUM pmedium,
|
|
BOOL fHere)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPCLASSDESCRIPTOR pClass = pDV->_pClass;
|
|
|
|
//
|
|
//REVIEW: what's the best display name for the OBJECTDESCRIPTOR?
|
|
//
|
|
|
|
OLECHAR achDisplay[256];
|
|
|
|
ostrcpy(achDisplay, OLETEXT("Microsoft "));
|
|
ostrcat(achDisplay, pClass->_szUserClassType[USERCLASSTYPE_FULL]);
|
|
|
|
LPOLESTR lpstrDisplay = achDisplay;
|
|
DWORD dwDisplay = lpstrDisplay ?
|
|
(ostrlen(lpstrDisplay) + 1) * sizeof(OLECHAR) : 0;
|
|
LPOLESTR lpstrUserTypeFull = pClass->_szUserClassType[USERCLASSTYPE_FULL];
|
|
DWORD dwUserTypeFull = lpstrUserTypeFull ?
|
|
(ostrlen(lpstrUserTypeFull) + 1) * sizeof(OLECHAR) : 0;
|
|
|
|
DWORD dwSize = sizeof(OBJECTDESCRIPTOR) + dwUserTypeFull + dwDisplay;
|
|
|
|
if (!fHere)
|
|
{
|
|
// compute the amount of memory required
|
|
|
|
// fill in the pmedium structure
|
|
pmedium->tymed = TYMED_HGLOBAL;
|
|
pmedium->hGlobal = GlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE, dwSize);
|
|
if (pmedium->hGlobal == NULL)
|
|
{
|
|
DOUT(TEXT("o2base/SrvrDV::GetOBJECTDESCRIPTOR failed (pmedium)\r\n"));
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
pmedium->pUnkForRelease = NULL;
|
|
}
|
|
|
|
if (OK(hr) && (GlobalSize(pmedium->hGlobal) >= dwSize))
|
|
{
|
|
LPOBJECTDESCRIPTOR pObjDesc =
|
|
(LPOBJECTDESCRIPTOR)GlobalLock(pmedium->hGlobal);
|
|
if (pObjDesc == NULL)
|
|
{
|
|
DOUT(TEXT("o2base/SrvrDV::GetOBJECTDESCRIPTOR failed\r\n"));
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// fill in the object descriptor
|
|
//
|
|
pObjDesc->cbSize = dwSize;
|
|
pObjDesc->clsid = pClass->_clsid;
|
|
pObjDesc->dwDrawAspect = DVASPECT_CONTENT;
|
|
pObjDesc->sizel = pDV->_sizel;
|
|
pObjDesc->pointl.y = pObjDesc->pointl.x = 0;
|
|
pObjDesc->dwStatus = pClass->_dwMiscStatus;
|
|
|
|
LPOLESTR lpstrDest = (LPOLESTR)(pObjDesc + 1);
|
|
if(lpstrUserTypeFull)
|
|
{
|
|
ostrcpy(lpstrDest, lpstrUserTypeFull);
|
|
#if DBG
|
|
#if !defined(UNICODE) && !defined(OLE2ANSI)
|
|
LPSTR lp = ConvertOLESTRToMB(lpstrDest, -1);
|
|
OutputDebugStringA(lp);
|
|
OutputDebugStringA("\r\n");
|
|
TaskFreeMem(lp);
|
|
#endif
|
|
#endif
|
|
pObjDesc->dwFullUserTypeName = sizeof(OBJECTDESCRIPTOR);
|
|
lpstrDest += ostrlen(lpstrUserTypeFull) + 1;
|
|
|
|
}
|
|
else
|
|
pObjDesc->dwFullUserTypeName = 0;
|
|
|
|
if (lpstrDisplay)
|
|
{
|
|
pObjDesc->dwSrcOfCopy = pObjDesc->dwFullUserTypeName + dwUserTypeFull;
|
|
ostrcpy(lpstrDest, lpstrDisplay);
|
|
#if DBG
|
|
#if !defined(UNICODE) && !defined(OLE2ANSI)
|
|
LPSTR lp = ConvertOLESTRToMB(lpstrDest, -1);
|
|
OutputDebugStringA(lp);
|
|
OutputDebugStringA("\r\n");
|
|
TaskFreeMem(lp);
|
|
#endif
|
|
#endif
|
|
}
|
|
else
|
|
pObjDesc->dwSrcOfCopy = 0;
|
|
|
|
GlobalUnlock(pmedium->hGlobal);
|
|
hr = NOERROR;
|
|
}
|
|
|
|
// if we failed somehow and yet allocated memory,
|
|
// then we will release it here...
|
|
if (!OK(hr) && !fHere)
|
|
{
|
|
GlobalFree(pmedium->hGlobal);
|
|
pmedium->hGlobal = NULL;
|
|
}
|
|
}
|
|
else
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetLINKSOURCE, public
|
|
//
|
|
// Synopsis: Implementation of IDataObject::GetData and GetDataHere
|
|
// for the standard Link Source clipboard format
|
|
//
|
|
// Arguments: [pv] -- pointer to a SrvrDV object
|
|
// [pformatetc] -- as in IDataObject::GetData, GetDataHere
|
|
// [pmedium] -- as in IDataObject::GetData, GetDataHere
|
|
// [fHere] -- TRUE for GetDataHere, FALSE for GetData
|
|
//
|
|
// Returns: Success if the clipboard format could be dispensed
|
|
//
|
|
// Notes: This method uses the moniker cached by the data/view
|
|
// object.
|
|
// This and the other static GetXXX methods are for use
|
|
// in the server's Get format tables.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::GetLINKSOURCE( LPSRVRDV pDV,
|
|
LPFORMATETC pformatetc,
|
|
LPSTGMEDIUM pmedium,
|
|
BOOL fHere)
|
|
{
|
|
LPMONIKER pmk;
|
|
HRESULT hr;
|
|
|
|
if (OK(hr = pDV->GetMoniker(OLEGETMONIKER_FORCEASSIGN, &pmk)))
|
|
{
|
|
if (!fHere)
|
|
{
|
|
pmedium->tymed = TYMED_ISTREAM;
|
|
HRESULT hr = CreateStreamOnHGlobal(NULL, TRUE, &pmedium->pstm);
|
|
pmedium->pUnkForRelease = NULL;
|
|
}
|
|
|
|
if (OK(hr))
|
|
{
|
|
CLSID clsid;
|
|
if (OK(hr = pmk->GetClassID(&clsid)))
|
|
{
|
|
if (OK(hr = WriteClassStm(pmedium->pstm, clsid)))
|
|
hr = pmk->Save(pmedium->pstm, FALSE);
|
|
|
|
pDV->GetClassID(&clsid);
|
|
WriteClassStm(pmedium->pstm, clsid);
|
|
}
|
|
}
|
|
|
|
pmk->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+--------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetData, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method searches the server's Get format table
|
|
// for a compatible format. If one is found it calls
|
|
// the corresponding Get function.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetData(LPFORMATETC pformatetc, LPSTGMEDIUM pmedium)
|
|
{
|
|
DOUT(TEXT("SrvrDV:GetData\r\n"));
|
|
|
|
if (pformatetc == NULL || pmedium == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetData E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
HRESULT hr;
|
|
|
|
int i = FindCompatibleFormat(_pClass->_pGetFmtTable,
|
|
_pClass->_cGetFmtTable,
|
|
*pformatetc);
|
|
|
|
if (i < 0)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetData DV_E_FORMATETC\r\n"));
|
|
hr = DV_E_FORMATETC;
|
|
}
|
|
else
|
|
{
|
|
hr = (*_pGetFuncs[i]) (this, pformatetc, pmedium, FALSE);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetDataHere, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method searches the server's Get format table
|
|
// for a compatible format. If one is found it calls
|
|
// the corresponding Get function.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetDataHere(LPFORMATETC pformatetc, LPSTGMEDIUM pmedium)
|
|
{
|
|
DOUT(TEXT("SrvrDV:GetDataHere\r\n"));
|
|
|
|
HRESULT hr;
|
|
int i = FindCompatibleFormat(_pClass->_pGetFmtTable,
|
|
_pClass->_cGetFmtTable,
|
|
*pformatetc);
|
|
if (i < 0)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetDataHere DV_E_FORMATETC\r\n"));
|
|
hr = DV_E_FORMATETC;
|
|
}
|
|
else
|
|
{
|
|
hr = (*_pGetFuncs[i]) (this, pformatetc, pmedium, TRUE);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::QueryGetData, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method searches the server's Get format table
|
|
// for a compatible format. The return value indicates
|
|
// whether or not a compatible format was found.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::QueryGetData(LPFORMATETC pformatetc)
|
|
{
|
|
return FindCompatibleFormat(_pClass->_pGetFmtTable,
|
|
_pClass->_cGetFmtTable,
|
|
*pformatetc) >=0 ? NOERROR : DV_E_FORMATETC;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetCanonicalFormatEtc, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method returns DATA_S_SAMEFORMATETC assuming
|
|
// that each format the server dispenses is its own
|
|
// canonical format. If this is not the case then this
|
|
// method should be overridden.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetCanonicalFormatEtc(LPFORMATETC pformatetc,
|
|
LPFORMATETC pformatetcOut)
|
|
{
|
|
return DATA_S_SAMEFORMATETC;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SetData, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method searches the server's Set format table
|
|
// for a compatible format. If one is found it calls
|
|
// the corresponding Set function.
|
|
//
|
|
//---------------------------------------------------------------
|
|
STDMETHODIMP
|
|
SrvrDV::SetData(LPFORMATETC pformatetc, STGMEDIUM FAR *pmedium, BOOL fRelease)
|
|
{
|
|
DOUT(TEXT("SrvrDV:SetData\r\n"));
|
|
|
|
HRESULT hr;
|
|
int i = FindCompatibleFormat(_pClass->_pSetFmtTable,
|
|
_pClass->_cSetFmtTable,
|
|
*pformatetc);
|
|
if (i < 0)
|
|
{
|
|
DOUT(TEXT("SrvrDV::SetData DV_E_FORMATETC\r\n"));
|
|
hr = DV_E_FORMATETC;
|
|
}
|
|
else
|
|
{
|
|
hr = (*_pSetFuncs[i]) (this, pformatetc, pmedium);
|
|
}
|
|
|
|
if (fRelease)
|
|
{
|
|
ReleaseStgMedium(pmedium);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::EnumFormatEtc, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method creates an enumerator over the Get or
|
|
// Set format tables depending on the value of the
|
|
// dwDirection argument.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::EnumFormatEtc(DWORD dwDirection, LPENUMFORMATETC FAR* ppenumFormatEtc)
|
|
{
|
|
if (ppenumFormatEtc == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::EnumFormatEtc E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
*ppenumFormatEtc = NULL; // set out params to NULL
|
|
|
|
// create an enumerator over our static format table.
|
|
HRESULT hr = E_INVALIDARG;
|
|
switch (dwDirection)
|
|
{
|
|
case DATADIR_GET:
|
|
hr = CreateFORMATETCEnum(_pClass->_pGetFmtTable,
|
|
_pClass->_cGetFmtTable,
|
|
ppenumFormatEtc);
|
|
break;
|
|
case DATADIR_SET:
|
|
hr = CreateFORMATETCEnum(_pClass->_pSetFmtTable,
|
|
_pClass->_cSetFmtTable,
|
|
ppenumFormatEtc);
|
|
break;
|
|
default:
|
|
DOUT(TEXT("SrvrDV::EnumFormatEtc E_INVALIDARG (2)\r\n"));
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::DAdvise, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method uses the standard OLE data advise holder.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::DAdvise(FORMATETC FAR* pFormatetc,
|
|
DWORD advf,
|
|
LPADVISESINK pAdvSink,
|
|
DWORD FAR* pdwConnection)
|
|
{
|
|
if (pdwConnection == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::DAdvise E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*pdwConnection = NULL; // set out params to NULL
|
|
|
|
HRESULT hr = NOERROR;
|
|
if (_pDataAdviseHolder == NULL)
|
|
hr = CreateDataAdviseHolder(&_pDataAdviseHolder);
|
|
|
|
if (OK(hr))
|
|
{
|
|
hr = _pDataAdviseHolder->Advise((LPDATAOBJECT)this,
|
|
pFormatetc,
|
|
advf,
|
|
pAdvSink,
|
|
pdwConnection);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::DUnadvise, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method uses the standard OLE data advise holder.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::DUnadvise(DWORD dwConnection)
|
|
{
|
|
if (_pDataAdviseHolder == NULL)
|
|
return NOERROR;
|
|
|
|
return _pDataAdviseHolder->Unadvise(dwConnection);
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::EnumDAdvise, public
|
|
//
|
|
// Synopsis: Method of IDataObject interface
|
|
//
|
|
// Notes: This method uses the standard OLE data advise holder.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::EnumDAdvise(LPENUMSTATDATA FAR* ppenumAdvise)
|
|
{
|
|
if (ppenumAdvise == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::EnumDAdvise E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppenumAdvise = NULL; // set out params to NULL
|
|
|
|
HRESULT hr;
|
|
if (_pDataAdviseHolder == NULL)
|
|
hr = NOERROR;
|
|
else
|
|
hr = _pDataAdviseHolder->EnumAdvise(ppenumAdvise);
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::RenderContent, public
|
|
//
|
|
// Synopsis: Used to draw the content aspect of the object
|
|
//
|
|
// Notes: This method is used by the implementation of IViewObject::Draw
|
|
// when the content aspect is requested. The parameters are
|
|
// identical to those of IViewObject::Draw.
|
|
// All objects should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::RenderContent(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DVTARGETDEVICE FAR * ptd,
|
|
HDC hicTargetDev,
|
|
HDC hdcDraw,
|
|
LPCRECTL lprectl,
|
|
LPCRECTL lprcWBounds,
|
|
BOOL (CALLBACK * pfnContinue) (ULONG_PTR),
|
|
ULONG_PTR dwContinue)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::RenderPrint, public
|
|
//
|
|
// Synopsis: Used to draw the print aspect of the object
|
|
//
|
|
// Notes: This method is used by the implementation of IViewObject::Draw
|
|
// when the docprint aspect is requested. The parameters are
|
|
// identical to those of IViewObject::Draw.
|
|
// By default this method calls RenderContent. If the
|
|
// server has special processing for the print case then
|
|
// this method should be overridden.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::RenderPrint(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DVTARGETDEVICE FAR * ptd,
|
|
HDC hicTargetDev,
|
|
HDC hdcDraw,
|
|
LPCRECTL lprectl,
|
|
LPCRECTL lprcWBounds,
|
|
BOOL (CALLBACK * pfnContinue) (ULONG_PTR),
|
|
ULONG_PTR dwContinue)
|
|
{
|
|
return RenderContent(dwDrawAspect,
|
|
lindex,
|
|
pvAspect,
|
|
ptd,
|
|
hicTargetDev,
|
|
hdcDraw,
|
|
lprectl,
|
|
lprcWBounds,
|
|
pfnContinue,
|
|
dwContinue);
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::RenderThumbnail, public
|
|
//
|
|
// Synopsis: Used to draw the thumbnail aspect of the object
|
|
//
|
|
// Notes: This method is used by the implementation of IViewObject::Draw
|
|
// when the thumbnail aspect is requested. The parameters are
|
|
// identical to those of IViewObject::Draw.
|
|
// All objects should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::RenderThumbnail(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DVTARGETDEVICE FAR * ptd,
|
|
HDC hicTargetDev,
|
|
HDC hdcDraw,
|
|
LPCRECTL lprectl,
|
|
LPCRECTL lprcWBounds,
|
|
BOOL (CALLBACK * pfnContinue) (ULONG_PTR),
|
|
ULONG_PTR dwContinue)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Draw, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
// Notes: This method calls RenderContent/Print/Thumbnail for
|
|
// those respective aspects. It handles the icon aspect
|
|
// automatically using the icon found in the class descriptor
|
|
// indicated by the _pClass member.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Draw(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DVTARGETDEVICE FAR * ptd,
|
|
HDC hicTargetDev,
|
|
HDC hdcDraw,
|
|
LPCRECTL lprectl,
|
|
LPCRECTL lprcWBounds,
|
|
BOOL (CALLBACK * pfnContinue) (ULONG_PTR),
|
|
ULONG_PTR dwContinue)
|
|
{
|
|
HRESULT hr;
|
|
switch(dwDrawAspect) {
|
|
case DVASPECT_CONTENT:
|
|
hr = RenderContent(dwDrawAspect,
|
|
lindex,
|
|
pvAspect,
|
|
ptd,
|
|
hicTargetDev,
|
|
hdcDraw,
|
|
lprectl,
|
|
lprcWBounds,
|
|
pfnContinue,
|
|
dwContinue);
|
|
break;
|
|
|
|
case DVASPECT_DOCPRINT:
|
|
hr = RenderPrint(dwDrawAspect,
|
|
lindex,
|
|
pvAspect,
|
|
ptd,
|
|
hicTargetDev,
|
|
hdcDraw,
|
|
lprectl,
|
|
lprcWBounds,
|
|
pfnContinue,
|
|
dwContinue);
|
|
break;
|
|
|
|
case DVASPECT_THUMBNAIL:
|
|
hr = RenderThumbnail(dwDrawAspect,
|
|
lindex,
|
|
pvAspect,
|
|
ptd,
|
|
hicTargetDev,
|
|
hdcDraw,
|
|
lprectl,
|
|
lprcWBounds,
|
|
pfnContinue,
|
|
dwContinue);
|
|
break;
|
|
|
|
case DVASPECT_ICON:
|
|
{
|
|
// This is not the right way to do iconic aspect rendering!
|
|
RECT rc;
|
|
RECTL rcTemp = *lprectl;
|
|
RECTLtoRECT(rcTemp, &rc);
|
|
DrawIcon(hdcDraw, rc.left, rc.top, _pClass->_hicon);
|
|
hr = NOERROR;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
DOUT(TEXT("SrvrDV::Draw E_INVALIDARG\r\n"));
|
|
hr = E_INVALIDARG;
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetColorSet, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
// Notes: This method returns S_FALSE indicating the server
|
|
// does not support this functionality. Server's that
|
|
// wish to support it should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetColorSet(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DVTARGETDEVICE FAR * ptd,
|
|
HDC hicTargetDev,
|
|
LPLOGPALETTE FAR* ppColorSet)
|
|
{
|
|
if (ppColorSet == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetColorSet E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppColorSet = NULL; //set out params to NULL
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Freeze, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
// Notes: This method sets the frozen flag, _fFrozen.
|
|
// The derived class must pay attention to this flag
|
|
// and not allow any modifications that would change
|
|
// the current rendering.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Freeze(DWORD dwDrawAspect,
|
|
LONG lindex,
|
|
void FAR* pvAspect,
|
|
DWORD FAR* pdwFreeze)
|
|
{
|
|
if (pdwFreeze == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::Freeze E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*pdwFreeze = 0; //set out params to NULL
|
|
|
|
_fFrozen = TRUE;
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Unfreeze, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
// Notes: This method clears the frozen flag, _fFrozen.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Unfreeze(DWORD dwFreeze)
|
|
{
|
|
_fFrozen = FALSE;
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SetAdvise, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
// Notes: This method implements an advise holder for the view
|
|
// advise.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::SetAdvise(DWORD aspects, DWORD advf, LPADVISESINK pAdvSink)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
if (_pViewAdviseHolder == NULL)
|
|
hr = CreateViewAdviseHolder(&_pViewAdviseHolder);
|
|
|
|
if (OK(hr))
|
|
hr = _pViewAdviseHolder->SetAdvise(aspects, advf, pAdvSink);
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetAdvise, public
|
|
//
|
|
// Synopsis: Method of IViewObject interface
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetAdvise(DWORD FAR* pAspects,
|
|
DWORD FAR* pAdvf,
|
|
LPADVISESINK FAR* ppAdvSink)
|
|
{
|
|
if (ppAdvSink == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetAdvise E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppAdvSink = NULL; // set out params to NULL
|
|
|
|
HRESULT hr;
|
|
if (_pViewAdviseHolder==NULL)
|
|
hr = NOERROR;
|
|
else
|
|
hr = _pViewAdviseHolder->GetAdvise(pAspects, pAdvf, ppAdvSink);
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::LoadFromStream, protected
|
|
//
|
|
// Synopsis: Loads the object's persistent state from a stream
|
|
//
|
|
// Arguments: [pStrm] -- stream to load from
|
|
//
|
|
// Returns: Success iff persistent state was read
|
|
//
|
|
// Notes: This function is used in the implementation of
|
|
// IPersistStream::Load and IPersistFile::Load when
|
|
// the file is not a docfile.
|
|
// All objects should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::LoadFromStream(LPSTREAM pStrm)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SaveToStream, protected
|
|
//
|
|
// Synopsis: Saves the object's persistent state to a stream
|
|
//
|
|
// Arguments: [pStrm] -- stream to save to
|
|
//
|
|
// Returns: Success iff persistent state was written
|
|
//
|
|
// Notes: This function is used in the implementation of
|
|
// IPersistStream::Save and IPersistFile::Save when
|
|
// the file is not a docfile.
|
|
// All objects should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::SaveToStream(LPSTREAM pStrm)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetStreamSizeMax, protected
|
|
//
|
|
// Synopsis: Returns the number of bytes required to serialize object
|
|
//
|
|
// Notes: This function is used in the implementation of
|
|
// IPersistStream::GetSizeMax.
|
|
// All objects should override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
DWORD
|
|
SrvrDV::GetStreamSizeMax(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetClassID, public
|
|
//
|
|
// Synopsis: Method of IPersist interface
|
|
//
|
|
// Notes: This method uses the class id in the class descriptor.
|
|
//
|
|
//---------------------------------------------------------------
|
|
STDMETHODIMP
|
|
SrvrDV::GetClassID(LPCLSID lpClassID)
|
|
{
|
|
if (lpClassID == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetClassID E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*lpClassID = _pClass->_clsid;
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::IsDirty, public
|
|
//
|
|
// Synopsis: Method of IPersistStream/Storage/File interface
|
|
//
|
|
// Notes: This method uses the dirty flag, _fDirty.
|
|
// Objects should not set the _fDirty flag directly
|
|
// but instead call the OnDataChange method to set the
|
|
// flag.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::IsDirty(void)
|
|
{
|
|
return (_fDirty ? NOERROR : S_FALSE);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Load, public
|
|
//
|
|
// Synopsis: Method of IPersistStream interface
|
|
//
|
|
// Notes: This function uses the LoadFromStream method and
|
|
// transitions the object to the loaded state if the
|
|
// load was successful.
|
|
//
|
|
//---------------------------------------------------------------
|
|
STDMETHODIMP
|
|
SrvrDV::Load(LPSTREAM pStrm)
|
|
{
|
|
// object can be loaded only once!
|
|
if (_pCtrl->State() != OS_PASSIVE)
|
|
{
|
|
DOUT(TEXT("SrvrDV::Load E_FAIL\r\n"));
|
|
return E_FAIL;
|
|
}
|
|
|
|
HRESULT hr;
|
|
if (OK(hr = LoadFromStream(pStrm)))
|
|
hr = _pCtrl->TransitionTo(OS_LOADED);
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Save, public
|
|
//
|
|
// Synopsis: Method of IPersistStream interface
|
|
//
|
|
// Notes: This method uses the SaveToStream method and
|
|
// clears the _fDirty flag as appropriate.
|
|
// Containers that have nonserializeable embeddings can
|
|
// override this method and return STG_E_CANTSAVE
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Save(LPSTREAM pStrm, BOOL fClearDirty)
|
|
{
|
|
HRESULT hr;
|
|
if (OK(hr = SaveToStream(pStrm)))
|
|
{
|
|
if (fClearDirty)
|
|
{
|
|
_fDirty = FALSE;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetSizeMax
|
|
//
|
|
// Synopsis: Method of IPersistStream interface
|
|
//
|
|
// Notes:
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetSizeMax(ULARGE_INTEGER FAR * pcbSize)
|
|
{
|
|
if (pcbSize == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetSizeMax E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ULISet32(*pcbSize, GetStreamSizeMax());
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::LoadFromStorage, protected
|
|
//
|
|
// Synopsis: Loads the object's persistent state from a storage
|
|
//
|
|
// Arguments: [pSg] -- storage to load from
|
|
//
|
|
// Returns: Success iff persistent state was read
|
|
//
|
|
// Notes: This function is used in the implementation of
|
|
// IPersistStorage::Load and IPersistFile::Load when
|
|
// the file is a docfile.
|
|
// This method opens a stream, "CONTENTS", and uses
|
|
// method LoadFromStream to complete the load.
|
|
// Servers that do more sophisticated loading will want
|
|
// to override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::LoadFromStorage(LPSTORAGE pStg)
|
|
{
|
|
LPSTREAM pStrm;
|
|
HRESULT hr;
|
|
if (OK(hr = pStg->OpenStream(szContents, NULL, STGM_SRO, 0, &pStrm)))
|
|
{
|
|
hr = LoadFromStream(pStrm);
|
|
pStrm->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SaveToStorage
|
|
//
|
|
// Synopsis: Saves the object's persistent state to a storage
|
|
//
|
|
// Arguments: [pSg] -- storage to save to
|
|
//
|
|
// Returns: Success iff persistent state was written
|
|
//
|
|
// Notes: This function is used in the implementation of
|
|
// IPersistStorage::Save and IPersistFile::Save when
|
|
// the file is a docfile.
|
|
// This method opens a stream, "CONTENTS", and uses
|
|
// method SaveToStream to complete the save.
|
|
// Servers that do more sophisticated saving will want
|
|
// to override this method.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
HRESULT
|
|
SrvrDV::SaveToStorage(LPSTORAGE pStg, BOOL fSameAsLoad)
|
|
{
|
|
// write our native data stream
|
|
HRESULT hr;
|
|
LPSTREAM pStrm;
|
|
hr = pStg->CreateStream(szContents, STGM_SALL|STGM_CREATE, 0L, 0L, &pStrm);
|
|
if (OK(hr))
|
|
{
|
|
hr = SaveToStream(pStrm);
|
|
pStrm->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::InitNew
|
|
//
|
|
// Synopsis: IPersistStorage Method
|
|
//
|
|
// Notes: This method transitions the object to loaded.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::InitNew(LPSTORAGE pStg)
|
|
{
|
|
//
|
|
//REVIEW: what happens if we attempt to load the same ctrl more than once?
|
|
//
|
|
if (pStg == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::InitNew E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
HRESULT hr;
|
|
if (OK(hr = _pCtrl->TransitionTo(OS_LOADED)))
|
|
(_pStg = pStg)->AddRef(); // hold on to the storage
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Load
|
|
//
|
|
// Synopsis: IPersistStorage Method
|
|
//
|
|
// Notes: This method loads the object using LoadFromStorage and
|
|
// then transitions the object to loaded.
|
|
// A pointer to our storage is maintained in member variable
|
|
// _pStg.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Load(LPSTORAGE pStg)
|
|
{
|
|
// object can be loaded only once!
|
|
if (_pCtrl->State() != OS_PASSIVE)
|
|
{
|
|
DOUT(TEXT("SrvrDV::Load E_FAIL\r\n"));
|
|
return E_FAIL;
|
|
}
|
|
|
|
if (pStg == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::Load E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// do the load and move to the loaded state
|
|
HRESULT hr;
|
|
if (OK(hr = LoadFromStorage(pStg)))
|
|
{
|
|
if (OK(hr = _pCtrl->TransitionTo(OS_LOADED)))
|
|
{
|
|
(_pStg = pStg)->AddRef();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Save
|
|
//
|
|
// Synopsis: Method of IPersistStorage interface
|
|
//
|
|
// Notes: This method uses SaveToStorage to write the persistent
|
|
// state. It also writes the full user type string to the
|
|
// storage as is required.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Save(LPSTORAGE pStg, BOOL fSameAsLoad)
|
|
{
|
|
if (pStg == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::Save E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// write our native data stream
|
|
HRESULT hr;
|
|
if (OK(hr = SaveToStorage(pStg, fSameAsLoad)))
|
|
{
|
|
// Write the UserType string. We don't let this fail the operation.
|
|
WriteFmtUserTypeStg(pStg,
|
|
0,
|
|
_pClass->_szUserClassType[USERCLASSTYPE_FULL]);
|
|
|
|
_fNoScribble = TRUE;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SaveCompleted
|
|
//
|
|
// Synopsis: Method of IPersistStorage interface
|
|
//
|
|
// Notes: This method clears the dirty flag and updates our
|
|
// storage pointer, _pStg, if required.
|
|
// Servers that are also containers will want to override
|
|
// this method to pass the call recursively to all loaded
|
|
// embeddings.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::SaveCompleted(LPSTORAGE pStg)
|
|
{
|
|
// if pStg is non-null then we are coming out of HANDS-OFF mode,
|
|
// otherwise we are coming out of NO-SCRIBBLE mode.
|
|
if (pStg != NULL)
|
|
{
|
|
// We should be in HANDS-OFF mode and hence able to Assert that
|
|
// _pStg is NULL here by virtue of the HandsOffStorage call.
|
|
// However, the official OLE sample container app "Outline"
|
|
// fail to make the HandsOffStorage call.
|
|
// In order to be robust we release our _pStg handle if it is
|
|
// "illegally" NON-NULL
|
|
//
|
|
if (_pStg != NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV: WARNING! SaveCompleted: "));
|
|
DOUT(TEXT("SrvrDV: Container failed to make required HandsOffStorage call.\n"));
|
|
_pStg->Release();
|
|
}
|
|
|
|
(_pStg = pStg)->AddRef(); // hold on to the new storage
|
|
}
|
|
|
|
_fDirty = FALSE; // clear our dirty flag
|
|
_fNoScribble = FALSE; // we are out of NO-SCRIBBLE mode
|
|
|
|
//REVIEW: should we advise in the case we are not fRemembering?
|
|
if (_pCtrl != NULL)
|
|
_pCtrl->OnSave(); // and notify any advises that we have saved
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::HandsOffStorage
|
|
//
|
|
// Synopsis: Method of IPersistStorage interface
|
|
//
|
|
// Notes: This method releases the storage we are holding on to.
|
|
// Servers that are also containers will want to override
|
|
// this method to pass the call recursively to all loaded
|
|
// embeddings.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::HandsOffStorage(void)
|
|
{
|
|
if (_pStg != NULL)
|
|
_pStg->Release();
|
|
_pStg = NULL;
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Load, public
|
|
//
|
|
// Synopsis: Method of IPersistFile interface
|
|
//
|
|
// Notes: This opens the file as a docfile and uses IPersistStorage::Load
|
|
// to complete the operation.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Load(LPCOLESTR lpstrFile, DWORD grfMode)
|
|
{
|
|
|
|
// use the default storage modes if no flags were specified
|
|
if (grfMode == 0)
|
|
grfMode = STGM_DFRALL;
|
|
|
|
// if they didn't specify a share mode then use deny-write
|
|
if ( (grfMode & STGM_SHARE) == 0)
|
|
grfMode |= STGM_SHARE_DENY_WRITE;
|
|
|
|
// of course, we use transacted mode
|
|
grfMode |= STGM_TRANSACTED;
|
|
|
|
HRESULT hr;
|
|
if (lpstrFile == NULL)
|
|
{
|
|
// lpstrFile NULL is a special-case indicating that we should
|
|
// create a temporary docfile for the new file case
|
|
//
|
|
grfMode |= STGM_CREATE | STGM_DELETEONRELEASE;
|
|
|
|
LPSTORAGE pStg;
|
|
if (OK(hr = StgCreateDocfile(NULL, grfMode, 0L, &pStg)))
|
|
{
|
|
hr = InitNew(pStg);
|
|
|
|
// IPersistStorage::InitNew will hold on to the pStg
|
|
pStg->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
LPSTORAGE pStg;
|
|
if (OK(hr = StgOpenStorage(lpstrFile, NULL, grfMode, NULL, 0L, &pStg)))
|
|
{
|
|
hr = Load(pStg);
|
|
|
|
// IPersistStorage::Load will hold on to the pStg
|
|
pStg->Release();
|
|
}
|
|
//REVIEW: Is the first SetMoniker happening correctly?
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::Save, public
|
|
//
|
|
// Synopsis: Method of IPersistFile interface
|
|
//
|
|
// Notes: If a file is specified then this creates a docfile and
|
|
// uses IPersistStorage::Save to complete the operation.
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::Save(LPCOLESTR lpstrFile, BOOL fRemember)
|
|
{
|
|
// if lpstrFile is NULL that means that we should save from where we
|
|
// loaded. Otherwise create a docfile with the specified name
|
|
HRESULT hr = NOERROR;
|
|
LPSTORAGE pStg;
|
|
|
|
if (lpstrFile == NULL)
|
|
(pStg = _pStg)->AddRef();
|
|
else
|
|
hr = StgCreateDocfile(lpstrFile, STGM_DFRALL|STGM_CREATE, 0L, &pStg);
|
|
|
|
if (OK(hr))
|
|
{
|
|
hr = OleSave((LPPERSISTSTORAGE)this, pStg, pStg == _pStg);
|
|
|
|
if (OK(hr))
|
|
{
|
|
// if we are to remember this storage then release our old
|
|
// storage and hold on to the new.
|
|
// Otherwise, wrap up a storage save by the usual SaveCompleted.
|
|
if (lpstrFile != NULL && fRemember)
|
|
{
|
|
// release our previous storage or stream
|
|
// and hold on to our new
|
|
HandsOffStorage();
|
|
((LPPERSISTSTORAGE)this)->SaveCompleted(pStg);
|
|
}
|
|
else
|
|
{
|
|
// If we did a storage save and we are not switching to a new
|
|
// storage then we complete the transaction with a SaveCompleted.
|
|
((LPPERSISTSTORAGE)this)->SaveCompleted(NULL);
|
|
}
|
|
}
|
|
|
|
// Release the storage. If we are supposed to hold on
|
|
// to it then we have already add-ref'd it.
|
|
pStg->Release();
|
|
}
|
|
|
|
// if we have renamed then
|
|
if (lpstrFile != NULL)
|
|
{
|
|
// TBD: Send On_Renamed advise?
|
|
//
|
|
// inform our object of its new moniker
|
|
//
|
|
LPMONIKER pmk;
|
|
if (OK(CreateFileMoniker(lpstrFile, &pmk)))
|
|
{
|
|
_pCtrl->SetMoniker(OLEWHICHMK_OBJFULL, pmk);
|
|
pmk->Release();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::SaveCompleted, public
|
|
//
|
|
// Synopsis: Method of IPersistFile interface
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::SaveCompleted(LPCOLESTR lpstrFile)
|
|
{
|
|
//REVIEW: should we launch advise in the case we are not fRemembering?
|
|
if (_pCtrl != NULL)
|
|
_pCtrl->OnSave(); // and notify any advises that we have saved
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------
|
|
//
|
|
// Member: SrvrDV::GetCurFile, public
|
|
//
|
|
// Synopsis: Method of IPersistFile interface
|
|
//
|
|
//---------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
SrvrDV::GetCurFile(LPOLESTR FAR * ppstrFile)
|
|
{
|
|
if (ppstrFile == NULL)
|
|
{
|
|
DOUT(TEXT("SrvrDV::GetCurFile E_INVALIDARG\r\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppstrFile = 0; //set out params to NULL
|
|
|
|
HRESULT hr;
|
|
|
|
// if we don't currently have a file then return the default filename
|
|
if (_pStg == NULL)
|
|
{
|
|
// the default filename is *.ext where ext is our docfile extension
|
|
OLECHAR szDefault[6];
|
|
#if defined(OLE2ANSI)
|
|
wsprintf(szDefault, TEXT("*%s"), _pClass->_szDocfileExt);
|
|
#else
|
|
wsprintfW(szDefault, OLETEXT("*%ws"), _pClass->_szDocfileExt);
|
|
#endif
|
|
hr = TaskAllocString(szDefault, ppstrFile);
|
|
}
|
|
else
|
|
{
|
|
// the caller will free the task-allocated file name
|
|
STATSTG statstg;
|
|
if (OK(hr = _pStg->Stat(&statstg, STATFLAG_DEFAULT)))
|
|
*ppstrFile = statstg.pwcsName;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|