|
|
//+-------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1993.
//
// File: api.cpp
//
// Contents: OLE2 api definitions.
//
// Classes: none
//
// Functions: OleRun
// OleIsRunning
// OleLockRunning
// OleSetContainedObject
// OleNoteObjectVisible
// OleGetData
// OleSetData
// OleSave
// ReadClassStg
// WriteClassStg
// WriteFmtUserTypeStg
// ReadFmtUserTypeStg
// ReadM1ClassStm (internal)
// WriteM1ClassStm (internal)
// ReadClassStm
// WriteClassStm
// ReleaseStgMedium
// OleDuplicateData
// ReadOleStg (internal)
// WriteOleStg (internal)
// GetDocumentBitStg (internal and unused)
// GetConvertStg
// SetConvertStg
// ReadClipformatStm
// WriteClipformatStm
// WriteMonikerStm
// ReadMonikerStm
// OleDraw
// CreateObjectDescriptor (internal (for now))
//
// History: dd-mmm-yy Author Comment
// 20-Feb-95 KentCe Buffer version of Read/WriteM1ClassStm.
// 04-Jun-94 alexgo added CreateObjectDescriptor and
// enhanced metafile support
// 25-Jan-94 alexgo first pass at Cairo-style memory allocation
// 11-Jan-94 chriswe fixed broken asserts
// 11-Jan-94 alexgo added VDATEHEAP macros to every function
// and fixed compile warnings
// 08-Dec-93 ChrisWe added necessary casts to GlobalLock() calls
// resulting from removing bogus GlobalLock() macros in
// le2int.h
// 21-Oct-93 Alex Gounares (alexgo) 32-bit port, commented
// and substantial cleanup
// (curts) 11/01/92 Added OleDuplicateMedium
// (srinik) 06/22/92 Moved ReadStringStream, WriteStringStream
// to "utstream.cpp"
// (barrym) 06/02/92 Moved OleSave, ReadClassStg,
// WriteClassStg, added
// OleSaveCompleted, OleIsDirty
// 28-May-92 Srini Koppolu (srinik) Original Author
//
//--------------------------------------------------------------------------
// REVIEW FINAL: probably want to change all pstm->Read into StRead(pstm...)
// except if spec issue 313 is accepted in which case we change StRead into
// pstm->Read.
#include <le2int.h>
#pragma SEG(api)
#define COMPOBJSTM_HEADER_SIZE 7
FARINTERNAL_(HBITMAP) BmDuplicate(HBITMAP hold, DWORD FAR* lpdwSize, LPBITMAP lpBm);
NAME_SEG(Api) ASSERTDATA
#define MAX_STR 512
DWORD gdwFirstDword = (DWORD)MAKELONG(COMPOBJ_STREAM_VERSION, BYTE_ORDER_INDICATOR); DWORD gdwOleVersion = MAKELONG(OLE_STREAM_VERSION, OLE_PRODUCT_VERSION);
//+-------------------------------------------------------------------------
//
// Function: OleRun
//
// Synopsis: Calls IRunnableObject->Run on a given object
//
// Effects: Usually puts on object in the RunningObjectTable
//
// Arguments: [lpUnkown] -- Pointer to the object
//
// Requires:
//
// Returns: The HRESULT from the Run method.
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 21-Oct-93 alexgo ported to 32bit
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleRun)
STDAPI OleRun(IUnknown FAR* lpUnknown) { OLETRACEIN((API_OleRun, PARAMFMT("lpUnknown= %p"), lpUnknown));
VDATEHEAP();
HRESULT hresult; IRunnableObject FAR* pRO;
VDATEIFACE_LABEL(lpUnknown, errRtn, hresult);
if (lpUnknown->QueryInterface(IID_IRunnableObject, (LPLPVOID)&pRO) != NOERROR) { // if no IRunnableObject, assume already running
hresult = NOERROR; goto errRtn; }
hresult = pRO->Run(NULL); pRO->Release();
errRtn: OLETRACEOUT((API_OleRun, hresult));
return hresult; }
//+-------------------------------------------------------------------------
//
// Function: OleIsRunning
//
// Synopsis: calls IRunnableObject->IsRunning on the given object
//
// Effects: Usually returns whether or not an object is in the
// Running Object Table.
//
// Arguments: [lpOleObj] -- pointer to the object
//
// Requires:
//
// Returns: TRUE or FALSE (from IRO->IsRunning)
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 21-Oct-93 alexgo ported to 32bit
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleIsRunning)
STDAPI_(BOOL) OleIsRunning(IOleObject FAR* lpOleObj) { OLETRACEIN((API_OleIsRunning, PARAMFMT("lpOleObj= %p"), lpOleObj));
VDATEHEAP();
IRunnableObject FAR* pRO; BOOL bRetval;
GEN_VDATEIFACE_LABEL(lpOleObj, FALSE, errRtn, bRetval);
if (lpOleObj->QueryInterface(IID_IRunnableObject, (LPLPVOID)&pRO) != NOERROR) { // if no IRunnableObject, assume already running
bRetval = TRUE; goto errRtn; }
bRetval = pRO->IsRunning(); pRO->Release();
errRtn: OLETRACEOUTEX((API_OleIsRunning, RETURNFMT("%B"), bRetval));
return bRetval; }
//+-------------------------------------------------------------------------
//
// Function: OleLockRunning
//
// Synopsis: calls IRunnableObject->LockRunning on the given object
//
// Effects: The object usually ends up calling CoLockObjectExternal
// on itself
//
// Arguments: [lpUnknown] -- pointer to the object
// [fLock] -- TRUE == lock running
// FALSE == unlock running
// [fLastUnlockCloses] -- if TRUE, IRO->LockRunning
// is supposed to call IOO->Close
// if this was the last unlock
//
// Requires:
//
// Returns: HRESULT from IRunnableObject->LockRunning()
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 21-Oct-93 alexgo 32bit port, changed GEN_VDATEIFACE
// to VDATEIFACE to fix a bug
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleLockRunning)
STDAPI OleLockRunning(LPUNKNOWN lpUnknown, BOOL fLock, BOOL fLastUnlockCloses) { OLETRACEIN((API_OleLockRunning, PARAMFMT("lpUnknown= %p, fLock= %B, fLastUnlockCloses= %B"), lpUnknown, fLock, fLastUnlockCloses));
VDATEHEAP();
IRunnableObject FAR* pRO; HRESULT hresult;
VDATEIFACE_LABEL(lpUnknown, errRtn, hresult);
if (lpUnknown->QueryInterface(IID_IRunnableObject, (LPLPVOID)&pRO) != NOERROR) { // if no IRunnableObject, no locks
hresult = NOERROR; goto errRtn; }
hresult = pRO->LockRunning(fLock, fLastUnlockCloses); pRO->Release();
errRtn: OLETRACEOUT((API_OleLockRunning, hresult));
return hresult; }
//+-------------------------------------------------------------------------
//
// Function: OleSetContainedObject
//
// Synopsis: calls IRunnableObject->SetContainedObject on the given object
//
// Effects: Usually has the effect of calling CoLockObjectExternal
// (lpUnkown, !fContained, FALSE).
//
// Arguments: [lpUnknown] -- pointer to the object
// [fContained] -- if TRUE, the object is an embedding
//
// Requires:
//
// Returns: HRESULT from the IRO->SetContainedObject call
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 21-Oct-92 alexgo 32bit port, changed GEN_VDATEIFACE to
// VDATEIFACE to fix a bug
//
// Notes: Containers usually call OleSetContainedObject(..,TRUE) after
// OleLoad or OleCreate. The basic idea is to tell OLE that
// the object is an embedding. The real effect is to unlock
// the object (since all objects start out locked) so that
// other connections may determine it's fate while invisible.
// OleNoteObjectVisible, for instance, would be called to lock
// the object when it become visible.
//
//
//--------------------------------------------------------------------------
#pragma SEG(OleSetContainedObject)
STDAPI OleSetContainedObject(LPUNKNOWN lpUnknown, BOOL fContained) { OLETRACEIN((API_OleSetContainedObject, PARAMFMT("lpUnknown= %p, fContained= %B"), lpUnknown, fContained));
VDATEHEAP();
IRunnableObject FAR* pRO; HRESULT hresult;
VDATEIFACE_LABEL(lpUnknown, errRtn, hresult);
if (lpUnknown->QueryInterface(IID_IRunnableObject, (LPLPVOID)&pRO) != NOERROR) { // if no IRunnableObject, assume container-ness doesn't matter
hresult = NOERROR; goto errRtn; }
hresult = pRO->SetContainedObject(fContained); pRO->Release();
errRtn: OLETRACEOUT((API_OleSetContainedObject, hresult));
return hresult; }
//+-------------------------------------------------------------------------
//
// Function: OleNoteObjectVisible
//
// Synopsis: Simple calls CoLockObjectExternal
//
// Effects:
//
// Arguments: [lpUnknown] -- pointer to the object
// [fVisible] -- if TRUE, then lock the object,
// if false, then unlock
//
// Requires:
//
// Returns: HRESULT from CoLockObjectExternal
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 21-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleNoteObjectVisible)
STDAPI OleNoteObjectVisible(LPUNKNOWN pUnknown, BOOL fVisible) { OLETRACEIN((API_OleNoteObjectVisible, PARAMFMT("pUnknown= %p, fVisible= %B"), pUnknown, fVisible));
VDATEHEAP();
// NOTE: we as fLastUnlockReleases=TRUE here because there would
// otherwise be no other way to fully release the stubmgr. This
// means that objects can't use this mechanism to hold invisible
// objects alive.
HRESULT hr;
hr = CoLockObjectExternal(pUnknown, fVisible, TRUE);
OLETRACEOUT((API_OleNoteObjectVisible, hr));
return hr; }
//+-------------------------------------------------------------------------
//
// Function: OleSave
//
// Synopsis: Writes the CLSID to the storage and calls IPersistStorage->
// Save()
//
// Effects:
//
// Arguments: [pPS] -- pointer to the IPersistStorage interface
// on the object to be saved
// [pstgSave] -- pointer to the storage to which the object
// should be saved
// [fSameAsLoad] -- FALSE indicates a SaveAs operation
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 22-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleSave)
STDAPI OleSave( IPersistStorage FAR* pPS, IStorage FAR* pstgSave, BOOL fSameAsLoad ) { OLETRACEIN((API_OleSave, PARAMFMT("pPS= %p, pstgSave= %p, fSameAsLoad= %B"), pPS, pstgSave, fSameAsLoad));
VDATEHEAP();
HRESULT hresult; CLSID clsid;
VDATEIFACE_LABEL(pPS, errRtn, hresult); VDATEIFACE_LABEL(pstgSave, errRtn, hresult);
if (hresult = pPS->GetClassID(&clsid)) { goto errRtn; }
if (hresult = WriteClassStg(pstgSave, clsid)) { goto errRtn; }
if ((hresult = pPS->Save(pstgSave, fSameAsLoad)) == NOERROR) { hresult = pstgSave->Commit(0); }
errRtn: OLETRACEOUT((API_OleSave, hresult));
return hresult; }
//+-------------------------------------------------------------------------
//
// Function: ReadClassStg
//
// Synopsis: Calls IStorage->Stat to get the CLSID from the given storage
//
// Effects:
//
// Arguments: [pstg] -- pointer to the storage
// [pclsid] -- place to return the CLSID
//
// Requires:
//
// Returns: HRESULT from the IS->Stat call
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 22-Oct-93 alexgo 32bit port, fixed bug with invalid
// [pclsid] and error on IS->Stat
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(ReadClassStg)
STDAPI ReadClassStg( IStorage FAR * pstg, LPCLSID pclsid) { OLETRACEIN((API_ReadClassStg, PARAMFMT("pstg= %p, pclsid= %p"), pstg, pclsid));
VDATEHEAP();
HRESULT hresult; STATSTG statstg;
VDATEIFACE_LABEL(pstg, errRtn, hresult); VDATEPTROUT_LABEL(pclsid, CLSID, errRtn, hresult);
if ((hresult = pstg->Stat(&statstg, STATFLAG_NONAME)) != NOERROR) { *pclsid = CLSID_NULL; goto errRtn; }
*pclsid = statstg.clsid;
errRtn: OLETRACEOUT((API_ReadClassStg, hresult));
return hresult; }
//+-------------------------------------------------------------------------
//
// Function: WriteClassStg
//
// Synopsis: Calls IStorage->SetClass to store the CLSID in the given
// storage
//
// Effects:
//
// Arguments: [pstg] -- pointer to the storage
// [clsid] -- the CLSID to write into the storage
//
// Requires:
//
// Returns: HRESULT from the IS->SetClass call
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 22-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(WriteClassStg)
STDAPI WriteClassStg( IStorage FAR * pstg, REFCLSID clsid) { OLETRACEIN((API_WriteClassStg, PARAMFMT("pstg= %p, clsid= %I"), pstg, &clsid));
VDATEHEAP();
HRESULT hr;
VDATEIFACE_LABEL(pstg, errRtn, hr);
// write clsid in storage (what is read above)
hr = pstg->SetClass(clsid);
errRtn: OLETRACEOUT((API_WriteClassStg, hr));
return hr; }
//+-------------------------------------------------------------------------
//
// Function: ReadM1ClassStm
//
// Synopsis: Reads -1L, CLSID from the given stream
//
// Effects:
//
// Arguments: [pStm] -- pointer to the stream
// [pclsid] -- where to put the clsid
//
// Requires:
//
// Returns: HRESULT from the ReadM1ClassStm.
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 24-Oct-93 alexgo 32bit port
// 20-Feb-95 KentCe Convert to buffered stream reads.
//
// Notes: Internal API.
//
// Reads -1L and CLSID from stream swapping bytes on
// big-endian machines
//
//--------------------------------------------------------------------------
STDAPI ReadM1ClassStm(LPSTREAM pStm, LPCLSID pclsid) { VDATEHEAP(); CStmBufRead StmRead; HRESULT error;
StmRead.Init(pStm);
error = ReadM1ClassStmBuf(StmRead, pclsid);
if (error != NOERROR) *pclsid = CLSID_NULL;
StmRead.Release();
return error; }
//+-------------------------------------------------------------------------
//
// Function: WriteM1ClassStm
//
// Synopsis: Writes -1L, CLSID to the given stream
//
// Effects:
//
// Arguments: [pStm] -- pointer to the stream
// [clsid] -- CLSID to be written
//
// Requires:
//
// Returns: HRESULT from the WriteM1ClassStm
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 11-Jan-84 alexgo changed dw from a DWORD to a LONG
// 24-Oct-93 alexgo 32bit port
// 20-Feb-95 KentCe Convert to buffered stream writes.
//
// Notes: Internal API.
//
// Writess -1L and CLSID from stream swapping bytes on
// big-endian machines
//
//--------------------------------------------------------------------------
STDAPI WriteM1ClassStm(LPSTREAM pStm, REFCLSID clsid) { VDATEHEAP();
CStmBufWrite StmWrite; HRESULT error;
VDATEIFACE( pStm );
StmWrite.Init(pStm);
error = WriteM1ClassStmBuf(StmWrite, clsid); if (FAILED(error)) { goto errRtn; }
error = StmWrite.Flush();
errRtn: StmWrite.Release();
return error; }
//+-------------------------------------------------------------------------
//
// Function: ReadM1ClassStmBuf
//
// Synopsis: Reads -1L and CLSID from the given buffered stream.
//
// Arguments: [StmRead] -- Stream Read Object.
// [pclsid] -- Where to put the clsid
//
// Returns: HRESULT from the StmRead.Read's
//
//
// History: dd-mmm-yy Author Comment
// 20-Feb-95 KentCe Convert to buffered stream reads.
// 24-Oct-93 alexgo 32bit port
//
// Notes: Internal API.
//
// Reads -1L and CLSID from stream swapping bytes on
// big-endian machines
//
//--------------------------------------------------------------------------
STDAPI ReadM1ClassStmBuf(CStmBufRead & StmRead, LPCLSID pclsid) { VDATEHEAP();
HRESULT error; LONG lValue;
if ((error = StmRead.ReadLong(&lValue)) != NOERROR) { goto errRtn; }
if (lValue == -1) { // have a GUID
error = StmRead.Read((void FAR *)pclsid, sizeof(CLSID)); } else { // this is now an error; we don't allow string form
// of clsid anymore
error = ResultFromScode(E_UNSPEC); }
errRtn: if (error != NOERROR) { *pclsid = CLSID_NULL; }
return error; }
//+-------------------------------------------------------------------------
//
// Function: WriteM1ClassStmBuf
//
// Synopsis: Writes -1L and CLSID to the given buffered stream.
//
// Arguments: [StmRead] -- Stream Write Object.
// [pclsid] -- Where to read the clsid
//
// Returns: HRESULT from the StmWrite.Write's
//
//
// History: dd-mmm-yy Author Comment
// 20-Feb-95 KentCe Convert to buffered stream reads.
// 24-Oct-93 alexgo 32bit port
//
// Notes: Internal API.
//
// Writess -1L and CLSID from stream swapping bytes on
// big-endian machines
//
//--------------------------------------------------------------------------
STDAPI WriteM1ClassStmBuf(CStmBufWrite & StmWrite, REFCLSID clsid) { VDATEHEAP();
HRESULT error;
// format is -1L followed by GUID
if ((error = StmWrite.WriteLong(-1)) != NOERROR) return error;
return StmWrite.Write((LPVOID)&clsid, sizeof(clsid)); }
//+-------------------------------------------------------------------------
//
// Function: ReadClassStm
//
// Synopsis: Reads the CLSID from the given stream
//
// Effects:
//
// Arguments: [pStm] -- pointer to the stream
// [pclsid] -- where to put the clsid
//
// Requires:
//
// Returns: HRESULT from the IStream->Read
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 24-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(ReadClassStm)
// reads CLSID from stream swapping bytes on big-endian machines
STDAPI ReadClassStm(LPSTREAM pStm, LPCLSID pclsid) { OLETRACEIN((API_ReadClassStm, PARAMFMT("pStm= %p, pclsid= %p"), pStm, pclsid));
VDATEHEAP(); HRESULT error;
VDATEIFACE_LABEL( pStm, errRtn, error ); VDATEPTROUT_LABEL(pclsid, CLSID, errRtn, error);
if ((error = StRead(pStm, (void FAR *)pclsid, sizeof(CLSID))) != NOERROR) *pclsid = CLSID_NULL;
errRtn: OLETRACEOUT((API_ReadClassStm, error));
return error; }
//+-------------------------------------------------------------------------
//
// Function: WriteClassStm
//
// Synopsis: Writes the class ID to the given stream
//
// Effects:
//
// Arguments: [pStm] -- pointer to the stream
// [clsid] -- CLSID to write to the stream
//
// Requires:
//
// Returns: HRESULT from the IStream->Write call
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 24-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(WriteClassStm)
// writes CLSID to stream swapping bytes on big-endian machines
STDAPI WriteClassStm(LPSTREAM pStm, REFCLSID clsid) { OLETRACEIN((API_WriteClassStm, PARAMFMT("pStm= %p, clsid= %I"), pStm, &clsid));
VDATEHEAP();
HRESULT hr;
VDATEIFACE_LABEL( pStm, errRtn, hr);
hr = pStm->Write(&clsid, sizeof(clsid), NULL);
errRtn: OLETRACEOUT((API_WriteClassStm, hr));
return hr; }
//+-------------------------------------------------------------------------
//
// Function: ReleaseStgMedium
//
// Synopsis: Releases any resources held by a storage medium
//
// Arguments: [pMedium] -- pointer to the storage medium
//
// Returns: nothing
//
// History: dd-mmm-yy Author Comment
// 24-Oct-93 alexgo 32-bit port
// 15-May-94 DavePl Added EMF support
//
//--------------------------------------------------------------------------
#pragma SEG(ReleaseStgMedium)
STDAPI_(void) ReleaseStgMedium( LPSTGMEDIUM pMedium ) { OLETRACEIN((API_ReleaseStgMedium, PARAMFMT("pMedium= %p"), pMedium));
VDATEHEAP();
if (pMedium) {
BOOL fPunkRel;
//VDATEPTRIN rejects NULL
VOID_VDATEPTRIN_LABEL( pMedium, STGMEDIUM, errRtn); fPunkRel = pMedium->pUnkForRelease != NULL;
switch (pMedium->tymed) { case TYMED_HGLOBAL: if (pMedium->hGlobal != NULL && !fPunkRel) Verify(GlobalFree(pMedium->hGlobal) == 0); break;
case TYMED_GDI: if (pMedium->hGlobal != NULL && !fPunkRel) DeleteObject(pMedium->hGlobal); break;
case TYMED_ENHMF: if (pMedium->hEnhMetaFile != NULL && !fPunkRel) { Verify(DeleteEnhMetaFile(pMedium->hEnhMetaFile)); }; break;
case TYMED_MFPICT: if (pMedium->hGlobal != NULL && !fPunkRel) { LPMETAFILEPICT pmfp;
if ((pmfp = (LPMETAFILEPICT)GlobalLock(pMedium->hGlobal)) == NULL) break;
DeleteMetaFile(pmfp->hMF); GlobalUnlock(pMedium->hGlobal); Verify(GlobalFree(pMedium->hGlobal) == 0); } break;
case TYMED_FILE: if (pMedium->lpszFileName != NULL) { if (!IsValidPtrIn(pMedium->lpszFileName, 1)) break; if (!fPunkRel) { DeleteFile(pMedium->lpszFileName); }
// WARNING: there was a bug in the 16bit code that the filename
// string was not being freed if pUnkForRelease was NULL. the
// spec says it should delete the string, so we follow the spec
// here.
PubMemFree(pMedium->lpszFileName); pMedium->lpszFileName = NULL;
} break;
case TYMED_ISTREAM: if (pMedium->pstm != NULL && IsValidInterface(pMedium->pstm)) pMedium->pstm->Release(); break;
case TYMED_ISTORAGE: if (pMedium->pstg != NULL && IsValidInterface(pMedium->pstg)) pMedium->pstg->Release(); break;
case TYMED_NULL: break;
default: AssertSz(FALSE, "Invalid medium in ReleaseStgMedium"); }
if (pMedium->pUnkForRelease) { if (IsValidInterface(pMedium->pUnkForRelease)) pMedium->pUnkForRelease->Release(); pMedium->pUnkForRelease = NULL; }
// NULL out to prevent unwanted use of just freed data.
// Note: this must be done AFTER punkForRelease is called
// because our special punkForRelease used in remoting
// needs the tymed value.
pMedium->tymed = TYMED_NULL;
}
errRtn: OLETRACEOUTEX((API_ReleaseStgMedium, NORETURN));
return; }
#ifdef MAC_REVIEW
This API must be written for MAC and PICT format. #endif
//+-------------------------------------------------------------------------
//
// Function: OleDuplicateData
//
// Synopsis: Duplicates data from the given handle and clipboard format
//
// Effects:
//
// Arguments: [hSrc] -- handle to the data to be duplicated
// [cfFormat] -- format of [hSrc]
// [uiFlags] -- any flags (such a GMEM_MOVEABLE) for
// memory allocation
//
// Requires:
//
// Returns: a HANDLE to the duplicated resource
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 30-May-94 alexgo added support for enhanced metafiles
// 24-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(OleDuplicateData)
STDAPI_(HANDLE) OleDuplicateData (HANDLE hSrc, CLIPFORMAT cfFormat, UINT uiFlags) { HANDLE hDup;
OLETRACEIN((API_OleDuplicateData, PARAMFMT("hSrc= %h, cfFormat= %d, uiFlags= %x"), hSrc, cfFormat, uiFlags));
VDATEHEAP();
if (!hSrc) { hDup = NULL; goto errRtn; }
switch( cfFormat ) { case CF_BITMAP: hDup = (HANDLE) BmDuplicate ((HBITMAP)hSrc, NULL, NULL); break;
case CF_PALETTE: hDup = (HANDLE) UtDupPalette ((HPALETTE)hSrc); break;
case CF_ENHMETAFILE: hDup = (HANDLE) CopyEnhMetaFile((HENHMETAFILE)hSrc, NULL); break;
case CF_METAFILEPICT: if (uiFlags == NULL) { uiFlags = GMEM_MOVEABLE; }
LPMETAFILEPICT lpmfpSrc; LPMETAFILEPICT lpmfpDst;
if (!(lpmfpSrc = (LPMETAFILEPICT) GlobalLock (hSrc))) { hDup = NULL; goto errRtn; }
if (!(hDup = UtDupGlobal (hSrc, uiFlags))) { GlobalUnlock(hSrc); hDup = NULL; goto errRtn; }
if (!(lpmfpDst = (LPMETAFILEPICT) GlobalLock (hDup))) { GlobalUnlock(hSrc); GlobalFree (hDup); hDup = NULL; goto errRtn; }
*lpmfpDst = *lpmfpSrc; lpmfpDst->hMF = CopyMetaFile (lpmfpSrc->hMF, NULL); GlobalUnlock (hSrc); GlobalUnlock (hDup); break;
default: if (uiFlags == NULL) { uiFlags = GMEM_MOVEABLE; }
hDup = UtDupGlobal (hSrc, uiFlags); }
errRtn: OLETRACEOUTEX((API_OleDuplicateData, RETURNFMT("%h"), hDup));
return hDup; }
//+-------------------------------------------------------------------------
//
// Function: BmDuplicate
//
// Synopsis: Duplicates a bitmap
//
// Effects:
//
// Arguments: [hold] -- the source bitmap
// [lpdwSize] -- where to put the bitmap size
// [lpBm] -- where to put the new bitmap
//
// Requires:
//
// Returns: A handle to the new bitmap
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 25-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(BmDuplicate)
FARINTERNAL_(HBITMAP) BmDuplicate (HBITMAP hold, DWORD FAR* lpdwSize, LPBITMAP lpBm) { VDATEHEAP();
HBITMAP hnew = NULL; HANDLE hMem; LPVOID lpMem; DWORD dwSize; BITMAP bm; SIZE extents;
extents.cx = extents.cy = 0;
// REVIEW (davepl): The bitmap pointer here was being cast to LPOLESTR
// for some reason. It's takes a void pointer!
GetObject (hold, sizeof(BITMAP), &bm); dwSize = ((DWORD) bm.bmHeight) * ((DWORD) bm.bmWidthBytes) * ((DWORD) bm.bmPlanes);
if (!(hMem = GlobalAlloc (GMEM_MOVEABLE, dwSize))) return NULL;
if (!(lpMem = GlobalLock (hMem))) goto errRtn;
GlobalUnlock (hMem);
// REVIEW(davepl): This should probably use GetDIBits() instead
GetBitmapBits (hold, dwSize, lpMem); if (hnew = CreateBitmap (bm.bmWidth, bm.bmHeight, bm.bmPlanes, bm.bmBitsPixel, NULL)) { if (!SetBitmapBits (hnew, dwSize, lpMem)) { DeleteObject (hnew); hnew = NULL; goto errRtn; } }
if (lpdwSize) *lpdwSize = dwSize;
if (lpBm) *lpBm = bm;
if (hnew && GetBitmapDimensionEx(hold, &extents) && extents.cx && extents.cy) SetBitmapDimensionEx(hnew, extents.cx, extents.cy, NULL);
errRtn: if (hMem) GlobalFree (hMem);
return hnew; }
//+-------------------------------------------------------------------------
//
// Function: ReadOleStg
//
// Synopsis: Internal API to read private OLE information from
// the OLE_STREAM in the given storage
//
// Effects:
//
// Arguments: [pstg] -- pointer to the storage
// [pdwFlags] -- where to put flags stored in the
// the stream (may be NULL)
// [pdwOptUpdate] -- where to put the update flags
// (may be NULL)
// [pdwReserved] -- where to put the reserved value
// (may be NULL)
// [ppmk] -- where to put the moniker
// (may be NULL)
// [ppstmOut] -- where to put the OLE_STREAM pointer
// (may be NULL)
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(ReadOleStg)
STDAPI ReadOleStg (LPSTORAGE pstg, DWORD FAR* pdwFlags, DWORD FAR* pdwOptUpdate, DWORD FAR* pdwReserved, LPMONIKER FAR* ppmk, LPSTREAM FAR* ppstmOut) { OLETRACEIN((API_ReadOleStg, PARAMFMT("pdwFlags= %p, pdwOptUpdate= %p, pdwReserved= %p, ppmk= %p, ppstmOut= %p"), pdwFlags, pdwOptUpdate, pdwReserved, ppmk, ppstmOut));
VDATEHEAP();
HRESULT error; IStream FAR * pstm; DWORD dwBuf[4]; LPMONIKER pmk; LPOLESTR szClassName = OLE_STREAM;
if (ppmk) { VDATEPTROUT_LABEL( ppmk, LPMONIKER, errNoFreeRtn, error); *ppmk = NULL; }
if (ppstmOut){ VDATEPTROUT_LABEL( ppstmOut, LPSTREAM, errNoFreeRtn, error); *ppstmOut = NULL; } VDATEIFACE_LABEL( pstg, errNoFreeRtn, error);
if ((error = pstg->OpenStream(szClassName, NULL, (STGM_READ | STGM_SHARE_EXCLUSIVE), 0, &pstm)) != NOERROR) { // This error is OK for some callers (ex: default handler)
// of this function. They depend on NOERROR or this error
// code. So, don't change the error code.
error = ReportResult(0, STG_E_FILENOTFOUND, 0, 0); goto errNoFreeRtn; }
// read Ole version number, flags, Update options, reserved field
if ((error = StRead (pstm, dwBuf, 4*sizeof(DWORD))) != NOERROR) goto errRtn;
if (dwBuf[0] != gdwOleVersion) { error = ResultFromScode(DV_E_CLIPFORMAT); goto errRtn; }
if (pdwFlags) *pdwFlags = dwBuf[1];
if (pdwOptUpdate) *pdwOptUpdate = dwBuf[2];
AssertSz(dwBuf[3] == NULL,"Reserved field in OLE STREAM is not NULL");
if (dwBuf[3] != NULL) { error = ResultFromScode(DV_E_CLIPFORMAT); goto errRtn; }
if (pdwReserved) *pdwReserved = dwBuf[3];
if ((error = ReadMonikerStm (pstm, &pmk)) != NOERROR) goto errRtn;
if (ppmk) *ppmk = pmk; else if (pmk) pmk->Release();
errRtn: if (pstm) { if ((error == NOERROR) && (ppstmOut != NULL)) *ppstmOut = pstm; else pstm->Release(); }
errNoFreeRtn: OLETRACEOUT((API_ReadOleStg, error));
return error; }
//+-------------------------------------------------------------------------
//
// Function: WriteOleStg
//
// Synopsis: Writes private OLE info into an OLE_STREAM in the given storage.
//
// Arguments: [pstg] [in] -- pointer to the storage
// [pOleObj] [in] -- object from which to get info to write
// (may be NULL)
// [dwReserved] [in] -- reserved
// [ppstmOut] [out] -- pointer to return the private stream
// (may be NULL)
// Returns: HRESULT
//
// History: dd-mmm-yy Author Comment
// Oct 27, 93 alexgo 32bit port
// Oct 23, 96 gopalk Changed to call WriteOleStgEx
//
//--------------------------------------------------------------------------
#pragma SEG(WriteOleStg)
STDAPI WriteOleStg(LPSTORAGE pstg, IOleObject* pOleObj, DWORD dwReserved, LPSTREAM* ppstmOut) { OLETRACEIN((API_WriteOleStg, PARAMFMT("pstg=%p, pOleObj=%p, dwReserved=%x, ppstmOut=%p, "), pstg, pOleObj, dwReserved, ppstmOut));
// Local variable
HRESULT error;
do { // Validation Checks
VDATEHEAP(); if(ppstmOut && !IsValidPtrOut(ppstmOut, sizeof(LPSTREAM))) { error = ResultFromScode(E_INVALIDARG); break; } if(!IsValidInterface(pstg)) { error = ResultFromScode(E_INVALIDARG); break; } if(pOleObj && !IsValidInterface(pOleObj)) { error = ResultFromScode(E_INVALIDARG); break; } // Call WriteOleStgEx
error = WriteOleStgEx(pstg, pOleObj, dwReserved, 0, ppstmOut); } while(FALSE);
OLETRACEOUT((API_WriteOleStg, error)); return error; }
//+-------------------------------------------------------------------------
//
// Function: WriteOleStgEx (Internal)
//
// Synopsis: Writes private OLE info into an OLE_STREAM in the given storage.
//
// Arguments: [pstg] [in] -- pointer to the storage
// [pOleObj] [in] -- object from which to get info to write
// (may be NULL)
// [dwReserved] [in] -- reserved
// [ppstmOut] [out] -- pointer to return the private stream
// (may be NULL)
// [dwGivenFlags] [in] -- Additional object flags to be set
//
// Returns: HRESULT
//
// History: dd-mmm-yy Author Comment
// Oct 23, 96 gopalk Creation
//
//--------------------------------------------------------------------------
STDAPI WriteOleStgEx(LPSTORAGE pstg, IOleObject* pOleObj, DWORD dwReserved, DWORD dwGivenFlags, LPSTREAM* ppstmOut) { // Local Variables
HRESULT error = NOERROR; IStream* pstm = NULL; IOleLink* pLink; LPMONIKER pmk; DWORD dwUpdOpt; ULONG cbRead; DWORD objflags; ULARGE_INTEGER ularge_integer; LARGE_INTEGER large_integer;
// Initialize out parameter
if(ppstmOut) *ppstmOut = NULL;
// Open or Create OLE_STREAM
error = OpenOrCreateStream(pstg, OLE_STREAM, &pstm); if(error == NOERROR) { // Write Ole version
error = pstm->Write(&gdwOleVersion, sizeof(DWORD), NULL); if(error == NOERROR) { // Read existing Objflags to preserve doc bit
if(pstm->Read(&objflags, sizeof(DWORD), &cbRead) != NOERROR || cbRead != sizeof(DWORD)) objflags = 0;
// Only preserve docbit
objflags &= OBJFLAGS_DOCUMENT; // Set the given flags
objflags |= dwGivenFlags;
// Obtain link update options
dwUpdOpt = 0L; if(pOleObj != NULL && pOleObj->QueryInterface(IID_IOleLink, (void **)&pLink) == NOERROR) { objflags |= OBJFLAGS_LINK; pLink->GetUpdateOptions(&dwUpdOpt); pLink->Release(); }
// Seek to the Objflags field. We could be off due to the above read
LISet32(large_integer, sizeof(DWORD)); error = pstm->Seek(large_integer, STREAM_SEEK_SET, NULL); if(error == NOERROR) { // Write Objflags and link update options
DWORD dwBuf[3];
dwBuf[0] = objflags; dwBuf[1] = dwUpdOpt; Win4Assert(dwReserved == NULL); dwBuf[2] = 0L;
error = pstm->Write(dwBuf, 3*sizeof(DWORD), NULL); if(error == NOERROR) { // Obtain object moniker
pmk = NULL; if(pOleObj != NULL) { error = pOleObj->GetMoniker(OLEGETMONIKER_ONLYIFTHERE, OLEWHICHMK_OBJREL, &pmk); if(SUCCEEDED(error) && !IsValidInterface(pmk)) { Win4Assert(FALSE); pmk = NULL; } else if(FAILED(error) && pmk) { Win4Assert(FALSE); if(!IsValidInterface(pmk)) pmk = NULL; }
// Write Object moniker
error = WriteMonikerStm(pstm, pmk); if(pmk) pmk->Release();
// Truncate the stream to remove any existing data
if(error == NOERROR) { LISet32(large_integer, 0); error = pstm->Seek(large_integer, STREAM_SEEK_CUR, &ularge_integer); if(error == NOERROR) pstm->SetSize(ularge_integer); } } } } } if(error==NOERROR && ppstmOut) *ppstmOut = pstm; else pstm->Release(); } return error; }
//+-------------------------------------------------------------------------
//
// Function: SetBitOleStg
//
// Synopsis: internal function to write private OLE info into
// OLE_STREAM on the given storage
//
// Effects:
//
// Arguments: [pstg] -- pointer to the storage
// [mask] -- mask for old values
// [value] -- values to write
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm: writes (old_values & mask ) | value into the stream
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32-bit port, fixed bugs
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(SetBitOleStg)
static INTERNAL SetBitOleStg(LPSTORAGE pstg, DWORD mask, DWORD value) { VDATEHEAP();
IStream FAR * pstm = NULL; HRESULT error; DWORD objflags = 0; LARGE_INTEGER large_integer;
VDATEIFACE( pstg );
if (error = pstg->OpenStream(OLE_STREAM, NULL, STGM_SALL, 0, &pstm)) { if (STG_E_FILENOTFOUND != GetScode(error)) goto errRtn;
if ((error = pstg->CreateStream(OLE_STREAM, STGM_SALL, 0, 0, &pstm)) != NOERROR) goto errRtn;
DWORD dwBuf[5];
dwBuf[0] = gdwOleVersion; dwBuf[1] = objflags; dwBuf[2] = 0L; dwBuf[3] = 0L; dwBuf[4] = 0L;
if ((error = pstm->Write(dwBuf, 5*sizeof(DWORD), NULL)) != NOERROR) goto errRtn; }
// seek directly to word, read, modify, seek back and write.
LISet32( large_integer, sizeof(DWORD) ); if ((error = pstm->Seek(large_integer, STREAM_SEEK_SET, NULL)) != NOERROR) goto errRtn;
if ((error = StRead(pstm, &objflags, sizeof(objflags))) != NOERROR) goto errRtn;
objflags = (objflags & mask) | value;
LISet32( large_integer, sizeof(DWORD) ); if ((error = pstm->Seek(large_integer, STREAM_SEEK_SET, NULL)) != NOERROR) goto errRtn;
error = pstm->Write(&objflags, sizeof(DWORD), NULL);
errRtn:// close and return error code.
if (pstm) pstm->Release(); return error; }
//+-------------------------------------------------------------------------
//
// Function: GetFlagsOleStg
//
// Synopsis: Internal function to get the private ole flags from a
// given storage
//
// Effects:
//
// Arguments: [pstg] -- pointer to the storage
// [lpobjflags] -- where to put the flags
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port, fixed bugs (error return)
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(GetFlagsOleStg)
static INTERNAL GetFlagsOleStg(LPSTORAGE pstg, LPDWORD lpobjflags) { VDATEHEAP();
IStream FAR * pstm = NULL; HRESULT error; LARGE_INTEGER large_integer;
VDATEIFACE( pstg );
if ((error = pstg->OpenStream(OLE_STREAM, NULL, (STGM_READ | STGM_SHARE_EXCLUSIVE), 0, &pstm)) != NOERROR) goto errRtn;
// seek directly to word, read, modify, seek back and write.
LISet32( large_integer, sizeof(DWORD) ); if ((error = pstm->Seek(large_integer, STREAM_SEEK_SET, NULL)) != NOERROR) goto errRtn;
error = StRead(pstm, lpobjflags, sizeof(*lpobjflags));
errRtn: if (pstm) pstm->Release(); return error; }
//+-------------------------------------------------------------------------
//
// Function: GetDocumentBitStg
//
// Synopsis: returns the doc bit from the given storage
//
// Effects:
//
// Arguments: [pStg] -- pointer to the storage
//
// Requires:
//
// Returns: NOERROR if the doc bit is set, S_FALSE if not
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
// REVIEW32:: Nobody seems to use this function. Nuke it.
//
//--------------------------------------------------------------------------
#pragma SEG(GetDocumentBitStg)
// get doc bit; return NOERROR if on; S_FALSE if off
STDAPI GetDocumentBitStg(LPSTORAGE pStg) { OLETRACEIN((API_GetDocumentBitStg, PARAMFMT("pStg= %p"), pStg));
VDATEHEAP();
DWORD objflags; HRESULT error;
if ((error = GetFlagsOleStg(pStg, &objflags)) == NOERROR) { if(!(objflags&OBJFLAGS_DOCUMENT)) { error = ResultFromScode(S_FALSE); } }
OLETRACEOUT((API_GetDocumentBitStg, error));
return error; }
//+-------------------------------------------------------------------------
//
// Function: SetDocumentBitStg
//
// Synopsis: Writes the document bit to the given storage
//
// Effects:
//
// Arguments: [pStg] -- pointer to the storage
// [fDocument] -- TRUE, storage is a document, false
// otherwise
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32-bit port
//
// Notes:
// REVIEW32: nobody seems to use this function, nuke it
//
//--------------------------------------------------------------------------
#pragma SEG(SetDocumentBitStg)
// set doc bit according to fDocument
STDAPI SetDocumentBitStg(LPSTORAGE pStg, BOOL fDocument) { OLETRACEIN((API_SetDocumentBitStg, PARAMFMT("pStg= %p, fDocument= %B"), pStg, fDocument));
VDATEHEAP();
HRESULT hr;
hr = SetBitOleStg(pStg, fDocument ? -1L : ~OBJFLAGS_DOCUMENT, fDocument ? OBJFLAGS_DOCUMENT : 0);
OLETRACEOUT((API_SetDocumentBitStg, hr));
return hr; }
//+-------------------------------------------------------------------------
//
// Function: GetConvertStg
//
// Synopsis: Gets the convert bit from the given storage
//
// Effects:
//
// Arguments: [pStg] -- pointer to the storage
//
// Requires:
//
// Returns: NOERROR if set, S_FALSE if not
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(GetConvertStg)
STDAPI GetConvertStg(LPSTORAGE pStg) { OLETRACEIN((API_GetConvertStg, PARAMFMT("pStg= %p"), pStg));
VDATEHEAP();
DWORD objflags; HRESULT error;
if ((error = GetFlagsOleStg(pStg, &objflags)) != NOERROR) { goto errRtn; }
if (objflags&OBJFLAGS_CONVERT) { error = NOERROR; } else { error = ResultFromScode(S_FALSE); }
errRtn: OLETRACEOUT((API_GetConvertStg, error));
return error; }
//+-------------------------------------------------------------------------
//
// Function: SetConvertStg
//
// Synopsis: Sets the convert bit in a storage
//
// Effects:
//
// Arguments: [pStg] -- pointer to the storage
// [fConvert] -- convert bit
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(SetConvertStg)
STDAPI SetConvertStg(LPSTORAGE pStg, BOOL fConvert) { OLETRACEIN((API_SetConvertStg, PARAMFMT("pStg= %p, fConvert= %B"), pStg, fConvert));
VDATEHEAP();
HRESULT hr;
hr = SetBitOleStg(pStg, fConvert ? -1L : ~OBJFLAGS_CONVERT, fConvert ? OBJFLAGS_CONVERT : 0);
OLETRACEOUT((API_SetConvertStg, hr));
return hr; }
//+-------------------------------------------------------------------------
//
// Function: ReadClipformatStm
//
// Synopsis: Reads the clipboard format from the given stream
//
// Effects: If the clipboard format is a length followed by a
// string, then the string is read and registered as a
// clipboard format (and the new format number is returned).
//
// Arguments: [lpstream] -- pointer to the stream
// [lpdwCf] -- where to put the clipboard format
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm: the format of the stream must be one of the following:
// 0 No clipboard format
// -1 DWORD predefined windows clipboard format in
// the second dword.
// -2 DWORD predefined mac clipboard format in the
// second dword. This may be obsolete or
// irrelevant for us. REVIEW32
// num STRING clipboard format name string (prefaced
// by length of string).
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port, fixed ifdef and NULL
// pointer bugs
//
// 17-Mar-94 davepl Revereted to ANSI string reads
// Notes:
//
//--------------------------------------------------------------------------
STDAPI ReadClipformatStm(LPSTREAM lpstream, DWORD FAR* lpdwCf) { VDATEHEAP();
HRESULT error; DWORD dwValue;
VDATEIFACE(lpstream); VDATEPTROUT(lpdwCf, DWORD);
if (error = StRead(lpstream, &dwValue, sizeof(DWORD))) { return error; }
if (dwValue == NULL) { // NULL cf value
*lpdwCf = NULL;
} else if (dwValue == -1L) { // Then this is a NON-NULL predefined windows clipformat.
// The clipformat values follows
if (error = StRead(lpstream, &dwValue, sizeof(DWORD))) return error;
*lpdwCf = dwValue;
} else if (dwValue == -2L) { // Then this is a NON-NULL MAC clipboard format.
// The clipformat value follows. For MAC the CLIPFORMAT
// is 4 bytes
if (error = StRead(lpstream, &dwValue, sizeof(DWORD))) { return error; } *lpdwCf = dwValue; return ResultFromScode(OLE_S_MAC_CLIPFORMAT); } else { char szACF[MAX_STR];
if (error = StRead(lpstream, szACF, dwValue)) { return error; }
if (((*lpdwCf = (DWORD) SSRegisterClipboardFormatA(szACF))) == 0) { return ResultFromScode(DV_E_CLIPFORMAT); } }
return NOERROR; }
//+-------------------------------------------------------------------------
//
// Function: WriteClipformatStm
//
// Synopsis: Writes the clipboard format the given stream
//
// Arguments: [lpstream] -- pointer to the stream
// [cf] -- the clipboard format
//
// Returns: HRESULT
//
// History: dd-mmm-yy Author Comment
// 11-Jan-94 alexgo cast -1 to a DWORD to remove compile
// warning
// 27-Oct-93 alexgo 32bit port
// 16-Mar-94 davepl Revereted to ANSI string writes
//
// Notes: see ReadClipformatStm for a description of the
// data layout in the stream
//
//--------------------------------------------------------------------------
#pragma SEG(WriteClipformatStm)
STDAPI WriteClipformatStm(LPSTREAM lpstream, CLIPFORMAT cf) { VDATEHEAP();
HRESULT error;
VDATEIFACE( lpstream );
//REVIEW32 where did 0xC000 come from??? Is this
//portable to NT && Chicago??? Try to replace with a constant.
//(although there don't seem to be any :( )
if (cf < 0xC000) { DWORD dwBuf[2]; DWORD dwSize = sizeof(DWORD);
if (cf == NULL) { dwBuf[0] = NULL; } else { // write -1L, to indicate NON NULL predefined
// clipboard format
dwBuf[0] = (DWORD)-1L; dwBuf[1] = (DWORD)cf; dwSize += sizeof(DWORD); }
if (error = StWrite(lpstream, dwBuf, dwSize)) { return error; }
} else { // it is a registerd clipboard format
char szACF[MAX_STR]; ULONG len;
// Get the name of the clipboard format
len = SSGetClipboardFormatNameA(cf, szACF, sizeof(szACF)); if (0 == len) { return ResultFromScode(E_UNSPEC); }
++len; // Account for NULL terminator
if (error = StWrite(lpstream, &len, sizeof(len))) { return error; }
// Write it (plus terminator) to the stream
if (error = StWrite(lpstream, szACF, len)) { return error; } }
return NOERROR; }
//+-------------------------------------------------------------------------
//
// Function: WriteMonikerStm
//
// Synopsis: Writes the persistent state of the given moniker to the
// given stream. Internal
//
// Effects:
//
// Arguments: [pstm] -- pointer to the stream
// [pmk] -- pointer to the moniker
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(WriteMonikerStm)
// write size long followed by persistent moniker
STDAPI WriteMonikerStm (LPSTREAM pstm, LPMONIKER pmk) { VDATEHEAP();
DWORD cb = NULL; ULARGE_INTEGER dwBegin; ULARGE_INTEGER dwEnd; HRESULT error; LARGE_INTEGER large_integer;
VDATEIFACE( pstm );
if (pmk == NULL) return pstm->Write(&cb, sizeof(DWORD), NULL); else { VDATEIFACE( pmk ); // get the begining position
LISet32( large_integer, 0 ); if ((error = pstm->Seek (large_integer, STREAM_SEEK_CUR, &dwBegin)) != NOERROR) return error;
// skip the moniker size DWORD
LISet32( large_integer, 4); if ((error = pstm->Seek (large_integer, STREAM_SEEK_CUR, NULL)) != NOERROR) return error;
if ((error = OleSaveToStream (pmk, pstm)) != NOERROR) return error;
// get the end position
LISet32( large_integer, 0); if ((error = pstm->Seek (large_integer, STREAM_SEEK_CUR, &dwEnd)) != NOERROR) return error;
// moniker data size
cb = dwEnd.LowPart - dwBegin.LowPart;
// seek to the begining position
LISet32( large_integer, dwBegin.LowPart); if ((error = pstm->Seek (large_integer, STREAM_SEEK_SET,NULL)) != NOERROR) return error;
// write moniker info size
if ((error = pstm->Write(&cb, sizeof(DWORD), NULL)) != NOERROR) return error;
// seek to the end position
LISet32( large_integer, dwEnd.LowPart); return pstm->Seek (large_integer, STREAM_SEEK_SET, NULL); } }
//+-------------------------------------------------------------------------
//
// Function: ReadMonikerStm
//
// Synopsis: Reads a moniker from the given stream (inverse of
// WriteMonikerStm)
//
// Effects:
//
// Arguments: [pstm] -- pointer to the stream
// [ppmk] -- where to put the moniker
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 27-Oct-93 alexgo 32bit port
//
// Notes:
//
//--------------------------------------------------------------------------
#pragma SEG(ReadMonikerStm)
// read size long followed by persistent moniker
STDAPI ReadMonikerStm (LPSTREAM pstm, LPMONIKER FAR* ppmk) { VDATEHEAP();
HRESULT error; DWORD cb;
VDATEPTROUT( ppmk, LPMONIKER ); *ppmk = NULL; VDATEIFACE( pstm );
if ((error = StRead (pstm, &cb, sizeof(DWORD))) != NOERROR) return error;
if (cb == NULL) return NOERROR;
return OleLoadFromStream (pstm, IID_IMoniker, (LPLPVOID) ppmk); }
//+-------------------------------------------------------------------------
//
// Function: OleDraw
//
// Synopsis: Calls IViewObject->Draw on the given object
//
// Effects: Draws something on the screen :)
//
// Arguments: [lpUnk] -- pointer to the object
// [dwAspect] -- aspect to draw (NORMAL, ICON, etc)
// [hdcDraw] -- the device context to use
// [lprcBounds] -- the rectangle in which to draw
//
// Requires:
//
// Returns: HRESULT
//
// Signals:
//
// Modifies:
//
// Algorithm:
//
// History: dd-mmm-yy Author Comment
// 01-Apr-94 alexgo fixed usage of MAKELONG (only
// for 16bit)
// 27-Oct-93 alexgo 32bit port
//
// Notes:
// On Win32, RECT and RECTL are identical structures, thus there
// is no need to convert from RECT to RECTL with MAKELONG.
//
//--------------------------------------------------------------------------
#pragma SEG(OleDraw)
STDAPI OleDraw (LPUNKNOWN lpUnk, DWORD dwAspect, HDC hdcDraw, LPCRECT lprcBounds) { HRESULT error; IViewObject FAR* lpView;
OLETRACEIN((API_OleDraw, PARAMFMT("lpUnk= %p, dwAspect= %x, hdcDraw= %h, lprcBounds= %tr"), lpUnk, dwAspect, hdcDraw, lprcBounds));
VDATEHEAP();
VDATEIFACE_LABEL( lpUnk, errRtn, error ); VDATEPTRIN_LABEL( lprcBounds, RECT, errRtn, error);
if ((error = lpUnk->QueryInterface (IID_IViewObject, (LPLPVOID)&lpView)) != NOERROR) { error = ResultFromScode(DV_E_NOIVIEWOBJECT); goto errRtn; }
error = lpView->Draw (dwAspect, DEF_LINDEX, 0, 0, 0, hdcDraw, (LPCRECTL)lprcBounds, 0, 0,0); lpView->Release();
errRtn: OLETRACEOUT((API_OleDraw, error));
return error; }
//+----------------------------------------------------------------------------
//
// Function:
// CreateObjectDescriptor, static
//
// Synopsis:
// Creates and initializes an OBJECTDESCRIPTOR from the given
// parameters
//
// Arguments:
// [clsid] -- the class ID of the object being transferred
// [dwAspect] -- the display aspect drawn by the source of the
// transfer
// [psizel] -- pointer to the size of the object
// [ppointl] -- pointer to the mouse offset in the object that
// initiated a drag-drop transfer
// [dwStatus] -- the OLEMISC status flags for the object
// being transferred
// [lpszFullUserTypeName] -- the full user type name of the
// object being transferred
// [lpszSrcOfCopy] -- a human readable name for the object
// being transferred
//
// Returns:
// If successful, A handle to the new OBJECTDESCRIPTOR; otherwise
// NULL.
//
// Notes:
// REVIEW, this seems generally useful for anyone using the
// clipboard, or drag-drop; perhaps it should be exported.
//
// History:
// 12/07/93 - ChrisWe - file inspection and cleanup
//
//-----------------------------------------------------------------------------
#pragma SEG(CreateObjectDescriptor)
INTERNAL_(HGLOBAL) CreateObjectDescriptor(CLSID clsid, DWORD dwAspect, const SIZEL FAR *psizel, const POINTL FAR *ppointl, DWORD dwStatus, LPOLESTR lpszFullUserTypeName, LPOLESTR lpszSrcOfCopy) { VDATEHEAP();
DWORD dwFullUserTypeNameBLen; // length of lpszFullUserTypeName in BYTES
DWORD dwSrcOfCopyBLen; // length of lpszSrcOfCopy in BYTES
HGLOBAL hMem; // handle to the object descriptor
LPOBJECTDESCRIPTOR lpOD; // the new object descriptor
// Get the length of Full User Type Name; Add 1 for the null terminator
if (!lpszFullUserTypeName) dwFullUserTypeNameBLen = 0; else dwFullUserTypeNameBLen = (_xstrlen(lpszFullUserTypeName) + 1) * sizeof(OLECHAR);
// Get the Source of Copy string and it's length; Add 1 for the null
// terminator
if (lpszSrcOfCopy) dwSrcOfCopyBLen = (_xstrlen(lpszSrcOfCopy) + 1) * sizeof(OLECHAR); else { // No src moniker so use user type name as source string.
lpszSrcOfCopy = lpszFullUserTypeName; dwSrcOfCopyBLen = dwFullUserTypeNameBLen; }
// allocate the memory where we'll put the object descriptor
hMem = GlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE, sizeof(OBJECTDESCRIPTOR) + dwFullUserTypeNameBLen + dwSrcOfCopyBLen); if (hMem == NULL) goto error;
lpOD = (LPOBJECTDESCRIPTOR)GlobalLock(hMem); if (lpOD == NULL) goto error;
// Set the FullUserTypeName offset and copy the string
if (!lpszFullUserTypeName) { // zero offset indicates that string is not present
lpOD->dwFullUserTypeName = 0; } else { lpOD->dwFullUserTypeName = sizeof(OBJECTDESCRIPTOR); _xmemcpy(((BYTE FAR *)lpOD)+lpOD->dwFullUserTypeName, (const void FAR *)lpszFullUserTypeName, dwFullUserTypeNameBLen); }
// Set the SrcOfCopy offset and copy the string
if (!lpszSrcOfCopy) { // zero offset indicates that string is not present
lpOD->dwSrcOfCopy = 0; } else { lpOD->dwSrcOfCopy = sizeof(OBJECTDESCRIPTOR) + dwFullUserTypeNameBLen; _xmemcpy(((BYTE FAR *)lpOD)+lpOD->dwSrcOfCopy, (const void FAR *)lpszSrcOfCopy, dwSrcOfCopyBLen); }
// Initialize the rest of the OBJECTDESCRIPTOR
lpOD->cbSize = sizeof(OBJECTDESCRIPTOR) + dwFullUserTypeNameBLen + dwSrcOfCopyBLen; lpOD->clsid = clsid; lpOD->dwDrawAspect = dwAspect; lpOD->sizel = *psizel; lpOD->pointl = *ppointl; lpOD->dwStatus = dwStatus;
GlobalUnlock(hMem); return(hMem);
error: if (hMem) { GlobalUnlock(hMem); GlobalFree(hMem); }
return(NULL); }
|