2020-09-30 17:12:32 +02:00

1404 lines
40 KiB
C++

#include "priv.h"
#include "privcpp.h"
#define CPP_FUNCTIONS
// #include <crtfree.h>
UINT g_cfFileContents;
UINT g_cfFileDescriptor;
UINT g_cfObjectDescriptor;
UINT g_cfEmbedSource;
UINT g_cfFileNameW;
INT g_cxIcon;
INT g_cyIcon;
INT g_cxArrange;
INT g_cyArrange;
HFONT g_hfontTitle;
static TCHAR szUserType[] = TEXT("Package");
static TCHAR szDefTempFile[] = TEXT("PKG");
CPackage::CPackage() :
_cRef(1)
{
DebugMsg(DM_TRACE, "pack - CPackage() called.");
g_cRefThisDll++;
ASSERT(_cf == 0);
ASSERT(_panetype == NOTHING);
ASSERT(_pEmbed == NULL);
ASSERT(_pCml == NULL);
ASSERT(_fLoaded == FALSE);
ASSERT(_lpszContainerApp == NULL);
ASSERT(_lpszContainerObj == NULL);
ASSERT(_fIsDirty == FALSE);
ASSERT(_pIStorage == NULL);
ASSERT(_pstmFileContents == NULL);
ASSERT(_pstm == NULL);
ASSERT(_pIPersistStorage == NULL);
ASSERT(_pIDataObject == NULL);
ASSERT(_pIOleObject == NULL);
ASSERT(_pIViewObject2 == NULL);
ASSERT(_pIAdviseSink == NULL);
ASSERT(_pIRunnableObject == NULL);
ASSERT(_pIDataAdviseHolder == NULL);
ASSERT(_pIOleAdviseHolder == NULL);
ASSERT(_pIOleClientSite == NULL);
ASSERT(_pViewSink == NULL);
ASSERT(_dwViewAspects == 0);
ASSERT(_dwViewAdvf == 0);
ASSERT(_cVerbs == 0);
ASSERT(_nCurVerb == 0);
ASSERT(_pVerbs == NULL);
ASSERT(_pcm == NULL);
}
CPackage::~CPackage()
{
DebugMsg(DM_TRACE, "pack - ~CPackage() called.");
// We should never be destroyed unless our ref count is zero.
ASSERT(_cRef == 0);
g_cRefThisDll--;
// Destroy our interfaces...
delete _pIOleObject;
delete _pIViewObject2;
delete _pIDataObject;
delete _pIPersistStorage;
delete _pIAdviseSink;
delete _pIRunnableObject;
// Destroy the packaged file structure...
DestroyIC();
// we destroy depending on which type of object we had packaged
switch (_panetype) {
case PEMBED:
if (_pEmbed->pszTempName) {
DeleteFile(_pEmbed->pszTempName);
delete _pEmbed->pszTempName;
}
delete _pEmbed;
break;
case CMDLINK:
delete _pCml;
break;
}
// Release Advise pointers...
if (_pIDataAdviseHolder)
_pIDataAdviseHolder->Release();
if (_pIOleAdviseHolder)
_pIOleAdviseHolder->Release();
if (_pIOleClientSite)
_pIOleClientSite->Release();
// Release Storage pointers...
if (_pIStorage)
_pIStorage->Release();
if (_pstmFileContents)
_pstmFileContents->Release();
if (_pstm)
_pstm->Release();
delete _lpszContainerApp;
delete _lpszContainerObj;
ReleaseContextMenu();
if (NULL != _pVerbs) {
for (ULONG i = 0; i < _cVerbs; i++) {
delete _pVerbs[i].lpszVerbName;
}
delete _pVerbs;
}
DebugMsg(DM_TRACE, "CPackage being destroyed. _cRef == %d", _cRef);
}
HRESULT CPackage::Init()
{
// initializes parts of a package object that have a potential to fail
// return: S_OK -- everything initialized
// E_FAIL -- error in initialzation
// E_OUTOFMEMORY -- out of memory
DebugMsg(DM_TRACE, "pack - Init() called.");
// Initialize all the interfaces...
if (!(_pIOleObject = new CPackage_IOleObject(this)))
return E_OUTOFMEMORY;
if (!(_pIViewObject2 = new CPackage_IViewObject2(this)))
return E_OUTOFMEMORY;
if (!(_pIDataObject = new CPackage_IDataObject(this)))
return E_OUTOFMEMORY;
if (!(_pIPersistStorage = new CPackage_IPersistStorage(this)))
return E_OUTOFMEMORY;
if (!(_pIPersistFile = new CPackage_IPersistFile(this)))
return E_OUTOFMEMORY;
if (!(_pIAdviseSink = new CPackage_IAdviseSink(this)))
return E_OUTOFMEMORY;
if (!(_pIRunnableObject = new CPackage_IRunnableObject(this)))
return E_OUTOFMEMORY;
// Get some system metrics that we'll need later...
LOGFONT lf;
SystemParametersInfo(SPI_GETICONTITLELOGFONT, sizeof(lf), &lf, FALSE);
SystemParametersInfo(SPI_ICONHORIZONTALSPACING, 0, &g_cxArrange, FALSE);
SystemParametersInfo(SPI_ICONVERTICALSPACING, 0, &g_cyArrange, FALSE);
g_cxIcon = GetSystemMetrics(SM_CXICON);
g_cyIcon = GetSystemMetrics(SM_CYICON);
g_hfontTitle = CreateFontIndirect(&lf);
// register some clipboard formats that we support...
g_cfFileContents = RegisterClipboardFormat(CFSTR_FILECONTENTS);
g_cfFileDescriptor = RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR);
g_cfObjectDescriptor = RegisterClipboardFormat(CFSTR_OBJECTDESCRIPTOR);
g_cfEmbedSource = RegisterClipboardFormat(CFSTR_EMBEDSOURCE);
g_cfFileNameW = RegisterClipboardFormat(TEXT("FileNameW"));
// Initialize a generic icon
_lpic = IconCreate();
IconRefresh();
return S_OK;
}
// IUnknown Methods...
HRESULT CPackage::QueryInterface(REFIID iid, void** ppvObj)
{
DebugMsg(DM_TRACE, "pack - QueryInterface() called.");
if (iid == IID_IUnknown) {
DebugMsg(DM_TRACE, " getting IID_IUnknown");
*ppvObj = (void*)this;
} else if (iid == IID_IOleObject) {
DebugMsg(DM_TRACE, " getting IID_IOleObject");
*ppvObj = (void*)_pIOleObject;
} else if ((iid == IID_IViewObject2) || (iid == IID_IViewObject)) {
DebugMsg(DM_TRACE, " getting IID_IViewObject");
*ppvObj = (void*)_pIViewObject2;
} else if (iid == IID_IDataObject) {
DebugMsg(DM_TRACE, " getting IID_IDataObject");
*ppvObj = (void*)_pIDataObject;
} else if ((iid == IID_IPersistStorage) || (iid == IID_IPersist)) {
DebugMsg(DM_TRACE, " getting IID_IPersistStorage");
*ppvObj = (void*)_pIPersistStorage;
} else if (iid == IID_IPersistFile) {
DebugMsg(DM_TRACE, " getting IID_IPersistFile");
*ppvObj = (void*)_pIPersistFile;
} else if (iid == IID_IAdviseSink) {
DebugMsg(DM_TRACE, " getting IID_IAdviseSink");
*ppvObj = (void*)_pIAdviseSink;
} else if (iid == IID_IRunnableObject) {
DebugMsg(DM_TRACE, " getting IID_IRunnableObject");
*ppvObj = (void*)_pIRunnableObject;
} else if (iid == IID_IEnumOLEVERB) {
DebugMsg(DM_TRACE, " getting IID_IEnumOLEVERB");
*ppvObj = (IEnumOLEVERB*)this;
} else {
*ppvObj = NULL;
return E_NOINTERFACE;
}
((LPUNKNOWN)*ppvObj)->AddRef();
return S_OK;
}
ULONG CPackage::AddRef()
{
_cRef++;
return _cRef;
}
ULONG CPackage::Release()
{
_cRef--;
if (_cRef > 0)
return _cRef;
delete this;
return 0;
}
HRESULT CPackage_CreateInstnace(LPUNKNOWN* ppunk)
{
DebugMsg(DM_TRACE, "pack - CreateInstance called");
*ppunk = NULL; // null the out param
CPackage* pPack = new CPackage;
if (!pPack)
return E_OUTOFMEMORY;
if (FAILED(pPack->Init())) {
delete pPack;
return E_OUTOFMEMORY;
}
*ppunk = pPack;
return S_OK;
}
STDMETHODIMP CPackage::Next(ULONG celt, OLEVERB* rgVerbs, ULONG* pceltFetched)
{
HRESULT hr;
if (NULL != rgVerbs) {
if (1 == celt) {
if (_nCurVerb < _cVerbs) {
ASSERT(NULL != _pVerbs);
*rgVerbs = _pVerbs[_nCurVerb];
if ((NULL != _pVerbs[_nCurVerb].lpszVerbName) &&
(NULL != (rgVerbs->lpszVerbName = (LPWSTR)CoTaskMemAlloc(
(lstrlenW(_pVerbs[_nCurVerb].lpszVerbName) + 1) * SIZEOF(WCHAR))))) {
StrCpyW(rgVerbs->lpszVerbName, _pVerbs[_nCurVerb].lpszVerbName);
}
_nCurVerb++;
hr = S_OK;
} else {
hr = S_FALSE;
}
if (NULL != pceltFetched) {
*pceltFetched = (S_OK == hr) ? 1 : 0;
}
} else if (NULL != pceltFetched) {
int cVerbsToCopy = min(celt, _cVerbs - _nCurVerb);
if (cVerbsToCopy > 0) {
ASSERT(NULL != _pVerbs);
CopyMemory(rgVerbs, &(_pVerbs[_nCurVerb]), cVerbsToCopy * sizeof(OLEVERB));
for (int i = 0; i < cVerbsToCopy; i++) {
if ((NULL != _pVerbs[_nCurVerb + i].lpszVerbName) &&
(NULL != (rgVerbs[i].lpszVerbName = (LPWSTR)CoTaskMemAlloc(
(lstrlenW(_pVerbs[_nCurVerb + i].lpszVerbName) + 1) * SIZEOF(WCHAR))))) {
StrCpyW(rgVerbs[i].lpszVerbName, _pVerbs[_nCurVerb + i].lpszVerbName);
}
}
_nCurVerb += cVerbsToCopy;
}
*pceltFetched = (ULONG)cVerbsToCopy;
hr = (celt == (ULONG)cVerbsToCopy) ? S_OK : S_FALSE;
} else {
hr = E_INVALIDARG;
}
} else {
hr = E_INVALIDARG;
}
return hr;
}
STDMETHODIMP CPackage::Skip(ULONG celt)
{
if (_nCurVerb + celt > _cVerbs) {
// there aren't enough elements, go to the end and return S_FALSE
_nCurVerb = _cVerbs;
return S_FALSE;
} else {
_nCurVerb += celt;
return S_OK;
}
}
STDMETHODIMP CPackage::Reset()
{
_nCurVerb = 0;
return S_OK;
}
STDMETHODIMP CPackage::Clone(IEnumOLEVERB** ppEnum)
{
if (NULL != ppEnum) {
*ppEnum = NULL;
}
return E_NOTIMPL;
}
// Package helper functions
HRESULT CPackage::EmbedInitFromFile(LPTSTR lpFileName, BOOL fInitFile)
{
// get's the file size of the packaged file and set's the name
// of the packaged file if fInitFile == TRUE.
// return: S_OK -- initialized ok
// E_FAIL -- error initializing file
DWORD dwSize;
// if this is the first time we've been called, then we need to allocate
// memory for the _pEmbed structure
if (_pEmbed == NULL) {
_pEmbed = new EMBED;
if (_pEmbed) {
_pEmbed->pszTempName = NULL;
_pEmbed->hTask = NULL;
_pEmbed->poo = NULL;
_pEmbed->fIsOleFile = TRUE;
}
}
if (_pEmbed == NULL)
return E_OUTOFMEMORY;
// open the file to package...
HANDLE fh = CreateFile(lpFileName, GENERIC_READ, FILE_SHARE_READWRITE,
NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL);
if (fh == INVALID_HANDLE_VALUE) {
DWORD dwError = GetLastError();
return E_FAIL;
}
_panetype = PEMBED;
// Get the size of the file
_pEmbed->fd.nFileSizeLow = GetFileSize(fh, &dwSize);
if (_pEmbed->fd.nFileSizeLow == 0xFFFFFFFF) {
DWORD dwError = GetLastError();
return E_FAIL;
}
ASSERT(dwSize == 0);
_pEmbed->fd.nFileSizeHigh = 0L;
_pEmbed->fd.dwFlags = FD_FILESIZE;
// We only want to set the filename if this is the file to be packaged.
// If it's only a temp file that we're reloading (fInitFile == FALSE) then
// don't bother setting the filename.
if (fInitFile) {
lstrcpy(_pEmbed->fd.cFileName, lpFileName);
DestroyIC();
_lpic = IconCreateFromFile(lpFileName);
if (_pIDataAdviseHolder)
_pIDataAdviseHolder->SendOnDataChange(_pIDataObject, 0, NULL);
if (_pViewSink)
_pViewSink->OnViewChange(_dwViewAspects, _dwViewAdvf);
}
_fIsDirty = TRUE;
CloseHandle(fh);
return S_OK;
}
HRESULT CPackage::CmlInitFromFile(LPTSTR lpFileName, BOOL fUpdateIcon)
{
// if this is the first time we've been called, then we need to allocate
// memory for the _pCml structure
if (_pCml == NULL) {
_pCml = new CML;
if (_pCml) {
// we don't use this, but an old packager accessing us might.
_pCml->fCmdIsLink = FALSE;
}
}
if (_pCml == NULL)
return E_OUTOFMEMORY;
_panetype = CMDLINK;
lstrcpy(_pCml->szCommandLine, lpFileName);
_fIsDirty = TRUE;
if (fUpdateIcon) {
DestroyIC();
_lpic = IconCreateFromFile(lpFileName);
if (_pIDataAdviseHolder)
_pIDataAdviseHolder->SendOnDataChange(_pIDataObject, 0, NULL);
if (_pViewSink)
_pViewSink->OnViewChange(_dwViewAspects, _dwViewAdvf);
}
return S_OK;
}
HRESULT CPackage::InitFromPackInfo(LPPACKAGER_INFO lppi)
{
HRESULT hr;
DWORD dwFileAttributes = GetFileAttributes(lppi->szFilename);
// Ok, we need to test whether the user tried to package a folder
// instead of a file. If s/he did, then we'll just create a link
// to that folder instead of an embedded file.
if ((dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == FILE_ATTRIBUTE_DIRECTORY) {
hr = CmlInitFromFile(lppi->szFilename, FALSE);
} else {
// we pass FALSE here, because we don't want to write the icon
// information.
hr = EmbedInitFromFile(lppi->szFilename, FALSE);
lstrcpy(_pEmbed->fd.cFileName, lppi->szFilename);
_panetype = PEMBED;
}
// set the icon information
lstrcpy(_lpic->szIconPath, lppi->szIconPath);
_lpic->iDlgIcon = lppi->iIcon;
lstrcpy(_lpic->szIconText, lppi->szLabel);
IconRefresh();
// we need to tell the client we want to be saved...it should be smart
// enough to do it anyway, but we can't take any chances.
if (_pIOleClientSite)
_pIOleClientSite->SaveObject();
return hr;
}
HRESULT CPackage::CreateTempFileName()
{
ASSERT(NULL != _pEmbed);
TCHAR szDefPath[MAX_PATH];
if (_pEmbed->pszTempName) {
return S_OK;
} else if (GetTempPath(ARRAYSIZE(szDefPath), szDefPath)) {
LPTSTR pszFile;
if ((NULL != _lpic) && (TEXT('\0') != _lpic->szIconText[0])) {
pszFile = _lpic->szIconText;
} else {
pszFile = PathFindFileName(_pEmbed->fd.cFileName);
}
PathAppend(szDefPath, pszFile);
if (PathFileExists(szDefPath)) {
TCHAR szOriginal[MAX_PATH];
lstrcpy(szOriginal, szDefPath);
PathYetAnotherMakeUniqueName(szDefPath, szOriginal, NULL, NULL);
}
_pEmbed->pszTempName = new TCHAR[lstrlen(szDefPath) + 1];
if (!_pEmbed->pszTempName) {
DebugMsg(DM_TRACE, " couldn't alloc memory for pszTempName!!");
return E_OUTOFMEMORY;
}
lstrcpy(_pEmbed->pszTempName, szDefPath);
return S_OK;
} else {
DebugMsg(DM_TRACE, " couldn't get temp path!!");
return E_FAIL;
}
}
HRESULT CPackage::CreateTempFile()
{
// used to create a temporary file that holds the file contents of the
// packaged file. the old packager used to keep the packaged file in
// memory which is just a total waste. so, being as we're much more
// efficient, we create a temp file whenever someone wants to do something
// with our contents. we initialze the temp file from the original file
// to package or our persistent storage depending on whether we are a new
// package or a loaded package
// return: S_OK -- temp file created
// E_FAIL -- error creating temp file
DebugMsg(DM_TRACE, " CreateTempFile() called.");
HRESULT hr = CreateTempFileName();
if (FAILED(hr)) {
return hr;
}
if (PathFileExists(_pEmbed->pszTempName)) {
DebugMsg(DM_TRACE, " already have a temp file!!");
return S_OK;
}
// if we weren't loaded from a storage then we're in the process of
// creating a package, and should be able to copy the packaged file
// to create a temp file
if (!_fLoaded) {
if (!(CopyFile(_pEmbed->fd.cFileName, _pEmbed->pszTempName, FALSE))) {
DebugMsg(DM_TRACE, " couldn't copy file!!");
return E_FAIL;
}
} else {
TCHAR szTempFile[MAX_PATH];
// copy the file name because _pEmbed may get re-created below,
// but we want to hold on to this filename:
lstrcpy(szTempFile, _pEmbed->pszTempName);
// if we have a valid stream, but not a file contents stream,
// it's because we went hands off and didn't know where to put
// the seek pointer to init the filecontents stream. so, we
// call PackageReadFromStream to create the FileContents stream
if (_pstm && !_pstmFileContents) {
if (FAILED(PackageReadFromStream(_pstm))) {
DebugMsg(DM_TRACE, " couldn't read from stream!!");
return E_FAIL;
}
}
IStream* pstm;
_pstmFileContents->Clone(&pstm); // we don't want to move the seek
// pointer on our FileContents stream
if (FAILED(CopyStreamToFile(pstm, szTempFile))) {
DebugMsg(DM_TRACE, " couldn't copy from stream!!");
pstm->Release();
return E_FAIL;
} else {
ASSERT(_pEmbed);
delete _pEmbed->pszTempName;
if (NULL != (_pEmbed->pszTempName = new TCHAR[lstrlen(szTempFile) + 1])) {
lstrcpy(_pEmbed->pszTempName, szTempFile);
} else {
return E_OUTOFMEMORY;
}
}
pstm->Release();
}
// whenever we create a tempfile we are activating the contents which
// means we are dirty until we get a save message
return S_OK;
}
// Data Transfer Functions
HRESULT CPackage::GetFileDescriptor(LPFORMATETC pFE, LPSTGMEDIUM pSTM)
{
FILEGROUPDESCRIPTOR* pfgd;
DebugMsg(DM_TRACE, " Getting File Descriptor");
// we only support HGLOBAL at this time
if (!(pFE->tymed & TYMED_HGLOBAL)) {
DebugMsg(DM_TRACE, " does not support HGLOBAL!");
return DATA_E_FORMATETC;
}
// Copy file descriptor to HGLOBAL //
pSTM->tymed = TYMED_HGLOBAL;
// render the file descriptor
if (!(pfgd = (FILEGROUPDESCRIPTOR*)GlobalAlloc(GPTR,
sizeof(FILEGROUPDESCRIPTOR))))
return E_OUTOFMEMORY;
pSTM->hGlobal = pfgd;
pfgd->cItems = 1;
switch (_panetype) {
case PEMBED:
pfgd->fgd[0] = _pEmbed->fd;
GetDisplayName(pfgd->fgd[0].cFileName, _pEmbed->fd.cFileName);
break;
case CMDLINK:
// the label for the package will serve as the filename for the
// shortcut we're going to create.
lstrcpy(pfgd->fgd[0].cFileName, _lpic->szIconText);
// BUGBUG: harcoded use of .lnk extension!!
lstrcat(pfgd->fgd[0].cFileName, TEXT(".lnk"));
// we want to add the little arrow to the shortcut.
pfgd->fgd[0].dwFlags = FD_LINKUI;
break;
}
return S_OK;
}
HRESULT CPackage::GetFileContents(LPFORMATETC pFE, LPSTGMEDIUM pSTM)
{
void* lpvDest = NULL;
DWORD dwSize;
HANDLE hFile = NULL;
DWORD cb;
HRESULT hr = E_FAIL;
DebugMsg(DM_TRACE, " Getting File Contents");
// Copy file contents to ISTREAM //
// NOTE: Hopefully, everyone using our object supports TYMED_ISTREAM,
// otherwise we could get some really slow behavior. We might later
// want to implement TYMED_ISTORAGE as well and shove our file contents
// into a single stream named CONTENTS.
if (pFE->tymed & TYMED_ISTREAM) {
DebugMsg(DM_TRACE, " using TYMED_ISTREAM");
pSTM->tymed = TYMED_ISTREAM;
switch (_panetype) {
case PEMBED:
if (_pstmFileContents)
hr = _pstmFileContents->Clone(&pSTM->pstm);
else
return E_FAIL;
break;
case CMDLINK:
hr = CreateStreamOnHGlobal(NULL, TRUE, &pSTM->pstm);
if (SUCCEEDED(hr)) {
hr = CreateShortcutOnStream(pSTM->pstm);
if (FAILED(hr)) {
pSTM->pstm->Release();
pSTM->pstm = NULL;
}
}
break;
}
return hr;
}
// Copy file contents to HGLOBAL //
// NOTE: This is really icky and could potentially be very slow if
// somebody decides to package really large files. Hopefully,
// everyone should be able to get the info it wants through TYMED_ISTREAM,
// but this is here as a common denominator
if (pFE->tymed & TYMED_HGLOBAL) {
DebugMsg(DM_TRACE, " using TYMED_HGLOBAL");
pSTM->tymed = TYMED_HGLOBAL;
if (_panetype == CMDLINK) {
DebugMsg(DM_TRACE,
" H_GLOBAL not supported for CMDLINK");
return DATA_E_FORMATETC;
}
dwSize = _pEmbed->fd.nFileSizeLow;
// caller is responsible for freeing this memory, even if we fail.
if (!(lpvDest = GlobalAlloc(GPTR, dwSize))) {
DebugMsg(DM_TRACE, " out o memory!!");
return E_OUTOFMEMORY;
}
pSTM->hGlobal = lpvDest;
// This will reinitialize our FileContents stream if we had to get
// rid of it. For instance, we have to get rid of all our storage
// pointers in HandsOffStorage, but there's no need to reinit our
// FileContents stream unless we need it again.
if (_pstm && !_pstmFileContents)
PackageReadFromStream(_pstm);
if (_pstmFileContents) {
IStream* pstm;
hr = _pstmFileContents->Clone(&pstm);
if (FAILED(hr))
return hr;
hr = pstm->Read(lpvDest, dwSize, &cb);
pstm->Release();
if (FAILED(hr))
return hr;
} else
return E_FAIL;
if (FAILED(hr) || cb != dwSize) {
DebugMsg(DM_TRACE, " error reading from stream!!");
return E_FAIL;
}
return hr;
}
return DATA_E_FORMATETC;
}
HRESULT CPackage::GetMetafilePict(LPFORMATETC pFE, LPSTGMEDIUM pSTM)
{
LPMETAFILEPICT lpmfpict;
RECT rcTemp;
LPIC lpic = _lpic;
HDC hdcMF = NULL;
HFONT hfont = NULL;
DebugMsg(DM_TRACE, " Getting MetafilePict");
if (!(pFE->tymed & TYMED_MFPICT)) {
DebugMsg(DM_TRACE, " does not support MFPICT!");
return DATA_E_FORMATETC;
}
pSTM->tymed = TYMED_MFPICT;
// Allocate memory for the metafilepict and get a pointer to it
// NOTE: the caller is responsible for freeing this memory, even on fail
if (!(pSTM->hMetaFilePict = GlobalAlloc(GPTR, sizeof(METAFILEPICT))))
return E_OUTOFMEMORY;
lpmfpict = (LPMETAFILEPICT)pSTM->hMetaFilePict;
// Create the metafile
if (!(hdcMF = CreateMetaFile(NULL)))
return E_OUTOFMEMORY;
// Initializae the metafile
SetWindowOrgEx(hdcMF, 0, 0, NULL);
SetWindowExtEx(hdcMF, lpic->rc.right - 1, lpic->rc.bottom - 1, NULL);
SetRect(&rcTemp, 0, 0, lpic->rc.right, lpic->rc.bottom);
hfont = SelectFont(hdcMF, g_hfontTitle);
// Center the icon
DrawIcon(hdcMF, (rcTemp.right - g_cxIcon) / 2, 0, lpic->hDlgIcon);
// Center the text below the icon
SetBkMode(hdcMF, TRANSPARENT);
SetTextAlign(hdcMF, TA_CENTER);
TextOut(hdcMF, rcTemp.right / 2, g_cxIcon + 1, lpic->szIconText,
lstrlen(lpic->szIconText));
if (hfont)
SelectObject(hdcMF, hfont);
// Map to device independent coordinates
rcTemp.right =
MulDiv((rcTemp.right - rcTemp.left), HIMETRIC_PER_INCH, DEF_LOGPIXELSX);
rcTemp.bottom =
MulDiv((rcTemp.bottom - rcTemp.top), HIMETRIC_PER_INCH, DEF_LOGPIXELSY);
// Finish filling in the metafile header
lpmfpict->mm = MM_ANISOTROPIC;
lpmfpict->xExt = rcTemp.right;
lpmfpict->yExt = rcTemp.bottom;
lpmfpict->hMF = CloseMetaFile(hdcMF);
return S_OK;
}
HRESULT CPackage::GetObjectDescriptor(LPFORMATETC pFE, LPSTGMEDIUM pSTM)
{
LPOBJECTDESCRIPTOR lpobj;
DWORD dwFullUserTypeNameLen;
DebugMsg(DM_TRACE, " Getting Object Descriptor");
// we only support HGLOBAL at this time
if (!(pFE->tymed & TYMED_HGLOBAL)) {
DebugMsg(DM_TRACE, " does not support HGLOBAL!");
return DATA_E_FORMATETC;
}
// Copy file descriptor to HGLOBAL //
dwFullUserTypeNameLen = 0; //lstrlen(szUserType) + 1;
pSTM->tymed = TYMED_HGLOBAL;
if (!(lpobj = (OBJECTDESCRIPTOR*)GlobalAlloc(GPTR,
sizeof(OBJECTDESCRIPTOR) + dwFullUserTypeNameLen)))
return E_OUTOFMEMORY;
pSTM->hGlobal = lpobj;
lpobj->cbSize = sizeof(OBJECTDESCRIPTOR) + dwFullUserTypeNameLen;
lpobj->clsid = CLSID_CPackage;
lpobj->dwDrawAspect = DVASPECT_CONTENT | DVASPECT_ICON;
_pIOleObject->GetMiscStatus(DVASPECT_CONTENT | DVASPECT_ICON, &(lpobj->dwStatus));
lpobj->dwFullUserTypeName = 0L; //sizeof(OBJECTDESCRIPTOR);
lpobj->dwSrcOfCopy = 0L;
// lstrcpy((LPTSTR)lpobj+lpobj->dwFullUserTypeName, szUserType);
return S_OK;
}
// Stream I/O Functions
HRESULT CPackage::PackageReadFromStream(IStream* pstm)
{
// initialize the package object from a stream
// return: s_OK - package properly initialized
// E_FAIL - error initializing package
WORD w;
DWORD dw;
DebugMsg(DM_TRACE, "pack - PackageReadFromStream called.");
// read in the package size, which we don't really need, but the old
// packager puts it there.
if (FAILED(pstm->Read(&dw, sizeof(dw), NULL)))
return E_FAIL;
// NOTE: Ok, this is really dumb. The old packager allowed the user
// to create packages without giving them icons or labels, which
// in my opinion is just dumb, it should have at least created a default
// icon and shoved it in the persistent storage...oh well...
// So if the appearance type comes back as NOTHING ( == 0)
// then we just won't read any icon information.
// read in the appearance type
pstm->Read(&w, sizeof(w), NULL);
// read in the icon information
if (w == (WORD)ICON) {
if (FAILED(IconReadFromStream(pstm))) {
DebugMsg(DM_TRACE, " error reading icon info!!");
return E_FAIL;
}
} else if (w == (WORD)PICTURE) {
DebugMsg(DM_TRACE, " old Packager Appearance, not supported!!");
// NOTE: Ideally, we could just ignore the appearance and continue, but to
// do so, we'll need to know how much information to skip over before continuing
// to read from the stream
ShellMessageBox(g_hinst,
NULL,
MAKEINTRESOURCE(IDS_OLD_FORMAT_ERROR),
MAKEINTRESOURCE(IDS_APP_TITLE),
MB_OK | MB_ICONERROR | MB_TASKMODAL);
return E_FAIL;
}
// read in the contents type
pstm->Read(&w, sizeof(w), NULL);
_panetype = (PANETYPE)w;
switch ((PANETYPE)w) {
case PEMBED:
// read in the contents information
return EmbedReadFromStream(pstm);
case CMDLINK:
// read in the contents information
return CmlReadFromStream(pstm);
default:
return E_FAIL;
}
}
// read the icon info from a stream
// return: S_OK -- icon read correctly
// E_FAIL -- error reading icon
HRESULT CPackage::IconReadFromStream(IStream* pstm)
{
LPIC lpic = IconCreate();
if (lpic) {
CHAR szTemp[MAX_PATH];
StringReadFromStream(pstm, szTemp, ARRAYSIZE(szTemp));
SHAnsiToTChar(szTemp, lpic->szIconText, ARRAYSIZE(lpic->szIconText));
StringReadFromStream(pstm, szTemp, ARRAYSIZE(szTemp));
SHAnsiToTChar(szTemp, lpic->szIconPath, ARRAYSIZE(lpic->szIconPath));
WORD wDlgIcon;
pstm->Read(&wDlgIcon, sizeof(wDlgIcon), NULL);
lpic->iDlgIcon = (INT)wDlgIcon;
GetCurrentIcon(lpic);
IconCalcSize(lpic);
}
DestroyIC();
_lpic = lpic;
return lpic ? S_OK : E_FAIL;
}
HRESULT CPackage::EmbedReadFromStream(IStream* pstm)
{
// reads embedded file contents from a stream
// return: S_OK - contents read succesfully
// E_FAIL - error reading contents
DWORD dwSize;
CHAR szFileName[MAX_PATH];
DebugMsg(DM_TRACE, "pack - EmbedReadFromStream called.");
pstm->Read(&dwSize, sizeof(dwSize), NULL); // get string size
pstm->Read(szFileName, dwSize, NULL); // get string
pstm->Read(&dwSize, sizeof(dwSize), NULL); // get file size
// we don't do anything with the file contents here, because anything
// we do could be a potentially expensive operation. so, we just clone
// the stream and hold onto it for future use.
if (_pstmFileContents)
_pstmFileContents->Release();
if (FAILED(pstm->Clone(&_pstmFileContents)))
return E_FAIL;
if (_pEmbed) {
if (_pEmbed->pszTempName) {
DeleteFile(_pEmbed->pszTempName);
delete _pEmbed->pszTempName;
}
delete _pEmbed;
}
_pEmbed = new EMBED;
if (NULL != _pEmbed) {
_pEmbed->fd.dwFlags = FD_FILESIZE;
_pEmbed->fd.nFileSizeLow = dwSize;
_pEmbed->fd.nFileSizeHigh = 0;
SHAnsiToTChar(szFileName, _pEmbed->fd.cFileName, ARRAYSIZE(_pEmbed->fd.cFileName));
DebugMsg(DM_TRACE, " %s\n\r %d", _pEmbed->fd.cFileName, _pEmbed->fd.nFileSizeLow);
return S_OK;
} else {
return E_OUTOFMEMORY;
}
}
HRESULT CPackage::CmlReadFromStream(IStream* pstm)
{
// reads command line contents from a stream
// return: S_OK - contents read succesfully
// E_FAIL - error reading contents
WORD w;
CHAR szCmdLink[CBCMDLINKMAX];
DebugMsg(DM_TRACE, "pack - CmlReadFromStream called.");
// read in the fCmdIsLink and the command line string
if (FAILED(pstm->Read(&w, sizeof(w), NULL)))
return E_FAIL;
StringReadFromStream(pstm, szCmdLink, ARRAYSIZE(szCmdLink));
if (_pCml != NULL)
delete _pCml;
_pCml = new CML;
SHAnsiToTChar(szCmdLink, _pCml->szCommandLine, ARRAYSIZE(_pCml->szCommandLine));
return S_OK;
}
HRESULT CPackage::PackageWriteToStream(IStream* pstm)
{
// write the package object to a stream
// return: s_OK - package properly written
// E_FAIL - error writing package
WORD w;
DWORD cb = 0L;
DWORD dwSize;
DebugMsg(DM_TRACE, "pack - PackageWriteToStream called.");
// write out a DWORD where the package size will go
if (FAILED(pstm->Write(&cb, sizeof(DWORD), NULL)))
return E_FAIL;
// write out the appearance type
w = (WORD)ICON;
if (FAILED(pstm->Write(&w, sizeof(WORD), NULL)))
return E_FAIL;
cb += 2 * sizeof(WORD); // for appearance type and contents type
// write out the icon information
if (FAILED(IconWriteToStream(pstm, &dwSize))) {
DebugMsg(DM_TRACE, " error writing icon info!!");
return E_FAIL;
}
cb += dwSize;
// write out the contents type
w = (WORD)_panetype;
if (FAILED(pstm->Write(&_panetype, sizeof(WORD), NULL)))
return E_FAIL;
switch (_panetype) {
case PEMBED:
// write out the contents information
if (FAILED(EmbedWriteToStream(pstm, &dwSize))) {
DebugMsg(DM_TRACE, " error writing embed info!!");
return E_FAIL;
}
cb += dwSize;
break;
case CMDLINK:
// write out the contents information
if (FAILED(CmlWriteToStream(pstm, &dwSize))) {
DebugMsg(DM_TRACE, " error writing cml info!!");
return E_FAIL;
}
cb += dwSize;
break;
}
LARGE_INTEGER li = {0, 0};
if (FAILED(pstm->Seek(li, STREAM_SEEK_SET, NULL)))
return E_FAIL;
if (FAILED(pstm->Write(&cb, sizeof(DWORD), NULL)))
return E_FAIL;
return S_OK;
}
// write the icon to a stream
// return: s_OK - icon properly written
// E_FAIL - error writing icon
HRESULT CPackage::IconWriteToStream(IStream* pstm, DWORD* pdw)
{
DWORD cb = 0;
CHAR szTemp[MAX_PATH];
SHTCharToAnsi(_lpic->szIconText, szTemp, ARRAYSIZE(szTemp));
HRESULT hr = StringWriteToStream(pstm, szTemp, &cb);
if (SUCCEEDED(hr)) {
SHTCharToAnsi(_lpic->szIconPath, szTemp, ARRAYSIZE(szTemp));
hr = StringWriteToStream(pstm, szTemp, &cb);
if (SUCCEEDED(hr)) {
DWORD dwWrite;
WORD wDlgIcon = (WORD)_lpic->iDlgIcon;
hr = pstm->Write(&wDlgIcon, sizeof(wDlgIcon), &dwWrite);
if (SUCCEEDED(hr)) {
cb += dwWrite;
if (pdw)
*pdw = cb;
}
}
}
return hr;
}
// write embedded file contents to a stream
// return: S_OK - contents written succesfully
// E_FAIL - error writing contents
HRESULT CPackage::EmbedWriteToStream(IStream* pstm, DWORD* pdw)
{
DWORD cb = 0;
CHAR szTemp[MAX_PATH];
SHTCharToAnsi(_pEmbed->fd.cFileName, szTemp, ARRAYSIZE(szTemp));
DWORD dwSize = lstrlenA(szTemp) + 1;
HRESULT hr = pstm->Write(&dwSize, sizeof(dwSize), &cb);
if (SUCCEEDED(hr)) {
DWORD dwWrite;
hr = StringWriteToStream(pstm, szTemp, &dwWrite);
if (SUCCEEDED(hr)) {
cb += dwWrite;
hr = pstm->Write(&_pEmbed->fd.nFileSizeLow, sizeof(_pEmbed->fd.nFileSizeLow), &dwWrite);
if (SUCCEEDED(hr)) {
cb += dwWrite;
// we want to make sure our file contents stream always points to latest
// saved file contents
// NOTE: If we're not saving to our loaded stream, we shouldn't keep a
// pointer to it, because we're in a SaveAs situation, and we don't want
// to be hanging onto pointers to other peoples streams.
if (_pstmFileContents && _pstm == pstm) {
_pstmFileContents->Release();
pstm->Clone(&_pstmFileContents);
}
// This is for screwy apps, like MSWorks that ask us to save ourselves
// before they've even told us to initialize ourselves.
if (_pEmbed->fd.cFileName[0]) {
hr = CopyFileToStream(_pEmbed->pszTempName, pstm);
if (SUCCEEDED(hr)) {
cb += _pEmbed->fd.nFileSizeLow;
}
}
if (pdw)
*pdw = cb;
}
}
}
return hr;
}
// write embedded file contents to a stream
// return: S_OK - contents written succesfully
// E_FAIL - error writing contents
HRESULT CPackage::CmlWriteToStream(IStream* pstm, DWORD* pdw)
{
DWORD cb = 0;
WORD w = _pCml->fCmdIsLink;
DebugMsg(DM_TRACE, "pack - CmlWriteToStream called.");
if (FAILED(pstm->Write(&w, sizeof(w), NULL)))
return E_FAIL; // write fCmdIsLink
cb += sizeof(w); // for fCmdIsLink
CHAR szTemp[MAX_PATH];
SHTCharToAnsi(_pCml->szCommandLine, szTemp, ARRAYSIZE(szTemp));
HRESULT hres = StringWriteToStream(pstm, szTemp, &cb);
if (FAILED(hres))
return hres; // write command link
// return the number of bytes written in the outparam
if (pdw)
*pdw = cb;
return S_OK;
}
HRESULT CPackage::CreateShortcutOnStream(IStream* pstm)
{
HRESULT hr;
IShellLink* psl;
TCHAR szArgs[CBCMDLINKMAX - MAX_PATH];
TCHAR szPath[MAX_PATH];
hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (void**)&psl);
if (SUCCEEDED(hr)) {
IPersistStream* pps;
lstrcpy(szPath, _pCml->szCommandLine);
PathSeparateArgs(szPath, szArgs);
psl->SetPath(szPath);
psl->SetIconLocation(_lpic->szIconPath, _lpic->iDlgIcon);
psl->SetShowCmd(SW_SHOW);
psl->SetArguments(szArgs);
hr = psl->QueryInterface(IID_IPersistStream, (void**)&pps);
if (SUCCEEDED(hr)) {
hr = pps->Save(pstm, TRUE);
pps->Release();
}
psl->Release();
}
LARGE_INTEGER li = {0,0};
pstm->Seek(li, STREAM_SEEK_SET, NULL);
return hr;
}
HRESULT CPackage::InitVerbEnum(OLEVERB* pVerbs, ULONG cVerbs)
{
if (NULL != _pVerbs) {
for (ULONG i = 0; i < _cVerbs; i++) {
delete _pVerbs[i].lpszVerbName;
}
delete _pVerbs;
}
_pVerbs = pVerbs;
_cVerbs = cVerbs;
_nCurVerb = 0;
return (NULL != pVerbs) ? S_OK : E_FAIL;
}
VOID CPackage::ReleaseContextMenu()
{
if (NULL != _pcm) {
_pcm->Release();
_pcm = NULL;
}
}
HRESULT CPackage::GetContextMenu(IContextMenu** ppcm)
{
HRESULT hr = E_FAIL;
ASSERT(NULL != ppcm);
if (NULL != _pcm) {
_pcm->AddRef();
*ppcm = _pcm;
hr = S_OK;
} else if ((PEMBED == _panetype) || (CMDLINK == _panetype)) {
if (PEMBED == _panetype) {
hr = CreateTempFileName();
} else {
hr = S_OK;
}
if (SUCCEEDED(hr)) {
LPITEMIDLIST pidl = SHSimpleIDListFromPath((PEMBED == _panetype) ?
_pEmbed->pszTempName :
_pCml->szCommandLine);
if (NULL != pidl) {
IShellFolder* psf;
LPCITEMIDLIST pidlChild;
if (SUCCEEDED(hr = SHBindToIDListParent(pidl, IID_IShellFolder, (void**)&psf, &pidlChild))) {
hr = psf->GetUIObjectOf(NULL, 1, &pidlChild, IID_IContextMenu, NULL, (void**)&_pcm);
if (SUCCEEDED(hr)) {
_pcm->AddRef();
*ppcm = _pcm;
}
psf->Release();
}
ILFree(pidl);
} else {
hr = E_OUTOFMEMORY;
}
}
}
return hr;
}
HRESULT CPackage::IconRefresh()
{
// we refresh the icon. typically, this will be called the first time
// the package is created to load the new icon and calculate how big
// it should be. this will also be called after we edit the package,
// since the user might have changed the icon.
// First, load the appropriate icon. We'll load the icon specified by
// lpic->szIconPath and lpic->iDlgIcon if possible, otherwise we'll just
// use the generic packager icon.
GetCurrentIcon(_lpic);
// Next, we need to have the icon recalculate its size, since it's text
// might have changed, causing it to get bigger or smaller.
IconCalcSize(_lpic);
// Next, notify our containers that our view has changed.
if (_pIDataAdviseHolder)
_pIDataAdviseHolder->SendOnDataChange(_pIDataObject, 0, NULL);
if (_pViewSink)
_pViewSink->OnViewChange(_dwViewAspects, _dwViewAdvf);
// Set our dirty flag
_fIsDirty = TRUE;
return S_OK;
}
int CPackage::RunWizard()
{
PACKAGER_INFO packInfo;
HRESULT hr;
PackWiz_CreateWizard(NULL, &packInfo);
if (*packInfo.szFilename == TEXT('\0')) {
ShellMessageBox(g_hinst,
NULL,
MAKEINTRESOURCE(IDS_CREATE_ERROR),
MAKEINTRESOURCE(IDS_APP_TITLE),
MB_ICONERROR | MB_TASKMODAL | MB_OK);
return E_FAIL;
}
InitFromPackInfo(&packInfo);
hr = OleSetClipboard(_pIDataObject);
if (FAILED(hr)) {
ShellMessageBox(g_hinst,
NULL,
MAKEINTRESOURCE(IDS_COPY_ERROR),
MAKEINTRESOURCE(IDS_APP_TITLE),
MB_ICONERROR | MB_TASKMODAL | MB_OK);
return -1;
}
// we need to do this. our OleUninitialze call at the end, free the
// libarary and our dataobject on the clipboard unless we flush
// the clipboard.
hr = OleFlushClipboard();
if (FAILED(hr)) {
ShellMessageBox(g_hinst,
NULL,
MAKEINTRESOURCE(IDS_COPY_ERROR),
MAKEINTRESOURCE(IDS_APP_TITLE),
MB_ICONERROR | MB_TASKMODAL | MB_OK);
return -1;
}
ShellMessageBox(g_hinst,
NULL,
MAKEINTRESOURCE(IDS_COPY_COMPLETE),
MAKEINTRESOURCE(IDS_APP_TITLE),
MB_ICONINFORMATION | MB_TASKMODAL | MB_OK);
return 0;
}
void CPackage::DestroyIC()
{
if (_lpic) {
if (_lpic->hDlgIcon)
DestroyIcon(_lpic->hDlgIcon);
GlobalFree(_lpic);
}
}
STDAPI_(BOOL) PackWizRunFromExe()
{
OleInitialize(NULL);
CPackage* pPackage = new CPackage;
if (pPackage) {
pPackage->Init();
pPackage->RunWizard();
pPackage->Release();
}
OleUninitialize();
return 0;
}