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.
5828 lines
156 KiB
5828 lines
156 KiB
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1993.
|
|
//
|
|
// File: ostm2stg.cpp
|
|
//
|
|
// Contents: OLE 1 - OLE 2 Stream/IStorage Interoperatbility
|
|
//
|
|
// Functions: Implements API functions:
|
|
// OleConvertOLESTREAMToIStorage
|
|
// OleConvertIStorageToOLESTREAM
|
|
// OleConvertOLESTREAMToIStorageEx
|
|
// OleConvertIStorageToOLESTREAMEx
|
|
//
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 03-Feb-92 jasonful original version
|
|
// 08-Aug-93 srinik added Ex functions
|
|
// 12-Feb-94 davepl 32-bit port
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#include <le2int.h>
|
|
#include "ostm2stg.h"
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
#include <string.h>
|
|
#include <ole1cls.h>
|
|
|
|
ASSERTDATA
|
|
|
|
// We need a ptr value which will indicate that the associated handle
|
|
// is a metafile handle, and therefore cannot be cleaned up as if
|
|
// it were a normal global memory handle
|
|
|
|
#define METADATAPTR ((void *) -1)
|
|
|
|
// This fn is not prototyped in any include file, since it was static
|
|
// to its file. Need to add the prototype to a common include file.
|
|
|
|
HRESULT STDAPICALLTYPE CreateOle1FileMoniker(LPWSTR,REFCLSID,LPMONIKER FAR*);
|
|
|
|
// This is defined in the new privstm.cpp; must be added to an include file.
|
|
|
|
STDAPI ReadFmtProgIdStg ( IStorage * pstg, LPOLESTR * pszProgID );
|
|
FARINTERNAL wWriteFmtUserType (LPSTORAGE, REFCLSID);
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CGenericObject::CGenericObject
|
|
//
|
|
// Synopsis: Constructor for CGenericObject class
|
|
//
|
|
// Effects: Initializes all child pointers to NULL and sets
|
|
// flags to FALSE
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Cleanup and document
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
CGenericObject::CGenericObject(void)
|
|
{
|
|
m_ppres = NULL; // Presentation data
|
|
m_fLink = FALSE; // Flag: Linked (T) or Embedded (F)
|
|
m_fStatic = FALSE; // Flag: Static object
|
|
m_fNoBlankPres = FALSE; // Flag: do not want a blank presentation
|
|
m_szTopic = NULL; // Topic string for this object
|
|
m_szItem = NULL; // Item (file) string for this object
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CGenericObject::~CGenericObject
|
|
//
|
|
// Synopsis: Desctuctor for CGenericObject class
|
|
//
|
|
// Effects: Removes children then self
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 12-Aug-94 alexgo check for NULL before delete
|
|
// 14-Feb-94 davepl Cleanup and document
|
|
//
|
|
// Notes: As much as I hated to do it, some of these strings
|
|
// have to be freed with PubMemFree because they are
|
|
// allocated by UtDupString, which allocates public memory.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CGenericObject::~CGenericObject (void)
|
|
{
|
|
if( m_ppres )
|
|
{
|
|
delete m_ppres; // Presentation data
|
|
}
|
|
|
|
if( m_szTopic )
|
|
{
|
|
PubMemFree(m_szTopic); // Topic string
|
|
}
|
|
|
|
if( m_szItem )
|
|
{
|
|
PubMemFree(m_szItem); // Item string
|
|
}
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CData::CData
|
|
//
|
|
// Synopsis: Constructor for a simple class which holds a piece of
|
|
// memory.
|
|
//
|
|
// Effects: Clears size, flags, and pointer
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
//
|
|
// Notes: 14-Feb-94 davepl Cleanup and document
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CData::CData (void)
|
|
{
|
|
m_cbSize = 0; // Count, in bytes, of data size
|
|
m_h = NULL; // Memory handke
|
|
m_pv= NULL; // Memory pointer
|
|
m_fNoFree = FALSE; // Flag: Should memory be freed in destructor
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CData::~CData
|
|
//
|
|
// Synopsis: Destructor for simple data class
|
|
//
|
|
// Effects: Unlocks and frees memory if m_fNoFree is not set
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Cleanup and document
|
|
//
|
|
// Notes: If a metafile handle is stored in the handle, the
|
|
// pointer will be marked with a special value, indicating
|
|
// that we must DeleteMetafile, not GlobalFree the handle.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CData::~CData (void)
|
|
{
|
|
if (m_h) // Do we have a handle?
|
|
{
|
|
if (m_pv == METADATAPTR)
|
|
{
|
|
LEVERIFY(DeleteMetaFile((HMETAFILE) m_h));
|
|
}
|
|
else
|
|
{
|
|
GlobalUnlock (m_h); // Dec lock count
|
|
if (!m_fNoFree) // Free this memory if we
|
|
{ // have been flagged to do so
|
|
LEVERIFY(0==GlobalFree (m_h));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CFormat::CFormat
|
|
//
|
|
// Synopsis: CFormat class constructor
|
|
//
|
|
// Effects: Initializes format tag and clipboard format
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
//
|
|
// Notes: 14-Feb-94 davepl Cleanup and document
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CFormat::CFormat (void)
|
|
{
|
|
m_ftag = ftagNone; // Format tag (string, clipformat, or none)
|
|
m_cf = 0; // Clipboard format
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CClass::CClass
|
|
//
|
|
// Synopsis: CClass constructor
|
|
//
|
|
// Effects: sets class ID and class ID string to NULL
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CClass::CClass (void)
|
|
{
|
|
m_szClsid = NULL;
|
|
m_clsid = CLSID_NULL;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CPres::CPres
|
|
//
|
|
// Synopsis: CPres constructor
|
|
//
|
|
// Effects: Initializes height & width of presentation data to zero.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
CPres::CPres (void)
|
|
{
|
|
m_ulHeight = 0L;
|
|
m_ulWidth = 0L;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CClass::Set, INTERNAL
|
|
//
|
|
// Synopsis: Sets the m_szClsid based on clsid
|
|
//
|
|
// Effects: Sets m_szClsid in the following order of preference:
|
|
// - ProgID obtained from ProgIDFromCLSID()
|
|
// - If it is a static type, m_szClsid is left blank
|
|
// - Tries to read it from [pstg]
|
|
// - Tries to obtain it from registry based on CLSID
|
|
//
|
|
//
|
|
// Arguments: [clsid] - class id object is to be set to
|
|
// [pstg] - storage which may contain info on the
|
|
// clipboard format as a last resort
|
|
//
|
|
// Returns: NOERROR on success
|
|
// REGDB_E_CLASSNOTREG unknown class
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes: Hard-coded maximum of 256 character clip format name.
|
|
// On failure, m_clsid has still been set to clsid.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
INTERNAL CClass::Set (REFCLSID clsid, LPSTORAGE pstg)
|
|
{
|
|
CLIPFORMAT cf;
|
|
unsigned short const ccBufSize = 256;
|
|
LPOLESTR szProgId = NULL;
|
|
|
|
Assert (m_clsid == CLSID_NULL && m_szClsid == NULL);
|
|
|
|
// set the m_clsid member in the object
|
|
m_clsid = clsid;
|
|
|
|
// If we can get it using ProgIDFromCLSID, that's the simplest case
|
|
if (NOERROR == wProgIDFromCLSID (clsid, &m_szClsid))
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
// If not, maybe the object is static, in which case we leave the
|
|
// class string NULL
|
|
|
|
if (IsEqualCLSID(CLSID_StaticMetafile, clsid) ||
|
|
IsEqualCLSID(CLSID_StaticDib, clsid))
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
// If still no luck, try to read the clipboard format from the storage
|
|
// and then look that up.
|
|
|
|
if (pstg &&
|
|
SUCCEEDED(ReadFmtUserTypeStg(pstg, &cf, NULL)) &&
|
|
SUCCEEDED(ReadFmtProgIdStg (pstg, &szProgId)))
|
|
{
|
|
// Last-ditch effort. If the class is an unregistered OLE1 class,
|
|
// the ProgID should still be obtainable from the format tag.
|
|
// If the class is an unregistered OLE2 class, the ProgId should be
|
|
// at the end of the CompObj stream.
|
|
|
|
if (CoIsOle1Class(clsid))
|
|
{
|
|
Verify (GetClipboardFormatName (cf, szProgId, ccBufSize));
|
|
}
|
|
else
|
|
{
|
|
// If its an OLE 2 object and we couldn't get the program ID from
|
|
// the storage, we're out of luck
|
|
|
|
if (szProgId == NULL || szProgId[0] == L'\0')
|
|
{
|
|
if (szProgId)
|
|
{
|
|
PubMemFree(szProgId);
|
|
}
|
|
return ResultFromScode (REGDB_E_CLASSNOTREG);
|
|
}
|
|
}
|
|
|
|
// At this point we know we have a program ID and that this is an
|
|
// OLE 2 object, so we use the program ID as the class name.
|
|
|
|
m_szClsid = szProgId;
|
|
return NOERROR;
|
|
}
|
|
else
|
|
{
|
|
// If we hit this path, we couldn't read from the storage
|
|
|
|
return ResultFromScode (REGDB_E_CLASSNOTREG);
|
|
}
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CClass:SetSz, INTERNAL
|
|
//
|
|
// Synopsis: Sets CGenericObject's CLASS member ID based on the class
|
|
// name passed in.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 15-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
INTERNAL CClass::SetSz (LPOLESTR sz)
|
|
{
|
|
HRESULT hr;
|
|
|
|
// The class info should be completely unset at this point
|
|
Assert (m_clsid==CLSID_NULL && m_szClsid==NULL);
|
|
|
|
m_szClsid = sz;
|
|
|
|
if (FAILED(hr = wCLSIDFromProgID (sz, &m_clsid, TRUE)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CClass::Reset
|
|
//
|
|
// Synopsis: Frees the Class ID string for CClass and resets the pointer,
|
|
// then sets the class ID and string bassed on the CLSID
|
|
// passed in.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes: Class ID must already be set before calling RESET
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
INTERNAL CClass::Reset (REFCLSID clsid)
|
|
{
|
|
m_clsid = clsid;
|
|
|
|
// We should already have a class ID string if we are _re_setting it
|
|
Assert(m_szClsid);
|
|
|
|
PubMemFree(m_szClsid);
|
|
|
|
HRESULT hr;
|
|
m_szClsid = NULL;
|
|
|
|
if (FAILED(hr = wProgIDFromCLSID (clsid, &m_szClsid)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Member: CClass::~CClass
|
|
//
|
|
// Synopsis: CClass destructor
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 12-Aug-94 alexgo check for NULL before free'ing memory
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
CClass::~CClass (void)
|
|
{
|
|
// The string is created by UtDupString, so its public memory
|
|
|
|
if( m_szClsid )
|
|
{
|
|
PubMemFree(m_szClsid);
|
|
}
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wConvertOLESTREAMToIStorage, INTERNAL
|
|
//
|
|
// Synopsis: Worker function. Ensures the OLESTREAM is correctly
|
|
// set up then calls OLESTREAMToGenericObject.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 15-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
INTERNAL wConvertOLESTREAMToIStorage(
|
|
LPOLESTREAM polestream,
|
|
LPSTORAGE pstg,
|
|
PGENOBJ pgenobj)
|
|
{
|
|
VDATEIFACE (pstg);
|
|
|
|
#if DBG==1
|
|
if (!IsValidReadPtrIn (polestream, sizeof(OLESTREAM)) ||
|
|
!IsValidReadPtrIn (polestream->lpstbl, sizeof(OLESTREAMVTBL)) ||
|
|
!IsValidCodePtr ((FARPROC)polestream->lpstbl->Get))
|
|
{
|
|
AssertSz (0, "Bad OLESTREAM");
|
|
return ResultFromScode (E_INVALIDARG);
|
|
}
|
|
#endif
|
|
|
|
return OLESTREAMToGenericObject (polestream, pgenobj);
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OleConvertOLESTREAMToIStorage, STDAPI
|
|
//
|
|
// Synopsis: Given an OLE 1 stream and an OLE 2 storage, reads an object
|
|
// from the OLE 1 stream into a CGenericObject. Once read in,
|
|
// the object is written from generic format back into the OLE 2
|
|
// storage object.
|
|
//
|
|
// Arguments: [polestream] -- OLE 1 stream to read object from
|
|
// [pstg] -- OLE 2 storage to write object to
|
|
// [ptd] -- Target device
|
|
//
|
|
// Requires: Streams should be set up, and OLE 1 stream should be
|
|
// positioned at the beginning of the next OLE 1 object
|
|
// to be read.
|
|
//
|
|
// Returns: [DV_E_DVTARGETDEVICE] Invalid write ptr to target device
|
|
// CONVERT10_E_OLESTREAM_FMT On unknown OLE 1 format
|
|
// CONVERT10_E_OLESTREAM_GET On stream read failue
|
|
// E_OUTOFMEMORY On stream I/O memory failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDAPI OleConvertOLESTREAMToIStorage(
|
|
LPOLESTREAM polestream,
|
|
LPSTORAGE pstg,
|
|
const DVTARGETDEVICE FAR* ptd)
|
|
{
|
|
|
|
OLETRACEIN((API_OleConvertOLESTREAMToIStorage,
|
|
PARAMFMT("polestream= %p, pstg= %p, ptd= %td"),
|
|
polestream, pstg, ptd));
|
|
|
|
LEDebugOut((DEB_TRACE, "%p _IN OleConvertOLESTREAMToIStorage ("
|
|
" %p , %p , %p)\n", 0 /*function*/,
|
|
polestream, pstg, ptd
|
|
));
|
|
CALLHOOKOBJECT(S_OK,CLSID_NULL,IID_IStorage,(IUnknown **)&pstg);
|
|
|
|
HRESULT hresult;
|
|
|
|
// This is the generic object we will use as intermediate storage to
|
|
// hold the contents of the OLESTREAM
|
|
|
|
CGenericObject genobj;
|
|
|
|
if (ptd)
|
|
{
|
|
// The side of the td is the first DWORD. Ensure that much is
|
|
// valid and then we can use it to check the whole structure.
|
|
if (!IsValidReadPtrIn (ptd, sizeof(DWORD)))
|
|
{
|
|
hresult = ResultFromScode (DV_E_DVTARGETDEVICE);
|
|
goto errRtn;
|
|
}
|
|
if (!IsValidReadPtrIn (ptd, (UINT) ptd->tdSize))
|
|
{
|
|
hresult = ResultFromScode (DV_E_DVTARGETDEVICE_SIZE);
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hresult=wConvertOLESTREAMToIStorage(polestream,pstg,&genobj)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// If we were able to read the object out of the stream, we can now try
|
|
// to write it back out to the storage
|
|
|
|
hresult = GenericObjectToIStorage (genobj, pstg, ptd);
|
|
|
|
errRtn:
|
|
LEDebugOut((DEB_TRACE, "%p OUT OleConvertOLESTREAMToIStorage ( %lx ) "
|
|
"\n", 0 /*function*/, hresult));
|
|
|
|
OLETRACEOUT((API_OleConvertOLESTREAMToIStorage, hresult));
|
|
return hresult;
|
|
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PrependUNCName
|
|
//
|
|
// Synopsis: Convert *pszFile to use szUNC=="\\server\share" instead
|
|
// of drive letter
|
|
//
|
|
// Effects: Deletes the UNC name and returns *ppszFile as a NEW string
|
|
// with the full UNC filename. The string originally held at
|
|
// *ppszFile is deleted by this function.
|
|
//
|
|
// Arguments: [ppszFile] Pointer to incoming filename string pointer
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleanup, documentation, allocation fixes
|
|
//
|
|
// Notes: This function does some frightening things by changing the
|
|
// caller's pointer and deleting various reference parameters.
|
|
// Be sure you know what's going on before turning this function
|
|
// loose on one of your own pointers.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PrependUNCName (LPOLESTR FAR* ppszFile, LPOLESTR szUNC)
|
|
{
|
|
HRESULT hresult = NOERROR;
|
|
LPOLESTR szNew;
|
|
|
|
// No place to put result, so nothing to do...
|
|
if (NULL==szUNC)
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// Ensure the caller's pointer is valid
|
|
if (NULL == *ppszFile)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
|
|
// Ensure the second letter of path is a colon (ie; X:\file)
|
|
if((*ppszFile)[1] != L':')
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
|
|
// Allocate enough space for new filename (we will be
|
|
// omitting the X: portion of the filename, so this calculation
|
|
// is _not_ short by 2 as it may appear)
|
|
|
|
szNew = (LPOLESTR)
|
|
PubMemAlloc((_xstrlen(*ppszFile)+_xstrlen (szUNC)) * sizeof(OLECHAR));
|
|
|
|
if (NULL == szNew)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Copy over the UNC name
|
|
_xstrcpy (szNew, szUNC);
|
|
|
|
// Add the original name, except for the X:
|
|
_xstrcat (szNew, (*ppszFile) + 2);
|
|
|
|
// Free the original name
|
|
PubMemFree(*ppszFile);
|
|
*ppszFile = szNew;
|
|
|
|
// Delete the UNC name
|
|
PubMemFree(szUNC);
|
|
return hresult;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLESTREAMToGenericObject, INTERNAL
|
|
//
|
|
// Synopsis: Reads and OLE 1.0 version of an object from an OLE 1 stream
|
|
// and stores it internally, including presentation and native
|
|
// data, in a GenericObject.
|
|
//
|
|
// Effects: Creates a GenericObject that can be written back in OLE 1
|
|
// or OLE 2 format
|
|
//
|
|
// Arguments: [pos] -- pointer to OLE 1 stream to read object from
|
|
// [pgenobj] -- pointer to generic object to read into
|
|
//
|
|
// Requires: Input stream setup and GenObj created
|
|
//
|
|
// Returns: NOERROR On success
|
|
// CONVERT10_E_OLESTREAM_FMT On unknown OLE 1 format
|
|
// CONVERT10_E_OLESTREAM_GET On stream read failue
|
|
// E_OUTOFMEMORY On stream I/O memory failure
|
|
//
|
|
// Signals: (none)
|
|
//
|
|
// Modifies: Stream position, GenObj
|
|
//
|
|
// Algorithm:
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Added Trace code
|
|
// davepl Cleaned up and documented
|
|
// davepl Rerouted errors through central return
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
#pragma SEG(OLESTREAMToGenericObject)
|
|
static INTERNAL OLESTREAMToGenericObject
|
|
(
|
|
LPOLESTREAM pos,
|
|
PGENOBJ pgenobj
|
|
)
|
|
{
|
|
HRESULT error = NOERROR;
|
|
ULONG ulFmtId;
|
|
LPOLESTR szClass = NULL;
|
|
|
|
// Read OLE Version # from the stream and discard it
|
|
if (FAILED(error = OLE1StreamToUL(pos, NULL)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read OLE ver# from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
|
|
// Get Format ID from the stream
|
|
if (FAILED(error = OLE1StreamToUL(pos, &ulFmtId)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read format ID from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
|
|
// If this is a static object, read it into the generic object and return
|
|
if (ulFmtId == FMTID_STATIC)
|
|
{
|
|
if (FAILED(error = GetStaticObject (pos, pgenobj)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read static object at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
}
|
|
goto errRtn;
|
|
}
|
|
|
|
// If this is neither a linked nor an embedded object, something
|
|
// is wrong
|
|
if (ulFmtId != FMTID_LINK && ulFmtId != FMTID_EMBED)
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Object is neither linked nor embedded at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
error = ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
goto errRtn;
|
|
}
|
|
|
|
// If this is a linked object, set our flag in GenericObject
|
|
if (FMTID_LINK == ulFmtId)
|
|
{
|
|
pgenobj->m_fLink = TRUE;
|
|
}
|
|
|
|
// Read the class name from the stream
|
|
|
|
if (FAILED(error = OLE1StmToString(pos, &szClass)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read the class name from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
if (NULL == szClass)
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Class name was returned NULL at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
error = CONVERT10_E_OLESTREAM_FMT;
|
|
goto errRtn;
|
|
}
|
|
|
|
// If this is an embedded object, set the class ID and class string
|
|
// If it is a linked object, set the class name but set the class ID
|
|
// to CLSID_StdOleLink
|
|
|
|
if (FMTID_EMBED == ulFmtId)
|
|
{
|
|
pgenobj->m_class.SetSz (szClass);
|
|
}
|
|
else
|
|
{
|
|
Assert (ulFmtId == FMTID_LINK);
|
|
pgenobj->m_classLast.SetSz (szClass);
|
|
pgenobj->m_class.Set (CLSID_StdOleLink, NULL);
|
|
}
|
|
|
|
// Read the Topic string from the stream
|
|
if (FAILED(error = OLE1StmToString(pos, &(pgenobj->m_szTopic))))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read topic string from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the Item string from the stream
|
|
if (FAILED(error = OLE1StmToString(pos, &(pgenobj->m_szItem))))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to get item string from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// If this is a linked object, set up the filename etc.
|
|
if (FMTID_LINK == ulFmtId)
|
|
{
|
|
LPOLESTR szUNCName = NULL;
|
|
|
|
// Read the network name from the stream
|
|
|
|
if (FAILED(error = OLE1StmToString(pos, &szUNCName)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to get network name from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// Convert a drive-letter based name to \\srv\share name
|
|
if (FAILED(error = PrependUNCName (&(pgenobj->m_szTopic), szUNCName)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to convert drv ltr to UNC name at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read network type and network driver version # from stream
|
|
// (They are both shorts and we discarding them, so read a LONG)
|
|
if (FAILED(error = OLE1StreamToUL (pos, NULL)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to get net type/ver from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the link-updating options from the stream. This field
|
|
// use OLE 1.0 enumeration values for the link update options
|
|
if (FAILED(error = OLE1StreamToUL(pos, &(pgenobj->m_lnkupdopt))))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to read link update opts at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
// OLE 1.0 duplicates the link update options in the highword
|
|
// of the LONG, and we don't want that, so clear the highword.
|
|
|
|
pgenobj->m_lnkupdopt &= 0x0000FFFF;
|
|
}
|
|
else // This path is taken to read in embedded objects
|
|
{
|
|
Assert (ulFmtId == FMTID_EMBED);
|
|
|
|
// Read and store the native data from the stream
|
|
if (FAILED(error = GetSizedDataOLE1Stm (pos, &(pgenobj->m_dataNative))))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to get native data from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
// For both linked and embedded objects, we need to read in any
|
|
// presentation data that may be present. Note that certain formats
|
|
// such as MS-Paint will not provide presentation data; this is OK
|
|
// since they can be rendered by native data alone (space saving measure)
|
|
|
|
if (FAILED(error = GetPresentationObject (pos, pgenobj)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to get presentation data from stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
|
|
errRtn:
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p OUT OLESTREAMToGenericObject ( %lx ) \n",
|
|
NULL /*function*/, error));
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetStaticObject, INTERNAL
|
|
//
|
|
// Synopsis: Reads the presentation data for a static object into the
|
|
// PPRES member of GenericObject, and sets format and class
|
|
// flags accordingly
|
|
//
|
|
// Effects:
|
|
//
|
|
// Arguments: [pos] -- stream we are reading OLE 1 object from
|
|
// [pgenobj] -- GenericObject we are reading into
|
|
// Requires:
|
|
//
|
|
// Returns: NOERROR On success
|
|
// CONVERT10_E_OLESTREAM_FMT On unknown OLE 1 format
|
|
// CONVERT10_E_OLESTREAM_GET On stream read failue
|
|
// E_OUTOFMEMORY On stream I/O memory failure
|
|
//
|
|
// Signals: (none)
|
|
//
|
|
// Modifies: Stream position, GenericObject
|
|
//
|
|
// Algorithm:
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Cleanup and documentation
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetStaticObject (LPOLESTREAM pos, PGENOBJ pgenobj)
|
|
{
|
|
HRESULT error;
|
|
|
|
// Read the presentation data, standard or generic, into the
|
|
// PPRES member of the GenericObject
|
|
if (FAILED(error = GetPresentationObject(pos, pgenobj, TRUE)))
|
|
{
|
|
return ResultFromScode(error);
|
|
}
|
|
|
|
// Ensure that the format tag is a clipboard format
|
|
if (ftagClipFormat != pgenobj->m_ppres->m_format.m_ftag)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
|
|
// If the clipboard format is a METAFILEPIC, set the CLASS
|
|
// member of GenericObject to CLSID_StaticMetafile
|
|
if (CF_METAFILEPICT == pgenobj->m_ppres->m_format.m_cf)
|
|
{
|
|
pgenobj->m_class.Set (CLSID_StaticMetafile, NULL);
|
|
}
|
|
|
|
// Otherwise, check to see if it is a DIB, and set the CLASS
|
|
// member accordingly
|
|
|
|
else if (CF_DIB == pgenobj->m_ppres->m_format.m_cf)
|
|
{
|
|
pgenobj->m_class.Set (CLSID_StaticDib, NULL);
|
|
}
|
|
|
|
// If it is neither a METAFILEPIC nor a DIB, we have a problem
|
|
|
|
else
|
|
{
|
|
AssertSz (0, "1.0 static object not in one of 3 standard formats");
|
|
return ResultFromScode (CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
|
|
// Flag the GenericObject as Static
|
|
pgenobj->m_fStatic = TRUE;
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: CreateBlankPres, INTERNAL
|
|
//
|
|
// Synopsis: Sets up the format in the PPRES struct as ClipFormat 0
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL CreateBlankPres(PPRES ppres)
|
|
{
|
|
Assert (ppres);
|
|
ppres->m_format.m_ftag = ftagClipFormat;
|
|
ppres->m_format.m_cf = 0;
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetPresentationObject, INTERNAL
|
|
//
|
|
// Synopsis: Reads the presentation data into the CGenericObject object
|
|
//
|
|
// Arguments: [pos] -- OLE 1 stream we are reading from
|
|
// [pgenobj] -- Generic object we are reading to
|
|
// [fStatic] -- Flag: getting a static pres object?
|
|
//
|
|
// Requires: stream open, object allocated
|
|
//
|
|
// Returns: CONVERT10_E_OLESTREAM_FMT unknown format id in stream
|
|
//
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetPresentationObject(
|
|
LPOLESTREAM pos,
|
|
PGENOBJ pgenobj,
|
|
BOOL fStatic)
|
|
{
|
|
LPOLESTR szClass = NULL;
|
|
HRESULT hresult = NOERROR;
|
|
|
|
Assert (pgenobj->m_ppres==NULL);
|
|
|
|
if (TRUE != fStatic) //FALSE!
|
|
{
|
|
// Pull the OLE version number out of the stream, we don't want it
|
|
|
|
if (FAILED(hresult = OLE1StreamToUL(pos, NULL)))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// Pull the OLE 1 format identifier out of the stream
|
|
|
|
ULONG ulFmtId;
|
|
if (FAILED(hresult = OLE1StreamToUL (pos, &ulFmtId)))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// If the format identifier is not FMTID_PRES, we've got a
|
|
// problem... unless it's 0 in which case it simply means
|
|
// that there _is no_ presentation data, ie: PBrush, Excel
|
|
|
|
if (ulFmtId != FMTID_PRES)
|
|
{
|
|
if (0==ulFmtId)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
else
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Pull in the type name for the OLE1 data
|
|
|
|
if (FAILED(hresult = OLE1StmToString (pos, &szClass)))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
if (0==_xstrcmp (szClass, OLESTR("METAFILEPICT")))
|
|
{
|
|
hresult = GetStandardPresentation (pos, pgenobj, CF_METAFILEPICT);
|
|
}
|
|
else if (0==_xstrcmp (szClass, OLESTR("BITMAP")))
|
|
{
|
|
hresult = GetStandardPresentation (pos, pgenobj, CF_BITMAP);
|
|
}
|
|
else if (0==_xstrcmp (szClass, OLESTR("DIB")))
|
|
{
|
|
hresult = GetStandardPresentation (pos, pgenobj, CF_DIB);
|
|
}
|
|
else if (0==_xstrcmp (szClass, OLESTR("ENHMETAFILE")))
|
|
{
|
|
Assert (0 && "Encountered an unsupported format: ENHMETAFILE");
|
|
}
|
|
else
|
|
{
|
|
// This is a Generic Presentation stream
|
|
|
|
#if DBG==1
|
|
Assert (!fStatic);
|
|
if (_xstrcmp (pgenobj->m_fLink
|
|
? pgenobj->m_classLast.m_szClsid
|
|
: pgenobj->m_class.m_szClsid, szClass))
|
|
{
|
|
Assert (0 && "Class name in embedded object stream does\n"
|
|
"not match class name in pres object stream");
|
|
}
|
|
#endif
|
|
hresult = GetGenericPresentation (pos, pgenobj);
|
|
}
|
|
|
|
if (szClass)
|
|
{
|
|
PubMemFree(szClass);
|
|
}
|
|
|
|
return hresult;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetBitmapAsDib, INTERNAL
|
|
//
|
|
// Synopsis: Reads a bitmap from the OLE1 stream, converts it to a DIB,
|
|
// and stores it in the DATA member of CGenericObject
|
|
//
|
|
// Arguments: [pos] -- The OLE 1 stream to read from
|
|
// [pdata] -- The DATA object to read into
|
|
//
|
|
// Requires:
|
|
//
|
|
// Returns: NOERROR success
|
|
// CONVERT10_E_OLESTREAM_GET I/O error
|
|
// CONVERT10_E_OLESTREAM_BITMAP_TO_DIB conversion error
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetBitmapAsDib(LPOLESTREAM pos, PDATA pdata)
|
|
{
|
|
HRESULT hresult= NOERROR;
|
|
HGLOBAL hBits = NULL;
|
|
HGLOBAL hDib = NULL;
|
|
LPVOID pBits = NULL;
|
|
WIN16BITMAP bm;
|
|
HBITMAP hBitmap = NULL;
|
|
ULONG cbBits;
|
|
ULONG ul;
|
|
|
|
|
|
Assert (pdata->m_h==NULL && pdata->m_pv==NULL && pdata->m_cbSize==0);
|
|
|
|
// Get size of all bitmap data, including the bitmap header struct
|
|
|
|
if (FAILED(hresult = OLE1StreamToUL(pos, &ul)))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// Read the bitmap header structure. Since this was stored as Win16
|
|
// BITMAP, we have to pull a structure of that size from the stream
|
|
// (A Win32 BITMAP uses LONGs and hence is larger).
|
|
|
|
if (pos->lpstbl->Get (pos, &bm, sizeof(WIN16BITMAP)) < sizeof(WIN16BITMAP))
|
|
{
|
|
return ResultFromScode (CONVERT10_E_OLESTREAM_GET);
|
|
}
|
|
|
|
// The bitmap data is total size - header size
|
|
// Allocate enough memory to hold the bitmap data
|
|
|
|
cbBits = ul - sizeof(WIN16BITMAP);
|
|
hBits = GlobalAlloc (GMEM_MOVEABLE, cbBits);
|
|
if (NULL == hBits)
|
|
{
|
|
hresult = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
pBits = (void FAR*) GlobalLock (hBits);
|
|
if (pBits == NULL)
|
|
{
|
|
hresult = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the header data into our allocated buffer
|
|
if (pos->lpstbl->Get (pos, pBits, cbBits) < cbBits)
|
|
{
|
|
hresult = ResultFromScode (CONVERT10_E_OLESTREAM_GET);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Turn that raw data into a bitmap
|
|
hBitmap = CreateBitmap (bm.bmWidth, bm.bmHeight, bm.bmPlanes,
|
|
bm.bmBitsPixel, pBits);
|
|
|
|
if (NULL == hBitmap)
|
|
{
|
|
hresult = ResultFromScode(CONVERT10_E_OLESTREAM_BITMAP_TO_DIB);
|
|
goto errRtn;
|
|
}
|
|
|
|
// NOTE: The following call gave only the first parameter in the
|
|
// (davepl) original source; The second is the palette handle, which
|
|
// I've passed as NULL to indicate the default stock palette.
|
|
|
|
hDib = UtConvertBitmapToDib (hBitmap, NULL);
|
|
if (NULL == hDib)
|
|
{
|
|
hresult = ResultFromScode(CONVERT10_E_OLESTREAM_BITMAP_TO_DIB);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Set the presentation data pointers to point to this new DIB
|
|
|
|
pdata->m_pv = GlobalLock (hDib);
|
|
if (NULL == pdata->m_pv)
|
|
{
|
|
hresult = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
pdata->m_cbSize = (ULONG) GlobalSize (hDib);
|
|
pdata->m_h = hDib;
|
|
|
|
// Free up allocations and resources, return result
|
|
|
|
errRtn:
|
|
|
|
if (pBits)
|
|
{
|
|
Verify (0==GlobalUnlock (hBits));
|
|
}
|
|
if (hBits)
|
|
{
|
|
Verify (0==GlobalFree (hBits));
|
|
}
|
|
if (hBitmap)
|
|
{
|
|
Verify (DeleteObject (hBitmap));
|
|
}
|
|
return hresult;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetMfBits, INTERNAL
|
|
//
|
|
// Synopsis: Strips the METAFILE header from the stream and then reads
|
|
// the metafile bits into an allocated memory area; the
|
|
// presentation data member of [pos] is then set to point
|
|
// to this memory.
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream to read from
|
|
// [pdata] -- the presentation data member of generic object
|
|
//
|
|
// Returns: NOERROR success
|
|
// CONVERT10_E_OLESTREAM_GET stream error
|
|
// E_OUTOFMEMORY allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetMfBits(LPOLESTREAM pos, PDATA pdata)
|
|
{
|
|
ULONG cbSize;
|
|
WIN16METAFILEPICT mfpictDummy;
|
|
HRESULT hresult = NOERROR;
|
|
|
|
Assert (0==pdata->m_cbSize && pdata->m_h==NULL && NULL==pdata->m_pv);
|
|
|
|
// Read the data size from the stream
|
|
|
|
if (FAILED(hresult = (OLE1StreamToUL (pos, &cbSize))))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// Now read the actual data
|
|
|
|
if (cbSize <= sizeof(WIN16METAFILEPICT))
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_FMT);
|
|
}
|
|
|
|
// An OLESTREAM contains a METAFILEPICT structure (with a meaningless
|
|
// handle) followed by the metafile bits. So consume the METAFILEPICT.
|
|
|
|
if (pos->lpstbl->Get (pos, &mfpictDummy, sizeof(WIN16METAFILEPICT))
|
|
< sizeof(WIN16METAFILEPICT))
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_GET);
|
|
}
|
|
|
|
// Deduct from our count of bytes to read the size of the header which
|
|
// we just consumed. Set the presentation data size to be this new size.
|
|
|
|
cbSize -= sizeof(WIN16METAFILEPICT);
|
|
pdata->m_cbSize = cbSize;
|
|
|
|
// Grad some memory to store the metafile bits
|
|
|
|
pdata->m_h = GlobalAlloc (GMEM_MOVEABLE, cbSize);
|
|
if (NULL==pdata->m_h)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
pdata->m_pv = GlobalLock (pdata->m_h);
|
|
if (NULL==pdata->m_pv)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
// Get the actual metafile bits
|
|
|
|
if (pos->lpstbl->Get (pos, pdata->m_pv, cbSize) < cbSize)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_GET);
|
|
}
|
|
|
|
return hresult;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetStandardPresentation, INTERNAL
|
|
//
|
|
// Synopsis: Allocates a PRES member for generic object, then reads
|
|
// whatever presentation may be found in the stream into
|
|
// that PRES.
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream to read from
|
|
// [pgenobj] -- the generic object we are going to set
|
|
// up with the presentation data
|
|
// [cf] -- the clipboad format we are to read
|
|
//
|
|
// Returns: NOERROR success
|
|
// E_OUTOFMEMORY allocation failure
|
|
//
|
|
// Modifies: [pgenobj] - sets up the m_ppres member
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetStandardPresentation(
|
|
LPOLESTREAM pos,
|
|
PGENOBJ pgenobj,
|
|
CLIPFORMAT cf)
|
|
{
|
|
HRESULT hresult = NOERROR;
|
|
|
|
// Allocate enough memory for the PRES object
|
|
pgenobj->m_ppres = new PRES;
|
|
if (NULL == pgenobj->m_ppres)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Set up the format tag and clipboard format
|
|
pgenobj->m_ppres->m_format.m_ftag = ftagClipFormat;
|
|
pgenobj->m_ppres->m_format.m_cf = cf;
|
|
|
|
// Get the width of the data from the stream
|
|
if (FAILED(hresult = OLE1StreamToUL(pos, &(pgenobj->m_ppres->m_ulWidth))))
|
|
{
|
|
return hresult;
|
|
}
|
|
// Get the height of the data from the stream
|
|
if (FAILED(hresult=OLE1StreamToUL(pos, &(pgenobj->m_ppres->m_ulHeight))))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// The height saved by OLE 1.0 objects into the stream is always a
|
|
// negative value (Y-increase in pixel is negative upward?) so we
|
|
// have to correct that value.
|
|
|
|
pgenobj->m_ppres->m_ulHeight
|
|
= (ULONG) -((LONG) pgenobj->m_ppres->m_ulHeight);
|
|
|
|
// Read the appropriate presentation data based on the clipboard
|
|
// format ID
|
|
|
|
switch(cf)
|
|
{
|
|
case CF_METAFILEPICT:
|
|
{
|
|
hresult = GetMfBits (pos, &(pgenobj->m_ppres->m_data));
|
|
break;
|
|
}
|
|
|
|
case CF_BITMAP:
|
|
{
|
|
// When reading a bitmap, we will convert from Bitmap to
|
|
// DIB in the process, so update the PRES clipboard format ID
|
|
|
|
pgenobj->m_ppres->m_format.m_cf = CF_DIB;
|
|
hresult = GetBitmapAsDib (pos, &(pgenobj->m_ppres->m_data));
|
|
break;
|
|
}
|
|
|
|
case CF_DIB:
|
|
{
|
|
Assert (CF_DIB==cf);
|
|
hresult = GetSizedDataOLE1Stm (pos, &(pgenobj->m_ppres->m_data));
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
Assert(0 && "Unexpected clipboard format reading PRES");
|
|
}
|
|
}
|
|
|
|
return hresult;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetGenericPresentation, INTERNAL
|
|
//
|
|
// Synopsis: Allocated the PRES member of the generic object and reads
|
|
// the generic presentation data into it.
|
|
//
|
|
// Effects: If the format is a known clipboard format, we set the
|
|
// format tag to indicate this, and set the format type
|
|
// to indicate the clipboard format type. If it is unknown,
|
|
// we set the format tag to string and read the description
|
|
// of the format.
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream we are reading from
|
|
// [pgenobj] -- the generic object we are reading to
|
|
//
|
|
// Returns: NOERROR on success
|
|
// E_OUTOFMEMORY on allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Code cleanup and document
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetGenericPresentation(
|
|
LPOLESTREAM pos,
|
|
PGENOBJ pgenobj)
|
|
{
|
|
ULONG ulClipFormat;
|
|
HRESULT hresult = NOERROR;
|
|
|
|
// The PRES member should not exist at this point
|
|
Assert (NULL==pgenobj->m_ppres);
|
|
|
|
// Allocate the PRES member of the generic object
|
|
|
|
pgenobj->m_ppres = new PRES;
|
|
if (NULL == pgenobj->m_ppres)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Read the clipboard format ID
|
|
|
|
if (FAILED(hresult = OLE1StreamToUL (pos, &ulClipFormat)))
|
|
{
|
|
delete (pgenobj->m_ppres);
|
|
return hresult;
|
|
}
|
|
|
|
// If the clipboard format is not 0, we have a known clipboard
|
|
// format and we should set the tag type and ID accordingly
|
|
|
|
if (ulClipFormat)
|
|
{
|
|
pgenobj->m_ppres->m_format.m_ftag = ftagClipFormat;
|
|
pgenobj->m_ppres->m_format.m_cf = (CLIPFORMAT) ulClipFormat;
|
|
}
|
|
else
|
|
{
|
|
// Otherwise, we have a custom format so we need to set the
|
|
// tag type to string and read in the data format string
|
|
|
|
pgenobj->m_ppres->m_format.m_ftag = ftagString;
|
|
if (FAILED(hresult = (GetSizedDataOLE1Stm
|
|
(pos, &(pgenobj->m_ppres->m_format.m_dataFormatString)))))
|
|
{
|
|
delete (pgenobj->m_ppres);
|
|
return hresult;
|
|
}
|
|
}
|
|
|
|
// We don't know the size, so reset to 0
|
|
|
|
pgenobj->m_ppres->m_ulHeight = 0;
|
|
pgenobj->m_ppres->m_ulWidth = 0;
|
|
|
|
// Read the raw generic presentation data into the PRES member
|
|
|
|
if (FAILED(hresult=GetSizedDataOLE1Stm(pos,&(pgenobj->m_ppres->m_data))))
|
|
{
|
|
delete (pgenobj->m_ppres);
|
|
return hresult;
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GetSizedDataOLE1Stm, INTERNAL
|
|
//
|
|
// Synopsis: Reads bytes from an OLE 1 stream into a CData object.
|
|
// Obtains the number of bytes to read from the first
|
|
// ULONG in the stream
|
|
//
|
|
// Arguments: [pos] -- the stream to read from
|
|
// [pdata] -- the CData object to read to
|
|
//
|
|
// Requires:
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_GET on stream read problem
|
|
// E_OUTOFMEMORY on allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GetSizedDataOLE1Stm(LPOLESTREAM pos, PDATA pdata)
|
|
{
|
|
ULONG cbSize;
|
|
HRESULT hr;
|
|
Assert (0==pdata->m_cbSize && pdata->m_h==NULL && NULL==pdata->m_pv);
|
|
|
|
// Read size of data
|
|
if (FAILED(hr = OLE1StreamToUL(pos, &cbSize)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (cbSize==0)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
// Allocate memory for data
|
|
pdata->m_cbSize = cbSize;
|
|
|
|
pdata->m_h = GlobalAlloc (GMEM_MOVEABLE, cbSize);
|
|
if (NULL==pdata->m_h)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
pdata->m_pv = GlobalLock (pdata->m_h);
|
|
|
|
if (NULL==pdata->m_pv)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Read data into allocated buffer
|
|
|
|
if (pos->lpstbl->Get (pos, pdata->m_pv, cbSize) < cbSize)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_GET);
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLE1StreamToUL, INTERNAL
|
|
//
|
|
// Synopsis: Reads a ULONG from an OLE1 stream
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream to read from
|
|
// [pul] -- the ULONG to read into
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_GET on stream read failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes: on failure [pul] is preserved
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL OLE1StreamToUL(LPOLESTREAM pos, ULONG FAR* pul)
|
|
{
|
|
ULONG ul;
|
|
|
|
// Read the data from the stream into the local ULONG
|
|
|
|
if (pos->lpstbl->Get (pos, &ul, sizeof(ULONG)) < sizeof(ULONG))
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_GET);
|
|
}
|
|
|
|
// If all went well, store the data into [pul]
|
|
|
|
if (pul != NULL)
|
|
{
|
|
Assert (IsValidPtrOut (pul, sizeof(ULONG)));
|
|
*pul = ul;
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: DataToOLE1Stm, INTERNAL INLINE
|
|
//
|
|
// Synopsis: Writes raw data out to an OLE 1 stream
|
|
//
|
|
// Arguments: [pos] -- the stream to write to
|
|
// [pvBuf] -- the buffer to write from
|
|
// [ulSize] -- the number of bytes to write
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_PUT on stream write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline static INTERNAL DataToOLE1Stm(LPOLESTREAM pos, LPVOID pvBuf, ULONG ulSize)
|
|
{
|
|
// Write the data out to the stream
|
|
|
|
if (pos->lpstbl->Put(pos, pvBuf, ulSize) < ulSize)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_PUT);
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: ULToOLE1Stream, INTERNAL INLINE
|
|
//
|
|
// Synopsis: Write a ULONG to the specified OLESTREAM via the Put()
|
|
// member of the stream's VTBL
|
|
//
|
|
// Effects: Advances stream position by sizeof(ULONG) on success.
|
|
//
|
|
// Arguments: [pos] -- The stream into which the ULONG is written
|
|
// [ul] -- The ULONG, passed by value
|
|
//
|
|
// Requires:
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_PUT on failure
|
|
//
|
|
// Signals: (none)
|
|
//
|
|
// Modifies: Stream position
|
|
//
|
|
// Algorithm:
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 11-Jan-93 davepl Cleaned up and documented
|
|
//
|
|
// Notes: On failure 0-3 bytes may have been written
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline static INTERNAL ULToOLE1Stream(LPOLESTREAM pos, ULONG ul)
|
|
{
|
|
if (pos->lpstbl->Put (pos, &ul, sizeof(ULONG)) < sizeof(ULONG))
|
|
{
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_PUT);
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: StringToOLE1Stm, INTERNAL
|
|
//
|
|
// Synopsis: Converts the input OLESTR to ANSI and writes it to an
|
|
// OLE 1 stream, preceded by a ULONG indicating the number
|
|
// of bytes in the ANSI representation (terminator included).
|
|
//
|
|
// Arguments: [pos] -- The stream into which the ULONG is written
|
|
// [szOleStr] -- The STR to be written
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_PUT on stream write failure
|
|
// E_NOMEMORY on allocation failure
|
|
//
|
|
// Modifies: Stream position
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 11-Feb-94 davepl Cleaned up and documented
|
|
// 15-Feb-94 davepl Re-write for ANSI/WCHAR handling
|
|
// 17-Feb-94 davepl Restructured error handling
|
|
//
|
|
// Notes: On failure, 0 to (cbSize-1) bytes may have been written
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL StringToOLE1Stm(LPOLESTREAM pos, LPCOLESTR pszOleStr)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPSTR pszAnsi = NULL; // Ansi version of OLE input string
|
|
|
|
if (pszOleStr)
|
|
{
|
|
// This handy function will calculate the size of the buffer we
|
|
// need to represent the OLESTR in ANSI format for us.
|
|
|
|
ULONG cbSize = WideCharToMultiByte(CP_ACP, // Code Page ANSI
|
|
0, // No flags
|
|
pszOleStr, // Input OLESTR
|
|
-1, // Input len (auto detect)
|
|
NULL, // Output buffer
|
|
0, // Output len (check only)
|
|
NULL, // Default char
|
|
NULL);// Flag: Default char used
|
|
|
|
if (cbSize == FALSE)
|
|
{
|
|
return ResultFromScode(E_UNSPEC);
|
|
}
|
|
|
|
// Now that we know the actual needed length, allocate a buffer
|
|
|
|
pszAnsi = (LPSTR) PrivMemAlloc(cbSize);
|
|
if (NULL == pszAnsi)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// We've got out buffer and our length, so do the conversion now
|
|
// We don't need to check for cbSize == FALSE since that was
|
|
// already done during the length test, but we need to check
|
|
// for substitution. Iff this call sets the fDefChar even when
|
|
// only doing a length check, these two tests could be merged,
|
|
// but I don't believe this is the case.
|
|
|
|
BOOL fDefUsed = 0;
|
|
cbSize = WideCharToMultiByte(CP_ACP, // Code Page ANSI
|
|
0, // No flags
|
|
pszOleStr, // Input OLESTR
|
|
-1, // Input len (auto detect)
|
|
pszAnsi, // Output buffer
|
|
cbSize, // Output len
|
|
NULL, // Default char (use system's)
|
|
&fDefUsed); // Flag: Default char used
|
|
|
|
// If number of bytes converted was 0, we failed
|
|
|
|
if (fDefUsed)
|
|
{
|
|
hr = ResultFromScode(E_UNSPEC);
|
|
}
|
|
|
|
// Write the size of the string (including null terminator) to stream
|
|
|
|
else if (FAILED(hr = ULToOLE1Stream(pos, cbSize)))
|
|
{
|
|
NULL;
|
|
}
|
|
|
|
// Write the Ansi version of the string into the stream
|
|
|
|
else if (pos->lpstbl->Put(pos, pszAnsi, cbSize) < cbSize)
|
|
{
|
|
hr = ResultFromScode(CONVERT10_E_OLESTREAM_PUT);
|
|
}
|
|
|
|
if (pszAnsi)
|
|
{
|
|
PrivMemFree(pszAnsi);
|
|
}
|
|
}
|
|
|
|
// If the pointer is not valid, we write a length of zero into
|
|
// the stream
|
|
|
|
else
|
|
{
|
|
hr = ULToOLE1Stream(pos, 0);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLE2StmToUL, INTERNAL
|
|
//
|
|
// Synopsis: Reads a ULONG from the specified ISTREAM and stores it at
|
|
// the ULONG deferenced by pul
|
|
//
|
|
// Effects: Writes the value read into memory at pul
|
|
//
|
|
// Arguments: [pstm] -- The stream from which the ULONG is read
|
|
// [pul] -- ULONG to hold the value read
|
|
//
|
|
// Requires:
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_PUT on failure
|
|
//
|
|
// Signals: (none)
|
|
//
|
|
// Modifies: Stream position
|
|
//
|
|
// Algorithm:
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 11-Feb-93 davepl Cleaned up and documented
|
|
//
|
|
// Notes: On failure, *pul is not disturbed regardless of how
|
|
// many bytes were actually read from the stream
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL OLE2StmToUL(LPSTREAM pstm, ULONG FAR* pul)
|
|
{
|
|
ULONG ul;
|
|
ULONG cbRead;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Attempt to read 4 bytes from the stream to form a ULONG.
|
|
|
|
if (FAILED(hr = pstm->Read (&ul, sizeof(ULONG), &cbRead)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (cbRead != sizeof(ULONG))
|
|
{
|
|
hr = STG_E_READFAULT;
|
|
}
|
|
// Ensure that the [pul] pointer is valid and that we have write
|
|
// access to all 4 bytes (assertion only). If OK, transfer the
|
|
// ULONG to [*pul]
|
|
else if (pul != NULL)
|
|
{
|
|
Assert (FALSE == !IsValidPtrOut(pul, sizeof(ULONG)));
|
|
*pul = ul;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLE1StmToString, INTERNAL
|
|
//
|
|
// Synopsis: Reads a cstr from the specified STREAM and stores it in
|
|
// a dynamically allocated buffer as an OLESTR; sets the
|
|
// user's pointer to point to this new buffer.
|
|
//
|
|
// Effects: Allocates memory on the input pointer, advances stream pos'n
|
|
//
|
|
// Arguments: [pos ] -- The stream from which the STR is read
|
|
// [ppsz] -- OLESTR ** which allows this fn to modify the
|
|
// caller's pointer to point to memory allocated
|
|
// by this fn to hold the OLESTR
|
|
//
|
|
// Requires: Stream must be set up. Caller's responsibilty to free memory.
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_OLESTREAM_GET on failure
|
|
// E_OUTOFMEMORY if buffers couldn't be allocated
|
|
//
|
|
// Signals: (none)
|
|
//
|
|
// Modifies: Stream position, caller's string pointer
|
|
//
|
|
// Algorithm: if ppsz == NULL, string is read from stream and discarded
|
|
// if ppsz != NULL, string is read and converted into a
|
|
// dynamically allocated buffer. *ppsz is set
|
|
// to point to this buffer, which must be later
|
|
// freed by the caller
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 12-Jan-93 davepl Cleaned up and documented
|
|
// 14-Jan-93 davepl Changed to return LPOLESTR
|
|
//
|
|
// Notes: [ppsz] may be NULL on entry; string is read and discarded
|
|
// with no cleanup required by the caller
|
|
//
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL OLE1StmToString(LPOLESTREAM pos, LPOLESTR FAR* ppsz)
|
|
{
|
|
ULONG cbSize; // Size in bytes of cstr
|
|
LPOLESTR pszOleStr = NULL;
|
|
LPSTR pszAnsiStr = NULL;
|
|
HRESULT error = NOERROR;
|
|
|
|
// if ppsz is valid, NULL out *ppsz as default out parameter
|
|
|
|
if (NULL != ppsz)
|
|
{
|
|
*ppsz = NULL;
|
|
}
|
|
|
|
// Retrieve the incoming string size from the stream
|
|
|
|
if (FAILED(error = OLE1StreamToUL (pos, &cbSize)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// If there are chars to be read, allocate memory for the
|
|
// ANSI and OLESTR versions. Read the string into the
|
|
// ANSI version and convert it to OLESTR
|
|
|
|
if (0 < cbSize)
|
|
{
|
|
// Allocate the ANSI buffer
|
|
pszAnsiStr = (LPSTR) PrivMemAlloc((size_t)cbSize);
|
|
if (NULL == pszAnsiStr)
|
|
{
|
|
error = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the string into the ANSI buffer
|
|
if (pos->lpstbl->Get (pos, pszAnsiStr, cbSize) < cbSize)
|
|
{
|
|
error = ResultFromScode(CONVERT10_E_OLESTREAM_GET);
|
|
goto errRtn;
|
|
}
|
|
|
|
// We only need to perform the ANSI->OLESTR conversion in those
|
|
// cases where the caller needs an out parameter
|
|
|
|
if (NULL != ppsz)
|
|
{
|
|
// Allocate the OLESTR buffer
|
|
pszOleStr = (LPOLESTR) PubMemAlloc((size_t)cbSize * 2);
|
|
if (NULL == pszOleStr)
|
|
{
|
|
error = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Convert from ANSI buffer to OLESTR buffer
|
|
if (FALSE==MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, pszAnsiStr,
|
|
cbSize, pszOleStr, cbSize *2))
|
|
{
|
|
error = HRESULT_FROM_WIN32(ERROR_NO_UNICODE_TRANSLATION);
|
|
PubMemFree(pszOleStr);
|
|
goto errRtn;
|
|
}
|
|
*ppsz = pszOleStr;
|
|
}
|
|
}
|
|
|
|
errRtn:
|
|
|
|
if (pszAnsiStr)
|
|
{
|
|
PrivMemFree(pszAnsiStr);
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GenericObjectToIStorage
|
|
//
|
|
// Synopsis: Write the generic object in memory out to an OLE 2 IStorage
|
|
// This invovles writing the class, native data, and
|
|
// presentation data out where applicable.
|
|
//
|
|
// Arguments: [genobj] -- the generic object holding the info
|
|
// [pstg] -- the IStorage object to write to
|
|
// [ptd] -- target device
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_S_NO_PRESENTATION in cases where the object did
|
|
// not have needed presentation data
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 17-Feb-94 davepl Cleanup and document
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
FARINTERNAL GenericObjectToIStorage(
|
|
const GENOBJ FAR& genobj,
|
|
LPSTORAGE pstg,
|
|
const DVTARGETDEVICE FAR* ptd)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Assert (genobj.m_class.m_clsid != CLSID_NULL);
|
|
|
|
// Write the class ID out to the storage
|
|
if (FAILED(hr = WriteClassStg (pstg, genobj.m_class.m_clsid)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to WriteClassStg at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
|
|
if (!genobj.m_fLink)
|
|
{
|
|
if (genobj.m_fStatic)
|
|
{
|
|
// If we are a static embedded object, get the format name from
|
|
// the registration database and write it out to the IStorage
|
|
|
|
LPOLESTR pszUserType = NULL;
|
|
|
|
OleRegGetUserType(genobj.m_class.m_clsid, USERCLASSTYPE_FULL,
|
|
&pszUserType);
|
|
|
|
WriteFmtUserTypeStg (pstg, genobj.m_ppres->m_format.m_cf,
|
|
pszUserType);
|
|
|
|
if (pszUserType)
|
|
{
|
|
PubMemFree(pszUserType);
|
|
}
|
|
}
|
|
else if (wWriteFmtUserType (pstg, genobj.m_class.m_clsid) != NOERROR)
|
|
{
|
|
// This happens when the class is not registered.
|
|
// Use ProgId as UserType.
|
|
|
|
WriteFmtUserTypeStg (pstg,
|
|
(CLIPFORMAT) RegisterClipboardFormat (genobj.m_class.m_szClsid),
|
|
genobj.m_class.m_szClsid);
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr = GenObjToOLE2Stm (pstg, genobj)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write gen obj to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
|
|
// If it's not a link and not a static object, dump its native
|
|
// data out to the storage
|
|
|
|
if (!genobj.m_fLink && !genobj.m_fStatic)
|
|
{
|
|
if (FAILED(hr=Write20NativeStreams (pstg, genobj)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write native stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
if (! genobj.m_fLink)
|
|
{
|
|
if (genobj.m_class.m_clsid == CLSID_PBrush)
|
|
{
|
|
if (! genobj.m_ppres || (genobj.m_ppres->m_format.m_cf == CF_DIB))
|
|
{
|
|
// If the object is not a link, and it is a PBrush object with
|
|
// either a DIB presentation or no presentation at all, we
|
|
// don't need to do anything.
|
|
|
|
return NOERROR;
|
|
}
|
|
}
|
|
|
|
if (genobj.m_class.m_clsid == CLSID_MSDraw)
|
|
{
|
|
if (! genobj.m_ppres ||
|
|
(genobj.m_ppres->m_format.m_cf == CF_METAFILEPICT))
|
|
{
|
|
// Similarly, if it is not a link, and it is an MSDraw object
|
|
// with no presentation or a METAFILEPICT presentation, we
|
|
// don't need to do anything.
|
|
|
|
return NOERROR;
|
|
}
|
|
}
|
|
}
|
|
|
|
// In all other cases, we have to dump the presenation data out to
|
|
// the storage.
|
|
|
|
if (FAILED(hr = PresToIStorage (pstg, genobj, ptd)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write pres to IStorage at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
|
|
// If we are a static object, copy the contents of the presentation
|
|
// stream over to the contents stream.
|
|
|
|
if (genobj.m_fStatic)
|
|
{
|
|
UINT uiStatus;
|
|
return UtOlePresStmToContentsStm(pstg, OLE_PRESENTATION_STREAM,
|
|
TRUE, &uiStatus);
|
|
}
|
|
|
|
// If we don't have a presentation (but weren't one of the special
|
|
// cases handled above), we have a problem
|
|
|
|
//
|
|
// We don't care if genobj.m_pres is NULL if a blank presentation is
|
|
// permited as the routine PresToIStorage will generate a blank pres.
|
|
//
|
|
if ((NULL == genobj.m_ppres) && genobj.m_fNoBlankPres)
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"We have no presentation at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return ResultFromScode(CONVERT10_S_NO_PRESENTATION);
|
|
}
|
|
|
|
return NOERROR;
|
|
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GenObjToOLE2Stm, INTERNAL
|
|
//
|
|
// Synopsis: Write the generic object out to the OLE 2 stream
|
|
//
|
|
// Effects: Write the whole object, including presentation data, etc.
|
|
//
|
|
// Arguments: [pstg] -- the IStorage to write to
|
|
// [genobj] -- the generic object to write
|
|
//
|
|
// Returns: NOERROR on success
|
|
// This is an upper level function, so there are numerous
|
|
// error that could be propagated up through it
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 14-Feb-94 davepl Code cleanup and document
|
|
//
|
|
// Notes: The code is enclosed in a do{}while(FALSE) block so that
|
|
// we can break out of it on any error and fall through to
|
|
// the cleanup and error return code.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GenObjToOLE2Stm(LPSTORAGE pstg, const GENOBJ FAR& genobj)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPSTREAM pstm=NULL;
|
|
|
|
do { // The do{}while(FALSE) allows us to break out on error
|
|
|
|
// Create a stream in the current IStorage
|
|
if (FAILED(hr = OpenOrCreateStream (pstg, OLE_STREAM, &pstm)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Can't create streamat line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
break;
|
|
}
|
|
|
|
// Write the Ole version out to that new stream
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, gdwOleVersion)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Write the object flags (for links only, otherwise 0) to the stream
|
|
if (FAILED(hr = ULToOLE2Stm
|
|
(pstm, genobj.m_fLink ? OBJFLAGS_LINK : 0L)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Write the update options out to the stream
|
|
if (genobj.m_fLink || genobj.m_class.m_clsid == CLSID_StdOleLink)
|
|
{
|
|
// If our object's link update options are UPDATE_ONCALL, we
|
|
// write out the corresponding OLE 2 flags, otherwise, we
|
|
// write out OLEUPDATE_ALWAYS
|
|
|
|
if (genobj.m_lnkupdopt==UPDATE_ONCALL)
|
|
{
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, OLEUPDATE_ONCALL)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, OLEUPDATE_ALWAYS)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
// We are neither a link nor a StdOleLink, so we have no
|
|
// update options.. just write out a 0.
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, 0L)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// This is a reserved filed (was View Format), just write a 0
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, 0L)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// We have no relative moniker, write out NULL
|
|
if (FAILED(hr = WriteMonikerStm (pstm, (LPMONIKER)NULL)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write moniker to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
break;
|
|
}
|
|
|
|
// If we are a link, we have to write out all of that information...
|
|
|
|
if (genobj.m_fLink || genobj.m_class.m_clsid == CLSID_StdOleLink)
|
|
{
|
|
// relative source moniker
|
|
if (FAILED(hr = WriteMonikerStm (pstm, (LPMONIKER)NULL)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write moniker to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
break;
|
|
}
|
|
|
|
// absolute source moniker
|
|
if (FAILED(hr = MonikerToOLE2Stm (pstm, genobj.m_szTopic,
|
|
genobj.m_szItem, genobj.m_classLast.m_clsid)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write moniker to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
break;
|
|
}
|
|
|
|
// write the classLast field to the stream
|
|
|
|
CLSID clsid;
|
|
|
|
// If we have the classLast already, use that clsid
|
|
if (genobj.m_classLast.m_szClsid)
|
|
{
|
|
clsid = genobj.m_classLast.m_clsid;
|
|
}
|
|
else
|
|
{
|
|
// Otherwise, if it's a StdOleLink, class id is NULL
|
|
if (genobj.m_class.m_clsid == CLSID_StdOleLink)
|
|
{
|
|
clsid = CLSID_NULL;
|
|
}
|
|
else
|
|
{
|
|
// If we don't have last class and not a link, use the
|
|
// class id of the generic object
|
|
clsid = genobj.m_class.m_clsid;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr = WriteM1ClassStm(pstm, clsid)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write M1 to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
break;
|
|
}
|
|
|
|
// last display == NULL string
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, 0L)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Last Change time
|
|
if (FAILED(hr = FTToOle2Stm (pstm)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Last known up to date
|
|
if (FAILED(hr = FTToOle2Stm (pstm)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// rtUpdate
|
|
if (FAILED(hr = FTToOle2Stm (pstm)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
// end marker
|
|
if (FAILED(hr = ULToOLE2Stm(pstm, (ULONG) -1L)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
} while (FALSE); // This do{}while(FALSE) is a once-through "loop"
|
|
// that we can break out of on error and fall
|
|
// through to the return.
|
|
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: MonikerToOLE2Stm, INTERNAL
|
|
//
|
|
// Synopsis: Write the file and item moniker as a composite to the stream
|
|
//
|
|
// Effects: Builds a composite of the file and item monikers, and then
|
|
// writes them out. If there is no file, a NULL moniker is
|
|
// written in its place
|
|
//
|
|
// Arguments: [pstm] -- The OLE2 storage we are writing to
|
|
// [pszFile] -- The file associated with the object
|
|
// [spzItem] -- The item
|
|
// [clsid] -- The class ID of the object
|
|
//
|
|
// Returns: NOERROR on success
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl Reworked, cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#pragma SEG(MonikerToOLE2Stm)
|
|
static INTERNAL MonikerToOLE2Stm(
|
|
LPSTREAM pstm,
|
|
LPOLESTR szFile,
|
|
LPOLESTR szItem,
|
|
CLSID clsid) // CLSID of the link source file, szFile
|
|
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPMONIKER pmkFile = NULL; // File moniker
|
|
LPMONIKER pmkItem = NULL; // Item moniker
|
|
LPMONIKER pmkComp = NULL; // Composite of file + item monikers
|
|
|
|
|
|
// If we don't have a file, write a NULL moniker
|
|
if (NULL == szFile)
|
|
{
|
|
if (FAILED(hr = WriteMonikerStm (pstm, NULL)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Otherwise, create a file moniker (OLE1 or OLE2 as appplicable)
|
|
|
|
if (CoIsOle1Class (clsid))
|
|
{
|
|
if (FAILED(hr = CreateOle1FileMoniker (szFile, clsid, &pmkFile)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Can't create OLE 1 moniker at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = CreateFileMoniker (szFile, &pmkFile)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Can't create file moniker at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
// If we don't have an Item, write just the file moniker
|
|
|
|
if (NULL==szItem)
|
|
{
|
|
if (FAILED(hr = WriteMonikerStm (pstm, pmkFile)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write moniker to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
|
|
}
|
|
|
|
// Otherwise, create a composite of the file + item monikers
|
|
// and write it out
|
|
|
|
else
|
|
{
|
|
if (FAILED(hr=CreateItemMoniker(OLESTR("!"), szItem, &pmkItem)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to create item moniker at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
|
|
if (FAILED(hr=CreateGenericComposite(pmkFile, pmkItem, &pmkComp)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to create generic pres at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
|
|
if (FAILED(hr = WriteMonikerStm (pstm, pmkComp)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write moniker to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
goto errRtn;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
errRtn:
|
|
if (pmkFile)
|
|
{
|
|
pmkFile->Release();
|
|
}
|
|
if (pmkItem)
|
|
{
|
|
pmkItem->Release();
|
|
}
|
|
if (pmkComp)
|
|
{
|
|
pmkComp->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: IsStandardFormat, INTERNAL
|
|
//
|
|
// Synopsis: Returns TRUE if object is in clipboard format and is one
|
|
// one of the three standard formats (METAFILE, DIB, BITMAP)
|
|
//
|
|
// Arguments: [format] -- the format object which contains the
|
|
// format tag and clipboard format type
|
|
//
|
|
// Returns: TRUE if METAFILE, DIB, or BITMAP
|
|
// FALSE if other format or not clipboard format at all
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl documented and chaged from big
|
|
// conditional to a switch()
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL_(BOOL) IsStandardFormat(const FORMAT FAR& format)
|
|
{
|
|
// First we must ensure that the format tag indicates that this
|
|
// object is in clipboard format at all...
|
|
|
|
if (format.m_ftag == ftagClipFormat)
|
|
{
|
|
// If so, there is a limited set of clipboard formats which
|
|
// we consider "standard". If it is not among these,
|
|
// we return FALSE.
|
|
|
|
switch(format.m_cf)
|
|
{
|
|
case CF_METAFILEPICT:
|
|
case CF_BITMAP:
|
|
case CF_DIB:
|
|
|
|
return TRUE;
|
|
|
|
|
|
default:
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PresToIStorage, INTERNAL
|
|
//
|
|
// Synopsis: Given an generic object and an IStorage, write genobj's
|
|
// presentation data out to the storage
|
|
//
|
|
// Effects: Will call PresToNewOLE2Stm to create a stream in this
|
|
// storage to hold the presentation data
|
|
//
|
|
// Arguments: [pstg] -- the storage to save to
|
|
// [genobj] -- the generic object holding the presenation
|
|
// [ptd] -- the target device for the presentation
|
|
//
|
|
// Returns: NOERROR on success
|
|
// Various other errors may propagate back up from I/O funcs
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl ARRGR! Cleanup and document
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PresToIStorage(
|
|
LPSTORAGE pstg,
|
|
const GENOBJ FAR& genobj,
|
|
const DVTARGETDEVICE FAR* ptd)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
|
|
if (genobj.m_fNoBlankPres)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
PRES pres;
|
|
|
|
if (NULL==genobj.m_ppres)
|
|
{
|
|
// If we're not a link, and we don't have a presentation, we will
|
|
// create a blank presentation and write it out. If we are a link,
|
|
// we will do nothing, and just fall through to the return.
|
|
|
|
if (!genobj.m_fLink)
|
|
{
|
|
if (FAILED(hr = CreateBlankPres (&pres)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to create blank pres at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED(hr = PresToNewOLE2Stm
|
|
(pstg, genobj.m_fLink, pres, ptd, OLE_PRESENTATION_STREAM)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write pres to new stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
return hr;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If the object did indeed have a presentation, we write it
|
|
// out to a new stream
|
|
|
|
if (IsStandardFormat (genobj.m_ppres->m_format))
|
|
{
|
|
// If the presentation is a standard clipboard
|
|
// format, we can write it out with no other work
|
|
|
|
if (FAILED(hr = PresToNewOLE2Stm ( pstg,
|
|
genobj.m_fLink,
|
|
*(genobj.m_ppres),
|
|
ptd,
|
|
OLE_PRESENTATION_STREAM)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write pres to new stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
// If the presentation is not a standard format,
|
|
// it may be a PBrush object (handled below), or if
|
|
// not, we write it as a generic presentation stream
|
|
|
|
if (genobj.m_classLast.m_clsid != CLSID_PBrush)
|
|
{
|
|
if(FAILED(hr = PresToNewOLE2Stm ( pstg,
|
|
genobj.m_fLink,
|
|
*(genobj.m_ppres),
|
|
ptd,
|
|
OLE_PRESENTATION_STREAM)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write pres to new stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
else // PBrush
|
|
{
|
|
BOOL fPBrushNative = FALSE;
|
|
|
|
// We know this is a PBrush object. If the
|
|
// format tag is a format string, check to see
|
|
// if that string is "Native", in which case
|
|
// we set the flag to indicate that this is
|
|
// native pbrush data.
|
|
|
|
if (genobj.m_ppres->m_format.m_ftag == ftagString)
|
|
{
|
|
if (!strcmp( (LPCSTR) genobj.m_ppres->
|
|
m_format.m_dataFormatString.m_pv,
|
|
"Native"
|
|
)
|
|
)
|
|
{
|
|
fPBrushNative = TRUE;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr = PresToNewOLE2Stm( pstg,
|
|
genobj.m_fLink,
|
|
*(genobj.m_ppres),
|
|
ptd,
|
|
OLE_PRESENTATION_STREAM,
|
|
fPBrushNative)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write pres to new stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PresToNewOLE2Stm, INTERNAL
|
|
//
|
|
// Synopsis: Creates a new stream within a storage and writes the
|
|
// generic object's presentation data out to it.
|
|
//
|
|
// Arguments: [pstg] -- the storage in which to create the stream
|
|
// [fLink] -- flag: is this object a link?
|
|
// [pres] -- the presentation data to be saved
|
|
// [ptd] -- the target render device
|
|
// [szStream] -- the name of the new stream
|
|
// [fPBrushNative] -- flag: is this native PBrush pres data?
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_WRITEFAULT on stream write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PresToNewOLE2Stm(
|
|
LPSTORAGE pstg,
|
|
BOOL fLink,
|
|
const PRES FAR& pres,
|
|
const DVTARGETDEVICE FAR* ptd,
|
|
LPOLESTR szStream,
|
|
BOOL fPBrushNative
|
|
)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPSTREAM pstm=NULL;
|
|
FORMATETC foretc;
|
|
|
|
|
|
|
|
// Create the new stream to hold the presentation data
|
|
if (FAILED(hr = OpenOrCreateStream (pstg, szStream, &pstm)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Fill in the FormatEtc structure
|
|
if (fPBrushNative)
|
|
{
|
|
foretc.cfFormat = CF_DIB;
|
|
}
|
|
else
|
|
{
|
|
switch( pres.m_format.m_ftag)
|
|
{
|
|
case ftagClipFormat:
|
|
foretc.cfFormat = pres.m_format.m_cf;
|
|
break;
|
|
case ftagString:
|
|
// m_dataFormatString is an ASCII string.
|
|
foretc.cfFormat = (CLIPFORMAT) SSRegisterClipboardFormatA( (LPCSTR) pres.m_format.m_dataFormatString.m_pv);
|
|
Assert(0 != foretc.cfFormat);
|
|
break;
|
|
default:
|
|
AssertSz(0,"Error in Format");
|
|
hr = E_UNEXPECTED;
|
|
goto errRtn;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
foretc.ptd = (DVTARGETDEVICE *) ptd;
|
|
foretc.dwAspect = DVASPECT_CONTENT;
|
|
foretc.lindex = -1;
|
|
foretc.tymed = TYMED_NULL; // tymed field is ignored by utWriteOlePresStmHeader.
|
|
|
|
if (FAILED(hr = UtWriteOlePresStmHeader(pstm,&foretc,(fLink) ? (ADVF_PRIMEFIRST) : (0L))))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
if (fPBrushNative)
|
|
{
|
|
if (FAILED(hr = UtHDIBFileToOlePresStm(pres.m_data.m_h, pstm)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Unable to write DIB to stream at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
goto errRtn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Compression
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, 0L)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Width / Height
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, pres.m_ulWidth)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, pres.m_ulHeight)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Presentation data
|
|
if (FAILED(hr = DataObjToOLE2Stm (pstm, pres.m_data)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
errRtn:
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: ULToOLE2Stm, INTERNAL
|
|
//
|
|
// Synopsis: Writes a ULONG out to an OLE2 stream
|
|
//
|
|
// Arguments: [pstm] -- the stream to write to
|
|
// [ul] -- the ULONG to write to that stream
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_WRITEFAULT on write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline static INTERNAL ULToOLE2Stm(LPSTREAM pstm, ULONG ul)
|
|
{
|
|
// Write the ULONG out
|
|
return pstm->Write (&ul, sizeof(ULONG), NULL);
|
|
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: FTToOLE2Stm, INTERNAL
|
|
//
|
|
// Synopsis: Writes a dummy filetime out to an OLE2 stream
|
|
//
|
|
// Arguments: [pstm] -- the stream to write to
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_WRITEFAULT on write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 31-Mar-95 scottsk Created
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline static INTERNAL FTToOle2Stm(LPSTREAM pstm)
|
|
{
|
|
FILETIME ft = { 0, 0 };
|
|
|
|
return pstm->Write (&ft, sizeof(FILETIME), NULL);
|
|
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: DataObjToOLE2Stm
|
|
//
|
|
// Synopsis: Writes a fixed-size data buffer to an OLE2 stream preceded
|
|
// by a ULONG indicating the number of bytes to follow.
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_WRITEFAULT on write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl Code cleanup
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL DataObjToOLE2Stm(LPSTREAM pstm, const DATA FAR& data)
|
|
{
|
|
HRESULT hr;
|
|
|
|
|
|
// Write a ULONG indicating the number of bytes to follow
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, data.m_cbSize)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// If there are any bytes to follow...
|
|
if (data.m_cbSize)
|
|
{
|
|
if (FAILED(hr = pstm->Write (data.m_pv, data.m_cbSize, NULL)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: SizedDataToOLE1Stm
|
|
//
|
|
// Synopsis: Writes a fixed-size data buffer to an OLE1 stream preceded
|
|
// by a ULONG indicating the number of bytes to follow.
|
|
//
|
|
// Parameters: [pos] -- The stream to write to
|
|
// [data] -- The data object to write out
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_WRITEFAULT on write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl Code cleanup
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL SizedDataToOLE1Stm(LPOLESTREAM pos, const DATA FAR& data)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Ensure the memory we are going to write out is valid
|
|
Assert (data.m_pv);
|
|
|
|
// Write the ULONG representing the byte count of the sized data
|
|
|
|
if (FAILED(hr = ULToOLE1Stream (pos, data.m_cbSize)))
|
|
{
|
|
Assert (0 && "Can't write UL to ole1 stream");
|
|
return hr;
|
|
}
|
|
|
|
if (pos->lpstbl->Put (pos, data.m_pv, data.m_cbSize) < data.m_cbSize)
|
|
{
|
|
Assert (0 && "Cant write sized data to ole1 stream");
|
|
return ResultFromScode(CONVERT10_E_OLESTREAM_PUT);
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: Write20NativeStreams, INTERNAL
|
|
//
|
|
// Synopsis: Writes the generic object's native data out to an OLE 2 stream
|
|
//
|
|
// Effects: Creates an ILockBytes on the handle to the native data, and
|
|
// then attempts to create a storage on it. If it can, it uses
|
|
// the CopyTo interface to write that storage into our OLE 2
|
|
// stream. Otherwise, it manually creates a stream in the OLE 2
|
|
// storage and dumps the native data into it.
|
|
//
|
|
// Arguments: [pstg] -- the OLE 2 storage we are saving genobj to
|
|
// [genobj] -- the generic object we are writing
|
|
//
|
|
// Returns: NOERROR on success
|
|
// E_OUTOFMEMORY on allocation failure
|
|
// STG_E_WRITEFAULT on storage write failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 18-Feb-94 davepl Removed 14 goto's (for better or worse)
|
|
// See "Notes" for new control flow
|
|
// 24-Mar-94 alext Fix OLE 1 native case (there was an
|
|
// extra stream open)
|
|
//
|
|
// Notes: There are two possible major codepaths based on the creation
|
|
// of the Stg on ILockBytes. The outcome is handled by a
|
|
// switch statement, and both the TRUE and FALSE cases are
|
|
// loaded with break statements that will bail out to the
|
|
// bottom of the function on any failure. This gives us a
|
|
// single entry and exit point, without all the gotos
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL Write20NativeStreams(LPSTORAGE pstg, const GENOBJ FAR& genobj)
|
|
{
|
|
LPLOCKBYTES plkbyt = NULL;
|
|
LPSTORAGE pstgNative = NULL;
|
|
LPSTREAM pstmNative = NULL;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Create an ILockBytes instance on our generic object's native data
|
|
|
|
if (SUCCEEDED(hr = CreateILockBytesOnHGlobal
|
|
(genobj.m_dataNative.m_h, FALSE, &plkbyt)))
|
|
{
|
|
// If the ILockBytes appears to contain an IStorage, then this was
|
|
// an OLE 2 object "hiding" within the OLE 1 stream as native data
|
|
|
|
switch ((DWORD)(S_OK == StgIsStorageILockBytes (plkbyt)))
|
|
{
|
|
case (TRUE):
|
|
|
|
// Open the IStorage contained in the ILockBytes
|
|
|
|
if (FAILED(hr = StgOpenStorageOnILockBytes (plkbyt,
|
|
(LPSTORAGE)NULL,
|
|
STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_DIRECT,
|
|
(SNB)NULL,
|
|
0,
|
|
&pstgNative)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Can't open storage on ILBytes at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
// Remove the stream from the native data
|
|
|
|
if (FAILED(hr = UtDoStreamOperation(pstgNative,
|
|
NULL, // pstgDst
|
|
OPCODE_REMOVE, // operation
|
|
STREAMTYPE_CACHE))) // stream
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"OPCODE REMOVE stream op failed at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
// Copy the "hidden" IStorage to our destination storage
|
|
|
|
if (FAILED(hr = pstgNative->CopyTo (0, NULL,(SNB)NULL, pstg)))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"CopyTo member fn failed at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
break; // end case TRUE
|
|
|
|
|
|
case FALSE:
|
|
|
|
// This is the typical case, where the OLE 1 stream had just
|
|
// plain old native data, so write it to a stream inside our
|
|
// output IStorage and call it OLE10_NATIVE_STREAM
|
|
|
|
ULONG cb;
|
|
LPVOID pv = genobj.m_dataNative.m_pv;
|
|
|
|
if (NULL == pv)
|
|
{
|
|
hr = ResultFromScode(E_OUTOFMEMORY);
|
|
break;
|
|
}
|
|
|
|
// Create the new stream to hold the native data
|
|
|
|
if (FAILED(hr = OpenOrCreateStream
|
|
(pstg, OLE10_NATIVE_STREAM, &pstmNative)))
|
|
{
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
// Write the length of the native data to the stream
|
|
|
|
if (FAILED(hr = pstmNative->Write
|
|
(&genobj.m_dataNative.m_cbSize, sizeof(ULONG), &cb)))
|
|
{
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
// Now write the actual native data
|
|
|
|
if (FAILED(hr = pstmNative->Write
|
|
(pv, genobj.m_dataNative.m_cbSize, &cb)))
|
|
{
|
|
break; // on failure fall through to error return
|
|
}
|
|
|
|
// Write out the item name
|
|
|
|
if (genobj.m_szItem)
|
|
{
|
|
ULONG cchItem;
|
|
LPSTR pszAnsiItem;
|
|
int cbWritten;
|
|
|
|
// We need to convert m_szItem from Wide to Ansi
|
|
|
|
// The ANSI string is bounded by the byte length of the
|
|
// Unicode string (one Unicode character maximally translates
|
|
// to one double-byte char, so we just use that length
|
|
cchItem = lstrlenW(genobj.m_szItem) + 1;
|
|
|
|
pszAnsiItem = (LPSTR) PrivMemAlloc(cchItem * sizeof(OLECHAR));
|
|
if (NULL == pszAnsiItem)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
break;
|
|
}
|
|
|
|
// We've got out buffer and our length, so do the conversion now
|
|
// We don't need to check for cbSize == FALSE since that was
|
|
// already done during the length test, but we need to check
|
|
// for substitution. Iff this call sets the fDefChar even when
|
|
// only doing a length check, these two tests could be merged,
|
|
// but I don't believe this is the case.
|
|
|
|
BOOL fDefUsed = 0;
|
|
cbWritten = WideCharToMultiByte(CP_ACP, // Code Page ANSI
|
|
0, // No flags
|
|
genobj.m_szItem, // Input OLESTR
|
|
cchItem, // Input len (auto detect)
|
|
pszAnsiItem, // Output buffer
|
|
cchItem * sizeof(OLECHAR), // Output len
|
|
NULL, // Default char (use system's)
|
|
&fDefUsed); // Flag: Default char used
|
|
|
|
// If number of bytes converted was 0, we failed
|
|
|
|
if ((FALSE == cbWritten) || fDefUsed)
|
|
{
|
|
hr = ResultFromScode(E_UNSPEC);
|
|
}
|
|
else
|
|
{
|
|
// Write the size of the string (including null terminator) to stream
|
|
hr = StSave10ItemName(pstg, pszAnsiItem);
|
|
}
|
|
|
|
PrivMemFree(pszAnsiItem);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
break; // on failure fall through to error return
|
|
}
|
|
}
|
|
break;
|
|
|
|
} // end switch
|
|
} // end if
|
|
|
|
// Free up any resources that may have been allocated in any of the
|
|
// code paths above
|
|
|
|
if (NULL != plkbyt)
|
|
{
|
|
plkbyt->Release();
|
|
}
|
|
|
|
if (NULL != pstgNative)
|
|
{
|
|
pstgNative->Release();
|
|
}
|
|
|
|
if (NULL != pstmNative)
|
|
{
|
|
pstmNative->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wConvertIStorageToOLESTREAM, INTERNAL
|
|
//
|
|
// Synopsis: Worker function; brings object from the IStorage into
|
|
// the internal generic object representation
|
|
//
|
|
// Arguments: [pstg] -- the IStorage the object resides in
|
|
// [polestream]-- the OLE 1 stream it will be going to
|
|
// [pgenobj] -- the generic object to hold the internal rep
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_FILENOTFOUND bad IStorage
|
|
// CONVERT10_E_STG_NO_STD_STREAM the IStorage was missing one
|
|
// of the required standard streams
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
INTERNAL wConvertIStorageToOLESTREAM (
|
|
LPSTORAGE pstg,
|
|
LPOLESTREAM polestream,
|
|
PGENOBJ pgenobj
|
|
)
|
|
{
|
|
SCODE scode = S_OK;
|
|
|
|
VDATEIFACE (pstg);
|
|
|
|
// Ensure that all of the pointers are valid
|
|
|
|
#if DBG==1
|
|
if (!IsValidReadPtrIn (polestream, sizeof(OLESTREAM)) ||
|
|
!IsValidReadPtrIn (polestream->lpstbl, sizeof(OLESTREAMVTBL)) ||
|
|
!IsValidCodePtr ((FARPROC)polestream->lpstbl->Put))
|
|
{
|
|
LEDebugOut(( DEB_ERROR,
|
|
"Bad OLESTREAM at line %d in %s\n",
|
|
__LINE__, __FILE__));
|
|
|
|
return ResultFromScode (E_INVALIDARG);
|
|
}
|
|
#endif
|
|
|
|
scode = GetScode (StorageToGenericObject (pstg, pgenobj));
|
|
|
|
// If the storage was not there, modify the return code to
|
|
// make it specific to the conversion process, otherwise just
|
|
// return whatever error code came back.
|
|
|
|
if (scode != S_OK)
|
|
{
|
|
if (scode == STG_E_FILENOTFOUND)
|
|
{
|
|
return ResultFromScode(CONVERT10_E_STG_NO_STD_STREAM);
|
|
}
|
|
else
|
|
{
|
|
return ResultFromScode(scode);
|
|
}
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OleConvertIStorageToOLESTREAM, STDAPI
|
|
//
|
|
// Synopsis: Reads an object from an IStorage into a generic internal
|
|
// representation, then writes it back out to an OLE 1 stream
|
|
//
|
|
// Arguments: [pstg] -- the IStorage to read from
|
|
// [polestream] -- the OLESTREAM to write to
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_STG_NO_STD_STREAM when one of the needed streams
|
|
// inside the IStorage was not
|
|
// present
|
|
// E_INVALIDARG bad input argument
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
STDAPI OleConvertIStorageToOLESTREAM(LPSTORAGE pstg, LPOLESTREAM polestream)
|
|
{
|
|
OLETRACEIN((API_OleConvertIStorageToOLESTREAM,
|
|
PARAMFMT("pstg= %p, polestream= %p"), pstg, polestream));
|
|
|
|
LEDebugOut((DEB_TRACE, "%p _IN OleConvertIStorageToOLESTREAM ("
|
|
" %p , %p )\n", 0 /*function*/,
|
|
pstg, polestream
|
|
));
|
|
CALLHOOKOBJECT(S_OK,CLSID_NULL,IID_IStorage,(IUnknown **)&pstg);
|
|
|
|
HRESULT hr;
|
|
CGenericObject genobj;
|
|
|
|
// Read from the IStorage into the generic object
|
|
hr = wConvertIStorageToOLESTREAM(pstg, polestream, &genobj);
|
|
if (FAILED(hr))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Write from the generic object out to the OLE 1 stream
|
|
hr = GenericObjectToOLESTREAM (genobj, polestream);
|
|
|
|
errRtn:
|
|
LEDebugOut((DEB_TRACE,"%p OUT OleConvertIStorageToOLESTREAM ( %lx ) "
|
|
"\n", 0 /*function*/, hr));
|
|
|
|
OLETRACEOUT((API_OleConvertIStorageToOLESTREAM, hr));
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wFillPpres, INTERNAL
|
|
//
|
|
// Synopsis: Fills in the generic object's presentation data by
|
|
// building a presentation out of the native data
|
|
//
|
|
// Arguments: [pstg] -- the IStorage we are reading from
|
|
// [pgenobj] -- the generic object
|
|
// [cfFormat] -- what clipboard format is being used
|
|
// [fOle10Native] -- flag: is this OLE 1 native data?
|
|
//
|
|
// Returns: NOERROR on success
|
|
// E_OUTOFMEMORY can't allocate mem for PRES member
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup, documentation
|
|
// 19-Jul-94 davepl Fixed HMETAFILE cases
|
|
//
|
|
// Notes: Since most of this code treats HMETAFILE handles and
|
|
// HGLOBALS indentically, we need to special case the
|
|
// the HMETAFILE case by marking the pointer with a
|
|
// special value
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
static INTERNAL wFillPpres(
|
|
LPSTORAGE pstg,
|
|
PGENOBJ pgenobj,
|
|
CLIPFORMAT cfFormat,
|
|
BOOL fOle10Native)
|
|
{
|
|
pgenobj->m_ppres = new PRES;
|
|
|
|
if (pgenobj->m_ppres == NULL)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Set the format tag and clipboard format in the PRES member
|
|
pgenobj->m_ppres->m_format.m_cf = cfFormat;
|
|
pgenobj->m_ppres->m_format.m_ftag = ftagClipFormat;
|
|
|
|
// Build the presentation based on the object's native data
|
|
HANDLE hpres = UtGetHPRESFromNative(pstg, NULL, pgenobj->m_ppres->m_format.m_cf,
|
|
fOle10Native);
|
|
|
|
void * lppres = NULL;
|
|
|
|
if (hpres == NULL)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
// Lock the DIB or the METAFILEPICT structure
|
|
|
|
lppres = GlobalLock(hpres);
|
|
if (NULL == lppres)
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
if (cfFormat == CF_DIB)
|
|
{
|
|
// If it's a DIB, fill in the extents
|
|
LPBITMAPINFOHEADER lpbmi = (LPBITMAPINFOHEADER) lppres;
|
|
UtGetDibExtents(lpbmi, (LPLONG) &(pgenobj->m_ppres->m_ulWidth),
|
|
(LPLONG) &(pgenobj->m_ppres->m_ulHeight));
|
|
|
|
GlobalUnlock(hpres);
|
|
pgenobj->m_ppres->m_data.m_h = hpres;
|
|
|
|
pgenobj->m_ppres->m_data.m_cbSize
|
|
= (ULONG) GlobalSize(pgenobj->m_ppres->m_data.m_h);
|
|
pgenobj->m_ppres->m_data.m_pv
|
|
= GlobalLock(pgenobj->m_ppres->m_data.m_h);
|
|
|
|
|
|
}
|
|
else if (cfFormat == CF_METAFILEPICT)
|
|
{
|
|
LPMETAFILEPICT lpmfp = (LPMETAFILEPICT) lppres;
|
|
|
|
// If it's a METAFILE, fill in the width, height
|
|
pgenobj->m_ppres->m_ulWidth = (ULONG) lpmfp->xExt;
|
|
pgenobj->m_ppres->m_ulHeight = (ULONG) lpmfp->yExt;
|
|
pgenobj->m_ppres->m_data.m_h = lpmfp->hMF;
|
|
GlobalFree(hpres);
|
|
hpres = NULL;
|
|
|
|
// We place a special known value in the pointer field
|
|
// to indicate that the associated handle is a metafile
|
|
// handle (as opposed to a global memory handle), which
|
|
// signals us to special case its cleanup.
|
|
|
|
pgenobj->m_ppres->m_data.m_pv = METADATAPTR;
|
|
|
|
// We cannot merely GlobalSize() the HMETAFILE, so we
|
|
// ask the GDI how many bytes we will need to store the
|
|
// data.
|
|
|
|
pgenobj->m_ppres->m_data.m_cbSize =
|
|
GetMetaFileBitsEx((HMETAFILE) pgenobj->m_ppres->m_data.m_h, 0, NULL);
|
|
|
|
if (0 == pgenobj->m_ppres->m_data.m_cbSize)
|
|
{
|
|
pgenobj->m_ppres->m_data.m_h = NULL;
|
|
goto errRtn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
return NOERROR;
|
|
|
|
|
|
errRtn:
|
|
if (hpres)
|
|
{
|
|
Verify(GlobalUnlock(hpres));
|
|
GlobalFree(hpres);
|
|
}
|
|
|
|
delete pgenobj->m_ppres;
|
|
pgenobj->m_ppres = NULL;
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: StorageToGenericObject, INTERNAL
|
|
//
|
|
// Synopsis: Read an object from an IStorage into the generic object,
|
|
// and set up the format type, native and pres data.
|
|
//
|
|
// Arguments: [pstg] -- the IStorage we are reading from
|
|
// [pgenobj] -- the generic object we are reading into
|
|
//
|
|
// Returns: NOERROR on success
|
|
// various possible errors from lower-level fns
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL StorageToGenericObject(LPSTORAGE pstg, PGENOBJ pgenobj)
|
|
{
|
|
CLSID clsid;
|
|
CLIPFORMAT cf = NULL;
|
|
BOOL fObjFmtKnown = FALSE;
|
|
HRESULT hr;
|
|
|
|
// Get the class ID from the IStorage
|
|
if (FAILED(hr = ReadRealClassStg (pstg, &clsid)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Set the class ID in our generic object
|
|
if (CLSID_StaticMetafile == clsid || CLSID_StaticDib == clsid)
|
|
{
|
|
if (CLSID_StaticMetafile == clsid)
|
|
{
|
|
cf = CF_METAFILEPICT;
|
|
}
|
|
else
|
|
{
|
|
cf = CF_DIB;
|
|
}
|
|
fObjFmtKnown = TRUE;
|
|
|
|
pgenobj->m_class.Set(clsid, NULL);
|
|
pgenobj->m_fStatic = TRUE;
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = pgenobj->m_class.Set (clsid, pstg)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
// Get the OLE version, flags, update opts, and moniker
|
|
|
|
SCODE sc = GetScode (Read20OleStream (pstg, pgenobj));
|
|
|
|
// It is okay for the Ole Stream to be missing.
|
|
if (sc != S_OK)
|
|
{
|
|
if (sc != STG_E_FILENOTFOUND)
|
|
{
|
|
return ResultFromScode (sc);
|
|
}
|
|
}
|
|
|
|
// Read the native data into the generic object
|
|
if (FAILED(hr = Read20NativeStreams (pstg, &(pgenobj->m_dataNative))))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Try to ascertain the clipboard format
|
|
if (cf == 0)
|
|
{
|
|
if (clsid == CLSID_PBrush)
|
|
{
|
|
cf = CF_DIB;
|
|
}
|
|
else if (clsid == CLSID_MSDraw)
|
|
{
|
|
cf = CF_METAFILEPICT;
|
|
}
|
|
else
|
|
{
|
|
ReadFmtUserTypeStg (pstg, &cf, NULL);
|
|
}
|
|
|
|
fObjFmtKnown = (cf == CF_METAFILEPICT || cf == CF_DIB);
|
|
}
|
|
|
|
// Read the presentation data if possible
|
|
if (FAILED(hr = Read20PresStream (pstg, pgenobj, fObjFmtKnown)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// If we don't have a presentation, it might be a PBrush object,
|
|
// which is OK because OLE 1 DLLs know how to draw them based on
|
|
// the native data. Otherwise, we will try and create a presentation
|
|
// based on the native data.
|
|
|
|
if (pgenobj->m_ppres == NULL)
|
|
{
|
|
if (clsid == CLSID_PBrush)
|
|
{
|
|
return NOERROR;
|
|
}
|
|
if (cf == CF_METAFILEPICT || cf == CF_DIB)
|
|
{
|
|
if (FAILED(hr=wFillPpres(pstg,pgenobj,cf,clsid == CLSID_MSDraw)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: GenericObjectToOLESTREAM, INTERNAL
|
|
//
|
|
// Synopsis: Writes the interal object representation out to an OLE1
|
|
// stream.
|
|
//
|
|
// Arguments: [genobj] -- the object to write out
|
|
// [pos] -- the OLE 1 stream to write to
|
|
//
|
|
// Returns: NOERROR on success
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 22-Feb-94 davepl 32-bit port
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL GenericObjectToOLESTREAM(
|
|
const GENOBJ FAR& genobj,
|
|
LPOLESTREAM pos)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (genobj.m_fStatic)
|
|
{
|
|
return PutPresentationObject (pos, genobj.m_ppres, genobj.m_class,
|
|
TRUE /* fStatic*/ );
|
|
}
|
|
|
|
// OLE version
|
|
if (FAILED(hr = ULToOLE1Stream (pos, dwVerToFile)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Format ID for embedded or linked object
|
|
if (FAILED(hr = ULToOLE1Stream
|
|
(pos, genobj.m_fLink ? FMTID_LINK : FMTID_EMBED)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// We must have the class id string by this point
|
|
Assert (genobj.m_class.m_szClsid);
|
|
|
|
// Write out the class ID string
|
|
if (FAILED(hr = StringToOLE1Stm (pos, genobj.m_class.m_szClsid)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Write out the topic string
|
|
if (FAILED(hr = StringToOLE1Stm (pos, genobj.m_szTopic)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Write out the item string
|
|
if (FAILED(hr = StringToOLE1Stm (pos, genobj.m_szItem)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Write out the update options, network info for a link,
|
|
// or the native data for an embedded object
|
|
if (genobj.m_fLink)
|
|
{
|
|
// Network information
|
|
if (FAILED(hr = PutNetworkInfo (pos, genobj.m_szTopic)))
|
|
{
|
|
return hr;
|
|
}
|
|
// Link update options
|
|
if (FAILED(hr = ULToOLE1Stream (pos, genobj.m_lnkupdopt)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = SizedDataToOLE1Stm (pos, genobj.m_dataNative)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
// Write out the presentation data
|
|
return PutPresentationObject (pos, genobj.m_ppres, genobj.m_class);
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PutNetworkInfo, INTERNAL
|
|
//
|
|
// Synopsis: If needed, converts a DOS-style path to a proper network
|
|
// path. In any case, writes network path to OLE 1 stream
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream we are writing to
|
|
// [szTopic] -- the topic string for this object
|
|
//
|
|
// Returns: NOERROR on success
|
|
// Various possible I/O errors on write
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PutNetworkInfo(LPOLESTREAM pos, LPOLESTR szTopic)
|
|
{
|
|
LPOLESTR szNetName = NULL;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// If we have an X:\ style path, we want to convert that
|
|
// to a proper network name
|
|
|
|
if (szTopic && IsCharAlphaW(szTopic[0]) && szTopic[1]==':')
|
|
{
|
|
OLECHAR szBuf[80];
|
|
DWORD u;
|
|
OLECHAR szDrive[3];
|
|
|
|
szDrive[0] = (OLECHAR)CharUpperW((LPWSTR)szTopic[0]);
|
|
szDrive[1] = ':' ;
|
|
szDrive[2] = '\0';
|
|
|
|
if (GetDriveType (szDrive) == DRIVE_REMOTE
|
|
&& WNetGetConnection (szDrive, szBuf, &u) == WN_SUCCESS)
|
|
{
|
|
szNetName =szBuf;
|
|
}
|
|
}
|
|
|
|
// We now have the network name, so write it out to OLE 1 stream
|
|
if (FAILED(hr = StringToOLE1Stm (pos, szNetName)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Network type, driver version number, but we have to pad for
|
|
// the space anyway
|
|
|
|
if (FAILED(hr = ULToOLE1Stream (pos, 0L)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
Assert (hr == NOERROR);
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OpenStream, INTERNAL
|
|
//
|
|
// Synopsis: Opens a stream in SHARE_EXCLUSIVE, READ mode
|
|
//
|
|
// Arguments: [pstg] -- the storage the stream resides in
|
|
// [szName] -- the name of the stream
|
|
// [ppstm] -- out parameter for stream
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and document
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static inline INTERNAL OpenStream(
|
|
LPSTORAGE pstg,
|
|
LPOLESTR szName,
|
|
LPSTREAM FAR* ppstm)
|
|
{
|
|
return pstg->OpenStream
|
|
(szName, NULL, STGM_SHARE_EXCLUSIVE| STGM_READ, 0, ppstm);
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: ReadRealClassStg, INTERNAL
|
|
//
|
|
// Synopsis: Reads the _real_ class of the object. ie: if the class is
|
|
// StdOleLink, we need to find out the class of the object
|
|
// to which this is linked
|
|
//
|
|
// Arguments: pstg -- the storage to read from
|
|
// pclsid -- caller's CLSID holder
|
|
//
|
|
// Returns: NOERROR on success
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 04-Mar-04 davepl 32-bit port
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL ReadRealClassStg(LPSTORAGE pstg, LPCLSID pclsid)
|
|
{
|
|
LPSTREAM pstm = NULL;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Get the class ID from the IStorage
|
|
if (FAILED(hr = ReadClassStg (pstg, pclsid)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// If it's a link, we have to figure out what class its a link _to_
|
|
if (CLSID_StdOleLink == *pclsid)
|
|
{
|
|
LPMONIKER pmk = NULL;
|
|
|
|
if (FAILED(hr = ReadOleStg (pstg, NULL, NULL, NULL, NULL, &pstm)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED(hr = ReadMonikerStm (pstm, &pmk)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
if (pmk)
|
|
{
|
|
pmk->Release();
|
|
}
|
|
|
|
if (FAILED(hr = ReadMonikerStm (pstm, &pmk)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
if (pmk)
|
|
{
|
|
pmk->Release();
|
|
}
|
|
|
|
// Read "last class"
|
|
if (FAILED(hr = ReadM1ClassStm (pstm, pclsid)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
errRtn:
|
|
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: Read20OleStream, INTERNAL
|
|
//
|
|
// Synopsis: Reads the update options and absolute source class from
|
|
// an OLE 2 object
|
|
//
|
|
// Arguments: pstg -- the IStorage to read from
|
|
// pgenobj -- the genobj we are reading into
|
|
//
|
|
// Returns: NOERROR on success
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 06-Mar-94 davepl 32-bit port
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL Read20OleStream(LPSTORAGE pstg, PGENOBJ pgenobj)
|
|
{
|
|
LPMONIKER pmk = NULL;
|
|
HRESULT hr = NOERROR;
|
|
LPSTREAM pstm = NULL;
|
|
ULONG ul = (ULONG) -1L;
|
|
CLSID clsidLast;
|
|
|
|
if (SUCCEEDED(hr = OpenStream (pstg, OLE_STREAM, &pstm)))
|
|
{
|
|
// OLE version
|
|
if (SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)))
|
|
{
|
|
// Object flags
|
|
if (SUCCEEDED(hr = OLE2StmToUL (pstm, &ul)))
|
|
{
|
|
if (ul & OBJFLAGS_LINK)
|
|
{
|
|
pgenobj->m_fLink = TRUE;
|
|
}
|
|
|
|
// Update options
|
|
hr = OLE2StmToUL (pstm, &ul);
|
|
}
|
|
}
|
|
}
|
|
|
|
// If no errors so far...
|
|
|
|
// If this is a link, get the update options
|
|
|
|
if (SUCCEEDED(hr) && pgenobj->m_fLink)
|
|
{
|
|
switch (ul)
|
|
{
|
|
case OLEUPDATE_ALWAYS:
|
|
pgenobj->m_lnkupdopt = UPDATE_ALWAYS;
|
|
break;
|
|
|
|
case OLEUPDATE_ONCALL:
|
|
pgenobj->m_lnkupdopt = UPDATE_ONCALL;
|
|
break;
|
|
|
|
default:
|
|
AssertSz (0, "Warning: Invalid update options in Storage");
|
|
hr = ResultFromScode(CONVERT10_E_STG_FMT);
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr)) // Only continue if no failures so far
|
|
{
|
|
// Reserved (was view format)
|
|
if (SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)))
|
|
{
|
|
if (pgenobj->m_fLink)
|
|
{
|
|
|
|
// All 4 of these calls must succeed or we simply fall
|
|
// through to the cleanup code
|
|
|
|
// ignore relative moniker
|
|
if (SUCCEEDED(hr = OLE2StmToMoniker (pstm, NULL)) &&
|
|
// ignore relative source moniker
|
|
SUCCEEDED(hr = OLE2StmToMoniker (pstm, NULL)) &&
|
|
// get absolute source moniker
|
|
SUCCEEDED(hr = OLE2StmToMoniker (pstm, &pmk)) &&
|
|
// get class from abs moniker
|
|
SUCCEEDED(hr = ReadM1ClassStm (pstm, &clsidLast)) )
|
|
{
|
|
hr = MonikerIntoGenObj (pgenobj, clsidLast, pmk);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Clean up any resources and return status to caller
|
|
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
if (pmk)
|
|
{
|
|
pmk->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLE2StmToMoniker, INTERNAL
|
|
//
|
|
// Synopsis: Calls ReadMonikerStm to get a moniker from a stream,
|
|
// and if the ppmk parameter was NULL, it does a Release()
|
|
// on the moniker object immediately, otherwise sets the
|
|
// caller's pointer to point to the moniker that was read.
|
|
//
|
|
// Arguments: [pstm] -- the stream to read the moniker from
|
|
// [ppmk] -- points to caller's moniker ptr
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL OLE2StmToMoniker(LPSTREAM pstm, LPMONIKER FAR* ppmk)
|
|
{
|
|
LPMONIKER pmk = NULL;
|
|
HRESULT hr = NOERROR;
|
|
|
|
if (FAILED(hr = ReadMonikerStm (pstm, &pmk)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (ppmk) // If the callers wanted a result, return the
|
|
{ // moniker as an out parameter
|
|
*ppmk = pmk;
|
|
}
|
|
else // Otherwise, release it immediately and
|
|
{ // return to caller
|
|
if (pmk)
|
|
{
|
|
pmk->Release();
|
|
}
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: ReadFormat, INTERNAL
|
|
//
|
|
// Synopsis: Reads the format ID type from the stream, and based on that,
|
|
// reads the format ID from the stream.
|
|
//
|
|
// Arguments: [pstm] -- the stream to read from
|
|
// [pformat] -- caller's format member object
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes: The first ULONG indicates the type (standard clipboard,
|
|
// Mac, NULL, or string) of the identifier
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL ReadFormat(LPSTREAM pstm, PFORMAT pformat)
|
|
{
|
|
ULONG ul;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Get the format ID type indicator
|
|
|
|
if (FAILED(hr = OLE2StmToUL (pstm, &ul)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// The first ULONG indicates what kind of format ID will
|
|
// found in the stream:
|
|
//
|
|
// -1 => A standard clipboard format ID
|
|
// -2 => A Macintosh format
|
|
// 0 => NULL format
|
|
// >0 => The number of bytes of the text string
|
|
// identifier to follow
|
|
|
|
switch ((signed long)ul)
|
|
{
|
|
case -1L: // Standard clipboard format
|
|
|
|
ULONG ulClipFormat;
|
|
pformat->m_ftag = ftagClipFormat;
|
|
if (FAILED(hr = OLE2StmToUL (pstm, &ulClipFormat)))
|
|
{
|
|
return hr;
|
|
}
|
|
pformat->m_cf = (CLIPFORMAT) ulClipFormat;
|
|
break;
|
|
|
|
|
|
case -2L: // Macintosh format
|
|
|
|
return ResultFromScode(CONVERT10_E_STG_FMT);
|
|
|
|
|
|
case 0: // NULL format
|
|
|
|
pformat->m_ftag = ftagNone;
|
|
pformat->m_cf = 0;
|
|
return NOERROR;
|
|
|
|
|
|
default: // ul == size of string (format name)
|
|
|
|
|
|
pformat->m_ftag = ftagString;
|
|
if (FAILED(hr = OLE2StmToSizedData
|
|
(pstm, &(pformat->m_dataFormatString), 0, ul)))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
#ifdef _OBSOLETE
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: WriteFormat, INTERNAL
|
|
//
|
|
// Synopsis: Depending on what kind of format (standard cf, string, etc)
|
|
// the format object holds, this fn writes out the appropriate
|
|
// information to the stream
|
|
//
|
|
// Arguments: [pstm] -- the stream to write to
|
|
// [format] -- the format object to get info from
|
|
//
|
|
// Returns: NOERROR on success
|
|
// E_UNEXPECTED for a NULL format tag
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
// Notes:
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL WriteFormat(LPSTREAM pstm, const FORMAT FAR& format)
|
|
{
|
|
HRESULT hr;
|
|
|
|
switch (format.m_ftag)
|
|
{
|
|
case ftagNone:
|
|
Assert (0 && "Cant write a NULL format tag");
|
|
return ResultFromScode (E_UNEXPECTED);
|
|
|
|
case ftagClipFormat:
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, (ULONG) -1L)))
|
|
{
|
|
return hr;
|
|
}
|
|
if (FAILED(hr = ULToOLE2Stm (pstm, format.m_cf)))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
case ftagString:
|
|
if (FAILED(hr=DataObjToOLE2Stm(pstm,format.m_dataFormatString)))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
AssertSz (0, "invalid m_ftag value");
|
|
return ResultFromScode (E_UNEXPECTED);
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
#endif // _OBSOLETE
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: ReadDibAsBitmap, INTERNAL
|
|
//
|
|
// Synopsis: Reads a DIB from an OLE 2 stream and stores it as a
|
|
// Bitmap in a DATA structure
|
|
//
|
|
// Arguments: [pstm] -- the OLE 2 stream to read from
|
|
// [pdata] -- the data object to hold the bitmap
|
|
//
|
|
// Returns: NOERROR on success
|
|
// CONVERT10_E_STG_DIB_TO_BITMAP conversion failure
|
|
// E_OUTOFMEMORY allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL ReadDibAsBitmap(LPSTREAM pstm, PDATA pdata)
|
|
{
|
|
DATA dataDib;
|
|
ULONG cb;
|
|
ULONG cbBits;
|
|
ULONG cbBitsFake;
|
|
BITMAP bm;
|
|
|
|
HBITMAP hBitmap = NULL;
|
|
HRESULT hr = NOERROR;
|
|
HGLOBAL hBits = NULL;
|
|
LPBYTE pBits = NULL;
|
|
|
|
Assert (pdata&&pdata->m_cbSize==0&&pdata->m_h==NULL&&pdata->m_pv==NULL);
|
|
|
|
// Read the DIB into our local DATA object
|
|
if (FAILED(hr = OLE2StmToSizedData (pstm, &dataDib)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Convert the DIB to a Bitmap
|
|
hBitmap = UtConvertDibToBitmap (dataDib.m_h);
|
|
if (NULL == hBitmap )
|
|
{
|
|
return ResultFromScode(CONVERT10_E_STG_DIB_TO_BITMAP);
|
|
}
|
|
|
|
if (0 == GetObject (hBitmap, sizeof(BITMAP), &bm))
|
|
{
|
|
return ResultFromScode(CONVERT10_E_STG_DIB_TO_BITMAP);
|
|
}
|
|
|
|
cbBits = (DWORD) bm.bmHeight * (DWORD) bm.bmWidthBytes
|
|
* (DWORD) bm.bmPlanes;
|
|
|
|
// There was a bug in OLE 1.0. It calculated the size of a bitmap
|
|
// as Height * WidthBytes * Planes * BitsPixel.
|
|
// So we need to put that many bytes here even if most of the end of that
|
|
// data block is garbage. Otherwise OLE 1.0 will try to read too many
|
|
// bytes of the OLESTREAM as bitmap bits.
|
|
|
|
cbBitsFake = cbBits * (DWORD) bm.bmBitsPixel;
|
|
|
|
// Allocate enough memory for our resultant BITMAP & header
|
|
hBits = GlobalAlloc (GMEM_MOVEABLE, cbBitsFake + sizeof (BITMAP));
|
|
if (NULL == hBits)
|
|
{
|
|
if (hBitmap)
|
|
{
|
|
Verify (DeleteObject (hBitmap));
|
|
}
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Get a pointer to the memory
|
|
pBits = (LPBYTE) GlobalLock (hBits);
|
|
if (NULL == pBits)
|
|
{
|
|
if (hBitmap)
|
|
{
|
|
Verify (DeleteObject (hBitmap));
|
|
}
|
|
GlobalFree(hBits);
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Copy the raw bitmap data
|
|
cb = GetBitmapBits (hBitmap, cbBits, pBits + sizeof(BITMAP));
|
|
if (cb != cbBits)
|
|
{
|
|
if (hBitmap)
|
|
{
|
|
Verify (DeleteObject (hBitmap));
|
|
}
|
|
GlobalFree(hBits);
|
|
return ResultFromScode(CONVERT10_E_STG_DIB_TO_BITMAP);
|
|
}
|
|
|
|
// Set the caller's pointer to point to the bitmap
|
|
|
|
*((BITMAP FAR*)pBits) = bm;
|
|
|
|
pdata->m_h = hBits;
|
|
pdata->m_pv = pBits;
|
|
pdata->m_cbSize = cbBitsFake + sizeof(BITMAP);
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: Read20PresStream, INTERNAL
|
|
//
|
|
// Synopsis: Reads presentation data from an IStorage into a
|
|
// generic object
|
|
//
|
|
// Arguments: [pstg] -- the IStorage holding the pres stream
|
|
// [pgenobj] -- the generic object to read to
|
|
// [fObjFmtKnown] -- flag: Do we know the object format?
|
|
//
|
|
// Returns: NOEROR on success
|
|
// E_OUTOFMEMORY on allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 22-Feb-94 davepl Code cleanup and documentation
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL Read20PresStream(
|
|
LPSTORAGE pstg,
|
|
PGENOBJ pgenobj,
|
|
BOOL fObjFmtKnown)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPSTREAM pstm = NULL;
|
|
|
|
// Find the best presentation stream in this IStorage
|
|
|
|
if (FAILED(hr = FindPresStream (pstg, &pstm, fObjFmtKnown)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (pstm)
|
|
{
|
|
// Allocate a generic presentation object
|
|
Assert (NULL==pgenobj->m_ppres);
|
|
pgenobj->m_ppres = new PRES;
|
|
if (NULL == pgenobj->m_ppres)
|
|
{
|
|
pstm->Release();
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No presentation stream
|
|
Assert (NULL == pgenobj->m_ppres);
|
|
return NOERROR;
|
|
}
|
|
|
|
// read the format
|
|
if (FAILED(hr = ReadFormat (pstm, &(pgenobj->m_ppres->m_format))))
|
|
{
|
|
pstm->Release();
|
|
return hr;
|
|
}
|
|
|
|
// This is the fix for Bug 4020, highly requested by Access
|
|
if (pgenobj->m_ppres->m_format.m_ftag == ftagNone)
|
|
{
|
|
// NULL format
|
|
delete pgenobj->m_ppres;
|
|
pgenobj->m_ppres = NULL;
|
|
Assert (hr == NOERROR);
|
|
pstm->Release();
|
|
return hr;
|
|
}
|
|
|
|
// Each of the following calls must succeed in order for the following
|
|
// one to be executed; if any fails, the if( .. && ..) will be false
|
|
// and hr will be set to the error that caused the failure
|
|
|
|
// target device
|
|
if (SUCCEEDED(hr = OLE2StmToSizedData (pstm, NULL, 4)) &&
|
|
// aspect
|
|
SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)) &&
|
|
// lIndex
|
|
SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)) &&
|
|
// cache flags
|
|
SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)) &&
|
|
// compression
|
|
SUCCEEDED(hr = OLE2StmToUL (pstm, NULL)) &&
|
|
// width
|
|
SUCCEEDED(hr = OLE2StmToUL (pstm, &(pgenobj->m_ppres->m_ulWidth))))
|
|
{ // height
|
|
hr = OLE2StmToUL (pstm, &(pgenobj->m_ppres->m_ulHeight));
|
|
}
|
|
|
|
// We only proceed if everything so far has suceeded
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (pgenobj->m_ppres->m_format.m_ftag == ftagClipFormat &&
|
|
pgenobj->m_ppres->m_format.m_cf == CF_DIB &&
|
|
!pgenobj->m_fStatic)
|
|
{
|
|
pgenobj->m_ppres->m_format.m_cf = CF_BITMAP;
|
|
hr = ReadDibAsBitmap (pstm, &(pgenobj->m_ppres->m_data));
|
|
}
|
|
else
|
|
{
|
|
// In most cases, we look for a sized block of data in the
|
|
// stream.
|
|
|
|
hr = OLE2StmToSizedData (pstm, &(pgenobj->m_ppres->m_data));
|
|
}
|
|
}
|
|
|
|
// Free up the stream and return status to caller
|
|
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OLE2StmToSizedData, INTERNAL
|
|
//
|
|
// Synopsis: Reads a set amount of data from an OLE 2 stream into a
|
|
// DATA structure. If the number of bytes are not known
|
|
// ahead of time, the data length is pulled as the first
|
|
// ULONG at the current stream position.
|
|
//
|
|
// Arguments: [pstm] -- the stream to read from
|
|
// [pdata] -- the DATA structure to read to
|
|
// [cbSizeDelta] -- amount to be subtracted from
|
|
// length; used to read target devices
|
|
// where the length of data includes
|
|
// prefixed length
|
|
// [cbSizeKnown] -- number of bytes to read if known
|
|
// ahead of time
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL OLE2StmToSizedData(
|
|
LPSTREAM pstm,
|
|
PDATA pdata,
|
|
ULONG cbSizeDelta, // default 0
|
|
ULONG cbSizeKnown) // default 0
|
|
{
|
|
ULONG cbSize;
|
|
ULONG cbRead;
|
|
LARGE_INTEGER large_integer;
|
|
HRESULT hr = NOERROR;
|
|
|
|
// If we don't know the data size ahead of time, read it from the stream;
|
|
// it will be the first ULONG at the current position
|
|
|
|
if (cbSizeKnown)
|
|
{
|
|
cbSize = cbSizeKnown;
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = (OLE2StmToUL (pstm, &cbSize))))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
cbSize -= cbSizeDelta;
|
|
|
|
// If pdata is set, it means we actually do want to read the
|
|
// data to a buffer, rather than just skip over it (the NULL case)
|
|
|
|
if (pdata)
|
|
{
|
|
Assert (pdata->m_cbSize==0 && pdata->m_h==NULL && pdata->m_pv==NULL);
|
|
|
|
// Set the number of bytes in the DATA structure
|
|
|
|
pdata->m_cbSize = cbSize;
|
|
|
|
// If there are any, allocate a buffer and read them.
|
|
|
|
if (cbSize)
|
|
{
|
|
// Allocate memory on the DATA handle
|
|
pdata->m_h = GlobalAlloc (GMEM_MOVEABLE, cbSize);
|
|
if (NULL == pdata->m_h)
|
|
{
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Lock memory in for the read
|
|
pdata->m_pv = GlobalLock (pdata->m_h);
|
|
if (NULL == pdata->m_pv)
|
|
{
|
|
GlobalFree(pdata->m_h);
|
|
return ResultFromScode(E_OUTOFMEMORY);
|
|
}
|
|
|
|
// Read the data to the buffer
|
|
if (FAILED(hr = pstm->Read (pdata->m_pv, cbSize, &cbRead)))
|
|
{
|
|
GlobalUnlock(pdata->m_h);
|
|
GlobalFree(pdata->m_h);
|
|
return hr;
|
|
}
|
|
|
|
// If we didn't get enough bytes, bail now
|
|
if (cbRead != cbSize)
|
|
{
|
|
GlobalUnlock(pdata->m_h);
|
|
GlobalFree(pdata->m_h);
|
|
return ResultFromScode(STG_E_READFAULT);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// We have 0 bytes to read, so mark the
|
|
// memory handle and ptr as NULL
|
|
pdata->m_h = NULL;
|
|
pdata->m_pv = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// we don't care what the data is, so just skip it
|
|
LISet32( large_integer, cbSize );
|
|
if (FAILED(hr = pstm->Seek (large_integer, STREAM_SEEK_CUR, NULL)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: RankOfPres, INTERNAL
|
|
//
|
|
// Synopsis: Returns a ULONG indicating the relative "goodness" of a
|
|
// presentation. The preference is, in descending order:
|
|
//
|
|
// Type Rank
|
|
// ---------- ----------
|
|
// METAFILE x30000
|
|
// DIB x20000
|
|
// none x10000
|
|
//
|
|
// Add x200 for fScreenTargDev being set
|
|
// Add x4 for Content aspect
|
|
// Add x3 for Thumbnail aspect
|
|
// Add x2 for Icon aspect
|
|
// Add x1 for Docprint aspect
|
|
//
|
|
// Eg: Metafile in Content aspect, with ScreenTargDev: 30204
|
|
//
|
|
// The whole point of this is that there may be many
|
|
// presentation streams available in the IStorage. This fn
|
|
// is used to select the best one.
|
|
//
|
|
// Arguments: [format] -- the format tag & type structure
|
|
// [fScreenTargDev]-- do we have a handle to the target dev
|
|
// [dwAspect] -- the aspect type
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL_(ULONG) RankOfPres(
|
|
const FORMAT FAR& format,
|
|
const BOOL fScreenTargDev,
|
|
const DWORD dwAspect)
|
|
{
|
|
ULONG ul = 0L;
|
|
|
|
if (format.m_cf==CF_METAFILEPICT)
|
|
{
|
|
ul += 0x030000;
|
|
}
|
|
else if (format.m_cf==CF_DIB)
|
|
{
|
|
ul += 0x020000;
|
|
}
|
|
else if (format.m_ftag != ftagNone)
|
|
{
|
|
ul += 0x010000;
|
|
}
|
|
|
|
ul += (fScreenTargDev + 1) * 0x0100;
|
|
|
|
switch (dwAspect)
|
|
{
|
|
case DVASPECT_CONTENT:
|
|
ul += 0x04;
|
|
break;
|
|
|
|
case DVASPECT_THUMBNAIL:
|
|
ul += 0x03;
|
|
break;
|
|
|
|
case DVASPECT_ICON:
|
|
ul += 0x02;
|
|
break;
|
|
|
|
case DVASPECT_DOCPRINT:
|
|
ul += 0x01;
|
|
break;
|
|
}
|
|
|
|
return ul;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: IsBetter, INTERNAL INLINE
|
|
//
|
|
// Synopsis: Calls RankOfPres to determine if one presentation is
|
|
// better than another
|
|
//
|
|
// Effects:
|
|
//
|
|
// Arguments: [format] -- the format tag and type
|
|
// [fScreenTargDev]-- do we have a handle to target device
|
|
// [dwAspect] -- the aspect of the presentation
|
|
// [formatBest] -- the best format seen so far
|
|
// [fScreenTargDevBest] -- flag for best format seen so far
|
|
// [dwAspectBest] -- the aspect of best format seen so far
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
/// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline static INTERNAL_(BOOL) IsBetter(
|
|
const FORMAT FAR& format,
|
|
const BOOL fScreenTargDev,
|
|
const DWORD dwAspect,
|
|
const FORMAT FAR& formatBest,
|
|
const BOOL fScreenTargDevBest,
|
|
const DWORD dwAspectBest)
|
|
{
|
|
return RankOfPres (format, fScreenTargDev, dwAspect) >
|
|
RankOfPres (formatBest, fScreenTargDevBest, dwAspectBest);
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: FindPresStream, INTERNAL
|
|
//
|
|
// Synopsis: Enumerates over the streams in an IStorage, looking for
|
|
// presentation streams. Selects the best stream among
|
|
// these based on the comparison fn, IsBetter(), which uses
|
|
// for comparison the criteria established in RankOfPres().
|
|
//
|
|
// Arguments: [pstg] -- the IStorage to look in
|
|
// [ppstmBest] -- out param for best pres stream
|
|
// [fObjFmtKnown] is the object format known
|
|
//
|
|
// Returns: NOERROR on success
|
|
// If no presentation is found, it is not an error but
|
|
// *ppstm is set to NULL.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup and documentation
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL FindPresStream(
|
|
LPSTORAGE pstg,
|
|
LPSTREAM FAR* ppstmBest,
|
|
BOOL fObjFmtKnown)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPSTREAM pstm = NULL;
|
|
IEnumSTATSTG FAR* penumStg = NULL;
|
|
DWORD dwAspectBest = 0;
|
|
BOOL fTargDevBest = -1;
|
|
STATSTG statstg;
|
|
FORMAT formatBest;
|
|
|
|
Assert (ppstmBest);
|
|
|
|
*ppstmBest = NULL;
|
|
|
|
// Set up the enumeration on the available IStreams in the storage
|
|
if (FAILED(hr = pstg->EnumElements (NULL, NULL, NULL, &penumStg)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Enumerate through them and search for the best among all
|
|
// presentation streams
|
|
|
|
while (penumStg->Next (1, &statstg, NULL) == NOERROR)
|
|
{
|
|
// Check to see if this a presentation stream
|
|
|
|
if (lstrlenW(statstg.pwcsName) >= 8 &&
|
|
0==memcmp(statstg.pwcsName, OLESTR("\2OlePres"), 8*sizeof(WCHAR)))
|
|
{
|
|
FORMAT format;
|
|
DATA dataTargDev;
|
|
DWORD dwAspect;
|
|
|
|
// Open the presentation stream
|
|
if (FAILED(hr = OpenStream (pstg, statstg.pwcsName, &pstm)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the format from the pres stream
|
|
if (FAILED(hr = ReadFormat (pstm, &format)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Read the target device from the pres stream
|
|
if (FAILED(hr = OLE2StmToSizedData (pstm, &dataTargDev, 4)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Get the aspect from the pres stream
|
|
if (FAILED(hr = OLE2StmToUL (pstm, &dwAspect)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Check to see if this presentation stream is better
|
|
// than the best seen so far
|
|
|
|
if (IsBetter (format, dataTargDev.m_h==NULL, dwAspect,
|
|
formatBest, fTargDevBest, dwAspectBest))
|
|
{
|
|
// If it is, we can release the "best"
|
|
if (*ppstmBest)
|
|
{
|
|
(*ppstmBest)->Release();
|
|
}
|
|
|
|
// The king is dead, long live the king
|
|
*ppstmBest = pstm;
|
|
pstm->AddRef();
|
|
|
|
formatBest = format;
|
|
fTargDevBest = (dataTargDev.m_h==NULL);
|
|
dwAspectBest = dwAspect;
|
|
}
|
|
pstm->Release();
|
|
pstm = NULL;
|
|
}
|
|
PubMemFree(statstg.pwcsName);
|
|
statstg.pwcsName = NULL;
|
|
}
|
|
|
|
// On Windows For Workgroups machines, statstg.pwcsName!=NULL when
|
|
// Next() returns S_FALSE. Bug 3370.
|
|
statstg.pwcsName = NULL;
|
|
|
|
errRtn:
|
|
|
|
if (statstg.pwcsName)
|
|
{
|
|
PubMemFree(statstg.pwcsName);
|
|
}
|
|
|
|
if (*ppstmBest)
|
|
{
|
|
if (dwAspectBest != DVASPECT_CONTENT && fObjFmtKnown)
|
|
{
|
|
// then don't use this stream, we will get the presentaion
|
|
// from the CONTENTS stream
|
|
(*ppstmBest)->Release();
|
|
*ppstmBest = NULL;
|
|
}
|
|
else
|
|
{
|
|
LARGE_INTEGER large_integer;
|
|
LISet32( large_integer, 0);
|
|
hr = (*ppstmBest)->Seek(large_integer, STREAM_SEEK_SET,NULL);
|
|
}
|
|
}
|
|
|
|
if (penumStg)
|
|
{
|
|
penumStg->Release();
|
|
}
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: Reads native data from an OLE 2 stream
|
|
//
|
|
// Synopsis: If the fn can find OLE 1 native data in the stream, it is
|
|
// read out; otherwise, it attempts to create an IStorage
|
|
// in memory on the data in the stream, and then uses the
|
|
// CopyTo interface to extract the data.
|
|
//
|
|
// Arguments: [pstg] -- The OLE 2 IStorage to look in
|
|
// [pdata] -- The DATA object to read native data to
|
|
//
|
|
// Returns: NOERROR on success
|
|
// STG_E_READFAULT on read failure
|
|
// E_OUTOFMEMORY on allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-feb-94 davepl Cleaned up and documented code
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL Read20NativeStreams(LPSTORAGE pstg, PDATA pdata)
|
|
{
|
|
LPSTREAM pstm = NULL;
|
|
LPLOCKBYTES plkbyt = NULL;
|
|
LPSTORAGE pstgNative= NULL;
|
|
|
|
HRESULT hr = NOERROR;
|
|
|
|
// There are two possible codepaths based on the success of
|
|
// OpenStream. If it is true, it is because we were able to
|
|
// open the OLE 1 presentation stream in the OLE 2 object.
|
|
// Thus, it must have been an OLE 1 object "hidden" in
|
|
// an OLE 2 IStream.
|
|
//
|
|
// If that fails, we create an in-memory IStorage based on
|
|
// the native data and use the CopyTo member to extract the
|
|
// natice data.
|
|
//
|
|
// If we experience a failure at any point, a "break" statement
|
|
// bails us out past everything to the error cleanup and return
|
|
// code following the closure of the switch() statement.
|
|
|
|
switch ((DWORD)(NOERROR==OpenStream (pstg, OLE10_NATIVE_STREAM, &pstm)))
|
|
{
|
|
case TRUE:
|
|
{
|
|
// This was a 1.0 object "hidden" inside a 2.0 IStorage
|
|
ULONG cbRead;
|
|
|
|
Assert (pdata->m_cbSize==0 && NULL==pdata->m_h && NULL==pdata->m_pv);
|
|
|
|
// read size
|
|
if (FAILED(hr = pstm->Read(&(pdata->m_cbSize),sizeof(DWORD),&cbRead)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (sizeof(DWORD) != cbRead)
|
|
{
|
|
hr = ResultFromScode (STG_E_READFAULT);
|
|
break;
|
|
}
|
|
|
|
// allocate memory to store copy of stream
|
|
pdata->m_h = GlobalAlloc (GMEM_MOVEABLE, pdata->m_cbSize);
|
|
if (NULL == pdata->m_h)
|
|
{
|
|
hr = ResultFromScode(E_OUTOFMEMORY);
|
|
break;
|
|
}
|
|
|
|
pdata->m_pv = GlobalLock (pdata->m_h);
|
|
if (NULL == pdata->m_pv)
|
|
{
|
|
hr = ResultFromScode(E_OUTOFMEMORY);
|
|
break;
|
|
}
|
|
|
|
// read stream
|
|
if (FAILED(hr = pstm->Read(pdata->m_pv,pdata->m_cbSize,&cbRead)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (pdata->m_cbSize != cbRead)
|
|
{
|
|
hr= ResultFromScode (STG_E_READFAULT);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case FALSE:
|
|
{
|
|
const DWORD grfCreateStg = STGM_READWRITE | STGM_SHARE_EXCLUSIVE
|
|
| STGM_DIRECT | STGM_CREATE ;
|
|
|
|
// Copy pstg into pstgNative, thereby removing slack and
|
|
// giving us access to the bits via an ILockBytes
|
|
if (FAILED(hr = CreateILockBytesOnHGlobal (NULL, FALSE, &plkbyt)))
|
|
{
|
|
break;
|
|
}
|
|
if (FAILED(hr = StgCreateDocfileOnILockBytes
|
|
(plkbyt, grfCreateStg, 0, &pstgNative)))
|
|
{
|
|
break;
|
|
}
|
|
if (FAILED(hr = pstg->CopyTo (0, NULL, 0, pstgNative)))
|
|
{
|
|
break;
|
|
}
|
|
|
|
|
|
// Set pdata->m_cbSize
|
|
STATSTG statstg;
|
|
if (FAILED(hr = plkbyt->Stat (&statstg, 0)))
|
|
{
|
|
break;
|
|
}
|
|
pdata->m_cbSize = statstg.cbSize.LowPart;
|
|
|
|
// Set pdata->m_h
|
|
if (FAILED(hr = GetHGlobalFromILockBytes (plkbyt, &(pdata->m_h))))
|
|
{
|
|
break;
|
|
}
|
|
Assert (GlobalSize (pdata->m_h) >= pdata->m_cbSize);
|
|
|
|
// Set pdata->m_pv
|
|
pdata->m_pv = GlobalLock (pdata->m_h);
|
|
if (NULL == pdata->m_pv)
|
|
{
|
|
hr = ResultFromScode(E_OUTOFMEMORY);
|
|
break;
|
|
}
|
|
} // end case
|
|
} // end switch
|
|
|
|
// Cleanup and return status to caller
|
|
if (pstm)
|
|
{
|
|
pstm->Release();
|
|
}
|
|
if (plkbyt)
|
|
{
|
|
plkbyt->Release();
|
|
}
|
|
if (pstgNative)
|
|
{
|
|
pstgNative->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PutPresentationObject, INTERNAL
|
|
//
|
|
// Synopsis: Writes a presentation to an OLE 1 stream.
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream to write to
|
|
// [ppres] -- the presentation object
|
|
// [cls] -- the class object
|
|
// [fStatic] -- flag: is this a static object
|
|
//
|
|
// Returns: NOERROR on success
|
|
// various possible I/O errors on failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PutPresentationObject(
|
|
LPOLESTREAM pos,
|
|
const PRES FAR* ppres,
|
|
const CLASS FAR& cls,
|
|
BOOL fStatic) // optional
|
|
{
|
|
HRESULT hr;
|
|
|
|
// Is there a real presentation?
|
|
|
|
BOOL fIsPres = FALSE;
|
|
if (ppres)
|
|
{
|
|
if (ppres->m_format.m_ftag != ftagClipFormat ||
|
|
ppres->m_format.m_cf != 0 )
|
|
{
|
|
fIsPres = TRUE;
|
|
}
|
|
}
|
|
|
|
// write the OLE version to the stream
|
|
if (FAILED(hr = ULToOLE1Stream (pos, dwVerToFile)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Calc format ID for presentation object, use 0 for no presentation
|
|
|
|
ULONG id = 0L;
|
|
|
|
if (fIsPres)
|
|
{
|
|
if (fStatic)
|
|
{
|
|
id = FMTID_STATIC;
|
|
}
|
|
else
|
|
{
|
|
id = FMTID_PRES;
|
|
}
|
|
}
|
|
if (FAILED(hr = ULToOLE1Stream(pos, id)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (!fIsPres)
|
|
{
|
|
// No presentation
|
|
return NOERROR;
|
|
}
|
|
|
|
if (IsStandardFormat (ppres->m_format))
|
|
{
|
|
return PutStandardPresentation (pos, ppres);
|
|
}
|
|
else
|
|
{
|
|
Assert (!fStatic);
|
|
return PutGenericPresentation (pos, ppres, cls.m_szClsid);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PutStandardPresentation, INTERNAL
|
|
//
|
|
// Synopsis: Writes a standard presentation (META, DIB, or BITMAP) out
|
|
// to an OLE 1 stream. Creates the METAFILEPICT header
|
|
// as required.
|
|
//
|
|
// Arguments: [pos] -- the OLE 1 stream to write to
|
|
// [ppres] -- the presentation to write
|
|
//
|
|
// Returns: NOERROR on success
|
|
// Various other errors are possible from I/O routines
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PutStandardPresentation(
|
|
LPOLESTREAM pos,
|
|
const PRES FAR* ppres)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
|
|
Assert (ppres->m_format.m_ftag == ftagClipFormat);
|
|
|
|
// Write the clipboard format string to the OLE 1 stream
|
|
// (Will be written in ANSI, not OLESTR format)
|
|
|
|
switch (ppres->m_format.m_cf)
|
|
{
|
|
case CF_METAFILEPICT:
|
|
if (FAILED(hr = StringToOLE1Stm (pos, OLESTR("METAFILEPICT"))))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
case CF_DIB:
|
|
if (FAILED(hr = StringToOLE1Stm (pos, OLESTR("DIB"))))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
case CF_BITMAP:
|
|
if (FAILED(hr = StringToOLE1Stm (pos, OLESTR("BITMAP"))))
|
|
{
|
|
return hr;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
Assert (0 && "Don't know how to write pres format");
|
|
}
|
|
|
|
// Write width
|
|
|
|
if (FAILED(hr = ULToOLE1Stream(pos, ppres->m_ulWidth)))
|
|
{
|
|
return hr;
|
|
}
|
|
// OLE 1.0 file format expects height to be saved as a negative value
|
|
if (FAILED(hr = ULToOLE1Stream(pos, - ((LONG)ppres->m_ulHeight))))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Do special handling for CF_METAFILEPICT
|
|
if (ppres->m_format.m_cf == CF_METAFILEPICT)
|
|
{
|
|
// Need a header to write, crete one here
|
|
|
|
WIN16METAFILEPICT mfpict =
|
|
{
|
|
MM_ANISOTROPIC,
|
|
(short) ppres->m_ulWidth,
|
|
(short) ppres->m_ulHeight,
|
|
0
|
|
};
|
|
|
|
// put size ater adjusting it for metafilepict
|
|
|
|
if (FAILED(hr = ULToOLE1Stream
|
|
(pos, (ppres->m_data.m_cbSize + sizeof(WIN16METAFILEPICT)))))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// put metafilepict
|
|
|
|
if (FAILED(hr = DataToOLE1Stm(pos, &mfpict, sizeof(mfpict))))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// put metafile bits
|
|
|
|
// There are two possible means by which we got these metafile
|
|
// bits: either we have an in-memory metafile, or raw bits
|
|
// which we read from disk. If it is an in-memory metafile,
|
|
// the m_pv ptr will have been set to METADATAPTR, and we need
|
|
// to extract the bits to our own buffer before saving them.
|
|
// If they came from disk, we can just re-write the buffer
|
|
// into which we read them.
|
|
|
|
if (METADATAPTR == ppres->m_data.m_pv)
|
|
{
|
|
BYTE *pb = (BYTE *) PrivMemAlloc(ppres->m_data.m_cbSize);
|
|
if (NULL == pb)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (0 == GetMetaFileBitsEx((HMETAFILE) ppres->m_data.m_h,
|
|
ppres->m_data.m_cbSize, pb))
|
|
{
|
|
PrivMemFree(pb);
|
|
return HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
|
|
if (FAILED(hr = DataToOLE1Stm(pos, pb, ppres->m_data.m_cbSize)))
|
|
{
|
|
PrivMemFree(pb);
|
|
return hr;
|
|
}
|
|
PrivMemFree(pb);
|
|
}
|
|
else // Bits were originally read into our buffer from disk
|
|
{
|
|
if (FAILED(hr = DataToOLE1Stm(pos, ppres->m_data.m_pv,
|
|
ppres->m_data.m_cbSize)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Not a METAFILE, just write the data
|
|
|
|
if (FAILED(hr = SizedDataToOLE1Stm (pos, ppres->m_data)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: PutGenericPresentation, INTERNAL
|
|
//
|
|
// Synopsis: Writes a generic presentation to the stream based on
|
|
// the clipboard format. (Dumps raw pres data to stm)
|
|
//
|
|
// Arguments: [pos] -- the stream to write to
|
|
// [ppres] -- the presentation
|
|
// [szClass] -- class name
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 16-Feb-94 davepl 32-bit port'n'doc
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL PutGenericPresentation(
|
|
LPOLESTREAM pos,
|
|
const PRES FAR* ppres,
|
|
LPCOLESTR szClass)
|
|
{
|
|
Assert (szClass);
|
|
HRESULT hr = NOERROR;
|
|
|
|
// Write the format class name out to the stream
|
|
|
|
if (FAILED(hr = StringToOLE1Stm(pos, szClass)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// This semi-mythical 0xC000 occurs in
|
|
// other code I've seen in this project also; if there's
|
|
// a constant defined, someone ought to fix this
|
|
|
|
if (ppres->m_format.m_ftag == ftagClipFormat)
|
|
{
|
|
if (ppres->m_format.m_cf < 0xc000)
|
|
{
|
|
if (FAILED(hr = ULToOLE1Stream (pos, ppres->m_format.m_cf)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (FAILED(hr = ULToOLE1Stream (pos, 0L)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
OLECHAR buf[256];
|
|
|
|
if (!GetClipboardFormatName(ppres->m_format.m_cf, buf,
|
|
sizeof(buf)/sizeof(OLECHAR)))
|
|
{
|
|
return ResultFromScode(DV_E_CLIPFORMAT);
|
|
}
|
|
|
|
if (FAILED(hr = StringToOLE1Stm (pos, buf)))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
}
|
|
else if (ppres->m_format.m_ftag == ftagString)
|
|
{
|
|
// Write the format string to the stream
|
|
|
|
if (FAILED(hr = ULToOLE1Stream (pos, 0L)))
|
|
{
|
|
return hr;
|
|
}
|
|
if (FAILED(hr = SizedDataToOLE1Stm
|
|
(pos, ppres->m_format.m_dataFormatString)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
AssertSz (0, "Bad format");
|
|
}
|
|
|
|
Assert (ppres->m_data.m_cbSize && ppres->m_data.m_h);
|
|
|
|
// Write the raw presentation data out
|
|
|
|
if (FAILED(hr = SizedDataToOLE1Stm (pos, ppres->m_data)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wClassesMatchW, INTERNAL INLINE
|
|
//
|
|
// Synopsis: Worker function to compare classes. Special case for
|
|
// handling when the class of the file cannot be determined
|
|
// because it is not a real file; this returns NOERROR
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
inline INTERNAL wClassesMatchW(REFCLSID clsidIn, LPOLESTR szFile)
|
|
{
|
|
CLSID clsid;
|
|
|
|
// If we can get the CLSID for the code that works with this file,
|
|
// compare it to the CLSID passed in, and return the result of
|
|
// that comparison
|
|
|
|
if (NOERROR==GetClassFile (szFile, &clsid))
|
|
{
|
|
if (IsEqualCLSID(clsid, clsidIn))
|
|
{
|
|
return NOERROR;
|
|
}
|
|
else
|
|
{
|
|
return ResultFromScode(S_FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If we can't determine the class of the file (because it's
|
|
// not a real file) then OK.
|
|
// Bug 3937.
|
|
|
|
return NOERROR;
|
|
}
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: MonikerIntoGenObj, INTERNAL
|
|
//
|
|
// Synopsis: Merges an OLE 2.0 moniker into a generic object
|
|
//
|
|
// Effects: Sets ths Topic, Item, and class members
|
|
//
|
|
// Arguments: [pgenobj] -- the generic object to receive moniker
|
|
// [clsidLast] -- if a link, what its a link to
|
|
// [pmk] -- the moniker to merge in
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Code cleanup
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
static INTERNAL MonikerIntoGenObj(
|
|
PGENOBJ pgenobj,
|
|
REFCLSID clsidLast,
|
|
LPMONIKER pmk )
|
|
{
|
|
LPOLESTR szFile=NULL;
|
|
LPOLESTR szItem=NULL;
|
|
BOOL fClassesMatch = FALSE;
|
|
|
|
// If the classes match, that implies this is a link to a pseudo-object
|
|
// not to an embedded object. If GetClassFile fails because the file
|
|
// does not exist or is unsaved then we give the link the benefit
|
|
// of the doubt and let it stay a link. Only if we know the
|
|
// classes do NOT match do we change the link into an Ole2Link
|
|
// embedded object.
|
|
|
|
// Ole10_PareMoniker returns S_FALSE in the FileMoniker - ItemMoniker - ItemMoniker... case
|
|
// so check for NOERROR explicitly.
|
|
if (NOERROR == Ole10_ParseMoniker (pmk, &szFile, &szItem))
|
|
{
|
|
if (szFile)
|
|
{
|
|
SCODE sc = GetScode(wClassesMatchW(clsidLast, szFile));
|
|
if (sc == S_OK || sc == MK_E_CANTOPENFILE)
|
|
{
|
|
pgenobj->m_szTopic = szFile;
|
|
pgenobj->m_szItem = szItem;
|
|
fClassesMatch = TRUE;
|
|
}
|
|
}
|
|
}
|
|
if (FALSE == fClassesMatch)
|
|
{
|
|
// This moniker is either not a File or File::Item moniker,
|
|
// or is a link to an embedded object, so the only
|
|
// way we can convert it to OLE 1.0 is to make it an opaque Ole2Link
|
|
|
|
pgenobj->m_fLink = FALSE;
|
|
pgenobj->m_class.Reset (CLSID_StdOleLink);
|
|
}
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OleConvertIStorageToOLESTREAMEx, STDAPI
|
|
//
|
|
// Synopsis: Similar to OleConvertIStorageToOLESTREAM, except that the
|
|
// presentation data that needs to be written into OLESTREAM
|
|
// is passed in. pmedium->tymed can only be TYMED_HGLOBAL
|
|
// or TYMED_ISTREAM and the medium will not be released by the
|
|
// api. cfFormat can be NULL, If it is NULL then the other
|
|
// parameters (lWidth, lHeight, dwSize, pmedium) will be ignored.
|
|
//
|
|
// Arguments: [pstg] -- the storage object to convert from
|
|
// [cfFormat] -- clipboard format
|
|
// [lWidth] -- width
|
|
// [lHeight] -- height
|
|
// [dwSize] -- size in bytes
|
|
// [pmedium] -- serialized bytes
|
|
// [polestm] -- the OLE 1 stream to write to
|
|
//
|
|
// Returns: NOERROR on success
|
|
// DV_E_TYMED invalid clipboard format
|
|
// E_INVALIDARG invalid arg, normally stg or stm
|
|
// DV_E_STGMEDIUM bad medium ptr
|
|
// E_OUTOFMEMORY allocation failure
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleaned up and documented
|
|
//
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
STDAPI OleConvertIStorageToOLESTREAMEx
|
|
(
|
|
LPSTORAGE pstg,
|
|
CLIPFORMAT cfFormat,
|
|
LONG lWidth,
|
|
LONG lHeight,
|
|
DWORD dwSize,
|
|
LPSTGMEDIUM pmedium,
|
|
LPOLESTREAM polestm
|
|
)
|
|
{
|
|
|
|
OLETRACEIN((API_OleConvertIStorageToOLESTREAMEx,
|
|
PARAMFMT("pstg= %p, cfFormat= %x, lWidth= %d, lHeight= %d, dwSize= %ud, pmedium= %ts, polestm= %p"),
|
|
pstg, cfFormat, lWidth, lHeight, dwSize, pmedium, polestm));
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p _IN OleConvertIStorageToOLESTREAMEx ("
|
|
" %p, %x , %lx , %lx , %x , %p , %p )\n", 0 /*function*/,
|
|
pstg, cfFormat, lWidth, lHeight, dwSize, pmedium, polestm
|
|
));
|
|
CALLHOOKOBJECT(S_OK,CLSID_NULL,IID_IStorage,(IUnknown **)&pstg);
|
|
|
|
HGLOBAL hGlobal = NULL;
|
|
HRESULT hr = NOERROR;
|
|
BOOL fFree = FALSE;
|
|
CGenericObject genobj;
|
|
|
|
// If we are given a clipboard format...
|
|
|
|
if (cfFormat) {
|
|
|
|
VDATEPTRIN_LABEL(pmedium, STGMEDIUM, errRtn, hr);
|
|
|
|
// Check that the medium ptr is valid
|
|
if (pmedium->hGlobal == NULL)
|
|
{
|
|
hr = ResultFromScode(DV_E_STGMEDIUM);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Cannot have a 0 sized clipboard representation
|
|
if (dwSize == 0)
|
|
{
|
|
hr = ResultFromScode(E_INVALIDARG);
|
|
goto errRtn;
|
|
}
|
|
|
|
switch (pmedium->tymed)
|
|
{
|
|
case TYMED_HGLOBAL:
|
|
hGlobal = pmedium->hGlobal;
|
|
break;
|
|
|
|
case TYMED_ISTREAM:
|
|
VDATEIFACE_LABEL(pmedium->pstm, errRtn, hr);
|
|
if ((hr = UtGetHGLOBALFromStm(pmedium->pstm, dwSize,
|
|
&hGlobal)) != NOERROR)
|
|
{
|
|
goto errRtn;
|
|
}
|
|
fFree = TRUE;
|
|
break;
|
|
|
|
default:
|
|
hr = ResultFromScode(DV_E_TYMED);
|
|
goto errRtn;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr = wConvertIStorageToOLESTREAM(pstg, polestm, &genobj)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Clean m_ppres
|
|
if (genobj.m_ppres)
|
|
{
|
|
delete genobj.m_ppres;
|
|
genobj.m_ppres = NULL;
|
|
}
|
|
|
|
if (cfFormat)
|
|
{
|
|
// fill genobj.m_ppres
|
|
|
|
PPRES ppres;
|
|
|
|
if ((genobj.m_ppres = ppres = new PRES) == NULL)
|
|
{
|
|
hr = ResultFromScode(E_OUTOFMEMORY);
|
|
goto errRtn;
|
|
}
|
|
|
|
ppres->m_ulWidth = (ULONG) lWidth;
|
|
ppres->m_ulHeight = (ULONG) lHeight;
|
|
ppres->m_data.m_cbSize = dwSize;
|
|
ppres->m_data.m_fNoFree = !fFree;
|
|
ppres->m_data.m_h = hGlobal;
|
|
ppres->m_data.m_pv = GlobalLock(hGlobal);
|
|
ppres->m_format.m_ftag = ftagClipFormat;
|
|
ppres->m_format.m_cf = cfFormat;
|
|
|
|
}
|
|
else
|
|
{
|
|
genobj.m_fNoBlankPres = TRUE;
|
|
}
|
|
|
|
// REVIEW: We may not want to allow NULL cfFormat with static object
|
|
|
|
hr = GenericObjectToOLESTREAM (genobj, polestm);
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p OUT OleConvertIStorageToOLESTREAMEx ( %lx ) "
|
|
"\n", 0 /*function*/, hr));
|
|
|
|
OLETRACEOUT((API_OleConvertIStorageToOLESTREAMEx, hr));
|
|
|
|
return hr;
|
|
|
|
errRtn:
|
|
|
|
if (fFree && hGlobal != NULL)
|
|
{
|
|
GlobalFree(hGlobal);
|
|
}
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p OUT OleConvertIStorageToOLESTREAMEx ( %lx ) "
|
|
"\n", 0 /*function*/, hr));
|
|
|
|
OLETRACEOUT((API_OleConvertIStorageToOLESTREAMEx, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OleConvertOLESTREAMToIStorageEx, STDAPI
|
|
//
|
|
// Synopsis: Similar to OleConvertOLESTREAMToIStorage, except that the
|
|
// presentation data that is read from OLESTREAM is passed out.
|
|
// And no presentation stream will written in to the storage.
|
|
// pmedium->tymed can be TYMED_ISTREAM ot TYMED_NULL. If
|
|
// TYMED_NULL, then the bits will be returned in a global
|
|
// handle through pmedium->hGlobal. Otherwise data will be
|
|
// written into pmedium->pstm. NULL will be returned through
|
|
// *pcfFormat, if there is no presentation in the OLESTREAM.
|
|
//
|
|
// Arguments: [pstg] -- the storage object to convert to
|
|
// [cfFormat] -- clipboard format
|
|
// [lWidth] -- width
|
|
// [lHeight] -- height
|
|
// [dwSize] -- size in bytes
|
|
// [pmedium] -- serialized bytes
|
|
// [polestm] -- the OLE 1 stream to write from
|
|
//
|
|
// Returns: DV_E_TYMED invalid clipboard format
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
|
|
STDAPI OleConvertOLESTREAMToIStorageEx
|
|
(
|
|
LPOLESTREAM polestm,
|
|
LPSTORAGE pstg,
|
|
CLIPFORMAT FAR* pcfFormat,
|
|
LONG FAR* plWidth,
|
|
LONG FAR* plHeight,
|
|
DWORD FAR* pdwSize,
|
|
LPSTGMEDIUM pmedium
|
|
)
|
|
{
|
|
OLETRACEIN((API_OleConvertOLESTREAMToIStorageEx,
|
|
PARAMFMT("polestm= %p, pstg= %p, pcfFormat= %p, plWidth= %p, plHeight= %p, pdwSize= %p, pmedium= %p"),
|
|
polestm, pstg, pcfFormat, plWidth, plHeight, pdwSize, pmedium));
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p _IN OleConvertOLESTREAMToIStorageEx ("
|
|
" %p , %p , %p , %p , %p , %p , %p )\n", 0 /*function*/,
|
|
polestm, pstg, pcfFormat,plWidth,plHeight,pdwSize,pmedium
|
|
));
|
|
|
|
HRESULT hr;
|
|
PPRES ppres = NULL;
|
|
GENOBJ genobj;
|
|
|
|
VDATEPTROUT_LABEL(pcfFormat, CLIPFORMAT, errRtn, hr);
|
|
VDATEPTROUT_LABEL(plWidth, LONG, errRtn, hr);
|
|
VDATEPTROUT_LABEL(plHeight, LONG, errRtn, hr);
|
|
VDATEPTROUT_LABEL(pdwSize, DWORD, errRtn, hr);
|
|
VDATEPTROUT_LABEL(pmedium, STGMEDIUM, errRtn, hr);
|
|
|
|
CALLHOOKOBJECT(S_OK,CLSID_NULL,IID_IStorage,(IUnknown **)&pstg);
|
|
|
|
if (pmedium->tymed == TYMED_ISTREAM)
|
|
{
|
|
VDATEIFACE_LABEL(pmedium->pstm, errRtn, hr);
|
|
}
|
|
else if (pmedium->tymed != TYMED_NULL)
|
|
{
|
|
hr = ResultFromScode(DV_E_TYMED);
|
|
goto errRtn;
|
|
}
|
|
|
|
// Bring the object into genobj
|
|
|
|
if (FAILED((hr = wConvertOLESTREAMToIStorage(polestm, pstg, &genobj))))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
ppres = genobj.m_ppres;
|
|
genobj.m_ppres = NULL;
|
|
|
|
if (FAILED(hr = GenericObjectToIStorage (genobj, pstg, NULL)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// If no presentation is available, clear our all the pres
|
|
// dimensions and format
|
|
|
|
if (ppres == NULL)
|
|
{
|
|
*pcfFormat = 0;
|
|
*plWidth = 0L;
|
|
*plHeight = 0L;
|
|
*pdwSize = 0L;
|
|
|
|
// Don't worry about the pmedium, it is already in the proper state
|
|
|
|
hr = NOERROR;
|
|
goto errRtn;
|
|
}
|
|
|
|
// If we reach here, we have a presentation, so set the OUT
|
|
// parameters accordingly
|
|
|
|
*plWidth = (LONG) ppres->m_ulWidth;
|
|
*plHeight = (LONG) ppres->m_ulHeight;
|
|
*pdwSize = ppres->m_data.m_cbSize;
|
|
|
|
Assert(ppres->m_format.m_ftag != ftagNone);
|
|
|
|
// If we have a clipboard format ID, return that in the OUT paramter,
|
|
// otherwise return whatever we get back from an attempt to register
|
|
// the format string
|
|
|
|
if (ppres->m_format.m_ftag == ftagClipFormat)
|
|
{
|
|
*pcfFormat = ppres->m_format.m_cf;
|
|
}
|
|
else
|
|
{
|
|
// m_dataFormatString is an ASCII string.
|
|
*pcfFormat = (CLIPFORMAT) SSRegisterClipboardFormatA( (LPCSTR) ppres->m_format.m_dataFormatString.m_pv);
|
|
Assert(0 != *pcfFormat);
|
|
}
|
|
|
|
if (pmedium->tymed == TYMED_NULL)
|
|
{
|
|
if (ppres->m_data.m_h)
|
|
{
|
|
Assert(ppres->m_data.m_pv != NULL);
|
|
GlobalUnlock(ppres->m_data.m_h);
|
|
}
|
|
|
|
// transfer the ownership
|
|
pmedium->tymed = TYMED_HGLOBAL;
|
|
pmedium->hGlobal = ppres->m_data.m_h;
|
|
|
|
// Null out the handle and pointer so that destructor of PRES will not
|
|
// free it.
|
|
ppres->m_data.m_h = NULL;
|
|
ppres->m_data.m_pv = NULL;
|
|
|
|
}
|
|
else
|
|
{
|
|
hr = pmedium->pstm->Write(ppres->m_data.m_pv, *pdwSize, NULL);
|
|
}
|
|
|
|
errRtn:
|
|
|
|
if (ppres)
|
|
{
|
|
delete ppres;
|
|
}
|
|
|
|
LEDebugOut((DEB_ITRACE, "%p OUT OleConvertOLESTREAMToIStorageEx ( %lx ) "
|
|
"\n", 0 /*function*/, hr));
|
|
|
|
OLETRACEOUT((API_OleConvertOLESTREAMToIStorageEx, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wWriteFmtUserType, INTERNAL
|
|
//
|
|
// Synopsis: Gets the user type for a class ID and writes it to
|
|
// an IStorage
|
|
//
|
|
//
|
|
// Arguments: [pstg] -- the storage to write to
|
|
// [clsid] -- the class ID
|
|
//
|
|
//
|
|
// Returns: NOERROR on success
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 21-Feb-94 davepl Cleaned up and documented
|
|
// Notes:
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
FARINTERNAL wWriteFmtUserType(LPSTORAGE pstg, REFCLSID clsid)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
LPOLESTR szProgID = NULL;
|
|
LPOLESTR szUserType = NULL;
|
|
|
|
// Get the program ID
|
|
if (FAILED(hr = ProgIDFromCLSID (clsid, &szProgID)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Get the user type
|
|
if (FAILED(hr = OleRegGetUserType(clsid,USERCLASSTYPE_FULL,&szUserType)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Write the user type out to the storage
|
|
if (FAILED(hr = WriteFmtUserTypeStg
|
|
(pstg, (CLIPFORMAT) RegisterClipboardFormat (szProgID), szUserType)))
|
|
{
|
|
goto errRtn;
|
|
}
|
|
|
|
// Clean up and return status
|
|
|
|
errRtn:
|
|
|
|
if (szProgID)
|
|
{
|
|
PubMemFree(szProgID);
|
|
}
|
|
if (szUserType)
|
|
{
|
|
PubMemFree(szUserType);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wCLSIDFromProgID
|
|
//
|
|
// Synopsis: Looks for the key HKEY_CLASSES_ROOT\{ProgID}\Clsid\ to get
|
|
// the string version of the class ID, then returns the CLSID
|
|
// value of whatever it found.
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 25-Jun-94 alexgo fixed Ole1 CLSID creation
|
|
// 15-Apr-94 davepl Rewrite
|
|
//
|
|
// Notes: Used to be in clipboard code, but used in this file
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
INTERNAL wCLSIDFromProgID(LPOLESTR szProgID, LPCLSID pclsid, BOOL fForceAssign)
|
|
{
|
|
VDATEHEAP();
|
|
|
|
// Apparently some optimization. If the class name is "OLE2Link", we can
|
|
// return CLSID_StdOleLInk without even bothering to check the registry.
|
|
|
|
if (0 == _xstrcmp(szProgID, OLESTR("OLE2Link")))
|
|
{
|
|
*pclsid = CLSID_StdOleLink;
|
|
return NOERROR;
|
|
}
|
|
else
|
|
{
|
|
// this function will look for a CLSID under the ProgID entry in
|
|
// the registry or manufacture one if none present.
|
|
|
|
return CLSIDFromOle1Class(szProgID, pclsid, fForceAssign);
|
|
}
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: wProgIDFromCLSID
|
|
//
|
|
// Synopsis: A wrapper for ProgIDFromCLSID. The only change in
|
|
// functionality is to check and see if this is a
|
|
// CLSID_StdOleLink, and if so, return a prog ID of
|
|
// "OLE2Link" rather than failing.
|
|
//
|
|
//
|
|
// History: dd-mmm-yy Author Comment
|
|
// 15-Feb-94 davepl Rewrite
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
FARINTERNAL wProgIDFromCLSID(REFCLSID clsid, LPOLESTR FAR* psz)
|
|
{
|
|
VDATEHEAP();
|
|
|
|
HRESULT hresult;
|
|
|
|
// If we can get the ProgID by conventional methods, great, just
|
|
// return it.
|
|
|
|
if (NOERROR == (hresult = ProgIDFromCLSID(clsid, psz)))
|
|
{
|
|
return hresult;
|
|
}
|
|
|
|
// If we failed, it might be because this is a standard OLE link, which
|
|
// will not have a ProgID entry in the registry, so we fake it out by
|
|
// returning the ProgID manually.
|
|
|
|
if (IsEqualCLSID(clsid, CLSID_StdOleLink))
|
|
{
|
|
*psz = UtDupString(OLESTR("OLE2Link"));
|
|
|
|
if (*psz == NULL)
|
|
{
|
|
hresult = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
hresult = NOERROR;
|
|
}
|
|
}
|
|
|
|
// Must not have been able to resolve for ProgID, so return the error.
|
|
return(hresult);
|
|
}
|
|
|
|
|
|
#if 0
|
|
|
|
|
|
// We don't need these conversion fns yet, but we likely will soon.
|
|
|
|
inline INTERNAL_(VOID) ConvertBM32to16(LPBITMAP lpsrc, LPWIN16BITMAP lpdest)
|
|
{
|
|
lpdest->bmType = (short)lpsrc->bmType;
|
|
lpdest->bmWidth = (short)lpsrc->bmWidth;
|
|
lpdest->bmHeight = (short)lpsrc->bmHeight;
|
|
lpdest->bmWidthBytes = (short)lpsrc->bmWidthBytes;
|
|
lpdest->bmPlanes = (BYTE)lpsrc->bmPlanes;
|
|
lpdest->bmBitsPixel = (BYTE)lpsrc->bmBitsPixel;
|
|
}
|
|
|
|
inline INTERNAL_(VOID) ConvertBM16to32(LPWIN16BITMAP lpsrc, LPBITMAP lpdest)
|
|
{
|
|
lpdest->bmType = MAKELONG(lpsrc->bmType,NULL_WORD);
|
|
lpdest->bmWidth = MAKELONG(lpsrc->bmWidth,NULL_WORD);
|
|
lpdest->bmHeight = MAKELONG(lpsrc->bmHeight,NULL_WORD);
|
|
lpdest->bmWidthBytes = MAKELONG(lpsrc->bmWidthBytes,NULL_WORD);
|
|
lpdest->bmPlanes = (WORD)lpsrc->bmPlanes;
|
|
lpdest->bmBitsPixel = (WORD)lpsrc->bmBitsPixel;
|
|
}
|
|
|
|
inline INTERNAL_(VOID) ConvertMF16to32(
|
|
LPWIN16METAFILEPICT lpsrc,
|
|
LPMETAFILEPICT lpdest )
|
|
{
|
|
lpdest->mm = (DWORD)lpsrc->mm;
|
|
lpdest->xExt = (DWORD)MAKELONG(lpsrc->xExt,NULL_WORD);
|
|
lpdest->yExt = (DWORD)MAKELONG(lpsrc->yExt,NULL_WORD);
|
|
}
|
|
|
|
inline INTERNAL_(VOID) ConvertMF32to16(
|
|
LPMETAFILEPICT lpsrc,
|
|
LPWIN16METAFILEPICT lpdest )
|
|
{
|
|
lpdest->mm = (short)lpsrc->mm;
|
|
lpdest->xExt = (short)lpsrc->xExt;
|
|
lpdest->yExt = (short)lpsrc->yExt;
|
|
}
|
|
|
|
#endif
|