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.
3048 lines
90 KiB
3048 lines
90 KiB
/*************************************************************************
|
|
**
|
|
** OLE 2 Standard Utilities
|
|
**
|
|
** olestd.c
|
|
**
|
|
** This file contains utilities that are useful for most standard
|
|
** OLE 2.0 compound document type applications.
|
|
**
|
|
** (c) Copyright Microsoft Corp. 1992 All Rights Reserved
|
|
**
|
|
*************************************************************************/
|
|
|
|
// #define NONAMELESSUNION // use strict ANSI standard (for DVOBJ.H)
|
|
|
|
#define STRICT 1
|
|
#include "ole2ui.h"
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <shellapi.h>
|
|
#include "regdb.h"
|
|
#include "geticon.h"
|
|
#include "common.h"
|
|
|
|
OLEDBGDATA
|
|
|
|
static TCHAR szAssertMemAlloc[] = TEXT("CoGetMalloc failed");
|
|
|
|
static int IsCloseFormatEtc(FORMATETC FAR* pFetcLeft, FORMATETC FAR* pFetcRight);
|
|
|
|
|
|
/* OleStdSetupAdvises
|
|
** ------------------
|
|
** Setup the standard View advise required by a standard,
|
|
** compound document-oriented container. Such a container relies on
|
|
** Ole to manage the presentation of the Ole object. The container
|
|
** call IViewObject::Draw to render (display) the object.
|
|
**
|
|
** This helper routine performs the following tasks:
|
|
** setup View advise
|
|
** Call IOleObject::SetHostNames
|
|
** Call OleSetContainedObject
|
|
**
|
|
** fCreate should be set to TRUE if the object is being created. if
|
|
** an existing object is being loaded, then fCreate should be FALSE.
|
|
** if it is a creation situation, then the ADVF_PRIMEFIRST flag is
|
|
** used when settinp up the IViewObject::Advise. This will result in
|
|
** the immediate sending of the initial picture.
|
|
**
|
|
** OLE2NOTE: the standard container does NOT need to set up an OLE
|
|
** Advise (IOleObject::Advise). this routine does NOT set up an OLE
|
|
** Advise (a previous version of this function used to setup this
|
|
** advise, but it was not useful).
|
|
*/
|
|
STDAPI_(BOOL) OleStdSetupAdvises(LPOLEOBJECT lpOleObject, DWORD dwDrawAspect,
|
|
LPTSTR lpszContainerApp, LPTSTR lpszContainerObj,
|
|
LPADVISESINK lpAdviseSink, BOOL fCreate)
|
|
{
|
|
LPVIEWOBJECT lpViewObject;
|
|
HRESULT hrErr;
|
|
BOOL fStatus = TRUE;
|
|
#if defined( SPECIAL_CONTAINER )
|
|
DWORD dwTemp;
|
|
#endif
|
|
|
|
hrErr = lpOleObject->lpVtbl->QueryInterface(
|
|
lpOleObject,
|
|
&IID_IViewObject,
|
|
(LPVOID FAR*)&lpViewObject
|
|
);
|
|
|
|
/* Setup View advise */
|
|
if (hrErr == NOERROR) {
|
|
|
|
OLEDBG_BEGIN2(TEXT("IViewObject::SetAdvise called\r\n"))
|
|
lpViewObject->lpVtbl->SetAdvise(
|
|
lpViewObject,
|
|
dwDrawAspect,
|
|
(fCreate ? ADVF_PRIMEFIRST : 0),
|
|
lpAdviseSink
|
|
);
|
|
OLEDBG_END2
|
|
|
|
OleStdRelease((LPUNKNOWN)lpViewObject);
|
|
} else {
|
|
fStatus = FALSE;
|
|
}
|
|
|
|
#if defined( SPECIAL_CONTAINER )
|
|
/* Setup OLE advise.
|
|
** OLE2NOTE: normally containers do NOT need to setup an OLE
|
|
** advise. this advise connection is only useful for the OLE's
|
|
** DefHandler and the OleLink object implementation. some
|
|
** special container's might need to setup this advise for
|
|
** programatic reasons.
|
|
**
|
|
** NOTE: this advise will be torn down automatically by the
|
|
** server when we release the object, therefore we do not need
|
|
** to store the connection id.
|
|
*/
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::Advise called\r\n"))
|
|
hrErr = lpOleObject->lpVtbl->Advise(
|
|
lpOleObject,
|
|
lpAdviseSink,
|
|
(DWORD FAR*)&dwTemp
|
|
);
|
|
OLEDBG_END2
|
|
if (hrErr != NOERROR) fStatus = FALSE;
|
|
#endif
|
|
|
|
/* Setup the host names for the OLE object. */
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::SetHostNames called\r\n"))
|
|
|
|
hrErr = CallIOleObjectSetHostNamesA(
|
|
lpOleObject,
|
|
lpszContainerApp,
|
|
lpszContainerObj
|
|
);
|
|
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR) fStatus = FALSE;
|
|
|
|
/* Inform the loadded object's handler/inproc-server that it is in
|
|
** its embedding container's process.
|
|
*/
|
|
OLEDBG_BEGIN2(TEXT("OleSetContainedObject(TRUE) called\r\n"))
|
|
OleSetContainedObject((LPUNKNOWN)lpOleObject, TRUE);
|
|
OLEDBG_END2
|
|
|
|
return fStatus;
|
|
}
|
|
|
|
|
|
/* OleStdSwitchDisplayAspect
|
|
** -------------------------
|
|
** Switch the currently cached display aspect between DVASPECT_ICON
|
|
** and DVASPECT_CONTENT.
|
|
**
|
|
** NOTE: when setting up icon aspect, any currently cached content
|
|
** cache is discarded and any advise connections for content aspect
|
|
** are broken.
|
|
**
|
|
** RETURNS:
|
|
** S_OK -- new display aspect setup successfully
|
|
** E_INVALIDARG -- IOleCache interface is NOT supported (this is
|
|
** required).
|
|
** <other SCODE> -- any SCODE that can be returned by
|
|
** IOleCache::Cache method.
|
|
** NOTE: if an error occurs then the current display aspect and
|
|
** cache contents unchanged.
|
|
*/
|
|
STDAPI OleStdSwitchDisplayAspect(
|
|
LPOLEOBJECT lpOleObj,
|
|
LPDWORD lpdwCurAspect,
|
|
DWORD dwNewAspect,
|
|
HGLOBAL hMetaPict,
|
|
BOOL fDeleteOldAspect,
|
|
BOOL fSetupViewAdvise,
|
|
LPADVISESINK lpAdviseSink,
|
|
BOOL FAR* lpfMustUpdate
|
|
)
|
|
{
|
|
LPOLECACHE lpOleCache = NULL;
|
|
LPVIEWOBJECT lpViewObj = NULL;
|
|
LPENUMSTATDATA lpEnumStatData = NULL;
|
|
STATDATA StatData;
|
|
FORMATETC FmtEtc;
|
|
STGMEDIUM Medium;
|
|
DWORD dwAdvf;
|
|
DWORD dwNewConnection;
|
|
DWORD dwOldAspect = *lpdwCurAspect;
|
|
HRESULT hrErr;
|
|
|
|
if (lpfMustUpdate)
|
|
*lpfMustUpdate = FALSE;
|
|
|
|
lpOleCache = (LPOLECACHE)OleStdQueryInterface(
|
|
(LPUNKNOWN)lpOleObj,&IID_IOleCache);
|
|
|
|
// if IOleCache* is NOT available, do nothing
|
|
if (! lpOleCache)
|
|
return ResultFromScode(E_INVALIDARG);
|
|
|
|
// Setup new cache with the new aspect
|
|
FmtEtc.cfFormat = (CLIPFORMAT) NULL; // whatever is needed to draw
|
|
FmtEtc.ptd = NULL;
|
|
FmtEtc.dwAspect = dwNewAspect;
|
|
FmtEtc.lindex = -1;
|
|
FmtEtc.tymed = TYMED_NULL;
|
|
|
|
/* OLE2NOTE: if we are setting up Icon aspect with a custom icon
|
|
** then we do not want DataAdvise notifications to ever change
|
|
** the contents of the data cache. thus we set up a NODATA
|
|
** advise connection. otherwise we set up a standard DataAdvise
|
|
** connection.
|
|
*/
|
|
if (dwNewAspect == DVASPECT_ICON && hMetaPict)
|
|
dwAdvf = ADVF_NODATA;
|
|
else
|
|
dwAdvf = ADVF_PRIMEFIRST;
|
|
|
|
OLEDBG_BEGIN2(TEXT("IOleCache::Cache called\r\n"))
|
|
hrErr = lpOleCache->lpVtbl->Cache(
|
|
lpOleCache,
|
|
(LPFORMATETC)&FmtEtc,
|
|
dwAdvf,
|
|
(LPDWORD)&dwNewConnection
|
|
);
|
|
OLEDBG_END2
|
|
|
|
if (! SUCCEEDED(hrErr)) {
|
|
OleDbgOutHResult(TEXT("IOleCache::Cache returned"), hrErr);
|
|
OleStdRelease((LPUNKNOWN)lpOleCache);
|
|
return hrErr;
|
|
}
|
|
|
|
*lpdwCurAspect = dwNewAspect;
|
|
|
|
/* OLE2NOTE: if we are setting up Icon aspect with a custom icon,
|
|
** then stuff the icon into the cache. otherwise the cache must
|
|
** be forced to be updated. set the *lpfMustUpdate flag to tell
|
|
** caller to force the object to Run so that the cache will be
|
|
** updated.
|
|
*/
|
|
if (dwNewAspect == DVASPECT_ICON && hMetaPict) {
|
|
|
|
FmtEtc.cfFormat = CF_METAFILEPICT;
|
|
FmtEtc.ptd = NULL;
|
|
FmtEtc.dwAspect = DVASPECT_ICON;
|
|
FmtEtc.lindex = -1;
|
|
FmtEtc.tymed = TYMED_MFPICT;
|
|
|
|
Medium.tymed = TYMED_MFPICT;
|
|
Medium.hGlobal = hMetaPict;
|
|
Medium.pUnkForRelease = NULL;
|
|
|
|
OLEDBG_BEGIN2(TEXT("IOleCache::SetData called\r\n"))
|
|
hrErr = lpOleCache->lpVtbl->SetData(
|
|
lpOleCache,
|
|
(LPFORMATETC)&FmtEtc,
|
|
(LPSTGMEDIUM)&Medium,
|
|
FALSE /* fRelease */
|
|
);
|
|
OLEDBG_END2
|
|
} else {
|
|
if (lpfMustUpdate)
|
|
*lpfMustUpdate = TRUE;
|
|
}
|
|
|
|
if (fSetupViewAdvise && lpAdviseSink) {
|
|
/* OLE2NOTE: re-establish the ViewAdvise connection */
|
|
lpViewObj = (LPVIEWOBJECT)OleStdQueryInterface(
|
|
(LPUNKNOWN)lpOleObj,&IID_IViewObject);
|
|
|
|
if (lpViewObj) {
|
|
|
|
OLEDBG_BEGIN2(TEXT("IViewObject::SetAdvise called\r\n"))
|
|
lpViewObj->lpVtbl->SetAdvise(
|
|
lpViewObj,
|
|
dwNewAspect,
|
|
0,
|
|
lpAdviseSink
|
|
);
|
|
OLEDBG_END2
|
|
|
|
OleStdRelease((LPUNKNOWN)lpViewObj);
|
|
}
|
|
}
|
|
|
|
/* OLE2NOTE: remove any existing caches that are set up for the old
|
|
** display aspect. It WOULD be possible to retain the caches set
|
|
** up for the old aspect, but this would increase the storage
|
|
** space required for the object and possibly require additional
|
|
** overhead to maintain the unused cachaes. For these reasons the
|
|
** strategy to delete the previous caches is prefered. if it is a
|
|
** requirement to quickly switch between Icon and Content
|
|
** display, then it would be better to keep both aspect caches.
|
|
*/
|
|
|
|
if (fDeleteOldAspect) {
|
|
OLEDBG_BEGIN2(TEXT("IOleCache::EnumCache called\r\n"))
|
|
hrErr = lpOleCache->lpVtbl->EnumCache(
|
|
lpOleCache,
|
|
(LPENUMSTATDATA FAR*)&lpEnumStatData
|
|
);
|
|
OLEDBG_END2
|
|
|
|
while(hrErr == NOERROR) {
|
|
hrErr = lpEnumStatData->lpVtbl->Next(
|
|
lpEnumStatData,
|
|
1,
|
|
(LPSTATDATA)&StatData,
|
|
NULL
|
|
);
|
|
if (hrErr != NOERROR)
|
|
break; // DONE! no more caches.
|
|
|
|
if (StatData.formatetc.dwAspect == dwOldAspect) {
|
|
|
|
// Remove previous cache with old aspect
|
|
OLEDBG_BEGIN2(TEXT("IOleCache::Uncache called\r\n"))
|
|
lpOleCache->lpVtbl->Uncache(lpOleCache,StatData.dwConnection);
|
|
OLEDBG_END2
|
|
}
|
|
}
|
|
|
|
if (lpEnumStatData) {
|
|
OleStdVerifyRelease(
|
|
(LPUNKNOWN)lpEnumStatData,
|
|
TEXT("OleStdSwitchDisplayAspect: Cache enumerator NOT released")
|
|
);
|
|
}
|
|
}
|
|
|
|
if (lpOleCache)
|
|
OleStdRelease((LPUNKNOWN)lpOleCache);
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
/* OleStdSetIconInCache
|
|
** --------------------
|
|
** SetData a new icon into the existing DVASPECT_ICON cache.
|
|
**
|
|
** RETURNS:
|
|
** HRESULT returned from IOleCache::SetData
|
|
*/
|
|
STDAPI OleStdSetIconInCache(LPOLEOBJECT lpOleObj, HGLOBAL hMetaPict)
|
|
{
|
|
LPOLECACHE lpOleCache = NULL;
|
|
FORMATETC FmtEtc;
|
|
STGMEDIUM Medium;
|
|
HRESULT hrErr;
|
|
|
|
if (! hMetaPict)
|
|
return FALSE; // invalid icon
|
|
|
|
lpOleCache = (LPOLECACHE)OleStdQueryInterface(
|
|
(LPUNKNOWN)lpOleObj,&IID_IOleCache);
|
|
if (! lpOleCache)
|
|
return FALSE; // if IOleCache* is NOT available, do nothing
|
|
|
|
FmtEtc.cfFormat = CF_METAFILEPICT;
|
|
FmtEtc.ptd = NULL;
|
|
FmtEtc.dwAspect = DVASPECT_ICON;
|
|
FmtEtc.lindex = -1;
|
|
FmtEtc.tymed = TYMED_MFPICT;
|
|
|
|
// stuff the icon into the cache.
|
|
Medium.tymed = TYMED_MFPICT;
|
|
Medium.hGlobal = hMetaPict;
|
|
Medium.pUnkForRelease = NULL;
|
|
|
|
OLEDBG_BEGIN2(TEXT("IOleCache::SetData called\r\n"))
|
|
hrErr = lpOleCache->lpVtbl->SetData(
|
|
lpOleCache,
|
|
(LPFORMATETC)&FmtEtc,
|
|
(LPSTGMEDIUM)&Medium,
|
|
FALSE /* fRelease */
|
|
);
|
|
OLEDBG_END2
|
|
|
|
OleStdRelease((LPUNKNOWN)lpOleCache);
|
|
|
|
return hrErr;
|
|
}
|
|
|
|
|
|
|
|
/* OleStdDoConvert
|
|
** ---------------
|
|
** Do the container-side responsibilities for converting an object.
|
|
** This function would be used in conjunction with the OleUIConvert
|
|
** dialog. If the user selects to convert an object then the
|
|
** container must do the following:
|
|
** 1. unload the object.
|
|
** 2. write the NEW CLSID and NEW user type name
|
|
** string into the storage of the object,
|
|
** BUT write the OLD format tag.
|
|
** 3. force an update of the object to force the actual
|
|
** conversion of the data bits.
|
|
**
|
|
** This function takes care of step 2.
|
|
*/
|
|
STDAPI OleStdDoConvert(LPSTORAGE lpStg, REFCLSID rClsidNew)
|
|
{
|
|
HRESULT error;
|
|
CLSID clsidOld;
|
|
CLIPFORMAT cfOld;
|
|
LPTSTR lpszOld = NULL;
|
|
TCHAR szNew[OLEUI_CCHKEYMAX];
|
|
|
|
if ((error = ReadClassStg(lpStg, &clsidOld)) != NOERROR) {
|
|
clsidOld = CLSID_NULL;
|
|
goto errRtn;
|
|
}
|
|
|
|
// read old fmt/old user type; sets out params to NULL on error
|
|
{
|
|
LPOLESTR polestr;
|
|
|
|
error = ReadFmtUserTypeStg(lpStg, &cfOld, &polestr);
|
|
|
|
CopyAndFreeOLESTR(polestr, &lpszOld);
|
|
}
|
|
|
|
OleDbgAssert(error == NOERROR || (cfOld == 0 && lpszOld == NULL));
|
|
|
|
// get new user type name; if error, set to NULL string
|
|
if (OleStdGetUserTypeOfClass(
|
|
// (LPCLSID)
|
|
rClsidNew, szNew,sizeof(szNew),NULL /* hKey */) == 0)
|
|
szNew[0] = TEXT('\0');
|
|
|
|
// write class stg
|
|
if ((error = WriteClassStg(lpStg, rClsidNew)) != NOERROR)
|
|
goto errRtn;
|
|
|
|
// write old fmt/new user type;
|
|
#ifdef UNICODE
|
|
if ((error = WriteFmtUserTypeStg(lpStg, cfOld, szNew)) != NOERROR)
|
|
goto errRewriteInfo;
|
|
#else
|
|
{
|
|
// Chicago OLE is using UNICODE, so we need to convert the string to
|
|
// UNICODE.
|
|
WCHAR szNewT[OLEUI_CCHKEYMAX];
|
|
mbstowcs(szNewT, szNew, sizeof(szNew));
|
|
if ((error = WriteFmtUserTypeStg(lpStg, cfOld, szNewT)) != NOERROR)
|
|
goto errRewriteInfo;
|
|
}
|
|
#endif
|
|
|
|
// set convert bit
|
|
if ((error = SetConvertStg(lpStg, TRUE)) != NOERROR)
|
|
goto errRewriteInfo;
|
|
|
|
goto okRtn;
|
|
|
|
errRewriteInfo:
|
|
(void)WriteClassStg(lpStg, &clsidOld);
|
|
|
|
(void)WriteFmtUserTypeStgA(lpStg, cfOld, lpszOld);
|
|
|
|
errRtn:
|
|
|
|
okRtn:
|
|
OleStdFreeString(lpszOld, NULL);
|
|
return error;
|
|
}
|
|
|
|
|
|
/* OleStdGetTreatAsFmtUserType
|
|
** ---------------------------
|
|
** Determine if the application should perform a TreatAs (ActivateAs
|
|
** object or emulation) operation for the object that is stored in
|
|
** the storage.
|
|
**
|
|
** if the CLSID written in the storage is not the same as the
|
|
** application's own CLSID (clsidApp), then a TreatAs operation
|
|
** should take place. if so determine the format the data should be
|
|
** written and the user type name of the object the app should
|
|
** emulate (ie. pretend to be). if this information is not written
|
|
** in the storage then it is looked up in the REGDB. if it can not
|
|
** be found in the REGDB, then the TreatAs operation can NOT be
|
|
** executed.
|
|
**
|
|
** RETURNS: TRUE -- if TreatAs should be performed.
|
|
** valid lpclsid, lplpszType, lpcfFmt to TreatAs are returned
|
|
** (NOTE: lplpszType must be freed by caller)
|
|
** FALSE -- NO TreatAs. lpszType will be NULL.
|
|
** lpclsid = CLSID_NULL; lplpszType = lpcfFmt = NULL;
|
|
*/
|
|
STDAPI_(BOOL) OleStdGetTreatAsFmtUserType(
|
|
REFCLSID rclsidApp,
|
|
LPSTORAGE lpStg,
|
|
CLSID FAR* lpclsid,
|
|
CLIPFORMAT FAR* lpcfFmt,
|
|
LPTSTR FAR* lplpszType
|
|
)
|
|
{
|
|
HRESULT hrErr;
|
|
HKEY hKey;
|
|
LONG lRet;
|
|
UINT lSize;
|
|
TCHAR szBuf[OLEUI_CCHKEYMAX];
|
|
|
|
*lpclsid = CLSID_NULL;
|
|
*lpcfFmt = 0;
|
|
*lplpszType = NULL;
|
|
|
|
hrErr = ReadClassStg(lpStg, lpclsid);
|
|
if (hrErr == NOERROR &&
|
|
! IsEqualCLSID(lpclsid, &CLSID_NULL) &&
|
|
! IsEqualCLSID(lpclsid, rclsidApp)) {
|
|
|
|
hrErr = ReadFmtUserTypeStgA(lpStg,(CLIPFORMAT FAR*)lpcfFmt, lplpszType);
|
|
|
|
if (hrErr == NOERROR && lplpszType && *lpcfFmt != 0)
|
|
return TRUE; // Do TreatAs. info was in lpStg.
|
|
|
|
/* read info from REGDB
|
|
** *lpcfFmt = value of field: CLSID\{...}\DataFormats\DefaultFile
|
|
** *lplpszType = value of field: CLSID\{...}
|
|
*/
|
|
//Open up the root key.
|
|
lRet=RegOpenKey(HKEY_CLASSES_ROOT, NULL, &hKey);
|
|
if (lRet != (LONG)ERROR_SUCCESS)
|
|
return FALSE;
|
|
*lpcfFmt = OleStdGetDefaultFileFormatOfClass(lpclsid, hKey);
|
|
if (*lpcfFmt == 0)
|
|
return FALSE;
|
|
lSize = OleStdGetUserTypeOfClass(lpclsid,szBuf,sizeof(szBuf),hKey);
|
|
if (lSize == 0)
|
|
return FALSE;
|
|
*lplpszType = OleStdCopyString(szBuf, NULL);
|
|
} else {
|
|
return FALSE; // NO TreatAs
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/* OleStdDoTreatAsClass
|
|
** --------------------
|
|
** Do the container-side responsibilities for "ActivateAs" (aka.
|
|
** TreatAs) for an object.
|
|
** This function would be used in conjunction with the OleUIConvert
|
|
** dialog. If the user selects to ActivateAs an object then the
|
|
** container must do the following:
|
|
** 1. unload ALL objects of the OLD class that app knows about
|
|
** 2. add the TreatAs tag in the registration database
|
|
** by calling CoTreatAsClass().
|
|
** 3. lazily it can reload the objects; when the objects
|
|
** are reloaded the TreatAs will take effect.
|
|
**
|
|
** This function takes care of step 2.
|
|
*/
|
|
STDAPI OleStdDoTreatAsClass(LPTSTR lpszUserType, REFCLSID rclsid, REFCLSID rclsidNew)
|
|
{
|
|
HRESULT hrErr;
|
|
LPTSTR lpszCLSID = NULL;
|
|
LONG lRet;
|
|
HKEY hKey;
|
|
|
|
OLEDBG_BEGIN2(TEXT("CoTreatAsClass called\r\n"))
|
|
hrErr = CoTreatAsClass(rclsid, rclsidNew);
|
|
OLEDBG_END2
|
|
|
|
if ((hrErr != NOERROR) && lpszUserType) {
|
|
lRet = RegOpenKey(HKEY_CLASSES_ROOT, (LPCTSTR) TEXT("CLSID"),
|
|
(HKEY FAR *)&hKey);
|
|
StringFromCLSIDA(rclsid, &lpszCLSID);
|
|
|
|
RegSetValue(hKey, lpszCLSID, REG_SZ, lpszUserType,
|
|
lstrlen(lpszUserType));
|
|
|
|
if (lpszCLSID)
|
|
OleStdFreeString(lpszCLSID, NULL);
|
|
|
|
hrErr = CoTreatAsClass(rclsid, rclsidNew);
|
|
RegCloseKey(hKey);
|
|
}
|
|
|
|
return hrErr;
|
|
}
|
|
|
|
|
|
|
|
/* OleStdIsOleLink
|
|
** ---------------
|
|
** Returns TRUE if the OleObject is infact an OLE link object. this
|
|
** checks if IOleLink interface is supported. if so, the object is a
|
|
** link, otherwise not.
|
|
*/
|
|
STDAPI_(BOOL) OleStdIsOleLink(LPUNKNOWN lpUnk)
|
|
{
|
|
LPOLELINK lpOleLink;
|
|
|
|
lpOleLink = (LPOLELINK)OleStdQueryInterface(lpUnk, &IID_IOleLink);
|
|
|
|
if (lpOleLink) {
|
|
OleStdRelease((LPUNKNOWN)lpOleLink);
|
|
return TRUE;
|
|
} else
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* OleStdQueryInterface
|
|
** --------------------
|
|
** Returns the desired interface pointer if exposed by the given object.
|
|
** Returns NULL if the interface is not available.
|
|
** eg.:
|
|
** lpDataObj = OleStdQueryInterface(lpOleObj, &IID_DataObject);
|
|
*/
|
|
STDAPI_(LPUNKNOWN) OleStdQueryInterface(LPUNKNOWN lpUnk, REFIID riid)
|
|
{
|
|
LPUNKNOWN lpInterface;
|
|
HRESULT hrErr;
|
|
|
|
hrErr = lpUnk->lpVtbl->QueryInterface(
|
|
lpUnk,
|
|
riid,
|
|
(LPVOID FAR*)&lpInterface
|
|
);
|
|
|
|
if (hrErr == NOERROR)
|
|
return lpInterface;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* OleStdGetData
|
|
** -------------
|
|
** Retrieve data from an IDataObject in a specified format on a
|
|
** global memory block. This function ALWAYS returns a private copy
|
|
** of the data to the caller. if necessary a copy is made of the
|
|
** data (ie. if lpMedium->pUnkForRelease != NULL). The caller assumes
|
|
** ownership of the data block in all cases and must free the data
|
|
** when done with it. The caller may directly free the data handle
|
|
** returned (taking care whether it is a simple HGLOBAL or a HANDLE
|
|
** to a MetafilePict) or the caller may call
|
|
** ReleaseStgMedium(lpMedium). this OLE helper function will do the
|
|
** right thing.
|
|
**
|
|
** PARAMETERS:
|
|
** LPDATAOBJECT lpDataObj -- object on which GetData should be
|
|
** called.
|
|
** CLIPFORMAT cfFormat -- desired clipboard format (eg. CF_TEXT)
|
|
** DVTARGETDEVICE FAR* lpTargetDevice -- target device for which
|
|
** the data should be composed. This may
|
|
** be NULL. NULL can be used whenever the
|
|
** data format is insensitive to target
|
|
** device or when the caller does not care
|
|
** what device is used.
|
|
** LPSTGMEDIUM lpMedium -- ptr to STGMEDIUM struct. the
|
|
** resultant medium from the
|
|
** IDataObject::GetData call is
|
|
** returned.
|
|
**
|
|
** RETURNS:
|
|
** HGLOBAL -- global memory handle of retrieved data block.
|
|
** NULL -- if error.
|
|
*/
|
|
STDAPI_(HGLOBAL) OleStdGetData(
|
|
LPDATAOBJECT lpDataObj,
|
|
CLIPFORMAT cfFormat,
|
|
DVTARGETDEVICE FAR* lpTargetDevice,
|
|
DWORD dwDrawAspect,
|
|
LPSTGMEDIUM lpMedium
|
|
)
|
|
{
|
|
HRESULT hrErr;
|
|
FORMATETC formatetc;
|
|
HGLOBAL hGlobal = NULL;
|
|
HGLOBAL hCopy;
|
|
LPVOID lp;
|
|
|
|
formatetc.cfFormat = cfFormat;
|
|
formatetc.ptd = lpTargetDevice;
|
|
formatetc.dwAspect = dwDrawAspect;
|
|
formatetc.lindex = -1;
|
|
|
|
switch (cfFormat) {
|
|
case CF_METAFILEPICT:
|
|
formatetc.tymed = TYMED_MFPICT;
|
|
break;
|
|
|
|
case CF_BITMAP:
|
|
formatetc.tymed = TYMED_GDI;
|
|
break;
|
|
|
|
default:
|
|
formatetc.tymed = TYMED_HGLOBAL;
|
|
break;
|
|
}
|
|
|
|
OLEDBG_BEGIN2(TEXT("IDataObject::GetData called\r\n"))
|
|
hrErr = lpDataObj->lpVtbl->GetData(
|
|
lpDataObj,
|
|
(LPFORMATETC)&formatetc,
|
|
lpMedium
|
|
);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR)
|
|
return NULL;
|
|
|
|
if ((hGlobal = lpMedium->hGlobal) == NULL)
|
|
return NULL;
|
|
|
|
// Check if hGlobal really points to valid memory
|
|
if ((lp = GlobalLock(hGlobal)) != NULL) {
|
|
if (IsBadReadPtr(lp, 1)) {
|
|
GlobalUnlock(hGlobal);
|
|
return NULL; // ERROR: memory is NOT valid
|
|
}
|
|
GlobalUnlock(hGlobal);
|
|
}
|
|
|
|
if (hGlobal != NULL && lpMedium->pUnkForRelease != NULL) {
|
|
/* OLE2NOTE: the callee wants to retain ownership of the data.
|
|
** this is indicated by passing a non-NULL pUnkForRelease.
|
|
** thus, we will make a copy of the data and release the
|
|
** callee's copy.
|
|
*/
|
|
|
|
hCopy = OleDuplicateData(hGlobal, cfFormat, GHND|GMEM_SHARE);
|
|
ReleaseStgMedium(lpMedium); // release callee's copy of data
|
|
|
|
hGlobal = hCopy;
|
|
lpMedium->hGlobal = hCopy;
|
|
lpMedium->pUnkForRelease = NULL;
|
|
}
|
|
return hGlobal;
|
|
}
|
|
|
|
|
|
/* OleStdMalloc
|
|
** ------------
|
|
** allocate memory using the currently active IMalloc* allocator
|
|
*/
|
|
STDAPI_(LPVOID) OleStdMalloc(ULONG ulSize)
|
|
{
|
|
LPVOID pout;
|
|
LPMALLOC pmalloc;
|
|
|
|
if (CoGetMalloc(MEMCTX_TASK, &pmalloc) != NOERROR) {
|
|
OleDbgAssertSz(0, szAssertMemAlloc);
|
|
return NULL;
|
|
}
|
|
|
|
pout = (LPVOID)pmalloc->lpVtbl->Alloc(pmalloc, ulSize);
|
|
|
|
if (pmalloc != NULL) {
|
|
ULONG refs = pmalloc->lpVtbl->Release(pmalloc);
|
|
}
|
|
|
|
return pout;
|
|
}
|
|
|
|
|
|
/* OleStdRealloc
|
|
** -------------
|
|
** re-allocate memory using the currently active IMalloc* allocator
|
|
*/
|
|
STDAPI_(LPVOID) OleStdRealloc(LPVOID pmem, ULONG ulSize)
|
|
{
|
|
LPVOID pout;
|
|
LPMALLOC pmalloc;
|
|
|
|
if (CoGetMalloc(MEMCTX_TASK, &pmalloc) != NOERROR) {
|
|
OleDbgAssertSz(0, szAssertMemAlloc);
|
|
return NULL;
|
|
}
|
|
|
|
pout = (LPVOID)pmalloc->lpVtbl->Realloc(pmalloc, pmem, ulSize);
|
|
|
|
if (pmalloc != NULL) {
|
|
ULONG refs = pmalloc->lpVtbl->Release(pmalloc);
|
|
}
|
|
|
|
return pout;
|
|
}
|
|
|
|
|
|
/* OleStdFree
|
|
** ----------
|
|
** free memory using the currently active IMalloc* allocator
|
|
*/
|
|
STDAPI_(void) OleStdFree(LPVOID pmem)
|
|
{
|
|
LPMALLOC pmalloc;
|
|
|
|
if (pmem == NULL)
|
|
return;
|
|
|
|
if (CoGetMalloc(MEMCTX_TASK, &pmalloc) != NOERROR) {
|
|
OleDbgAssertSz(0, szAssertMemAlloc);
|
|
return;
|
|
}
|
|
|
|
pmalloc->lpVtbl->Free(pmalloc, pmem);
|
|
|
|
if (pmalloc != NULL) {
|
|
ULONG refs = pmalloc->lpVtbl->Release(pmalloc);
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdGetSize
|
|
** -------------
|
|
** Get the size of a memory block that was allocated using the
|
|
** currently active IMalloc* allocator.
|
|
*/
|
|
STDAPI_(ULONG) OleStdGetSize(LPVOID pmem)
|
|
{
|
|
ULONG ulSize;
|
|
LPMALLOC pmalloc;
|
|
|
|
if (CoGetMalloc(MEMCTX_TASK, &pmalloc) != NOERROR) {
|
|
OleDbgAssertSz(0, szAssertMemAlloc);
|
|
return (ULONG)-1;
|
|
}
|
|
|
|
ulSize = pmalloc->lpVtbl->GetSize(pmalloc, pmem);
|
|
|
|
if (pmalloc != NULL) {
|
|
ULONG refs = pmalloc->lpVtbl->Release(pmalloc);
|
|
}
|
|
|
|
return ulSize;
|
|
}
|
|
|
|
|
|
/* OleStdFreeString
|
|
** ----------------
|
|
** Free a string that was allocated with the currently active
|
|
** IMalloc* allocator.
|
|
**
|
|
** if the caller has the current IMalloc* handy, then it can be
|
|
** passed as a argument, otherwise this function will retrieve the
|
|
** active allocator and use it.
|
|
*/
|
|
STDAPI_(void) OleStdFreeString(LPTSTR lpsz, LPMALLOC lpMalloc)
|
|
{
|
|
BOOL fMustRelease = FALSE;
|
|
|
|
if (! lpMalloc) {
|
|
if (CoGetMalloc(MEMCTX_TASK, &lpMalloc) != NOERROR)
|
|
return;
|
|
fMustRelease = TRUE;
|
|
}
|
|
|
|
lpMalloc->lpVtbl->Free(lpMalloc, lpsz);
|
|
|
|
if (fMustRelease)
|
|
lpMalloc->lpVtbl->Release(lpMalloc);
|
|
}
|
|
|
|
|
|
/* OleStdCopyString
|
|
** ----------------
|
|
** Copy a string into memory allocated with the currently active
|
|
** IMalloc* allocator.
|
|
**
|
|
** if the caller has the current IMalloc* handy, then it can be
|
|
** passed as a argument, otherwise this function will retrieve the
|
|
** active allocator and use it.
|
|
*/
|
|
STDAPI_(LPTSTR) OleStdCopyString(LPTSTR lpszSrc, LPMALLOC lpMalloc)
|
|
{
|
|
LPTSTR lpszDest = NULL;
|
|
BOOL fMustRelease = FALSE;
|
|
UINT lSize = lstrlen(lpszSrc);
|
|
|
|
if (! lpMalloc) {
|
|
if (CoGetMalloc(MEMCTX_TASK, &lpMalloc) != NOERROR)
|
|
return NULL;
|
|
fMustRelease = TRUE;
|
|
}
|
|
|
|
lpszDest = lpMalloc->lpVtbl->Alloc(lpMalloc, (lSize+1)*sizeof(TCHAR));
|
|
|
|
if (lpszDest)
|
|
lstrcpy(lpszDest, lpszSrc);
|
|
|
|
if (fMustRelease)
|
|
lpMalloc->lpVtbl->Release(lpMalloc);
|
|
return lpszDest;
|
|
}
|
|
|
|
|
|
/*
|
|
* OleStdCreateStorageOnHGlobal()
|
|
*
|
|
* Purpose:
|
|
* Create a memory based IStorage*.
|
|
*
|
|
* OLE2NOTE: if fDeleteOnRelease==TRUE, then the ILockBytes is created
|
|
* such that it will delete them memory on its last release.
|
|
* the IStorage on created on top of the ILockBytes in NOT
|
|
* created with STGM_DELETEONRELEASE. when the IStorage receives
|
|
* its last release, it will release the ILockBytes which will
|
|
* in turn free the memory. it is in fact an error to specify
|
|
* STGM_DELETEONRELEASE in this situation.
|
|
*
|
|
* Parameters:
|
|
* hGlobal -- handle to MEM_SHARE allocated memory. may be NULL and
|
|
* memory will be automatically allocated.
|
|
* fDeleteOnRelease -- controls if the memory is freed on the last release.
|
|
* grfMode -- flags passed to StgCreateDocfileOnILockBytes
|
|
*
|
|
* NOTE: if hGlobal is NULL, then a new IStorage is created and
|
|
* STGM_CREATE flag is passed to StgCreateDocfileOnILockBytes.
|
|
* if hGlobal is non-NULL, then it is assumed that the hGlobal already
|
|
* has an IStorage inside it and STGM_CONVERT flag is passed
|
|
* to StgCreateDocfileOnILockBytes.
|
|
*
|
|
* Return Value:
|
|
* SCODE - S_OK if successful
|
|
*/
|
|
STDAPI_(LPSTORAGE) OleStdCreateStorageOnHGlobal(
|
|
HANDLE hGlobal,
|
|
BOOL fDeleteOnRelease,
|
|
DWORD grfMode
|
|
)
|
|
{
|
|
DWORD grfCreateMode=grfMode | (hGlobal==NULL ? STGM_CREATE:STGM_CONVERT);
|
|
HRESULT hrErr;
|
|
LPLOCKBYTES lpLockBytes = NULL;
|
|
DWORD reserved = 0;
|
|
LPSTORAGE lpStg = NULL;
|
|
|
|
hrErr = CreateILockBytesOnHGlobal(
|
|
hGlobal,
|
|
fDeleteOnRelease,
|
|
(LPLOCKBYTES FAR*)&lpLockBytes
|
|
);
|
|
if (hrErr != NOERROR)
|
|
return NULL;
|
|
|
|
hrErr = StgCreateDocfileOnILockBytes(
|
|
lpLockBytes,
|
|
grfCreateMode,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpStg
|
|
);
|
|
if (hrErr != NOERROR) {
|
|
OleStdRelease((LPUNKNOWN)lpLockBytes);
|
|
return NULL;
|
|
}
|
|
return lpStg;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* OleStdCreateTempStorage()
|
|
*
|
|
* Purpose:
|
|
* Create a temporay IStorage* that will DeleteOnRelease.
|
|
* this can be either memory based or file based.
|
|
*
|
|
* Parameters:
|
|
* fUseMemory -- controls if memory-based or file-based stg is created
|
|
* grfMode -- storage mode flags
|
|
*
|
|
* Return Value:
|
|
* LPSTORAGE - if successful, NULL otherwise
|
|
*/
|
|
STDAPI_(LPSTORAGE) OleStdCreateTempStorage(BOOL fUseMemory, DWORD grfMode)
|
|
{
|
|
LPSTORAGE lpstg;
|
|
HRESULT hrErr;
|
|
DWORD reserved = 0;
|
|
|
|
if (fUseMemory) {
|
|
lpstg = OleStdCreateStorageOnHGlobal(
|
|
NULL, /* auto allocate */
|
|
TRUE, /* delete on release */
|
|
grfMode
|
|
);
|
|
} else {
|
|
/* allocate a temp docfile that will delete on last release */
|
|
hrErr = StgCreateDocfile(
|
|
NULL,
|
|
grfMode | STGM_DELETEONRELEASE | STGM_CREATE,
|
|
reserved,
|
|
&lpstg
|
|
);
|
|
if (hrErr != NOERROR)
|
|
return NULL;
|
|
}
|
|
return lpstg;
|
|
}
|
|
|
|
|
|
/* OleStdGetOleObjectData
|
|
** ----------------------
|
|
** Render CF_EMBEDSOURCE/CF_EMBEDDEDOBJECT data on an TYMED_ISTORAGE
|
|
** medium by asking the object to save into the storage.
|
|
** the object must support IPersistStorage.
|
|
**
|
|
** if lpMedium->tymed == TYMED_NULL, then a delete-on-release
|
|
** storage is allocated (either file-based or memory-base depending
|
|
** the value of fUseMemory). this is useful to support an
|
|
** IDataObject::GetData call where the callee must allocate the
|
|
** medium.
|
|
**
|
|
** if lpMedium->tymed == TYMED_ISTORAGE, then the data is writen
|
|
** into the passed in IStorage. this is useful to support an
|
|
** IDataObject::GetDataHere call where the caller has allocated his
|
|
** own IStorage.
|
|
*/
|
|
STDAPI OleStdGetOleObjectData(
|
|
LPPERSISTSTORAGE lpPStg,
|
|
LPFORMATETC lpformatetc,
|
|
LPSTGMEDIUM lpMedium,
|
|
BOOL fUseMemory
|
|
)
|
|
{
|
|
LPSTORAGE lpstg = NULL;
|
|
DWORD reserved = 0;
|
|
SCODE sc = S_OK;
|
|
HRESULT hrErr;
|
|
|
|
lpMedium->pUnkForRelease = NULL;
|
|
|
|
if (lpMedium->tymed == TYMED_NULL) {
|
|
|
|
if (lpformatetc->tymed & TYMED_ISTORAGE) {
|
|
|
|
/* allocate a temp docfile that will delete on last release */
|
|
lpstg = OleStdCreateTempStorage(
|
|
TRUE /*fUseMemory*/,
|
|
STGM_READWRITE | STGM_TRANSACTED | STGM_SHARE_EXCLUSIVE
|
|
);
|
|
if (!lpstg)
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
|
|
lpMedium->pstg = lpstg;
|
|
lpMedium->tymed = TYMED_ISTORAGE;
|
|
lpMedium->pUnkForRelease = NULL;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
} else if (lpMedium->tymed == TYMED_ISTORAGE) {
|
|
lpMedium->tymed = TYMED_ISTORAGE;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
|
|
// OLE2NOTE: even if OleSave returns an error you should still call
|
|
// SaveCompleted.
|
|
|
|
OLEDBG_BEGIN2(TEXT("OleSave called\r\n"))
|
|
hrErr = OleSave(lpPStg, lpMedium->pstg, FALSE /* fSameAsLoad */);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR) {
|
|
OleDbgOutHResult(TEXT("WARNING: OleSave returned"), hrErr);
|
|
sc = GetScode(hrErr);
|
|
}
|
|
OLEDBG_BEGIN2(TEXT("IPersistStorage::SaveCompleted called\r\n"))
|
|
hrErr = lpPStg->lpVtbl->SaveCompleted(lpPStg, NULL);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR) {
|
|
OleDbgOutHResult(TEXT("WARNING: SaveCompleted returned"),hrErr);
|
|
if (sc == S_OK)
|
|
sc = GetScode(hrErr);
|
|
}
|
|
|
|
return ResultFromScode(sc);
|
|
}
|
|
|
|
|
|
STDAPI OleStdGetLinkSourceData(
|
|
LPMONIKER lpmk,
|
|
LPCLSID lpClsID,
|
|
LPFORMATETC lpformatetc,
|
|
LPSTGMEDIUM lpMedium
|
|
)
|
|
{
|
|
LPSTREAM lpstm = NULL;
|
|
DWORD reserved = 0;
|
|
HRESULT hrErr;
|
|
|
|
if (lpMedium->tymed == TYMED_NULL) {
|
|
if (lpformatetc->tymed & TYMED_ISTREAM) {
|
|
hrErr = CreateStreamOnHGlobal(
|
|
NULL, /* auto allocate */
|
|
TRUE, /* delete on release */
|
|
(LPSTREAM FAR*)&lpstm
|
|
);
|
|
if (hrErr != NOERROR) {
|
|
lpMedium->pUnkForRelease = NULL;
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
lpMedium->pstm = lpstm;
|
|
lpMedium->tymed = TYMED_ISTREAM;
|
|
lpMedium->pUnkForRelease = NULL;
|
|
} else {
|
|
lpMedium->pUnkForRelease = NULL;
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
} else {
|
|
if (lpMedium->tymed == TYMED_ISTREAM) {
|
|
lpMedium->tymed = TYMED_ISTREAM;
|
|
lpMedium->pstm = lpMedium->pstm;
|
|
lpMedium->pUnkForRelease = NULL;
|
|
} else {
|
|
lpMedium->pUnkForRelease = NULL;
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
}
|
|
|
|
hrErr = OleSaveToStream((LPPERSISTSTREAM)lpmk, lpMedium->pstm);
|
|
if (hrErr != NOERROR) return hrErr;
|
|
return WriteClassStm(lpMedium->pstm, lpClsID);
|
|
}
|
|
|
|
/*
|
|
* OleStdGetObjectDescriptorData
|
|
*
|
|
* Purpose:
|
|
* Fills and returns a OBJECTDESCRIPTOR structure.
|
|
* See OBJECTDESCRIPTOR for more information.
|
|
*
|
|
* Parameters:
|
|
* clsid CLSID CLSID of object being transferred
|
|
* dwDrawAspect DWORD Display Aspect of object
|
|
* sizel SIZEL Size of object in HIMETRIC
|
|
* pointl POINTL Offset from upper-left corner of object where mouse went
|
|
* down for drag. Meaningful only when drag-drop is used.
|
|
* dwStatus DWORD OLEMISC flags
|
|
* lpszFullUserTypeName LPSTR Full User Type Name
|
|
* lpszSrcOfCopy LPSTR Source of Copy
|
|
*
|
|
* Return Value:
|
|
* HBGLOBAL Handle to OBJECTDESCRIPTOR structure.
|
|
*/
|
|
STDAPI_(HGLOBAL) OleStdGetObjectDescriptorData(
|
|
CLSID clsid,
|
|
DWORD dwDrawAspect,
|
|
SIZEL sizel,
|
|
POINTL pointl,
|
|
DWORD dwStatus,
|
|
LPTSTR lpszFullUserTypeNameA,
|
|
LPTSTR lpszSrcOfCopyA
|
|
)
|
|
{
|
|
HGLOBAL hMem = NULL;
|
|
IBindCtx FAR *pbc = NULL;
|
|
LPOBJECTDESCRIPTOR lpOD;
|
|
DWORD dwObjectDescSize, dwFullUserTypeNameLen, dwSrcOfCopyLen;
|
|
LPOLESTR lpszFullUserTypeName,
|
|
lpszSrcOfCopy;
|
|
|
|
// convert out strings to UNICODE
|
|
|
|
if( lpszSrcOfCopyA )
|
|
{
|
|
lpszSrcOfCopy = CreateOLESTR(lpszSrcOfCopyA);
|
|
}
|
|
|
|
lpszFullUserTypeName = CreateOLESTR(lpszFullUserTypeNameA);
|
|
|
|
// Get the length of Full User Type Name; Add 1 for the null terminator
|
|
dwFullUserTypeNameLen = lpszFullUserTypeName ? wcslen(lpszFullUserTypeName)+1 : 0;
|
|
|
|
// Get the Source of Copy string and it's length; Add 1 for the null terminator
|
|
if (lpszSrcOfCopy)
|
|
dwSrcOfCopyLen = wcslen(lpszSrcOfCopy)+1;
|
|
else {
|
|
// No src moniker so use user type name as source string.
|
|
lpszSrcOfCopy = lpszFullUserTypeName;
|
|
dwSrcOfCopyLen = dwFullUserTypeNameLen;
|
|
}
|
|
|
|
// Allocate space for OBJECTDESCRIPTOR and the additional string data
|
|
dwObjectDescSize = sizeof(OBJECTDESCRIPTOR);
|
|
hMem = GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE, dwObjectDescSize +
|
|
(dwFullUserTypeNameLen + dwSrcOfCopyLen)*sizeof(OLECHAR));
|
|
if (NULL == hMem)
|
|
goto error;
|
|
|
|
lpOD = (LPOBJECTDESCRIPTOR)GlobalLock(hMem);
|
|
|
|
// Set the FullUserTypeName offset and copy the string
|
|
if (lpszFullUserTypeName)
|
|
{
|
|
lpOD->dwFullUserTypeName = dwObjectDescSize;
|
|
wcscpy((LPOLESTR)(((BYTE FAR *)lpOD)+lpOD->dwFullUserTypeName),
|
|
lpszFullUserTypeName);
|
|
}
|
|
else lpOD->dwFullUserTypeName = 0; // zero offset indicates that string is not present
|
|
|
|
// Set the SrcOfCopy offset and copy the string
|
|
if (lpszSrcOfCopy)
|
|
{
|
|
lpOD->dwSrcOfCopy = dwObjectDescSize +
|
|
dwFullUserTypeNameLen*sizeof(OLECHAR);
|
|
wcscpy((LPOLESTR)(((BYTE FAR *)lpOD)+lpOD->dwSrcOfCopy), lpszSrcOfCopy);
|
|
}
|
|
else lpOD->dwSrcOfCopy = 0; // zero offset indicates that string is not present
|
|
|
|
// Initialize the rest of the OBJECTDESCRIPTOR
|
|
lpOD->cbSize = dwObjectDescSize +
|
|
(dwFullUserTypeNameLen + dwSrcOfCopyLen)*sizeof(OLECHAR);
|
|
lpOD->clsid = clsid;
|
|
lpOD->dwDrawAspect = dwDrawAspect;
|
|
lpOD->sizel = sizel;
|
|
lpOD->pointl = pointl;
|
|
lpOD->dwStatus = dwStatus;
|
|
|
|
GlobalUnlock(hMem);
|
|
|
|
FREEOLESTR(lpszFullUserTypeName);
|
|
FREEOLESTR(lpszSrcOfCopy);
|
|
|
|
return hMem;
|
|
|
|
error:
|
|
if (hMem)
|
|
{
|
|
GlobalUnlock(hMem);
|
|
GlobalFree(hMem);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* OleStdGetObjectDescriptorDataFromOleObject
|
|
*
|
|
* Purpose:
|
|
* Fills and returns a OBJECTDESCRIPTOR structure. Information for the structure is
|
|
* obtained from an OLEOBJECT.
|
|
* See OBJECTDESCRIPTOR for more information.
|
|
*
|
|
* Parameters:
|
|
* lpOleObj LPOLEOBJECT OleObject from which ONJECTDESCRIPTOR info
|
|
* is obtained.
|
|
* lpszSrcOfCopy LPSTR string to identify source of copy.
|
|
* May be NULL in which case IOleObject::GetMoniker is called
|
|
* to get the moniker of the object. if the object is loaded
|
|
* as part of a data transfer document, then usually
|
|
* lpOleClientSite==NULL is passed to OleLoad when loading
|
|
* the object. in this case the IOleObject:GetMoniker call
|
|
* will always fail (it tries to call back to the object's
|
|
* client site). in this situation a non-NULL lpszSrcOfCopy
|
|
* parameter should be passed.
|
|
* dwDrawAspect DWORD Display Aspect of object
|
|
* pointl POINTL Offset from upper-left corner of object where
|
|
* mouse went down for drag. Meaningful only when drag-drop
|
|
* is used.
|
|
* lpSizelHim SIZEL (optional) If the object is being scaled in its
|
|
* container, then the container should pass the extents
|
|
* that it is using to display the object.
|
|
* May be NULL if the object is NOT being scaled. in this
|
|
* case, IViewObject2::GetExtent will be called to get the
|
|
* extents from the object.
|
|
*
|
|
* Return Value:
|
|
* HBGLOBAL Handle to OBJECTDESCRIPTOR structure.
|
|
*/
|
|
|
|
STDAPI_(HGLOBAL) OleStdGetObjectDescriptorDataFromOleObject(
|
|
LPOLEOBJECT lpOleObj,
|
|
LPTSTR lpszSrcOfCopy,
|
|
DWORD dwDrawAspect,
|
|
POINTL pointl,
|
|
LPSIZEL lpSizelHim
|
|
)
|
|
{
|
|
CLSID clsid;
|
|
LPTSTR lpszFullUserTypeName = NULL;
|
|
LPMONIKER lpSrcMonikerOfCopy = NULL;
|
|
HGLOBAL hObjDesc;
|
|
IBindCtx FAR *pbc = NULL;
|
|
HRESULT hrErr;
|
|
SIZEL sizelHim;
|
|
BOOL fFreeSrcOfCopy = FALSE;
|
|
LPOLELINK lpOleLink = (LPOLELINK)
|
|
OleStdQueryInterface((LPUNKNOWN)lpOleObj,&IID_IOleLink);
|
|
|
|
#ifdef OLE201
|
|
LPVIEWOBJECT2 lpViewObj2 = (LPVIEWOBJECT2)
|
|
OleStdQueryInterface((LPUNKNOWN)lpOleObj, &IID_IViewObject2);
|
|
#endif
|
|
|
|
BOOL fIsLink = (lpOleLink ? TRUE : FALSE);
|
|
TCHAR szLinkedTypeFmt[80];
|
|
LPTSTR lpszBuf = NULL;
|
|
DWORD dwStatus = 0;
|
|
|
|
// Get CLSID
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::GetUserClassID called\r\n"))
|
|
hrErr = lpOleObj->lpVtbl->GetUserClassID(lpOleObj, &clsid);
|
|
OLEDBG_END2
|
|
if (hrErr != NOERROR)
|
|
clsid = CLSID_NULL;
|
|
|
|
// Get FullUserTypeName
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::GetUserType called\r\n"))
|
|
{
|
|
LPOLESTR polestr;
|
|
|
|
hrErr = lpOleObj->lpVtbl->GetUserType(
|
|
lpOleObj,
|
|
USERCLASSTYPE_FULL,
|
|
&polestr
|
|
);
|
|
|
|
CopyAndFreeOLESTR(polestr, &lpszFullUserTypeName);
|
|
}
|
|
|
|
OLEDBG_END2
|
|
|
|
// REVIEW: added IDS_OLE2UILINKEDTYPE to strings.rc
|
|
/* if object is a link, then expand usertypename to be "Linked %s" */
|
|
if (fIsLink && lpszFullUserTypeName) {
|
|
if (0 == LoadString(ghInst, IDS_OLE2UIPASTELINKEDTYPE,
|
|
(LPTSTR)szLinkedTypeFmt, sizeof(szLinkedTypeFmt)/sizeof(TCHAR)))
|
|
lstrcpy(szLinkedTypeFmt, (LPTSTR) TEXT("Linked %s"));
|
|
lpszBuf = OleStdMalloc(
|
|
(lstrlen(lpszFullUserTypeName)+lstrlen(szLinkedTypeFmt)+1) *
|
|
sizeof(TCHAR));
|
|
if (lpszBuf) {
|
|
wsprintf(lpszBuf, szLinkedTypeFmt, lpszFullUserTypeName);
|
|
OleStdFreeString(lpszFullUserTypeName, NULL);
|
|
lpszFullUserTypeName = lpszBuf;
|
|
}
|
|
}
|
|
|
|
/* Get Source Of Copy
|
|
** if the object is an embedding, then get the object's moniker
|
|
** if the object is a link, then get the link source moniker
|
|
*/
|
|
if (fIsLink) {
|
|
|
|
OLEDBG_BEGIN2(TEXT("IOleLink::GetSourceDisplayName called\r\n"))
|
|
|
|
{
|
|
LPOLESTR polestr;
|
|
|
|
hrErr = lpOleLink->lpVtbl->GetSourceDisplayName(
|
|
lpOleLink, &polestr );
|
|
|
|
CopyAndFreeOLESTR(polestr, &lpszSrcOfCopy);
|
|
}
|
|
OLEDBG_END2
|
|
fFreeSrcOfCopy = TRUE;
|
|
|
|
} else {
|
|
|
|
if (lpszSrcOfCopy == NULL) {
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::GetMoniker called\r\n"))
|
|
hrErr = lpOleObj->lpVtbl->GetMoniker(
|
|
lpOleObj,
|
|
OLEGETMONIKER_TEMPFORUSER,
|
|
OLEWHICHMK_OBJFULL,
|
|
(LPMONIKER FAR*)&lpSrcMonikerOfCopy
|
|
);
|
|
OLEDBG_END2
|
|
if (hrErr == NOERROR)
|
|
{
|
|
#ifdef OLE201
|
|
CreateBindCtx(0, (LPBC FAR*)&pbc);
|
|
#endif
|
|
CallIMonikerGetDisplayNameA(
|
|
lpSrcMonikerOfCopy, pbc, NULL, &lpszSrcOfCopy);
|
|
|
|
pbc->lpVtbl->Release(pbc);
|
|
fFreeSrcOfCopy = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get SIZEL
|
|
if (lpSizelHim) {
|
|
// Use extents passed by the caller
|
|
sizelHim = *lpSizelHim;
|
|
} else if (lpViewObj2) {
|
|
// Get the current extents from the object
|
|
OLEDBG_BEGIN2(TEXT("IViewObject2::GetExtent called\r\n"))
|
|
hrErr = lpViewObj2->lpVtbl->GetExtent(
|
|
lpViewObj2,
|
|
dwDrawAspect,
|
|
-1, /*lindex*/
|
|
NULL, /*ptd*/
|
|
(LPSIZEL)&sizelHim
|
|
);
|
|
OLEDBG_END2
|
|
if (hrErr != NOERROR)
|
|
sizelHim.cx = sizelHim.cy = 0;
|
|
} else {
|
|
sizelHim.cx = sizelHim.cy = 0;
|
|
}
|
|
|
|
// Get DWSTATUS
|
|
OLEDBG_BEGIN2(TEXT("IOleObject::GetMiscStatus called\r\n"))
|
|
hrErr = lpOleObj->lpVtbl->GetMiscStatus(
|
|
lpOleObj,
|
|
dwDrawAspect,
|
|
&dwStatus
|
|
);
|
|
OLEDBG_END2
|
|
if (hrErr != NOERROR)
|
|
dwStatus = 0;
|
|
|
|
// Get OBJECTDESCRIPTOR
|
|
hObjDesc = OleStdGetObjectDescriptorData(
|
|
clsid,
|
|
dwDrawAspect,
|
|
sizelHim,
|
|
pointl,
|
|
dwStatus,
|
|
lpszFullUserTypeName,
|
|
lpszSrcOfCopy
|
|
);
|
|
if (! hObjDesc)
|
|
goto error;
|
|
|
|
// Clean up
|
|
if (lpszFullUserTypeName)
|
|
OleStdFreeString(lpszFullUserTypeName, NULL);
|
|
if (fFreeSrcOfCopy && lpszSrcOfCopy)
|
|
OleStdFreeString(lpszSrcOfCopy, NULL);
|
|
if (lpSrcMonikerOfCopy)
|
|
OleStdRelease((LPUNKNOWN)lpSrcMonikerOfCopy);
|
|
if (lpOleLink)
|
|
OleStdRelease((LPUNKNOWN)lpOleLink);
|
|
if (lpViewObj2)
|
|
OleStdRelease((LPUNKNOWN)lpViewObj2);
|
|
|
|
return hObjDesc;
|
|
|
|
error:
|
|
if (lpszFullUserTypeName)
|
|
OleStdFreeString(lpszFullUserTypeName, NULL);
|
|
if (fFreeSrcOfCopy && lpszSrcOfCopy)
|
|
OleStdFreeString(lpszSrcOfCopy, NULL);
|
|
if (lpSrcMonikerOfCopy)
|
|
OleStdRelease((LPUNKNOWN)lpSrcMonikerOfCopy);
|
|
if (lpOleLink)
|
|
OleStdRelease((LPUNKNOWN)lpOleLink);
|
|
if (lpViewObj2)
|
|
OleStdRelease((LPUNKNOWN)lpViewObj2);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* OleStdFillObjectDescriptorFromData
|
|
*
|
|
* Purpose:
|
|
* Fills and returns a OBJECTDESCRIPTOR structure. The source object will
|
|
* offer CF_OBJECTDESCRIPTOR if it is an OLE2 object, CF_OWNERLINK if it
|
|
* is an OLE1 object, or CF_FILENAME if it has been copied to the clipboard
|
|
* by FileManager.
|
|
*
|
|
* Parameters:
|
|
* lpDataObject LPDATAOBJECT Source object
|
|
* lpmedium LPSTGMEDIUM Storage medium
|
|
* lpcfFmt CLIPFORMAT FAR * Format offered by lpDataObject
|
|
* (OUT parameter)
|
|
*
|
|
* Return Value:
|
|
* HBGLOBAL Handle to OBJECTDESCRIPTOR structure.
|
|
*/
|
|
|
|
STDAPI_(HGLOBAL) OleStdFillObjectDescriptorFromData(
|
|
LPDATAOBJECT lpDataObject,
|
|
LPSTGMEDIUM lpmedium,
|
|
CLIPFORMAT FAR* lpcfFmt
|
|
)
|
|
{
|
|
CLSID clsid;
|
|
SIZEL sizelHim;
|
|
POINTL pointl;
|
|
LPTSTR lpsz, szFullUserTypeName, szSrcOfCopy, szClassName, szDocName, szItemName;
|
|
int nClassName, nDocName, nItemName, nFullUserTypeName;
|
|
LPTSTR szBuf = NULL;
|
|
HGLOBAL hMem = NULL;
|
|
HKEY hKey = NULL;
|
|
LPMALLOC pIMalloc = NULL;
|
|
DWORD dw = OLEUI_CCHKEYMAX_SIZE;
|
|
HGLOBAL hObjDesc;
|
|
HRESULT hrErr;
|
|
|
|
|
|
// GetData CF_OBJECTDESCRIPTOR format from the object on the clipboard.
|
|
// Only OLE 2 objects on the clipboard will offer CF_OBJECTDESCRIPTOR
|
|
if (hMem = OleStdGetData(
|
|
lpDataObject,
|
|
(CLIPFORMAT) cfObjectDescriptor,
|
|
NULL,
|
|
DVASPECT_CONTENT,
|
|
lpmedium))
|
|
{
|
|
*lpcfFmt = cfObjectDescriptor;
|
|
return hMem; // Don't drop to clean up at the end of this function
|
|
}
|
|
// If CF_OBJECTDESCRIPTOR is not available, i.e. if this is not an OLE2 object,
|
|
// check if this is an OLE 1 object. OLE 1 objects will offer CF_OWNERLINK
|
|
else if (hMem = OleStdGetData(
|
|
lpDataObject,
|
|
(CLIPFORMAT) cfOwnerLink,
|
|
NULL,
|
|
DVASPECT_CONTENT,
|
|
lpmedium))
|
|
{
|
|
*lpcfFmt = cfOwnerLink;
|
|
// CF_OWNERLINK contains null-terminated strings for class name, document name
|
|
// and item name with two null terminating characters at the end
|
|
szClassName = (LPTSTR)GlobalLock(hMem);
|
|
nClassName = lstrlen(szClassName);
|
|
szDocName = szClassName + nClassName + 1;
|
|
nDocName = lstrlen(szDocName);
|
|
szItemName = szDocName + nDocName + 1;
|
|
nItemName = lstrlen(szItemName);
|
|
|
|
hrErr = CoGetMalloc(MEMCTX_TASK, &pIMalloc);
|
|
if (hrErr != NOERROR)
|
|
goto error;
|
|
|
|
// Find FullUserTypeName from Registration database using class name
|
|
if (RegOpenKey(HKEY_CLASSES_ROOT, NULL, &hKey) != ERROR_SUCCESS)
|
|
goto error;
|
|
|
|
// Allocate space for szFullUserTypeName & szSrcOfCopy. Maximum length of FullUserTypeName
|
|
// is OLEUI_CCHKEYMAX_SIZE. SrcOfCopy is constructed by concatenating FullUserTypeName, Document
|
|
// Name and ItemName separated by spaces.
|
|
szBuf = (LPTSTR)pIMalloc->lpVtbl->Alloc(pIMalloc,
|
|
(DWORD)2*OLEUI_CCHKEYMAX_SIZE+
|
|
(nDocName+nItemName+4)*sizeof(TCHAR));
|
|
if (NULL == szBuf)
|
|
goto error;
|
|
szFullUserTypeName = szBuf;
|
|
szSrcOfCopy = szFullUserTypeName+OLEUI_CCHKEYMAX_SIZE+1;
|
|
|
|
// Get FullUserTypeName
|
|
if (RegQueryValue(hKey, NULL, szFullUserTypeName, &dw) != ERROR_SUCCESS)
|
|
goto error;
|
|
|
|
// Build up SrcOfCopy string from FullUserTypeName, DocumentName & ItemName
|
|
lpsz = szSrcOfCopy;
|
|
lstrcpy(lpsz, szFullUserTypeName);
|
|
nFullUserTypeName = lstrlen(szFullUserTypeName);
|
|
lpsz[nFullUserTypeName]= TEXT(' ');
|
|
lpsz += nFullUserTypeName+1;
|
|
lstrcpy(lpsz, szDocName);
|
|
lpsz[nDocName] = TEXT(' ');
|
|
lpsz += nDocName+1;
|
|
lstrcpy(lpsz, szItemName);
|
|
|
|
sizelHim.cx = sizelHim.cy = 0;
|
|
pointl.x = pointl.y = 0;
|
|
|
|
CLSIDFromProgIDA(szClassName, &clsid);
|
|
|
|
hObjDesc = OleStdGetObjectDescriptorData(
|
|
clsid,
|
|
DVASPECT_CONTENT,
|
|
sizelHim,
|
|
pointl,
|
|
0,
|
|
szFullUserTypeName,
|
|
szSrcOfCopy
|
|
);
|
|
if (!hObjDesc)
|
|
goto error;
|
|
}
|
|
// Check if object is CF_FILENAME
|
|
else if (hMem = OleStdGetData(
|
|
lpDataObject,
|
|
(CLIPFORMAT) cfFileName,
|
|
NULL,
|
|
DVASPECT_CONTENT,
|
|
lpmedium))
|
|
{
|
|
*lpcfFmt = cfFileName;
|
|
lpsz = (LPTSTR)GlobalLock(hMem);
|
|
|
|
hrErr = GetClassFileA(lpsz, &clsid);
|
|
|
|
/* OLE2NOTE: if the file does not have an OLE class
|
|
** associated, then use the OLE 1 Packager as the class of
|
|
** the object to be created. this is the behavior of
|
|
** OleCreateFromData API
|
|
*/
|
|
if (hrErr != NOERROR)
|
|
CLSIDFromProgIDA("Package", &clsid);
|
|
sizelHim.cx = sizelHim.cy = 0;
|
|
pointl.x = pointl.y = 0;
|
|
|
|
hrErr = CoGetMalloc(MEMCTX_TASK, &pIMalloc);
|
|
if (hrErr != NOERROR)
|
|
goto error;
|
|
szBuf = (LPTSTR)pIMalloc->lpVtbl->Alloc(pIMalloc, (DWORD)OLEUI_CCHKEYMAX_SIZE);
|
|
if (NULL == szBuf)
|
|
goto error;
|
|
|
|
OleStdGetUserTypeOfClass(&clsid, szBuf, OLEUI_CCHKEYMAX_SIZE, NULL);
|
|
|
|
hObjDesc = OleStdGetObjectDescriptorData(
|
|
clsid,
|
|
DVASPECT_CONTENT,
|
|
sizelHim,
|
|
pointl,
|
|
0,
|
|
szBuf,
|
|
lpsz
|
|
);
|
|
if (!hObjDesc)
|
|
goto error;
|
|
}
|
|
else goto error;
|
|
|
|
// Clean up
|
|
if (szBuf)
|
|
pIMalloc->lpVtbl->Free(pIMalloc, (LPVOID)szBuf);
|
|
if (pIMalloc)
|
|
pIMalloc->lpVtbl->Release(pIMalloc);
|
|
if (hMem)
|
|
{
|
|
GlobalUnlock(hMem);
|
|
GlobalFree(hMem);
|
|
}
|
|
if (hKey)
|
|
RegCloseKey(hKey);
|
|
return hObjDesc;
|
|
|
|
error:
|
|
if (szBuf)
|
|
pIMalloc->lpVtbl->Free(pIMalloc, (LPVOID)szBuf);
|
|
if (pIMalloc)
|
|
pIMalloc->lpVtbl->Release(pIMalloc);
|
|
if (hMem)
|
|
{
|
|
GlobalUnlock(hMem);
|
|
GlobalFree(hMem);
|
|
}
|
|
if (hKey)
|
|
RegCloseKey(hKey);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
#if defined( OBSOLETE )
|
|
|
|
/*************************************************************************
|
|
** The following API's have been converted into macros:
|
|
** OleStdQueryOleObjectData
|
|
** OleStdQueryLinkSourceData
|
|
** OleStdQueryObjectDescriptorData
|
|
** OleStdQueryFormatMedium
|
|
** OleStdCopyMetafilePict
|
|
** OleStdGetDropEffect
|
|
**
|
|
** These macros are defined in olestd.h
|
|
*************************************************************************/
|
|
|
|
STDAPI OleStdQueryOleObjectData(LPFORMATETC lpformatetc)
|
|
{
|
|
if (lpformatetc->tymed & TYMED_ISTORAGE) {
|
|
return NOERROR;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
}
|
|
|
|
|
|
STDAPI OleStdQueryLinkSourceData(LPFORMATETC lpformatetc)
|
|
{
|
|
if (lpformatetc->tymed & TYMED_ISTREAM) {
|
|
return NOERROR;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
}
|
|
|
|
|
|
STDAPI OleStdQueryObjectDescriptorData(LPFORMATETC lpformatetc)
|
|
{
|
|
if (lpformatetc->tymed & TYMED_HGLOBAL) {
|
|
return NOERROR;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
}
|
|
|
|
|
|
STDAPI OleStdQueryFormatMedium(LPFORMATETC lpformatetc, TYMED tymed)
|
|
{
|
|
if (lpformatetc->tymed & tymed) {
|
|
return NOERROR;
|
|
} else {
|
|
return ResultFromScode(DATA_E_FORMATETC);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* OleStdCopyMetafilePict()
|
|
*
|
|
* Purpose:
|
|
* Make an independent copy of a MetafilePict
|
|
* Parameters:
|
|
*
|
|
* Return Value:
|
|
* TRUE if successful, else FALSE.
|
|
*/
|
|
STDAPI_(BOOL) OleStdCopyMetafilePict(HANDLE hpictin, HANDLE FAR* phpictout)
|
|
{
|
|
HANDLE hpictout;
|
|
LPMETAFILEPICT ppictin, ppictout;
|
|
|
|
if (hpictin == NULL || phpictout == NULL) {
|
|
OleDbgAssert(hpictin == NULL || phpictout == NULL);
|
|
return FALSE;
|
|
}
|
|
|
|
*phpictout = NULL;
|
|
|
|
if ((ppictin = (LPMETAFILEPICT)GlobalLock(hpictin)) == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
hpictout = GlobalAlloc(GHND|GMEM_SHARE, sizeof(METAFILEPICT));
|
|
|
|
if (hpictout && (ppictout = (LPMETAFILEPICT)GlobalLock(hpictout))){
|
|
ppictout->hMF = CopyMetaFile(ppictin->hMF, NULL);
|
|
ppictout->xExt = ppictin->xExt;
|
|
ppictout->yExt = ppictin->yExt;
|
|
ppictout->mm = ppictin->mm;
|
|
GlobalUnlock(hpictout);
|
|
}
|
|
|
|
*phpictout = hpictout;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
/* OleStdGetDropEffect
|
|
** -------------------
|
|
**
|
|
** Convert a keyboard state into a DROPEFFECT.
|
|
**
|
|
** returns the DROPEFFECT value derived from the key state.
|
|
** the following is the standard interpretation:
|
|
** no modifier -- Default Drop (NULL is returned)
|
|
** CTRL -- DROPEFFECT_COPY
|
|
** SHIFT -- DROPEFFECT_MOVE
|
|
** CTRL-SHIFT -- DROPEFFECT_LINK
|
|
**
|
|
** Default Drop: this depends on the type of the target application.
|
|
** this is re-interpretable by each target application. a typical
|
|
** interpretation is if the drag is local to the same document
|
|
** (which is source of the drag) then a MOVE operation is
|
|
** performed. if the drag is not local, then a COPY operation is
|
|
** performed.
|
|
*/
|
|
STDAPI_(DWORD) OleStdGetDropEffect( DWORD grfKeyState )
|
|
{
|
|
|
|
if (grfKeyState & MK_CONTROL) {
|
|
|
|
if (grfKeyState & MK_SHIFT)
|
|
return DROPEFFECT_LINK;
|
|
else
|
|
return DROPEFFECT_COPY;
|
|
|
|
} else if (grfKeyState & MK_SHIFT)
|
|
return DROPEFFECT_MOVE;
|
|
|
|
return 0; // no modifier -- do default operation
|
|
}
|
|
#endif // OBSOLETE
|
|
|
|
|
|
/*
|
|
* OleStdGetMetafilePictFromOleObject()
|
|
*
|
|
* Purpose:
|
|
* Generate a MetafilePict by drawing the OLE object.
|
|
* Parameters:
|
|
* lpOleObj LPOLEOBJECT pointer to OLE Object
|
|
* dwDrawAspect DWORD Display Aspect of object
|
|
* lpSizelHim SIZEL (optional) If the object is being scaled in its
|
|
* container, then the container should pass the extents
|
|
* that it is using to display the object.
|
|
* May be NULL if the object is NOT being scaled. in this
|
|
* case, IViewObject2::GetExtent will be called to get the
|
|
* extents from the object.
|
|
* ptd TARGETDEVICE FAR* (optional) target device to render
|
|
* metafile for. May be NULL.
|
|
*
|
|
* Return Value:
|
|
* HANDLE -- handle of allocated METAFILEPICT
|
|
*/
|
|
STDAPI_(HANDLE) OleStdGetMetafilePictFromOleObject(
|
|
LPOLEOBJECT lpOleObj,
|
|
DWORD dwDrawAspect,
|
|
LPSIZEL lpSizelHim,
|
|
DVTARGETDEVICE FAR* ptd
|
|
)
|
|
{
|
|
LPVIEWOBJECT2 lpViewObj2 = NULL;
|
|
HDC hDC;
|
|
HMETAFILE hmf;
|
|
HANDLE hMetaPict;
|
|
LPMETAFILEPICT lpPict;
|
|
RECT rcHim;
|
|
RECTL rclHim;
|
|
SIZEL sizelHim;
|
|
HRESULT hrErr;
|
|
SIZE size;
|
|
POINT point;
|
|
|
|
#ifdef OLE201
|
|
lpViewObj2 = (LPVIEWOBJECT2)OleStdQueryInterface(
|
|
(LPUNKNOWN)lpOleObj, &IID_IViewObject2);
|
|
#endif
|
|
|
|
if (! lpViewObj2)
|
|
return NULL;
|
|
|
|
// Get SIZEL
|
|
if (lpSizelHim) {
|
|
// Use extents passed by the caller
|
|
sizelHim = *lpSizelHim;
|
|
} else {
|
|
// Get the current extents from the object
|
|
OLEDBG_BEGIN2(TEXT("IViewObject2::GetExtent called\r\n"))
|
|
hrErr = lpViewObj2->lpVtbl->GetExtent(
|
|
lpViewObj2,
|
|
dwDrawAspect,
|
|
-1, /*lindex*/
|
|
ptd, /*ptd*/
|
|
(LPSIZEL)&sizelHim
|
|
);
|
|
OLEDBG_END2
|
|
if (hrErr != NOERROR)
|
|
sizelHim.cx = sizelHim.cy = 0;
|
|
}
|
|
|
|
hDC = CreateMetaFile(NULL);
|
|
|
|
rclHim.left = 0;
|
|
rclHim.top = 0;
|
|
rclHim.right = sizelHim.cx;
|
|
rclHim.bottom = sizelHim.cy;
|
|
|
|
rcHim.left = (int)rclHim.left;
|
|
rcHim.top = (int)rclHim.top;
|
|
rcHim.right = (int)rclHim.right;
|
|
rcHim.bottom = (int)rclHim.bottom;
|
|
|
|
SetWindowOrgEx(hDC, rcHim.left, rcHim.top, &point);
|
|
SetWindowExtEx(hDC, rcHim.right-rcHim.left, rcHim.bottom-rcHim.top,&size);
|
|
|
|
OLEDBG_BEGIN2(TEXT("IViewObject::Draw called\r\n"))
|
|
hrErr = lpViewObj2->lpVtbl->Draw(
|
|
lpViewObj2,
|
|
dwDrawAspect,
|
|
-1,
|
|
NULL,
|
|
ptd,
|
|
NULL,
|
|
hDC,
|
|
(LPRECTL)&rclHim,
|
|
(LPRECTL)&rclHim,
|
|
NULL,
|
|
0
|
|
);
|
|
OLEDBG_END2
|
|
|
|
OleStdRelease((LPUNKNOWN)lpViewObj2);
|
|
if (hrErr != NOERROR) {
|
|
OleDbgOutHResult(TEXT("IViewObject::Draw returned"), hrErr);
|
|
}
|
|
|
|
hmf = CloseMetaFile(hDC);
|
|
|
|
hMetaPict = GlobalAlloc(GHND|GMEM_SHARE, sizeof(METAFILEPICT));
|
|
|
|
if (hMetaPict && (lpPict = (LPMETAFILEPICT)GlobalLock(hMetaPict))){
|
|
lpPict->hMF = hmf;
|
|
lpPict->xExt = (int)sizelHim.cx ;
|
|
lpPict->yExt = (int)sizelHim.cy ;
|
|
lpPict->mm = MM_ANISOTROPIC;
|
|
GlobalUnlock(hMetaPict);
|
|
}
|
|
|
|
return hMetaPict;
|
|
}
|
|
|
|
|
|
/* Call Release on the object that is expected to go away.
|
|
** if the refcnt of the object did no go to 0 then give a debug message.
|
|
*/
|
|
STDAPI_(ULONG) OleStdVerifyRelease(LPUNKNOWN lpUnk, LPTSTR lpszMsg)
|
|
{
|
|
ULONG cRef;
|
|
|
|
cRef = lpUnk->lpVtbl->Release(lpUnk);
|
|
|
|
#if defined( _DEBUG )
|
|
if (cRef != 0) {
|
|
TCHAR szBuf[80];
|
|
if (lpszMsg)
|
|
MessageBox(NULL, lpszMsg, NULL, MB_ICONEXCLAMATION | MB_OK);
|
|
wsprintf(
|
|
(LPTSTR)szBuf,
|
|
TEXT("refcnt (%ld) != 0 after object (0x%lx) release\n"),
|
|
cRef,
|
|
lpUnk
|
|
);
|
|
if (lpszMsg)
|
|
OleDbgOut1(lpszMsg);
|
|
OleDbgOut1((LPTSTR)szBuf);
|
|
OleDbgAssertSz(cRef == 0, (LPTSTR)szBuf);
|
|
} else {
|
|
TCHAR szBuf[80];
|
|
wsprintf(
|
|
(LPTSTR)szBuf,
|
|
TEXT("refcnt = 0 after object (0x%lx) release\n"), lpUnk
|
|
);
|
|
OleDbgOut4((LPTSTR)szBuf);
|
|
}
|
|
#endif
|
|
return cRef;
|
|
}
|
|
|
|
|
|
/* Call Release on the object that is NOT necessarily expected to go away.
|
|
*/
|
|
STDAPI_(ULONG) OleStdRelease(LPUNKNOWN lpUnk)
|
|
{
|
|
ULONG cRef;
|
|
|
|
cRef = lpUnk->lpVtbl->Release(lpUnk);
|
|
|
|
#if defined( _DEBUG )
|
|
{
|
|
TCHAR szBuf[80];
|
|
wsprintf(
|
|
(LPTSTR)szBuf,
|
|
TEXT("refcnt = %ld after object (0x%lx) release\n"),
|
|
cRef,
|
|
lpUnk
|
|
);
|
|
OleDbgOut4((LPTSTR)szBuf);
|
|
}
|
|
#endif
|
|
return cRef;
|
|
}
|
|
|
|
|
|
/* OleStdInitVtbl
|
|
** --------------
|
|
**
|
|
** Initialize an interface Vtbl to ensure that there are no NULL
|
|
** function pointers in the Vtbl. All entries in the Vtbl are
|
|
** set to a valid funtion pointer (OleStdNullMethod) that issues
|
|
** debug assert message (message box) and returns E_NOTIMPL if called.
|
|
**
|
|
** NOTE: this funtion does not initialize the Vtbl with usefull
|
|
** function pointers, only valid function pointers to avoid the
|
|
** horrible run-time crash when a call is made through the Vtbl with
|
|
** a NULL function pointer. this API is only necessary when
|
|
** initializing the Vtbl's in C. C++ guarantees that all interface
|
|
** functions (in C++ terms -- pure virtual functions) are implemented.
|
|
*/
|
|
|
|
STDAPI_(void) OleStdInitVtbl(LPVOID lpVtbl, UINT nSizeOfVtbl)
|
|
{
|
|
LPVOID FAR* lpFuncPtrArr = (LPVOID FAR*)lpVtbl;
|
|
UINT nMethods = nSizeOfVtbl/sizeof(VOID FAR*);
|
|
UINT i;
|
|
|
|
for (i = 0; i < nMethods; i++) {
|
|
lpFuncPtrArr[i] = OleStdNullMethod;
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdCheckVtbl
|
|
** ---------------
|
|
**
|
|
** Check if all entries in the Vtbl are properly initialized with
|
|
** valid function pointers. If any entries are either NULL or
|
|
** OleStdNullMethod, then this function returns FALSE. If compiled
|
|
** for _DEBUG this function reports which function pointers are
|
|
** invalid.
|
|
**
|
|
** RETURNS: TRUE if all entries in Vtbl are valid
|
|
** FALSE otherwise.
|
|
*/
|
|
|
|
STDAPI_(BOOL) OleStdCheckVtbl(LPVOID lpVtbl, UINT nSizeOfVtbl, LPTSTR lpszIface)
|
|
{
|
|
LPVOID FAR* lpFuncPtrArr = (LPVOID FAR*)lpVtbl;
|
|
UINT nMethods = nSizeOfVtbl/sizeof(VOID FAR*);
|
|
UINT i;
|
|
BOOL fStatus = TRUE;
|
|
int nChar = 0;
|
|
|
|
for (i = 0; i < nMethods; i++) {
|
|
if (lpFuncPtrArr[i] == NULL || lpFuncPtrArr[i] == OleStdNullMethod) {
|
|
#if defined( _DEBUG )
|
|
TCHAR szBuf[256];
|
|
wsprintf(szBuf, TEXT("%s::method# %d NOT valid!"), lpszIface, i);
|
|
OleDbgOut1((LPTSTR)szBuf);
|
|
#endif
|
|
fStatus = FALSE;
|
|
}
|
|
}
|
|
return fStatus;
|
|
}
|
|
|
|
|
|
/* OleStdNullMethod
|
|
** ----------------
|
|
** Dummy method used by OleStdInitVtbl to initialize an interface
|
|
** Vtbl to ensure that there are no NULL function pointers in the
|
|
** Vtbl. All entries in the Vtbl are set to this function. this
|
|
** function issues a debug assert message (message box) and returns
|
|
** E_NOTIMPL if called. If all is done properly, this function will
|
|
** NEVER be called!
|
|
*/
|
|
STDMETHODIMP OleStdNullMethod(LPUNKNOWN lpThis)
|
|
{
|
|
MessageBox(
|
|
NULL,
|
|
TEXT("ERROR: INTERFACE METHOD NOT IMPLEMENTED!\r\n"),
|
|
NULL,
|
|
MB_SYSTEMMODAL | MB_ICONHAND | MB_OK
|
|
);
|
|
|
|
return ResultFromScode(E_NOTIMPL);
|
|
}
|
|
|
|
|
|
|
|
static BOOL GetFileTimes(LPTSTR lpszFileName, FILETIME FAR* pfiletime)
|
|
{
|
|
#ifdef WIN32
|
|
WIN32_FIND_DATA fd;
|
|
HANDLE hFind;
|
|
hFind = FindFirstFile(lpszFileName,&fd);
|
|
if (hFind == NULL || hFind == INVALID_HANDLE_VALUE) {
|
|
return FALSE;
|
|
}
|
|
FindClose(hFind);
|
|
*pfiletime = fd.ftLastWriteTime;
|
|
return TRUE;
|
|
#else // !Win32
|
|
static char sz[256];
|
|
static struct _find_t fileinfo;
|
|
|
|
LSTRCPYN((LPTSTR)sz, lpszFileName, sizeof(sz)-1);
|
|
sz[sizeof(sz)-1]= TEXT('\0');
|
|
AnsiToOem(sz, sz);
|
|
return (_dos_findfirst(sz,_A_NORMAL|_A_HIDDEN|_A_SUBDIR|_A_SYSTEM,
|
|
(struct _find_t *)&fileinfo) == 0 &&
|
|
CoDosDateTimeToFileTime(fileinfo.wr_date,fileinfo.wr_time,pfiletime));
|
|
#endif // Win32
|
|
}
|
|
|
|
|
|
|
|
/* OleStdRegisterAsRunning
|
|
** -----------------------
|
|
** Register a moniker in the RunningObjectTable.
|
|
** if there is an existing registration (*lpdwRegister!=NULL), then
|
|
** first revoke that registration.
|
|
**
|
|
** new dwRegister key is returned via *lpdwRegister parameter.
|
|
*/
|
|
STDAPI_(void) OleStdRegisterAsRunning(LPUNKNOWN lpUnk, LPMONIKER lpmkFull, DWORD FAR* lpdwRegister)
|
|
{
|
|
LPRUNNINGOBJECTTABLE lpROT;
|
|
HRESULT hrErr;
|
|
DWORD dwOldRegister = *lpdwRegister;
|
|
|
|
OLEDBG_BEGIN2(TEXT("OleStdRegisterAsRunning\r\n"))
|
|
|
|
OLEDBG_BEGIN2(TEXT("GetRunningObjectTable called\r\n"))
|
|
hrErr = GetRunningObjectTable(0,(LPRUNNINGOBJECTTABLE FAR*)&lpROT);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr == NOERROR) {
|
|
|
|
/* register as running if a valid moniker is passed
|
|
**
|
|
** OLE2NOTE: we deliberately register the new moniker BEFORE
|
|
** revoking the old moniker just in case the object
|
|
** currently has no external locks. if the object has no
|
|
** locks then revoking it from the running object table will
|
|
** cause the object's StubManager to initiate shutdown of
|
|
** the object.
|
|
*/
|
|
if (lpmkFull) {
|
|
|
|
OLEDBG_BEGIN2(TEXT("IRunningObjectTable::Register called\r\n"))
|
|
lpROT->lpVtbl->Register(lpROT, 0, lpUnk,lpmkFull,lpdwRegister);
|
|
OLEDBG_END2
|
|
|
|
#if defined(_DEBUG)
|
|
{
|
|
TCHAR szBuf[512];
|
|
LPTSTR lpszDisplay;
|
|
LPBC lpbc;
|
|
|
|
#ifdef OLE201
|
|
CreateBindCtx(0, (LPBC FAR*)&lpbc);
|
|
#endif
|
|
|
|
CallIMonikerGetDisplayNameA(
|
|
lpmkFull,
|
|
lpbc,
|
|
NULL,
|
|
&lpszDisplay
|
|
);
|
|
OleStdRelease((LPUNKNOWN)lpbc);
|
|
wsprintf(
|
|
szBuf,
|
|
TEXT("Moniker '%s' REGISTERED as [0x%lx] in ROT\r\n"),
|
|
lpszDisplay,
|
|
*lpdwRegister
|
|
);
|
|
OleDbgOut2(szBuf);
|
|
OleStdFreeString(lpszDisplay, NULL);
|
|
}
|
|
#endif // _DEBUG
|
|
|
|
}
|
|
|
|
// if already registered, revoke
|
|
if (dwOldRegister != 0) {
|
|
|
|
#if defined(_DEBUG)
|
|
{
|
|
TCHAR szBuf[512];
|
|
|
|
wsprintf(
|
|
szBuf,
|
|
TEXT("Moniker [0x%lx] REVOKED from ROT\r\n"),
|
|
dwOldRegister
|
|
);
|
|
OleDbgOut2(szBuf);
|
|
}
|
|
#endif // _DEBUG
|
|
|
|
OLEDBG_BEGIN2(TEXT("IRunningObjectTable::Revoke called\r\n"))
|
|
lpROT->lpVtbl->Revoke(lpROT, dwOldRegister);
|
|
OLEDBG_END2
|
|
}
|
|
|
|
OleStdRelease((LPUNKNOWN)lpROT);
|
|
} else {
|
|
OleDbgAssertSz(
|
|
lpROT != NULL,
|
|
TEXT("OleStdRegisterAsRunning: GetRunningObjectTable FAILED\r\n")
|
|
);
|
|
}
|
|
|
|
OLEDBG_END2
|
|
}
|
|
|
|
|
|
|
|
/* OleStdRevokeAsRunning
|
|
** ---------------------
|
|
** Revoke a moniker from the RunningObjectTable if there is an
|
|
** existing registration (*lpdwRegister!=NULL).
|
|
**
|
|
** *lpdwRegister parameter will be set to NULL.
|
|
*/
|
|
STDAPI_(void) OleStdRevokeAsRunning(DWORD FAR* lpdwRegister)
|
|
{
|
|
LPRUNNINGOBJECTTABLE lpROT;
|
|
HRESULT hrErr;
|
|
|
|
OLEDBG_BEGIN2(TEXT("OleStdRevokeAsRunning\r\n"))
|
|
|
|
// if still registered, then revoke
|
|
if (*lpdwRegister != 0) {
|
|
|
|
OLEDBG_BEGIN2(TEXT("GetRunningObjectTable called\r\n"))
|
|
hrErr = GetRunningObjectTable(0,(LPRUNNINGOBJECTTABLE FAR*)&lpROT);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr == NOERROR) {
|
|
|
|
#if defined(_DEBUG)
|
|
{
|
|
TCHAR szBuf[512];
|
|
|
|
wsprintf(
|
|
szBuf,
|
|
TEXT("Moniker [0x%lx] REVOKED from ROT\r\n"),
|
|
*lpdwRegister
|
|
);
|
|
OleDbgOut2(szBuf);
|
|
}
|
|
#endif // _DEBUG
|
|
|
|
OLEDBG_BEGIN2(TEXT("IRunningObjectTable::Revoke called\r\n"))
|
|
lpROT->lpVtbl->Revoke(lpROT, *lpdwRegister);
|
|
OLEDBG_END2
|
|
|
|
*lpdwRegister = 0;
|
|
|
|
OleStdRelease((LPUNKNOWN)lpROT);
|
|
} else {
|
|
OleDbgAssertSz(
|
|
lpROT != NULL,
|
|
TEXT("OleStdRevokeAsRunning: GetRunningObjectTable FAILED\r\n")
|
|
);
|
|
}
|
|
}
|
|
OLEDBG_END2
|
|
}
|
|
|
|
|
|
/* OleStdNoteFileChangeTime
|
|
** ------------------------
|
|
** Note the time a File-Based object has been saved in the
|
|
** RunningObjectTable. These change times are used as the basis for
|
|
** IOleObject::IsUpToDate.
|
|
** It is important to set the time of the file-based object
|
|
** following a save operation to exactly the time of the saved file.
|
|
** this helps IOleObject::IsUpToDate to give the correct answer
|
|
** after a file has been saved.
|
|
*/
|
|
STDAPI_(void) OleStdNoteFileChangeTime(LPTSTR lpszFileName, DWORD dwRegister)
|
|
{
|
|
if (dwRegister != 0) {
|
|
|
|
LPRUNNINGOBJECTTABLE lprot;
|
|
FILETIME filetime;
|
|
|
|
if (GetFileTimes(lpszFileName, &filetime) &&
|
|
GetRunningObjectTable(0,&lprot) == NOERROR)
|
|
{
|
|
lprot->lpVtbl->NoteChangeTime( lprot, dwRegister, &filetime );
|
|
lprot->lpVtbl->Release(lprot);
|
|
|
|
OleDbgOut2(TEXT("IRunningObjectTable::NoteChangeTime called\r\n"));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdNoteObjectChangeTime
|
|
** --------------------------
|
|
** Set the last change time of an object that is registered in the
|
|
** RunningObjectTable. These change times are used as the basis for
|
|
** IOleObject::IsUpToDate.
|
|
**
|
|
** every time the object sends out a OnDataChange notification, it
|
|
** should update the Time of last change in the ROT.
|
|
**
|
|
** NOTE: this function set the change time to the current time.
|
|
*/
|
|
STDAPI_(void) OleStdNoteObjectChangeTime(DWORD dwRegister)
|
|
{
|
|
if (dwRegister != 0) {
|
|
|
|
LPRUNNINGOBJECTTABLE lprot;
|
|
FILETIME filetime;
|
|
|
|
if (GetRunningObjectTable(0,&lprot) == NOERROR)
|
|
{
|
|
#ifdef OLE201
|
|
CoFileTimeNow( &filetime );
|
|
lprot->lpVtbl->NoteChangeTime( lprot, dwRegister, &filetime );
|
|
#endif
|
|
lprot->lpVtbl->Release(lprot);
|
|
|
|
OleDbgOut2(TEXT("IRunningObjectTable::NoteChangeTime called\r\n"));
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdCreateTempFileMoniker
|
|
** ---------------------------
|
|
** return the next available FileMoniker that can be used as the
|
|
** name of an untitled document.
|
|
** the FileMoniker is built of the form:
|
|
** <lpszPrefixString><number>
|
|
** eg. "Outline1", "Outline2", etc.
|
|
**
|
|
** The RunningObjectTable (ROT) is consulted to determine if a
|
|
** FileMoniker is in use. If the name is in use then the number is
|
|
** incremented and the ROT is checked again.
|
|
**
|
|
** Parameters:
|
|
** LPSTR lpszPrefixString - prefix used to build the name
|
|
** UINT FAR* lpuUnique - (IN-OUT) last used number.
|
|
** this number is used to make the
|
|
** name unique. on entry, the input
|
|
** number is incremented. on output,
|
|
** the number used is returned. this
|
|
** number should be passed again
|
|
** unchanged on the next call.
|
|
** LPSTR lpszName - (OUT) buffer used to build string.
|
|
** caller must be sure buffer is large
|
|
** enough to hold the generated string.
|
|
** LPMONIKER FAR* lplpmk - (OUT) next unused FileMoniker
|
|
**
|
|
** Returns:
|
|
** void
|
|
**
|
|
** Comments:
|
|
** This function is similar in spirit to the Windows API
|
|
** CreateTempFileName.
|
|
*/
|
|
STDAPI_(void) OleStdCreateTempFileMoniker(
|
|
LPTSTR lpszPrefixString,
|
|
UINT FAR* lpuUnique,
|
|
LPTSTR lpszName,
|
|
LPMONIKER FAR* lplpmk
|
|
)
|
|
{
|
|
LPRUNNINGOBJECTTABLE lpROT = NULL;
|
|
UINT i = (lpuUnique != NULL ? *lpuUnique : 1);
|
|
HRESULT hrErr;
|
|
|
|
wsprintf(lpszName, TEXT("%s%d"), lpszPrefixString, i++);
|
|
|
|
CreateFileMonikerA(lpszName, lplpmk);
|
|
|
|
|
|
OLEDBG_BEGIN2(TEXT("GetRunningObjectTable called\r\n"))
|
|
hrErr = GetRunningObjectTable(0,(LPRUNNINGOBJECTTABLE FAR*)&lpROT);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr == NOERROR) {
|
|
|
|
while (1) {
|
|
if (! *lplpmk)
|
|
break; // failed to create FileMoniker
|
|
|
|
OLEDBG_BEGIN2(TEXT("IRunningObjectTable::IsRunning called\r\n"))
|
|
hrErr = lpROT->lpVtbl->IsRunning(lpROT,*lplpmk);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR)
|
|
break; // the Moniker is NOT running; found unused one!
|
|
|
|
OleStdVerifyRelease(
|
|
(LPUNKNOWN)*lplpmk,
|
|
TEXT("OleStdCreateTempFileMoniker: Moniker NOT released")
|
|
);
|
|
|
|
wsprintf(lpszName, TEXT("%s%d"), lpszPrefixString, i++);
|
|
CreateFileMonikerA(lpszName, lplpmk);
|
|
}
|
|
|
|
OleStdRelease((LPUNKNOWN)lpROT);
|
|
}
|
|
|
|
if (lpuUnique != NULL)
|
|
*lpuUnique = i;
|
|
}
|
|
|
|
|
|
/* OleStdGetFirstMoniker
|
|
** ---------------------
|
|
** return the first piece of a moniker.
|
|
**
|
|
** NOTE: if the given moniker is not a generic composite moniker,
|
|
** then an AddRef'ed pointer to the given moniker is returned.
|
|
*/
|
|
STDAPI_(LPMONIKER) OleStdGetFirstMoniker(LPMONIKER lpmk)
|
|
{
|
|
LPMONIKER lpmkFirst = NULL;
|
|
LPENUMMONIKER lpenumMoniker;
|
|
DWORD dwMksys;
|
|
HRESULT hrErr;
|
|
|
|
if (! lpmk)
|
|
return NULL;
|
|
|
|
if (lpmk->lpVtbl->IsSystemMoniker(lpmk, (LPDWORD)&dwMksys) == NOERROR
|
|
&& dwMksys == MKSYS_GENERICCOMPOSITE) {
|
|
|
|
/* OLE2NOTE: the moniker is a GenericCompositeMoniker.
|
|
** enumerate the moniker to pull off the first piece.
|
|
*/
|
|
|
|
hrErr = lpmk->lpVtbl->Enum(
|
|
lpmk,
|
|
TRUE /* fForward */,
|
|
(LPENUMMONIKER FAR*)&lpenumMoniker
|
|
);
|
|
if (hrErr != NOERROR)
|
|
return NULL; // ERROR: give up!
|
|
|
|
hrErr = lpenumMoniker->lpVtbl->Next(
|
|
lpenumMoniker,
|
|
1,
|
|
(LPMONIKER FAR*)&lpmkFirst,
|
|
NULL
|
|
);
|
|
lpenumMoniker->lpVtbl->Release(lpenumMoniker);
|
|
return lpmkFirst;
|
|
|
|
} else {
|
|
/* OLE2NOTE: the moniker is NOT a GenericCompositeMoniker.
|
|
** return an AddRef'ed pointer to the input moniker.
|
|
*/
|
|
lpmk->lpVtbl->AddRef(lpmk);
|
|
return lpmk;
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdGetLenFilePrefixOfMoniker
|
|
** -------------------------------
|
|
** if the first piece of the Moniker is a FileMoniker, then return
|
|
** the length of the filename string.
|
|
**
|
|
** lpmk pointer to moniker
|
|
**
|
|
** Returns
|
|
** 0 if moniker does NOT start with a FileMoniker
|
|
** uLen string length of filename prefix of the display name
|
|
** retrieved from the given (lpmk) moniker.
|
|
*/
|
|
STDAPI_(ULONG) OleStdGetLenFilePrefixOfMoniker(LPMONIKER lpmk)
|
|
{
|
|
LPMONIKER lpmkFirst = NULL;
|
|
DWORD dwMksys;
|
|
LPTSTR lpsz = NULL;
|
|
LPBC lpbc = NULL;
|
|
ULONG uLen = 0;
|
|
HRESULT hrErr;
|
|
|
|
if (! lpmk)
|
|
return 0;
|
|
|
|
lpmkFirst = OleStdGetFirstMoniker(lpmk);
|
|
if (lpmkFirst) {
|
|
if ( (lpmkFirst->lpVtbl->IsSystemMoniker(
|
|
lpmkFirst, (LPDWORD)&dwMksys) == NOERROR)
|
|
&& dwMksys == MKSYS_FILEMONIKER) {
|
|
|
|
#ifdef OLE201
|
|
hrErr = CreateBindCtx(0, (LPBC FAR*)&lpbc);
|
|
#endif
|
|
if (hrErr == NOERROR) {
|
|
hrErr = CallIMonikerGetDisplayNameA(lpmkFirst, lpbc, NULL,
|
|
&lpsz);
|
|
|
|
if (hrErr == NOERROR && lpsz != NULL) {
|
|
uLen = lstrlen(lpsz);
|
|
OleStdFreeString(lpsz, NULL);
|
|
}
|
|
OleStdRelease((LPUNKNOWN)lpbc);
|
|
}
|
|
}
|
|
lpmkFirst->lpVtbl->Release(lpmkFirst);
|
|
}
|
|
return uLen;
|
|
}
|
|
|
|
|
|
/* OleStdMkParseDisplayName
|
|
** Parse a string into a Moniker by calling the OLE API
|
|
** MkParseDisplayName. if the original link source class was an OLE1
|
|
** class, then attempt the parsing assuming the same class applies.
|
|
**
|
|
** if the class of the previous link source was an OLE1 class,
|
|
** then first attempt to parse a string that it is qualified
|
|
** with the progID associated with the OLE1 class. this more
|
|
** closely matches the semantics of OLE1 where the class of
|
|
** link sources is not expected to change. prefixing the
|
|
** string with "@<ProgID -- OLE1 class name>!" will force the
|
|
** parsing of the string to assume the file is of that
|
|
** class.
|
|
** NOTE: this trick of prepending the string with "@<ProgID>
|
|
** only works for OLE1 classes.
|
|
**
|
|
** PARAMETERS:
|
|
** REFCLSID rClsid -- original class of link source.
|
|
** CLSID_NULL if class is unknown
|
|
** ... other parameters the same as MkParseDisplayName API ...
|
|
**
|
|
** RETURNS
|
|
** NOERROR if string parsed successfully
|
|
** else error code returned by MkParseDisplayName
|
|
*/
|
|
STDAPI OleStdMkParseDisplayName(
|
|
REFCLSID rClsid,
|
|
LPBC lpbc,
|
|
LPTSTR lpszUserName,
|
|
ULONG FAR* lpchEaten,
|
|
LPMONIKER FAR* lplpmk
|
|
)
|
|
{
|
|
HRESULT hrErr;
|
|
|
|
if (!IsEqualCLSID(rClsid,&CLSID_NULL) && CoIsOle1Class(rClsid) &&
|
|
lpszUserName[0] != '@') {
|
|
LPTSTR lpszBuf;
|
|
LPTSTR lpszProgID;
|
|
|
|
// Prepend "@<ProgID>!" to the input string
|
|
ProgIDFromCLSIDA(rClsid, &lpszProgID);
|
|
|
|
if (lpszProgID == NULL)
|
|
goto Cont1;
|
|
lpszBuf = OleStdMalloc(
|
|
((ULONG)lstrlen(lpszUserName)+
|
|
#ifdef UNICODE
|
|
// OLE in Win32 is always UNICODE
|
|
wcslen(lpszProgID)
|
|
#else
|
|
lstrlen(lpszProgID)
|
|
#endif
|
|
+3)*sizeof(TCHAR));
|
|
if (lpszBuf == NULL) {
|
|
if (lpszProgID)
|
|
OleStdFree(lpszProgID);
|
|
goto Cont1;
|
|
}
|
|
|
|
wsprintf(lpszBuf, TEXT("@%s!%s"), lpszProgID, lpszUserName);
|
|
|
|
OLEDBG_BEGIN2(TEXT("MkParseDisplayName called\r\n"))
|
|
|
|
hrErr = MkParseDisplayNameA(lpbc, lpszBuf, lpchEaten, lplpmk);
|
|
|
|
OLEDBG_END2
|
|
|
|
if (lpszProgID)
|
|
OleStdFree(lpszProgID);
|
|
if (lpszBuf)
|
|
OleStdFree(lpszBuf);
|
|
|
|
if (hrErr == NOERROR)
|
|
return NOERROR;
|
|
}
|
|
|
|
Cont1:
|
|
OLEDBG_BEGIN2(TEXT("MkParseDisplayName called\r\n"))
|
|
|
|
hrErr = MkParseDisplayNameA(lpbc, lpszUserName, lpchEaten, lplpmk);
|
|
|
|
OLEDBG_END2
|
|
|
|
return hrErr;
|
|
}
|
|
|
|
|
|
/*
|
|
* OleStdMarkPasteEntryList
|
|
*
|
|
* Purpose:
|
|
* Mark each entry in the PasteEntryList if its format is available from
|
|
* the source IDataObject*. the dwScratchSpace field of each PasteEntry
|
|
* is set to TRUE if available, else FALSE.
|
|
*
|
|
* Parameters:
|
|
* LPOLEUIPASTEENTRY array of PasteEntry structures
|
|
* int count of elements in PasteEntry array
|
|
* LPDATAOBJECT source IDataObject* pointer
|
|
*
|
|
* Return Value:
|
|
* none
|
|
*/
|
|
STDAPI_(void) OleStdMarkPasteEntryList(
|
|
LPDATAOBJECT lpSrcDataObj,
|
|
LPOLEUIPASTEENTRY lpPriorityList,
|
|
int cEntries
|
|
)
|
|
{
|
|
LPENUMFORMATETC lpEnumFmtEtc = NULL;
|
|
#define FORMATETC_MAX 20
|
|
FORMATETC rgfmtetc[FORMATETC_MAX];
|
|
int i;
|
|
HRESULT hrErr;
|
|
long j, cFetched;
|
|
|
|
// Clear all marks
|
|
for (i = 0; i < cEntries; i++) {
|
|
lpPriorityList[i].dwScratchSpace = FALSE;
|
|
|
|
if (! lpPriorityList[i].fmtetc.cfFormat) {
|
|
// caller wants this item always considered available
|
|
// (by specifying a NULL format)
|
|
lpPriorityList[i].dwScratchSpace = TRUE;
|
|
} else if (lpPriorityList[i].fmtetc.cfFormat == cfEmbeddedObject
|
|
|| lpPriorityList[i].fmtetc.cfFormat == cfEmbedSource
|
|
|| lpPriorityList[i].fmtetc.cfFormat == cfFileName) {
|
|
|
|
// if there is an OLE object format, then handle it
|
|
// specially by calling OleQueryCreateFromData. the caller
|
|
// need only specify one object type format.
|
|
OLEDBG_BEGIN2(TEXT("OleQueryCreateFromData called\r\n"))
|
|
hrErr = OleQueryCreateFromData(lpSrcDataObj);
|
|
OLEDBG_END2
|
|
if(NOERROR == hrErr) {
|
|
lpPriorityList[i].dwScratchSpace = TRUE;
|
|
}
|
|
} else if (lpPriorityList[i].fmtetc.cfFormat == cfLinkSource) {
|
|
|
|
// if there is OLE 2.0 LinkSource format, then handle it
|
|
// specially by calling OleQueryLinkFromData.
|
|
OLEDBG_BEGIN2(TEXT("OleQueryLinkFromData called\r\n"))
|
|
hrErr = OleQueryLinkFromData(lpSrcDataObj);
|
|
OLEDBG_END2
|
|
if(NOERROR == hrErr) {
|
|
lpPriorityList[i].dwScratchSpace = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
OLEDBG_BEGIN2(TEXT("IDataObject::EnumFormatEtc called\r\n"))
|
|
hrErr = lpSrcDataObj->lpVtbl->EnumFormatEtc(
|
|
lpSrcDataObj,
|
|
DATADIR_GET,
|
|
(LPENUMFORMATETC FAR*)&lpEnumFmtEtc
|
|
);
|
|
OLEDBG_END2
|
|
|
|
if (hrErr != NOERROR)
|
|
return; // unable to get format enumerator
|
|
|
|
// Enumerate the formats offered by the source
|
|
// Loop over all formats offered by the source
|
|
cFetched = 0;
|
|
_fmemset(rgfmtetc,0,sizeof(rgfmtetc[FORMATETC_MAX]) );
|
|
if (lpEnumFmtEtc->lpVtbl->Next(
|
|
lpEnumFmtEtc, FORMATETC_MAX, rgfmtetc, &cFetched) == NOERROR
|
|
|| (cFetched > 0 && cFetched <= FORMATETC_MAX) )
|
|
{
|
|
|
|
for (j = 0; j < cFetched; j++)
|
|
{
|
|
for (i = 0; i < cEntries; i++)
|
|
{
|
|
if (! lpPriorityList[i].dwScratchSpace &&
|
|
IsCloseFormatEtc(&rgfmtetc[j], &lpPriorityList[i].fmtetc))
|
|
{
|
|
lpPriorityList[i].dwScratchSpace = TRUE;
|
|
}
|
|
}
|
|
}
|
|
} // endif
|
|
|
|
// Clean up
|
|
if (lpEnumFmtEtc)
|
|
OleStdRelease((LPUNKNOWN)lpEnumFmtEtc);
|
|
}
|
|
|
|
|
|
/* OleStdGetPriorityClipboardFormat
|
|
** --------------------------------
|
|
**
|
|
** Retrieve the first clipboard format in a list for which data
|
|
** exists in the source IDataObject*.
|
|
**
|
|
** Returns -1 if no acceptable match is found.
|
|
** index of first acceptable match in the priority list.
|
|
**
|
|
*/
|
|
STDAPI_(int) OleStdGetPriorityClipboardFormat(
|
|
LPDATAOBJECT lpSrcDataObj,
|
|
LPOLEUIPASTEENTRY lpPriorityList,
|
|
int cEntries
|
|
)
|
|
{
|
|
int i;
|
|
int nFmtEtc = -1;
|
|
|
|
// Mark all entries that the Source provides
|
|
OleStdMarkPasteEntryList(lpSrcDataObj, lpPriorityList, cEntries);
|
|
|
|
// Loop over the target's priority list of formats
|
|
for (i = 0; i < cEntries; i++)
|
|
{
|
|
if (lpPriorityList[i].dwFlags != OLEUIPASTE_PASTEONLY &&
|
|
!(lpPriorityList[i].dwFlags & OLEUIPASTE_PASTE))
|
|
continue;
|
|
|
|
// get first marked entry
|
|
if (lpPriorityList[i].dwScratchSpace) {
|
|
nFmtEtc = i;
|
|
break; // Found priority format; DONE
|
|
}
|
|
}
|
|
|
|
return nFmtEtc;
|
|
}
|
|
|
|
|
|
/* OleStdIsDuplicateFormat
|
|
** -----------------------
|
|
** Returns TRUE if the lpFmtEtc->cfFormat is found in the array of
|
|
** FormatEtc structures.
|
|
*/
|
|
STDAPI_(BOOL) OleStdIsDuplicateFormat(
|
|
LPFORMATETC lpFmtEtc,
|
|
LPFORMATETC arrFmtEtc,
|
|
int nFmtEtc
|
|
)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < nFmtEtc; i++) {
|
|
if (IsEqualFORMATETC((*lpFmtEtc), arrFmtEtc[i]))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* OleStdGetItemToken
|
|
* ------------------
|
|
*
|
|
* LPTSTR lpszSrc - Pointer to a source string
|
|
* LPTSTR lpszDst - Pointer to destination buffer
|
|
*
|
|
* Will copy one token from the lpszSrc buffer to the lpszItem buffer.
|
|
* It considers all alpha-numeric and white space characters as valid
|
|
* characters for a token. the first non-valid character delimates the
|
|
* token.
|
|
*
|
|
* returns the number of charaters eaten.
|
|
*/
|
|
STDAPI_(ULONG) OleStdGetItemToken(LPTSTR lpszSrc, LPTSTR lpszDst, int nMaxChars)
|
|
{
|
|
ULONG chEaten = 0L;
|
|
|
|
// skip leading delimeter characters
|
|
while (*lpszSrc && --nMaxChars > 0
|
|
&& ((*lpszSrc==TEXT('/')) || (*lpszSrc==TEXT('\\')) ||
|
|
(*lpszSrc==TEXT('!')) || (*lpszSrc==TEXT(':')))) {
|
|
*lpszSrc++;
|
|
chEaten++;
|
|
}
|
|
|
|
// Extract token string (up to first delimeter char or EOS)
|
|
while (*lpszSrc && --nMaxChars > 0
|
|
&& !((*lpszSrc==TEXT('/')) || (*lpszSrc==TEXT('\\')) ||
|
|
(*lpszSrc==TEXT('!')) || (*lpszSrc==TEXT(':')))) {
|
|
*lpszDst++ = *lpszSrc++;
|
|
chEaten++;
|
|
}
|
|
*lpszDst = TEXT('\0');
|
|
return chEaten;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
** OleStdCreateRootStorage
|
|
** create a root level Storage given a filename that is compatible
|
|
** to be used by a top-level OLE container. if the filename
|
|
** specifies an existing file, then an error is returned.
|
|
** the root storage (Docfile) that is created by this function
|
|
** is suitable to be used to create child storages for embedings.
|
|
** (CreateChildStorage can be used to create child storages.)
|
|
** NOTE: the root-level storage is opened in transacted mode.
|
|
*************************************************************************/
|
|
|
|
STDAPI_(LPSTORAGE) OleStdCreateRootStorage(LPTSTR lpszStgName, DWORD grfMode)
|
|
{
|
|
HRESULT hr;
|
|
DWORD grfCreateMode = STGM_READWRITE | STGM_TRANSACTED;
|
|
DWORD reserved = 0;
|
|
LPSTORAGE lpRootStg;
|
|
TCHAR szMsg[64];
|
|
|
|
// if temp file is being created, enable delete-on-release
|
|
if (! lpszStgName)
|
|
grfCreateMode |= STGM_DELETEONRELEASE;
|
|
|
|
hr = StgCreateDocfileA(
|
|
lpszStgName,
|
|
grfMode | grfCreateMode,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpRootStg
|
|
);
|
|
|
|
if (hr == NOERROR)
|
|
return lpRootStg; // existing file successfully opened
|
|
|
|
OleDbgOutHResult(TEXT("StgCreateDocfile returned"), hr);
|
|
|
|
if (0 == LoadString(ghInst, IDS_OLESTDNOCREATEFILE, (LPTSTR)szMsg, 64))
|
|
return NULL;
|
|
|
|
MessageBox(NULL, (LPTSTR)szMsg, NULL,MB_ICONEXCLAMATION | MB_OK);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
** OleStdOpenRootStorage
|
|
** open a root level Storage given a filename that is compatible
|
|
** to be used by a top-level OLE container. if the file does not
|
|
** exist then an error is returned.
|
|
** the root storage (Docfile) that is opened by this function
|
|
** is suitable to be used to create child storages for embedings.
|
|
** (CreateChildStorage can be used to create child storages.)
|
|
** NOTE: the root-level storage is opened in transacted mode.
|
|
*************************************************************************/
|
|
|
|
STDAPI_(LPSTORAGE) OleStdOpenRootStorage(LPTSTR lpszStgName, DWORD grfMode)
|
|
{
|
|
HRESULT hr;
|
|
DWORD reserved = 0;
|
|
LPSTORAGE lpRootStg;
|
|
TCHAR szMsg[64];
|
|
|
|
if (lpszStgName) {
|
|
hr = StgOpenStorageA(
|
|
lpszStgName,
|
|
NULL,
|
|
grfMode | STGM_TRANSACTED,
|
|
NULL,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpRootStg
|
|
);
|
|
|
|
if (hr == NOERROR)
|
|
return lpRootStg; // existing file successfully opened
|
|
|
|
OleDbgOutHResult(TEXT("StgOpenStorage returned"), hr);
|
|
}
|
|
|
|
if (0 == LoadString(ghInst, IDS_OLESTDNOOPENFILE, szMsg, 64))
|
|
return NULL;
|
|
|
|
MessageBox(NULL, (LPTSTR)szMsg, NULL,MB_ICONEXCLAMATION | MB_OK);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
** OpenOrCreateRootStorage
|
|
** open a root level Storage given a filename that is compatible
|
|
** to be used by a top-level OLE container. if the filename
|
|
** specifies an existing file, then it is open, otherwise a new file
|
|
** with the given name is created.
|
|
** the root storage (Docfile) that is created by this function
|
|
** is suitable to be used to create child storages for embedings.
|
|
** (CreateChildStorage can be used to create child storages.)
|
|
** NOTE: the root-level storage is opened in transacted mode.
|
|
*************************************************************************/
|
|
|
|
STDAPI_(LPSTORAGE) OleStdOpenOrCreateRootStorage(LPTSTR lpszStgName, DWORD grfMode)
|
|
{
|
|
HRESULT hrErr;
|
|
SCODE sc;
|
|
DWORD reserved = 0;
|
|
LPSTORAGE lpRootStg;
|
|
TCHAR szMsg[64];
|
|
|
|
if (lpszStgName) {
|
|
|
|
hrErr = StgOpenStorageA(
|
|
lpszStgName,
|
|
NULL,
|
|
grfMode | STGM_READWRITE | STGM_TRANSACTED,
|
|
NULL,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpRootStg
|
|
);
|
|
|
|
if (hrErr == NOERROR)
|
|
return lpRootStg; // existing file successfully opened
|
|
|
|
OleDbgOutHResult(TEXT("StgOpenStorage returned"), hrErr);
|
|
sc = GetScode(hrErr);
|
|
|
|
if (sc!=STG_E_FILENOTFOUND && sc!=STG_E_FILEALREADYEXISTS) {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/* if file did not already exist, try to create a new one */
|
|
hrErr = StgCreateDocfileA(
|
|
lpszStgName,
|
|
grfMode | STGM_READWRITE | STGM_TRANSACTED,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpRootStg
|
|
);
|
|
|
|
if (hrErr == NOERROR)
|
|
return lpRootStg; // existing file successfully opened
|
|
|
|
OleDbgOutHResult(TEXT("StgCreateDocfile returned"), hrErr);
|
|
|
|
if (0 == LoadString(ghInst, IDS_OLESTDNOCREATEFILE, (LPTSTR)szMsg, 64))
|
|
return NULL;
|
|
|
|
MessageBox(NULL, (LPTSTR)szMsg, NULL, MB_ICONEXCLAMATION | MB_OK);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
** OleStdCreateChildStorage
|
|
** create a child Storage inside the given lpStg that is compatible
|
|
** to be used by an embedded OLE object. the return value from this
|
|
** function can be passed to OleCreateXXX functions.
|
|
** NOTE: the child storage is opened in transacted mode.
|
|
*/
|
|
STDAPI_(LPSTORAGE) OleStdCreateChildStorage(LPSTORAGE lpStg, LPTSTR lpszStgName)
|
|
{
|
|
if (lpStg != NULL) {
|
|
LPSTORAGE lpChildStg;
|
|
DWORD grfMode = (STGM_READWRITE | STGM_TRANSACTED |
|
|
STGM_SHARE_EXCLUSIVE);
|
|
DWORD reserved = 0;
|
|
|
|
HRESULT hrErr = CallIStorageCreateStorageA(
|
|
lpStg,
|
|
lpszStgName,
|
|
grfMode,
|
|
reserved,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpChildStg
|
|
);
|
|
|
|
if (hrErr == NOERROR)
|
|
return lpChildStg;
|
|
|
|
OleDbgOutHResult(TEXT("lpStg->lpVtbl->CreateStorage returned"), hrErr);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
** OleStdOpenChildStorage
|
|
** open a child Storage inside the given lpStg that is compatible
|
|
** to be used by an embedded OLE object. the return value from this
|
|
** function can be passed to OleLoad function.
|
|
** NOTE: the child storage is opened in transacted mode.
|
|
*/
|
|
STDAPI_(LPSTORAGE) OleStdOpenChildStorage(LPSTORAGE lpStg, LPTSTR lpszStgName, DWORD grfMode)
|
|
{
|
|
LPSTORAGE lpChildStg;
|
|
LPSTORAGE lpstgPriority = NULL;
|
|
DWORD reserved = 0;
|
|
HRESULT hrErr;
|
|
|
|
if (lpStg != NULL) {
|
|
|
|
hrErr = CallIStorageOpenStorageA(
|
|
lpStg,
|
|
lpszStgName,
|
|
lpstgPriority,
|
|
grfMode | STGM_TRANSACTED | STGM_SHARE_EXCLUSIVE,
|
|
NULL,
|
|
reserved,
|
|
(LPSTORAGE FAR*)&lpChildStg
|
|
);
|
|
|
|
if (hrErr == NOERROR)
|
|
return lpChildStg;
|
|
|
|
OleDbgOutHResult(TEXT("lpStg->lpVtbl->OpenStorage returned"), hrErr);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* OleStdCommitStorage
|
|
** -------------------
|
|
** Commit the changes to the given IStorage*. This routine can be
|
|
** called on either a root-level storage as used by an OLE-Container
|
|
** or by a child storage as used by an embedded object.
|
|
**
|
|
** This routine first attempts to perform this commit in a safe
|
|
** manner. if this fails it then attempts to do the commit in a less
|
|
** robust manner (STGC_OVERWRITE).
|
|
*/
|
|
STDAPI_(BOOL) OleStdCommitStorage(LPSTORAGE lpStg)
|
|
{
|
|
HRESULT hrErr;
|
|
|
|
// make the changes permanent
|
|
hrErr = lpStg->lpVtbl->Commit(lpStg, 0);
|
|
|
|
if (GetScode(hrErr) == STG_E_MEDIUMFULL) {
|
|
// try to commit changes in less robust manner.
|
|
OleDbgOut(TEXT("Warning: commiting with STGC_OVERWRITE specified\n"));
|
|
hrErr = lpStg->lpVtbl->Commit(lpStg, STGC_OVERWRITE);
|
|
}
|
|
|
|
if (hrErr != NOERROR)
|
|
{
|
|
TCHAR szMsg[64];
|
|
|
|
if (0 == LoadString(ghInst, IDS_OLESTDDISKFULL, (LPTSTR)szMsg, 64))
|
|
return FALSE;
|
|
|
|
MessageBox(NULL, (LPTSTR)szMsg, NULL, MB_ICONEXCLAMATION | MB_OK);
|
|
return FALSE;
|
|
}
|
|
else {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
/* OleStdDestroyAllElements
|
|
** ------------------------
|
|
** Destroy all elements within an open storage. this is subject
|
|
** to the current transaction.
|
|
*/
|
|
STDAPI OleStdDestroyAllElements(LPSTORAGE lpStg)
|
|
{
|
|
IEnumSTATSTG FAR* lpEnum;
|
|
STATSTG sstg;
|
|
HRESULT hrErr;
|
|
|
|
hrErr = lpStg->lpVtbl->EnumElements(
|
|
lpStg, 0, NULL, 0, (IEnumSTATSTG FAR* FAR*)&lpEnum);
|
|
|
|
if (hrErr != NOERROR)
|
|
return hrErr;
|
|
|
|
while (1) {
|
|
if (lpEnum->lpVtbl->Next(lpEnum, 1, &sstg, NULL) != NOERROR)
|
|
break;
|
|
lpStg->lpVtbl->DestroyElement(lpStg, sstg.pwcsName);
|
|
OleStdFree(sstg.pwcsName);
|
|
}
|
|
lpEnum->lpVtbl->Release(lpEnum);
|
|
return NOERROR;
|
|
}
|
|
|
|
// returns 1 for a close match
|
|
// (all fields match exactly except the tymed which simply overlaps)
|
|
// 0 for no match
|
|
|
|
int IsCloseFormatEtc(FORMATETC FAR* pFetcLeft, FORMATETC FAR* pFetcRight)
|
|
{
|
|
if (pFetcLeft->cfFormat != pFetcRight->cfFormat)
|
|
return 0;
|
|
else if (!OleStdCompareTargetDevice (pFetcLeft->ptd, pFetcRight->ptd))
|
|
return 0;
|
|
if (pFetcLeft->dwAspect != pFetcRight->dwAspect)
|
|
return 0;
|
|
return((pFetcLeft->tymed | pFetcRight->tymed) != 0);
|
|
}
|
|
|
|
|