2073 lines
57 KiB
C++
2073 lines
57 KiB
C++
/*
|
|
|
|
* ftpobj.cpp - IDataObject interface
|
|
|
|
*/
|
|
|
|
#include "priv.h"
|
|
#include "ftpobj.h"
|
|
#include "ftpurl.h"
|
|
#include <shlwapi.h>
|
|
|
|
|
|
// CLSIDs
|
|
// {299D0193-6DAA-11d2-B679-006097DF5BD4}
|
|
const GUID CLSID_FtpDataObject = { 0x299d0193, 0x6daa, 0x11d2, 0xb6, 0x79, 0x0, 0x60, 0x97, 0xdf, 0x5b, 0xd4 };
|
|
|
|
/*
|
|
|
|
* g_dropTypes conveniently mirrors our FORMATETCs.
|
|
|
|
* Hardly coincidence, of course. Enum_Fe did the real work.
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Preinitialized global data.
|
|
|
|
*/
|
|
FORMATETC g_formatEtcOffsets;
|
|
FORMATETC g_formatPasteSucceeded;
|
|
CLIPFORMAT g_cfTargetCLSID;
|
|
|
|
FORMATETC g_dropTypes[] =
|
|
{
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_ISTREAM }, // DROP_FCont
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_FGDW
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_FGDA
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_IDList
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_URL
|
|
// { 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_Offsets
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_PrefDe
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_PerfDe
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_FTP_PRIVATE
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_OLEPERSIST - see _RenderOlePersist() for desc.
|
|
{ CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_Hdrop
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL }, // DROP_FNMA
|
|
{ 0, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL } // DROP_FNMW
|
|
};
|
|
|
|
|
|
|
|
/**\
|
|
GLOBAL: c_stgInit
|
|
|
|
DESCRIPTION:
|
|
Mostly straightforward. The only major weirdness is that cfURL
|
|
is delay-rendered iff the m_pflHfpl contains only one object. Otherwise,
|
|
cfURL is not supported. (URLs can refer to only one object at a time.)
|
|
\**/
|
|
STGMEDIUM c_stgInit[] =
|
|
{
|
|
{ 0, 0, 0 }, // DROP_FCont
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_FGDW - delay-rendered
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_FGDA - delay-rendered
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_IDList - delay-rendered
|
|
{ 0, 0, 0 }, // DROP_URL - opt delay-rendered
|
|
// { 0, 0, 0 }, // DROP_Offsets
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_PrefDe - delay-rendered
|
|
{ 0, 0, 0 }, // DROP_PerfDe
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_FTP_PRIVATE
|
|
{ TYMED_HGLOBAL, 0, 0 }, // DROP_OLEPERSIST - see _RenderOlePersist() for desc.
|
|
{ 0, 0, 0 }, // DROP_Hdrop
|
|
{ 0, 0, 0 }, // DROP_FNMA
|
|
{ 0, 0, 0 } // DROP_FNMW
|
|
};
|
|
|
|
|
|
|
|
/**\
|
|
FUNCTION: TraceMsgWithFormatEtc
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
void TraceMsgWithFormat(DWORD dwFlags, LPCSTR pszBefore, LPFORMATETC pFormatEtc, LPCSTR pszAfter, HRESULT hr)
|
|
{
|
|
#ifdef DEBUG
|
|
TCHAR szFormatName[MAX_PATH];
|
|
TCHAR szMedium[MAX_PATH];
|
|
|
|
szFormatName[0] = 0;
|
|
szMedium[0] = 0;
|
|
if (pFormatEtc)
|
|
{
|
|
// This may fail if it's a basic format.
|
|
if (!GetClipboardFormatName(pFormatEtc->cfFormat, szFormatName, ARRAYSIZE(szFormatName)))
|
|
wnsprintf(szFormatName, ARRAYSIZE(szFormatName), TEXT("Pre-defined=%d"), pFormatEtc->cfFormat);
|
|
|
|
switch (pFormatEtc->tymed)
|
|
{
|
|
case TYMED_HGLOBAL: StrCpyN(szMedium, TEXT("HGLOBAL"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_FILE: StrCpyN(szMedium, TEXT("File"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_GDI: StrCpyN(szMedium, TEXT("GDI"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_MFPICT: StrCpyN(szMedium, TEXT("MFPICT"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_ENHMF: StrCpyN(szMedium, TEXT("ENHMF"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_ISTORAGE: StrCpyN(szMedium, TEXT("ISTORAGE"), ARRAYSIZE(szMedium)); break;
|
|
case TYMED_ISTREAM: StrCpyN(szMedium, TEXT("ISTREAM"), ARRAYSIZE(szMedium)); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
szMedium[0] = 0;
|
|
}
|
|
|
|
TraceMsg(dwFlags, "%hs [FRMTETC: %ls, lndx: %d, %ls] hr=%#08lx, %hs", pszBefore, szFormatName, pFormatEtc->lindex, szMedium, hr, pszAfter);
|
|
#endif // DEBUG
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _IsLindexOkay
|
|
|
|
DESCRIPTION:
|
|
If ife != DROP_FCont, then pfeWant->lindex must be -1.
|
|
|
|
If ife == DROP_FCont, then pfeWant->lindex must be in the range
|
|
0 ... m_pflHfpl->GetCount() - 1
|
|
\**/
|
|
BOOL CFtpObj::_IsLindexOkay(int ife, FORMATETC *pfeWant)
|
|
{
|
|
BOOL fResult;
|
|
|
|
if (ife != DROP_FCont)
|
|
fResult = pfeWant->lindex == -1;
|
|
else
|
|
fResult = (LONG)pfeWant->lindex < m_pflHfpl->GetCount();
|
|
|
|
return fResult;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _FindData
|
|
|
|
DESCRIPTION:
|
|
Locate our FORMATETC/STGMEDIUM given a FORMATETC from somebody else.
|
|
On success, stores the index found into *piOut.
|
|
|
|
We do not allow clients to change the TYMED of a FORMATETC, so
|
|
in fact checking the TYMED is what we want, even on a SetData.
|
|
\**/
|
|
HRESULT CFtpObj::_FindData(FORMATETC *pfe, PINT piOut)
|
|
{
|
|
int nIndex;
|
|
HRESULT hres = DV_E_FORMATETC;
|
|
|
|
*piOut = 0;
|
|
for (nIndex = DROP_FCont; nIndex < DROP_OFFERMAX; nIndex++)
|
|
{
|
|
ASSERT(0 == (g_dropTypes[nIndex]).ptd);
|
|
ASSERT(g_dropTypes[nIndex].dwAspect == DVASPECT_CONTENT);
|
|
|
|
if ((pfe->cfFormat == g_dropTypes[nIndex].cfFormat) && !ShouldSkipDropFormat(nIndex))
|
|
{
|
|
if (EVAL(g_dropTypes[nIndex].ptd == NULL))
|
|
{
|
|
if (EVAL(pfe->dwAspect == DVASPECT_CONTENT))
|
|
{
|
|
if (EVAL(g_dropTypes[nIndex].tymed & pfe->tymed))
|
|
{
|
|
if (EVAL(_IsLindexOkay(nIndex, pfe)))
|
|
{
|
|
*piOut = nIndex;
|
|
hres = S_OK;
|
|
}
|
|
else
|
|
hres = DV_E_LINDEX;
|
|
}
|
|
else
|
|
hres = DV_E_TYMED;
|
|
}
|
|
else
|
|
hres = DV_E_DVASPECT;
|
|
}
|
|
else
|
|
hres = DV_E_DVTARGETDEVICE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _FindDataForGet
|
|
|
|
DESCRIPTION:
|
|
Locate our FORMATETC/STGMEDIUM given a FORMATETC from somebody else.
|
|
On success, stores the index found into *piOut. Unlike _FindData, we will
|
|
fail the call if the data object doesn't currently have the clipboard format.
|
|
(Delayed render counts as "currently having it". What we are filtering out
|
|
are formats for which GetData will necessarily fail.)
|
|
\**/
|
|
HRESULT CFtpObj::_FindDataForGet(FORMATETC *pfe, PINT piOut)
|
|
{
|
|
HRESULT hr = _FindData(pfe, piOut);
|
|
|
|
// TODO: g_cfHIDA should return an array of pidls for each folder.
|
|
// If we do this, the caller will support creating Shortcuts
|
|
// (LNK files) that point to these pidls. We may want to do
|
|
// that later.
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (*piOut != DROP_FCont)
|
|
{
|
|
if (m_stgCache[*piOut].tymed)
|
|
{
|
|
// Do we have data at all?
|
|
// (possibly delay-rendered)
|
|
}
|
|
else
|
|
hr = DV_E_FORMATETC; // I guess not
|
|
}
|
|
else
|
|
{
|
|
// File contents always okay
|
|
}
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (FAILED(hr))
|
|
{
|
|
//TraceMsg(TF_FTPDRAGDROP, "CFtpObj::_FindDataForGet(FORMATETC.cfFormat=%d) Failed.", pfe->cfFormat);
|
|
*piOut = 0xBAADF00D;
|
|
}
|
|
#endif
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// The following are used to enumerate sub directories when creating a list of pidls for
|
|
// a directory download (Ftp->FileSys).
|
|
typedef struct tagGENPIDLLIST
|
|
{
|
|
CFtpPidlList * ppidlList;
|
|
IMalloc * pm;
|
|
IProgressDialog * ppd;
|
|
CWireEncoding * pwe;
|
|
} GENPIDLLIST;
|
|
|
|
|
|
/**\
|
|
FUNCTION: ProcessItemCB
|
|
|
|
DESCRIPTION:
|
|
This function will add the specified pidl to the list. It will then
|
|
detect if it's a folder and if so, will call EnumFolder() to recursively
|
|
enum it's contents and call ProcessItemCB() for each one.
|
|
|
|
PARAMETERS:
|
|
\**/
|
|
HRESULT ProcessItemCB(LPVOID pvFuncCB, HINTERNET hint, LPCITEMIDLIST pidlFull, BOOL * pfValidhinst, LPVOID pvData)
|
|
{
|
|
GENPIDLLIST * pGenPidlList = (GENPIDLLIST *) pvData;
|
|
HRESULT hr = S_OK;
|
|
|
|
// Does the user want to cancel?
|
|
if (pGenPidlList->ppd && pGenPidlList->ppd->HasUserCancelled())
|
|
{
|
|
EVAL(SUCCEEDED(pGenPidlList->ppd->StopProgressDialog()));
|
|
hr = HRESULT_FROM_WIN32(ERROR_CANCELLED);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// No, don't cancel so continue...
|
|
|
|
// Add everything except SoftLinks.
|
|
// This is because dir SoftLinks may cause infinite recurion.
|
|
// Someday, we may want to upload a shortcut but
|
|
// that's too much work for now.
|
|
if (0 != FtpPidl_GetAttributes(pidlFull))
|
|
{
|
|
// We exist to do this:
|
|
pGenPidlList->ppidlList->InsertSorted(pidlFull);
|
|
}
|
|
|
|
// Is this a dir/folder that we need to recurse into?
|
|
if (SUCCEEDED(hr) && (FILE_ATTRIBUTE_DIRECTORY & FtpPidl_GetAttributes(pidlFull)))
|
|
{
|
|
hr = EnumFolder((LPFNPROCESSITEMCB) pvFuncCB, hint, pidlFull, pGenPidlList->pwe, pfValidhinst, pvData);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _ExpandPidlListRecursively
|
|
|
|
DESCRIPTION:
|
|
This function will take the pidl list (ppidlListSrc) and call into it
|
|
to enumerate. It will provide ProcessItemCB as the callback function.
|
|
This function will help it create a new CFtpPidlList which will not only
|
|
contain the pidls in a base folder, but also all the pidls in any subfolders
|
|
that are in the original list.
|
|
|
|
Delay-render a file group descriptor.
|
|
\**/
|
|
CFtpPidlList * CFtpObj::_ExpandPidlListRecursively(CFtpPidlList * ppidlListSrc)
|
|
{
|
|
GENPIDLLIST pep = {0};
|
|
|
|
pep.ppidlList = NULL;
|
|
pep.ppd = m_ppd;
|
|
pep.pwe = m_pff->GetCWireEncoding();
|
|
if (SUCCEEDED(CFtpPidlList_Create(0, NULL, &pep.ppidlList)))
|
|
{
|
|
m_pff->GetItemAllocator(&pep.pm);
|
|
|
|
if (EVAL(m_pfd) && EVAL(pep.pm))
|
|
{
|
|
HINTERNET hint;
|
|
|
|
if (SUCCEEDED(m_pfd->GetHint(NULL, NULL, &hint, NULL, m_pff)))
|
|
{
|
|
LPITEMIDLIST pidlRoot = ILClone(m_pfd->GetPidlReference());
|
|
|
|
if (EVAL(pidlRoot))
|
|
{
|
|
HRESULT hr = ppidlListSrc->RecursiveEnum(pidlRoot, ProcessItemCB, hint, (LPVOID) &pep);
|
|
|
|
if (m_ppd)
|
|
EVAL(SUCCEEDED(m_ppd->StopProgressDialog()));
|
|
|
|
if (FAILED(hr) && (HRESULT_FROM_WIN32(ERROR_CANCELLED) != hr) && !m_fErrAlreadyDisplayed)
|
|
{
|
|
pep.ppidlList->Release();
|
|
pep.ppidlList = NULL;
|
|
|
|
// Oh, I want a real hwnd, but where or where can I get one?
|
|
DisplayWininetErrorEx(NULL, TRUE, HRESULT_CODE(hr), IDS_FTPERR_TITLE_ERROR, IDS_FTPERR_DROPFAIL, IDS_FTPERR_WININET, MB_OK, NULL, NULL);
|
|
hr = HRESULT_FROM_WIN32(ERROR_CANCELLED); // Wrong permissions
|
|
|
|
// We need to suppress subsequent error dlgs from this location
|
|
// because callers like to ask for FILEGROUPDESCRIPTORA and
|
|
// if that fails, ask for FILEGROUPDESCRIPTORW and we don't
|
|
// want an error dialog for each.
|
|
m_fErrAlreadyDisplayed = TRUE;
|
|
}
|
|
|
|
ILFree(pidlRoot);
|
|
}
|
|
|
|
m_pfd->ReleaseHint(hint);
|
|
pep.pm->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
return pep.ppidlList;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DelayRender_FGD
|
|
|
|
DESCRIPTION:
|
|
Delay-render a file group descriptor
|
|
\**/
|
|
HGLOBAL CFtpObj::_DelayRender_FGD(BOOL fUnicode)
|
|
{
|
|
HGLOBAL hGlobal = NULL;
|
|
|
|
if (m_fCheckSecurity &&
|
|
ZoneCheckPidlAction(SAFECAST(this, IInternetSecurityMgrSite *), URLACTION_SHELL_FILE_DOWNLOAD, m_pff->GetPrivatePidlReference(), (PUAF_DEFAULT | PUAF_WARN_IF_DENIED)))
|
|
{
|
|
m_pflHfpl->TraceDump(m_pff->GetPrivatePidlReference(), TEXT("_DelayRender_FGD() TraceDump before"));
|
|
CFtpPidlList * pPidlList;
|
|
|
|
if (!m_fFGDRendered)
|
|
{
|
|
pPidlList = _ExpandPidlListRecursively(m_pflHfpl);
|
|
if (pPidlList)
|
|
{
|
|
// We succeeded so now it's expanded.
|
|
m_fFGDRendered = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_pflHfpl->AddRef();
|
|
pPidlList = m_pflHfpl;
|
|
}
|
|
|
|
if (pPidlList)
|
|
{
|
|
hGlobal = Misc_HFGD_Create(pPidlList, m_pff->GetPrivatePidlReference(), fUnicode);
|
|
IUnknown_Set(&m_pflHfpl, pPidlList);
|
|
m_pflHfpl->TraceDump(m_pff->GetPrivatePidlReference(), TEXT("_DelayRender_FGD() TraceDump after"));
|
|
pPidlList->Release();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Suppress future UI. We don't need to check any more
|
|
// because our pidl won't change. We could not pass PUAF_WARN_IF_DENIED
|
|
// but that won't suppress the UI in the prompt case. (Only admins can
|
|
// turn on the prompt case).
|
|
m_fCheckSecurity = FALSE;
|
|
}
|
|
|
|
return hGlobal;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DelayRender_IDList
|
|
|
|
DESCRIPTION:
|
|
Delay-render an ID List Array (HIDA)
|
|
\**/
|
|
HRESULT CFtpObj::_DelayRender_IDList(STGMEDIUM * pStgMedium)
|
|
{
|
|
pStgMedium->hGlobal = Misc_HIDA_Create(m_pff->GetPublicRootPidlReference(), m_pflHfpl);
|
|
|
|
ASSERT(pStgMedium->hGlobal);
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DelayRender_URL
|
|
|
|
DESCRIPTION:
|
|
The caller wants an URL in an Ansi String
|
|
\**/
|
|
HRESULT CFtpObj::_DelayRender_URL(STGMEDIUM * pStgMedium)
|
|
{
|
|
LPSTR pszUrl = NULL;
|
|
LPITEMIDLIST pidlFull = NULL;
|
|
LPITEMIDLIST pidl = m_pflHfpl->GetPidl(0);
|
|
|
|
ASSERT(pidl); // We need this
|
|
// Sometimes m_pflHfpl->GetPidl(0) is fully qualified and
|
|
// sometimes it's not.
|
|
if (!FtpID_IsServerItemID(pidl))
|
|
{
|
|
pidlFull = ILCombine(m_pfd->GetPidlReference(), pidl);
|
|
pidl = pidlFull;
|
|
}
|
|
|
|
ASSERT(m_pflHfpl->GetCount() == 1); // How do we give them more than 1 URL?
|
|
if (pidl)
|
|
{
|
|
TCHAR szUrl[MAX_URL_STRING];
|
|
|
|
if (EVAL(SUCCEEDED(UrlCreateFromPidl(pidl, SHGDN_FORADDRESSBAR, szUrl, ARRAYSIZE(szUrl), (ICU_ESCAPE | ICU_USERNAME), TRUE))))
|
|
{
|
|
DWORD cchSize = (lstrlen(szUrl) + 1);
|
|
|
|
pszUrl = (LPSTR) LocalAlloc(LPTR, (cchSize * sizeof(CHAR)));
|
|
if (EVAL(pszUrl))
|
|
SHTCharToAnsi(szUrl, pszUrl, cchSize);
|
|
}
|
|
|
|
ILFree(pidlFull);
|
|
}
|
|
|
|
pStgMedium->hGlobal = (HGLOBAL) pszUrl;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
#pragma BEGIN_CONST_DATA
|
|
|
|
DROPEFFECT c_deCopyLink = DROPEFFECT_COPY | DROPEFFECT_LINK;
|
|
DROPEFFECT c_deLink = DROPEFFECT_LINK;
|
|
|
|
#pragma END_CONST_DATA
|
|
/**\
|
|
FUNCTION: _DelayRender_PrefDe
|
|
|
|
DESCRIPTION:
|
|
Delay-render a preferred drop effect.
|
|
|
|
The preferred drop effect is DROPEFFECT_COPY (with DROPEFFECT_LINK as fallback),
|
|
unless you are dragging an FTP site, in which case it's just DROPEFFECT_LINK.
|
|
|
|
DROPEFFECT_MOVE is never preferred. We can do it; it just isn't preferred.
|
|
|
|
BUGBUG/NOTES: About DROPEFFECT_MOVE
|
|
We cannot support Move on platforms before NT5 because of a Recycle Bin bug
|
|
were it would clain to have succeeded with the copy but it actually didn't
|
|
copy anything. On NT5, the Recycle Bin drop target will call pDataObject->SetData()
|
|
with a data type of "Dropped On" and the data being the CLSID of the drop
|
|
target in addition to really copying the files to the recycle bin. This will
|
|
let us delete the files knowing they are in the recycle bin.
|
|
\**/
|
|
HRESULT CFtpObj::_DelayRender_PrefDe(STGMEDIUM * pStgMedium)
|
|
{
|
|
DROPEFFECT * pde;
|
|
|
|
if (!m_pfd->IsRoot())
|
|
pde = &c_deCopyLink;
|
|
else
|
|
pde = &c_deLink;
|
|
|
|
return Misc_CreateHglob(sizeof(*pde), pde, &pStgMedium->hGlobal);
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _RenderOlePersist
|
|
|
|
DESCRIPTION:
|
|
When the copy source goes away (the process shuts down), it calls
|
|
OleFlushClipboard. OLE will then copy our data, release us, and then
|
|
give out our data later. This works for most things except for:
|
|
1. When lindex needs to very. This doesn't work because ole doesn't know
|
|
how to ask us how may lindexs they need to copy.
|
|
2. If this object has a private interface OLE doesn't know about. For us,
|
|
it's IAsyncOperation.
|
|
|
|
To get around this problem, we want OLE to recreate us when some possible
|
|
paste target calls OleGetClipboard. We want OLE to call OleLoadFromStream()
|
|
to have us CoCreated and reload our persisted data via IPersistStream.
|
|
OLE doesn't want to do this by default or they may have backward compat
|
|
problems so they want a sign from the heavens, or at least from us, that
|
|
we will work. They ping our "OleClipboardPersistOnFlush" clipboard format
|
|
to ask this.
|
|
\**/
|
|
HRESULT CFtpObj::_RenderOlePersist(STGMEDIUM * pStgMedium)
|
|
{
|
|
// The actual cookie value is opaque to the outside world. Since
|
|
// we don't use it either, we just leave it at zero in case we use
|
|
// it in the future. It's mere existence will cause OLE to do the
|
|
// use our IPersistStream, which is what we want.
|
|
DWORD dwCookie = 0;
|
|
return Misc_CreateHglob(sizeof(dwCookie), &dwCookie, &pStgMedium->hGlobal);
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _RenderFGD
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::_RenderFGD(int nIndex, STGMEDIUM * pStgMedium)
|
|
{
|
|
HRESULT hr = _DoProgressForLegacySystemsPre();
|
|
|
|
if (SUCCEEDED(hr))
|
|
pStgMedium->hGlobal = _DelayRender_FGD((DROP_FGDW == nIndex) ? TRUE : FALSE);
|
|
|
|
if (!pStgMedium->hGlobal)
|
|
hr = HRESULT_FROM_WIN32(ERROR_CANCELLED); // Probably failed because of Zones check.
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _ForceRender
|
|
|
|
DESCRIPTION:
|
|
We previously delayed rendering the data for perf reasons. This function
|
|
was called, so we now need to render the data.
|
|
\**/
|
|
HRESULT CFtpObj::_ForceRender(int nIndex)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// We only support HGLOBALs here, but the caller may be valid
|
|
// to ask for something we don't support or an extended data.
|
|
// ASSERT((m_stgCache[nIndex].tymed) == TYMED_HGLOBAL);
|
|
|
|
if (!m_stgCache[nIndex].hGlobal)
|
|
{
|
|
STGMEDIUM medium = {TYMED_HGLOBAL, 0, NULL};
|
|
|
|
switch (nIndex)
|
|
{
|
|
case DROP_FCont:
|
|
ASSERT(0);
|
|
break;
|
|
case DROP_FGDW:
|
|
case DROP_FGDA:
|
|
hr = _RenderFGD(nIndex, &medium);
|
|
break;
|
|
case DROP_IDList:
|
|
hr = _DelayRender_IDList(&medium);
|
|
break;
|
|
/* Nuke
|
|
case DROP_Offsets:
|
|
ASSERT(0);
|
|
// hglob = _DelayRender_Offsets();
|
|
break;
|
|
*/
|
|
case DROP_PrefDe:
|
|
hr = _DelayRender_PrefDe(&medium);
|
|
break;
|
|
case DROP_PerfDe:
|
|
ASSERT(0);
|
|
// hglob = _DelayRender_PerfDe();
|
|
break;
|
|
case DROP_FTP_PRIVATE:
|
|
hr = DV_E_FORMATETC;
|
|
break;
|
|
case DROP_OLEPERSIST:
|
|
hr = _RenderOlePersist(&medium);
|
|
break;
|
|
case DROP_Hdrop:
|
|
ASSERT(0);
|
|
// hglob = _DelayRender_Hdrop();
|
|
break;
|
|
case DROP_FNMA:
|
|
ASSERT(0);
|
|
// hglob = _DelayRender_FNM();
|
|
break;
|
|
case DROP_FNMW:
|
|
ASSERT(0);
|
|
// hglob = _DelayRender_FNM();
|
|
break;
|
|
case DROP_URL:
|
|
hr = _DelayRender_URL(&medium);
|
|
break;
|
|
default:
|
|
ASSERT(0); // Should never hit.
|
|
break;
|
|
}
|
|
|
|
if (medium.hGlobal) // Will fail if the Zones Security Check Fails.
|
|
{
|
|
m_stgCache[nIndex].pUnkForRelease = NULL;
|
|
m_stgCache[nIndex].hGlobal = medium.hGlobal;
|
|
}
|
|
else
|
|
{
|
|
if (S_OK == hr)
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
TraceMsg(TF_FTPDRAGDROP, "CFtpObj::_ForceRender() FAILED. hres=%#08lx", hr);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DoProgressForLegacySystemsPre
|
|
|
|
DESCRIPTION:
|
|
Shell's pre-NT5 didn't do progress on the File Contents drop, so we
|
|
will do it here. This function will display a progress dialog while we
|
|
walk the server and expand the pidls that are needed to be copied.
|
|
Later,
|
|
\**/
|
|
HRESULT CFtpObj::_DoProgressForLegacySystemsPre(void)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (DEBUG_LEGACY_PROGRESS || (SHELL_VERSION_NT5 > GetShellVersion()))
|
|
{
|
|
TraceMsg(TF_ALWAYS, "CFtpObj::_DoProgressForLegacySystemsPre() going to do the Legacy dialogs.");
|
|
|
|
// Do we need to initialize the list?
|
|
if (!m_ppd && (-1 == m_nStartIndex))
|
|
{
|
|
// Yes, so create the create the dialog and find the sizes of the list.
|
|
if (m_ppd)
|
|
_CloseProgressDialog();
|
|
|
|
m_uliCompleted.QuadPart = 0;
|
|
m_uliTotal.QuadPart = 0;
|
|
m_ppd = CProgressDialog_CreateInstance(IDS_COPY_TITLE, IDA_FTPDOWNLOAD);
|
|
if (EVAL(m_ppd))
|
|
{
|
|
WCHAR wzProgressDialogStr[MAX_PATH];
|
|
|
|
// Tell the user we are calculating how long it will take.
|
|
if (EVAL(LoadStringW(HINST_THISDLL, IDS_PROGRESS_DOWNLOADTIMECALC, wzProgressDialogStr, ARRAYSIZE(wzProgressDialogStr))))
|
|
EVAL(SUCCEEDED(m_ppd->SetLine(2, wzProgressDialogStr, FALSE, NULL)));
|
|
|
|
// We give a NULL punkEnableModless because we don't want to go modal.
|
|
EVAL(SUCCEEDED(m_ppd->StartProgressDialog(NULL, NULL, PROGDLG_AUTOTIME, NULL)));
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DoProgressForLegacySystemsStart
|
|
|
|
DESCRIPTION:
|
|
Shell's pre-NT5 didn't do progress on the File Contents drop, so we
|
|
will do it here. Only return FAILED(hr) if IProgressDialog::HasUserCancelled().
|
|
\**/
|
|
HRESULT CFtpObj::_DoProgressForLegacySystemsStart(LPCITEMIDLIST pidl, int nIndex)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (DEBUG_LEGACY_PROGRESS || (SHELL_VERSION_NT5 > GetShellVersion()))
|
|
{
|
|
TraceMsg(TF_ALWAYS, "CFtpObj::_DoProgressForLegacySystemsStart() going to do the Legacy dialogs.");
|
|
|
|
// Do we need to initialize the list?
|
|
if (-1 == m_nStartIndex)
|
|
hr = _SetProgressDialogValues(nIndex); // Yes, so do so.
|
|
|
|
if (EVAL(m_ppd))
|
|
{
|
|
WCHAR wzTemplate[MAX_PATH];
|
|
WCHAR wzPath[MAX_PATH];
|
|
WCHAR wzStatusText[MAX_PATH];
|
|
LPITEMIDLIST pidlBase = (LPITEMIDLIST) pidl;
|
|
|
|
EVAL(SUCCEEDED(m_ppd->StartProgressDialog(NULL, NULL, PROGDLG_AUTOTIME, NULL)));
|
|
|
|
// Generate the string "Downloading <FileName>..." status string
|
|
EVAL(LoadStringW(HINST_THISDLL, IDS_DOWNLOADING, wzTemplate, ARRAYSIZE(wzTemplate)));
|
|
wnsprintfW(wzStatusText, ARRAYSIZE(wzStatusText), wzTemplate, FtpPidl_GetLastItemDisplayName(pidl));
|
|
EVAL(SUCCEEDED(m_ppd->SetLine(1, wzStatusText, FALSE, NULL)));
|
|
|
|
if (FtpPidl_IsDirectory(pidl, FALSE))
|
|
{
|
|
pidlBase = ILClone(pidl);
|
|
ILRemoveLastID(pidlBase);
|
|
}
|
|
|
|
// Generate the string "From <SrcFileDir>" status string
|
|
GetDisplayPathFromPidl(pidlBase, wzPath, ARRAYSIZE(wzPath), TRUE);
|
|
EVAL(LoadStringW(HINST_THISDLL, IDS_DL_SRC_DIR, wzTemplate, ARRAYSIZE(wzTemplate)));
|
|
wnsprintfW(wzStatusText, ARRAYSIZE(wzStatusText), wzTemplate, wzPath);
|
|
EVAL(SUCCEEDED(m_ppd->SetLine(2, wzStatusText, FALSE, NULL)));
|
|
|
|
EVAL(SUCCEEDED(m_ppd->SetProgress64(m_uliCompleted.QuadPart, m_uliTotal.QuadPart)));
|
|
TraceMsg(TF_ALWAYS, "CFtpObj::_DoProgressForLegacySystemsStart() SetProgress64(%#08lx, %#08lx)", m_uliCompleted.LowPart, m_uliTotal.LowPart);
|
|
if (m_ppd->HasUserCancelled())
|
|
hr = HRESULT_FROM_WIN32(ERROR_CANCELLED);
|
|
|
|
if (pidlBase != pidl) // Did we allocated it?
|
|
ILFree(pidlBase);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _DoProgressForLegacySystemsPost
|
|
|
|
DESCRIPTION:
|
|
Shell's pre-NT5 didn't do progress on the File Contents drop, so we
|
|
will do it here. Only return FAILED(hr) if IProgressDialog::HasUserCancelled().
|
|
\**/
|
|
HRESULT CFtpObj::_DoProgressForLegacySystemsPost(LPCITEMIDLIST pidl, BOOL fLast)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if ((DEBUG_LEGACY_PROGRESS || (SHELL_VERSION_NT5 > GetShellVersion())) && EVAL(m_ppd))
|
|
{
|
|
if (pidl)
|
|
{
|
|
// Add the file size to the Completed.
|
|
m_uliCompleted.QuadPart += FtpPidl_GetFileSize(pidl);
|
|
}
|
|
|
|
TraceMsg(TF_ALWAYS, "CFtpObj::_DoProgressForLegacySystemsPost() Closing DLG");
|
|
|
|
if (fLast)
|
|
IUnknown_Set((IUnknown **)&m_ppd, NULL); // The stream will close the dialog and release it.
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CFtpObj::_SetProgressDialogValues(int nIndex)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
m_nStartIndex = nIndex;
|
|
if (EVAL(m_ppd))
|
|
{
|
|
// Calculate m_nEndIndex
|
|
while (nIndex < m_pflHfpl->GetCount())
|
|
{
|
|
if (!FtpPidl_IsDirectory(m_pflHfpl->GetPidl(nIndex), FALSE))
|
|
m_nEndIndex = nIndex;
|
|
nIndex++;
|
|
}
|
|
|
|
for (nIndex = 0; nIndex < m_pflHfpl->GetCount(); nIndex++)
|
|
{
|
|
LPCITEMIDLIST pidl = m_pflHfpl->GetPidl(nIndex);
|
|
m_uliTotal.QuadPart += FtpPidl_GetFileSize(pidl);
|
|
}
|
|
|
|
// Reset because the above for loop can take a long time and the estimated time
|
|
// is based on the time between ::StartProgressDialog() and the first
|
|
// ::SetProgress() call.
|
|
EVAL(SUCCEEDED(m_ppd->Timer(PDTIMER_RESET, NULL)));
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CFtpObj::_CloseProgressDialog(void)
|
|
{
|
|
m_nStartIndex = -1; // Indicate we haven't inited yet.
|
|
if (m_ppd)
|
|
{
|
|
EVAL(SUCCEEDED(m_ppd->StopProgressDialog()));
|
|
IUnknown_Set((IUnknown **)&m_ppd, NULL);
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CFtpObj::_RefThread(void)
|
|
{
|
|
if (NULL == m_punkThreadRef)
|
|
{
|
|
// This is valid to fail from some hosts who won't go away,
|
|
// so they don't need to support ref counting threads.
|
|
SHGetThreadRef(&m_punkThreadRef);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CFtpObj::_RenderFileContents(LPFORMATETC pfe, LPSTGMEDIUM pstg)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
// callers have a bad habit of asking for lindex == -1 because
|
|
// that means 'all' data. But how can you hand out one IStream* for
|
|
// all files?
|
|
if (-1 != pfe->lindex)
|
|
{
|
|
LPITEMIDLIST pidl = m_pflHfpl->GetPidl(pfe->lindex);
|
|
// FileContents are always regenerated afresh.
|
|
pstg->pUnkForRelease = 0;
|
|
pstg->tymed = TYMED_ISTREAM;
|
|
|
|
if (EVAL(pidl))
|
|
{
|
|
hr = _DoProgressForLegacySystemsStart(pidl, pfe->lindex);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// Is it a directory?
|
|
if (FtpPidl_IsDirectory(pidl, FALSE))
|
|
{
|
|
// Yes, so pack the name and attributes
|
|
hr = DV_E_LINDEX;
|
|
AssertMsg(0, TEXT("Someone is asking for a FILECONTENTs for a directory item."));
|
|
}
|
|
else
|
|
{
|
|
// No, so give them the stream.
|
|
|
|
// shell32 v5 will display progress dialogs, but we need to
|
|
// display progress dialogs for shell32 v3 or v4. We do this
|
|
// by creating the progress dialog when the caller asks for the
|
|
// first stream. We then need to find out when they call for
|
|
// the last stream and then hand off the IProgressDialog to the
|
|
// CFtpStm. The CFtpStm will then close down the dialog when the
|
|
// caller closes it.
|
|
hr = CFtpStm_Create(m_pfd, pidl, GENERIC_READ, &pstg->pstm, m_uliCompleted, m_uliTotal, m_ppd, (pfe->lindex == m_nEndIndex));
|
|
EVAL(SUCCEEDED(_DoProgressForLegacySystemsPost(pidl, (pfe->lindex == m_nEndIndex))));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// The user may have cancelled
|
|
ASSERT(HRESULT_FROM_WIN32(ERROR_CANCELLED) == hr);
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
_CloseProgressDialog();
|
|
}
|
|
|
|
//TraceMsg(TF_FTPDRAGDROP, "CFtpObj::GetData() CFtpStm_Create() returned hr=%#08lx", hr);
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IsEqualFORMATETC
|
|
|
|
DESCRIPTION:
|
|
The two fields of a FORMATETC that need to match to be equivalent are:
|
|
cfFormat and lindex.
|
|
\**/
|
|
BOOL IsEqualFORMATETC(FORMATETC * pfe1, FORMATETC * pfe2)
|
|
{
|
|
BOOL fIsEqual = FALSE;
|
|
|
|
if ((pfe1->cfFormat == pfe2->cfFormat) && (pfe1->lindex == pfe2->lindex))
|
|
{
|
|
fIsEqual = TRUE;
|
|
}
|
|
|
|
return fIsEqual;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _FreeExtraData
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
int CFtpObj::_DSA_FreeCB(LPVOID pvItem, LPVOID pvlparam)
|
|
{
|
|
FORMATETC_STGMEDIUM * pfs = (FORMATETC_STGMEDIUM *) pvItem;
|
|
|
|
if (EVAL(pfs))
|
|
ReleaseStgMedium(&(pfs->medium));
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _FindSetDataIndex
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
int CFtpObj::_FindExtraDataIndex(FORMATETC *pfe)
|
|
{
|
|
int nIndex;
|
|
|
|
for (nIndex = (DSA_GetItemCount(m_hdsaSetData) - 1); nIndex >= 0; nIndex--)
|
|
{
|
|
FORMATETC_STGMEDIUM * pfs = (FORMATETC_STGMEDIUM *) DSA_GetItemPtr(m_hdsaSetData, nIndex);
|
|
|
|
if (IsEqualFORMATETC(pfe, &pfs->formatEtc))
|
|
{
|
|
return nIndex;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: _SetExtraData
|
|
|
|
DESCRIPTION:
|
|
We don't render the data, but we will carry it because someone may need
|
|
or want it. This is the case with the drag source's defview pushing in
|
|
the icon points via CFSTR_SHELLIDLISTOFFSET for the drop target.
|
|
\**/
|
|
HRESULT CFtpObj::_SetExtraData(FORMATETC *pfe, STGMEDIUM *pstg, BOOL fRelease)
|
|
{
|
|
HRESULT hr;
|
|
int nIndex = _FindExtraDataIndex(pfe);
|
|
|
|
// Do we already have someone's copy?
|
|
if (-1 == nIndex)
|
|
{
|
|
FORMATETC_STGMEDIUM fs;
|
|
|
|
fs.formatEtc = *pfe;
|
|
|
|
// If there is a pointer, copy the data because we can't maintain the lifetime
|
|
// of the pointer.
|
|
if (fs.formatEtc.ptd)
|
|
{
|
|
fs.dvTargetDevice = *(pfe->ptd);
|
|
fs.formatEtc.ptd = &fs.dvTargetDevice;
|
|
}
|
|
|
|
hr = CopyStgMediumWrap(pstg, &fs.medium);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
// No, so just append it to the end.
|
|
DSA_AppendItem(m_hdsaSetData, &fs);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FORMATETC_STGMEDIUM fs;
|
|
|
|
DSA_GetItem(m_hdsaSetData, nIndex, &fs);
|
|
// Free the previous guy.
|
|
ReleaseStgMedium(&fs.medium);
|
|
|
|
// Yes, so Replace it.
|
|
hr = CopyStgMediumWrap(pstg, &fs.medium);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
// Replace the data.
|
|
DSA_SetItem(m_hdsaSetData, nIndex, &fs);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
typedef struct
|
|
{
|
|
DWORD dwVersion;
|
|
DWORD dwExtraSize; // After pidl list
|
|
BOOL fFGDRendered;
|
|
DWORD dwReserved1;
|
|
DWORD dwReserved2;
|
|
} FTPDATAOBJ_PERSISTSTRUCT;
|
|
|
|
|
|
/**\
|
|
FUNCTION: FormatEtcSaveToStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT FormatEtcSaveToStream(IStream *pStm, FORMATETC * pFormatEtc)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
// We don't support ptd because where would the allocation be
|
|
// on the load?
|
|
if (EVAL(NULL == pFormatEtc->ptd))
|
|
{
|
|
WCHAR szFormatName[MAX_PATH];
|
|
|
|
if (EVAL(GetClipboardFormatNameW(pFormatEtc->cfFormat, szFormatName, ARRAYSIZE(szFormatName))))
|
|
{
|
|
DWORD cbFormatNameSize = ((lstrlenW(szFormatName) + 1) * sizeof(szFormatName[0]));
|
|
|
|
hr = pStm->Write(pFormatEtc, SIZEOF(*pFormatEtc), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = pStm->Write(&cbFormatNameSize, SIZEOF(cbFormatNameSize), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = pStm->Write(szFormatName, cbFormatNameSize, NULL);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: FormatEtcLoadFromStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT FormatEtcLoadFromStream(IStream *pStm, FORMATETC * pFormatEtc)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
hr = pStm->Read(pFormatEtc, SIZEOF(*pFormatEtc), NULL);
|
|
ASSERT(NULL == pFormatEtc->ptd); // We don't support this.
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
DWORD cbFormatNameSize;
|
|
|
|
hr = pStm->Read(&cbFormatNameSize, SIZEOF(cbFormatNameSize), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
WCHAR szFormatName[MAX_PATH];
|
|
|
|
hr = pStm->Read(szFormatName, cbFormatNameSize, NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
pFormatEtc->cfFormat = (CLIPFORMAT)RegisterClipboardFormatW(szFormatName);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
typedef struct
|
|
{
|
|
DWORD dwVersion;
|
|
DWORD dwExtraSize; // After this struct
|
|
DWORD dwTymed; // What type of data is stored?
|
|
BOOL fUnkForRelease; // Did we save the object after this?
|
|
DWORD dwReserved1; //
|
|
DWORD dwReserved2; //
|
|
} STGMEDIUM_PERSISTSTRUCT;
|
|
|
|
/**\
|
|
FUNCTION: StgMediumSaveToStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT StgMediumSaveToStream(IStream *pStm, STGMEDIUM * pMedium)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
STGMEDIUM_PERSISTSTRUCT smps = {0};
|
|
|
|
smps.dwVersion = 1;
|
|
smps.dwTymed = pMedium->tymed;
|
|
|
|
switch (pMedium->tymed)
|
|
{
|
|
case TYMED_HGLOBAL:
|
|
{
|
|
IStream * pstmHGlobal;
|
|
|
|
hr = CreateStreamOnHGlobal(pMedium->hGlobal, FALSE, &pstmHGlobal);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
STATSTG statStg;
|
|
|
|
hr = pstmHGlobal->Stat(&statStg, STATFLAG_NONAME);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
ASSERT(!statStg.cbSize.HighPart);
|
|
smps.dwExtraSize = statStg.cbSize.LowPart;
|
|
hr = pStm->Write(&smps, SIZEOF(smps), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = pstmHGlobal->CopyTo(pStm, statStg.cbSize, NULL, NULL);
|
|
}
|
|
|
|
pstmHGlobal->Release();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TYMED_FILE:
|
|
smps.dwExtraSize = ((lstrlenW(pMedium->lpszFileName) + 1) * sizeof(WCHAR));
|
|
|
|
hr = pStm->Write(&smps, SIZEOF(smps), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = pStm->Write(pMedium->lpszFileName, smps.dwExtraSize, NULL);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
break;
|
|
|
|
case TYMED_GDI:
|
|
case TYMED_MFPICT:
|
|
case TYMED_ENHMF:
|
|
case TYMED_ISTORAGE:
|
|
case TYMED_ISTREAM:
|
|
default:
|
|
ASSERT(0); // What are you doing? Impl this if you need it.
|
|
hr = E_NOTIMPL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
LPWSTR OLESTRAlloc(DWORD cchSize)
|
|
{
|
|
return (LPWSTR) new WCHAR [cchSize + 1];
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: StgMediumLoadFromStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT StgMediumLoadFromStream(IStream *pStm, STGMEDIUM * pMedium)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm && pMedium)
|
|
{
|
|
STGMEDIUM_PERSISTSTRUCT smps;
|
|
|
|
pMedium->pUnkForRelease = NULL;
|
|
hr = pStm->Read(&smps, SIZEOF(smps), NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
pMedium->tymed = smps.dwTymed;
|
|
ASSERT(!pMedium->pUnkForRelease);
|
|
|
|
switch (pMedium->tymed)
|
|
{
|
|
case TYMED_HGLOBAL:
|
|
{
|
|
IStream * pstmTemp;
|
|
hr = CreateStreamOnHGlobal(NULL, FALSE, &pstmTemp);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
ULARGE_INTEGER uli = {0};
|
|
|
|
uli.LowPart = smps.dwExtraSize;
|
|
hr = pStm->CopyTo(pstmTemp, uli, NULL, NULL);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = GetHGlobalFromStream(pstmTemp, &pMedium->hGlobal);
|
|
}
|
|
|
|
pstmTemp->Release();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TYMED_FILE:
|
|
pMedium->lpszFileName = OLESTRAlloc(smps.dwExtraSize / sizeof(WCHAR));
|
|
if (pMedium->lpszFileName)
|
|
hr = pStm->Read(pMedium->lpszFileName, smps.dwExtraSize, NULL);
|
|
else
|
|
hr = E_OUTOFMEMORY;
|
|
break;
|
|
|
|
case TYMED_GDI:
|
|
case TYMED_MFPICT:
|
|
case TYMED_ENHMF:
|
|
case TYMED_ISTORAGE:
|
|
case TYMED_ISTREAM:
|
|
default:
|
|
ASSERT(0); // What are you doing? Impl this if you need it.
|
|
// Some future version must have done the save, so skip the
|
|
// data so we don't leave unread data.
|
|
if (0 != smps.dwExtraSize)
|
|
{
|
|
LARGE_INTEGER li = {0};
|
|
|
|
li.LowPart = smps.dwExtraSize;
|
|
EVAL(SUCCEEDED(pStm->Seek(li, STREAM_SEEK_CUR, NULL)));
|
|
}
|
|
hr = E_NOTIMPL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: FORMATETC_STGMEDIUMSaveToStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT FORMATETC_STGMEDIUMSaveToStream(IStream *pStm, FORMATETC_STGMEDIUM * pfdops)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
hr = FormatEtcSaveToStream(pStm, &pfdops->formatEtc);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = StgMediumSaveToStream(pStm, &pfdops->medium);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: FORMATETC_STGMEDIUMLoadFromStream
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT FORMATETC_STGMEDIUMLoadFromStream(IStream *pStm, FORMATETC_STGMEDIUM * pfdops)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
hr = FormatEtcLoadFromStream(pStm, &pfdops->formatEtc);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = StgMediumLoadFromStream(pStm, &pfdops->medium);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
|
|
// IAsynchDataObject Impl
|
|
|
|
|
|
|
|
/**\
|
|
FUNCTION: IAsyncOperation::GetAsyncMode
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::GetAsyncMode(BOOL * pfIsOpAsync)
|
|
{
|
|
*pfIsOpAsync = TRUE;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IAsyncOperation::StartOperation
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::StartOperation(IBindCtx * pbcReserved)
|
|
{
|
|
ASSERT(!pbcReserved);
|
|
m_fDidAsynchStart = TRUE;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IAsyncOperation::InOperation
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::InOperation(BOOL * pfInAsyncOp)
|
|
{
|
|
if (m_fDidAsynchStart)
|
|
*pfInAsyncOp = TRUE;
|
|
else
|
|
*pfInAsyncOp = FALSE;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IAsyncOperation::EndOperation
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::EndOperation(HRESULT hResult, IBindCtx * pbcReserved, DWORD dwEffects)
|
|
{
|
|
if (SUCCEEDED(hResult) &&
|
|
(DROPEFFECT_MOVE == dwEffects))
|
|
{
|
|
CFtpPidlList * pPidlListNew = CreateRelativePidlList(m_pff, m_pflHfpl);
|
|
|
|
if (pPidlListNew)
|
|
{
|
|
Misc_DeleteHfpl(m_pff, GetDesktopWindow(), pPidlListNew);
|
|
pPidlListNew->Release();
|
|
}
|
|
}
|
|
|
|
m_fDidAsynchStart = FALSE;
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
|
|
|
|
// IPersistStream Impl
|
|
|
|
|
|
|
|
/**\
|
|
FUNCTION: IPersistStream::Load
|
|
|
|
DESCRIPTION:
|
|
See IPersistStream::Save() for the layout of the stream.
|
|
\**/
|
|
HRESULT CFtpObj::Load(IStream *pStm)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
FTPDATAOBJ_PERSISTSTRUCT fdoss;
|
|
DWORD dwNumPidls;
|
|
DWORD dwNumStgMedium;
|
|
|
|
hr = pStm->Read(&fdoss, SIZEOF(fdoss), NULL); // #1
|
|
// If we rev the version, read it now (fdoss.dwVersion)
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
LPITEMIDLIST pidl = NULL; // ILLoadFromStream frees the param
|
|
|
|
ASSERT(!m_pff);
|
|
m_fFGDRendered = fdoss.fFGDRendered;
|
|
|
|
hr = ILLoadFromStream(pStm, &pidl); // #2
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = SHBindToIDList(pidl, NULL, IID_CFtpFolder, (void **)&m_pff);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
m_pfd = m_pff->GetFtpDir();
|
|
|
|
ASSERT(m_pfd);
|
|
ILFree(pidl);
|
|
}
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = pStm->Read(&dwNumPidls, SIZEOF(dwNumPidls), NULL); // #3
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = CFtpPidlList_Create(0, NULL, &m_pflHfpl);
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
for (int nIndex = 0; (nIndex < (int)dwNumPidls) && SUCCEEDED(hr); nIndex++)
|
|
{
|
|
LPITEMIDLIST pidl = NULL; // ILLoadFromStream frees the param
|
|
|
|
hr = ILLoadFromStream(pStm, &pidl); // #4
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
hr = m_pflHfpl->InsertSorted(pidl);
|
|
ILFree(pidl);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = pStm->Read(&dwNumStgMedium, SIZEOF(dwNumStgMedium), NULL); // #5
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
for (int nIndex = 0; (nIndex < (int)dwNumStgMedium) && SUCCEEDED(hr); nIndex++)
|
|
{
|
|
FORMATETC_STGMEDIUM fs;
|
|
|
|
hr = FORMATETC_STGMEDIUMLoadFromStream(pStm, &fs); // #6
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
DSA_AppendItem(m_hdsaSetData, &fs);
|
|
}
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
// We may be reading a version newer than us, so skip their data.
|
|
if (0 != fdoss.dwExtraSize)
|
|
{
|
|
LARGE_INTEGER li = {0};
|
|
|
|
li.LowPart = fdoss.dwExtraSize;
|
|
hr = pStm->Seek(li, STREAM_SEEK_CUR, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IPersistStream::Save
|
|
|
|
DESCRIPTION:
|
|
The stream will be layed out in the following way:
|
|
|
|
Version 1:
|
|
1. FTPDATAOBJ_PERSISTSTRUCT - Constant sized data.
|
|
<PidlList BEGIN>
|
|
2. PIDL pidl - Pidl for m_pff. It will be a public pidl (fully qualified
|
|
from the shell root)
|
|
3. DWORD dwNumPidls - Number of pidls coming.
|
|
4. PIDL pidl(n) - Pidl in slot (n) of m_pflHfpl
|
|
<PidlList END>
|
|
5. DWORD dwNumStgMedium - Number of FORMATETC_STGMEDIUMs coming
|
|
6. FORMATETC_STGMEDIUM fmtstg(n) - dwNumStgMedium FORMATETC_STGMEDIUMs.
|
|
\**/
|
|
HRESULT CFtpObj::Save(IStream *pStm, BOOL fClearDirty)
|
|
{
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
if (pStm)
|
|
{
|
|
FTPDATAOBJ_PERSISTSTRUCT fdoss = {0};
|
|
DWORD dwNumPidls = m_pflHfpl->GetCount();
|
|
DWORD dwNumStgMedium = DSA_GetItemCount(m_hdsaSetData);
|
|
|
|
fdoss.dwVersion = 1;
|
|
fdoss.fFGDRendered = m_fFGDRendered;
|
|
hr = pStm->Write(&fdoss, SIZEOF(fdoss), NULL); // #1
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
ASSERT(m_pff);
|
|
hr = ILSaveToStream(pStm, m_pff->GetPublicRootPidlReference()); // #2
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = pStm->Write(&dwNumPidls, SIZEOF(dwNumPidls), NULL); // #3
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
for (int nIndex = 0; (nIndex < (int)dwNumPidls) && SUCCEEDED(hr); nIndex++)
|
|
{
|
|
LPITEMIDLIST pidlCur = m_pflHfpl->GetPidl(nIndex);
|
|
|
|
ASSERT(pidlCur);
|
|
hr = ILSaveToStream(pStm, pidlCur); // #4
|
|
}
|
|
}
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
hr = pStm->Write(&dwNumStgMedium, SIZEOF(dwNumStgMedium), NULL); // #5
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
for (int nIndex = 0; (nIndex < (int)dwNumStgMedium) && SUCCEEDED(hr); nIndex++)
|
|
{
|
|
FORMATETC_STGMEDIUM fs;
|
|
|
|
DSA_GetItem(m_hdsaSetData, nIndex, &fs);
|
|
|
|
hr = FORMATETC_STGMEDIUMSaveToStream(pStm, &fs); // #6
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
#define MAX_STREAM_SIZE (500 * 1024) // 500k
|
|
/**\
|
|
FUNCTION: IPersistStream::GetSizeMax
|
|
|
|
DESCRIPTION:
|
|
Now this is tough. I can't calculate the real value because I don't know
|
|
how big the hglobals are going to be for the user provided data. I will
|
|
assume everything fits in
|
|
\**/
|
|
HRESULT CFtpObj::GetSizeMax(ULARGE_INTEGER * pcbSize)
|
|
{
|
|
if (pcbSize)
|
|
{
|
|
pcbSize->HighPart = 0;
|
|
pcbSize->LowPart = MAX_STREAM_SIZE;
|
|
}
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
|
|
|
|
// IDataObject Impl
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::GetData
|
|
|
|
DESCRIPTION:
|
|
Render the data in the requested format and put it into the
|
|
STGMEDIUM structure.
|
|
\**/
|
|
HRESULT CFtpObj::GetData(LPFORMATETC pfe, LPSTGMEDIUM pstg)
|
|
{
|
|
int ife;
|
|
HRESULT hr;
|
|
|
|
hr = _FindDataForGet(pfe, &ife);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (ife == DROP_FCont)
|
|
hr = _RenderFileContents(pfe, pstg);
|
|
else
|
|
{
|
|
hr = _ForceRender(ife);
|
|
if (SUCCEEDED(hr)) // May not succeed for security reasons.
|
|
{
|
|
ASSERT(m_stgCache[ife].hGlobal);
|
|
|
|
// It's possible to use the hacking STGMEDIUM.pUnkForRelease to give away
|
|
// pointers to our data, but we then need massive amounts of code to babysite
|
|
// the lifetime of those pointers. This becomes more work when ::SetData() can
|
|
// replace that data, so we just take the hit of the memcpy for less code.
|
|
hr = CopyStgMediumWrap(&m_stgCache[ife], pstg);
|
|
ASSERT(SUCCEEDED(hr));
|
|
ASSERT(NULL == pstg->pUnkForRelease);
|
|
//TraceMsg(TF_FTPDRAGDROP, "CFtpObj::GetData() pstg->hGlobal=%#08lx. pstg->pUnkForRelease=%#08lx.", pstg->hGlobal, pstg->pUnkForRelease);
|
|
}
|
|
}
|
|
|
|
TraceMsgWithFormat(TF_FTPDRAGDROP, "CFtpObj::GetData()", pfe, "Format in static list", hr);
|
|
}
|
|
else
|
|
{
|
|
int nIndex = _FindExtraDataIndex(pfe);
|
|
|
|
if (-1 == nIndex)
|
|
hr = E_FAIL;
|
|
else
|
|
{
|
|
FORMATETC_STGMEDIUM fs;
|
|
|
|
DSA_GetItem(m_hdsaSetData, nIndex, &fs);
|
|
hr = CopyStgMediumWrap(&fs.medium, pstg);
|
|
}
|
|
|
|
TraceMsgWithFormat(TF_FTPDRAGDROP, "CFtpObj::GetData()", pfe, "Looking in dyn list", hr);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
IDataObject::GetDataHere
|
|
|
|
Render the data in the requested format and put it into the
|
|
object provided by the caller.
|
|
\**/
|
|
HRESULT CFtpObj::GetDataHere(FORMATETC *pfe, STGMEDIUM *pstg)
|
|
{
|
|
TraceMsg(TF_FTPDRAGDROP, "CFtpObj::GetDataHere() pfe->cfFormat=%d.", pfe->cfFormat);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::QueryGetData
|
|
|
|
DESCRIPTION:
|
|
Indicate whether we could provide data in the requested format.
|
|
\**/
|
|
HRESULT CFtpObj::QueryGetData(FORMATETC *pfe)
|
|
{
|
|
int ife;
|
|
HRESULT hr = _FindDataForGet(pfe, &ife);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
// If it wasn't one of the types we offer, see if it was given to us via
|
|
// IDataObject::SetData().
|
|
int nIndex = _FindExtraDataIndex(pfe);
|
|
|
|
if (-1 != nIndex)
|
|
hr = S_OK;
|
|
}
|
|
|
|
TraceMsgWithFormat(TF_FTPDRAGDROP, "CFtpObj::QueryGetData()", pfe, "", hr);
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::GetCanonicalFormatEtc
|
|
|
|
DESCRIPTION:
|
|
Our data are not sensitive to device-specific renderings,
|
|
so we do what the book tells us to do.
|
|
|
|
Or we *try* to do what the book tells us to do.
|
|
|
|
OLE random documentation of the day:
|
|
IDataObject::GetCanonicalFormatEtc.
|
|
|
|
Turns out that the man page contradicts itself within sentences:
|
|
|
|
DATA_S_SAMEFORMATETC - The FORMATETC structures are the same
|
|
and NULL is returned in pfeOut.
|
|
|
|
If the data object never provides device-specific renderings,
|
|
the implementation of IDataObject::GetCanonicalFormatEtc
|
|
simply copies the input FORMATETC to the output FORMATETC,
|
|
stores a null in the ptd field, and returns DATA_S_SAMEFORMATETC.
|
|
|
|
And it turns out that the shell doesn't do *either* of these things.
|
|
It just returns DATA_S_SAMEFORMATETC and doesn't touch pfeOut.
|
|
|
|
The book is even more confused. Under pfeOut, it says
|
|
|
|
The value is NULL if the method returns DATA_S_SAMEFORMATETC.
|
|
|
|
This makes no sense. The caller provides the value of pfeOut.
|
|
How can the caller possibly know that the method is going to return
|
|
DATA_S_SAMEFORMATETC before it calls it? If you expect the
|
|
method to write "pfeOut = 0" before returning, you're nuts. That
|
|
communicates nothing to the caller.
|
|
|
|
I'll just do what the shell does.
|
|
\**/
|
|
HRESULT CFtpObj::GetCanonicalFormatEtc(FORMATETC *pfeIn, FORMATETC *pfeOut)
|
|
{
|
|
return DATA_S_SAMEFORMATETC;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::SetData
|
|
|
|
DESCRIPTION:
|
|
We let people change TYMED_HGLOBAL gizmos, but nothing else.
|
|
|
|
We need to do a careful two-step when replacing the HGLOBAL.
|
|
If the user gave us a plain HGLOBAL without a pUnkForRelease,
|
|
we need to invent our own pUnkForRelease to track it. But we
|
|
don't want to release the old STGMEDIUM until we're sure we
|
|
can accept the new one.
|
|
|
|
fRelease == 0 makes life doubly interesting, because we also
|
|
have to clone the HGLOBAL (and remember to free the clone on the
|
|
error path).
|
|
|
|
_SOMEDAY_/TODO -- Need to support PerformedDropEffect so we can
|
|
clean up stuff on a cut/paste.
|
|
\**/
|
|
HRESULT CFtpObj::SetData(FORMATETC *pfe, STGMEDIUM *pstg, BOOL fRelease)
|
|
{
|
|
int ife;
|
|
HRESULT hr;
|
|
|
|
hr = _FindData(pfe, &ife);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (ife == DROP_FCont)
|
|
{
|
|
TraceMsg(TF_FTPDRAGDROP, "CFtpObj::SetData(FORMATETC.cfFormat=%d) ife == DROP_FCont", pfe->cfFormat);
|
|
hr = DV_E_FORMATETC;
|
|
}
|
|
else
|
|
{
|
|
ASSERT(g_dropTypes[ife].tymed == TYMED_HGLOBAL);
|
|
ASSERT(pstg->tymed == TYMED_HGLOBAL);
|
|
if (EVAL(pstg->hGlobal))
|
|
{
|
|
STGMEDIUM stg = {0};
|
|
|
|
hr = CopyStgMediumWrap(pstg, &stg);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
ReleaseStgMedium(&m_stgCache[ife]);
|
|
m_stgCache[ife] = stg;
|
|
}
|
|
}
|
|
else
|
|
{ // Tried to SetData a _DelayRender
|
|
hr = DV_E_STGMEDIUM; // You idiot you
|
|
}
|
|
}
|
|
|
|
TraceMsgWithFormat(TF_FTPDRAGDROP, "CFtpObj::SetData()", pfe, "in static list", hr);
|
|
}
|
|
else
|
|
{
|
|
hr = _SetExtraData(pfe, pstg, fRelease);
|
|
TraceMsgWithFormat(TF_FTPDRAGDROP, "CFtpObj::SetData()", pfe, "in dyn list", hr);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::EnumFormatEtc
|
|
|
|
DESCRIPTION:
|
|
_UNDOCUMENTED_: If you drag something from a DefView, it will
|
|
check the data object to see if it has a hida. If so, then it
|
|
will cook up a CFSTR_SHELLIDLISTOFFSET *for you* and SetData
|
|
the information into the data object. So in order to get
|
|
position-aware drag/drop working, you must allow DefView to change
|
|
your CFSTR_SHELLIDLISTOFFSET.
|
|
|
|
We allow all FORMATETCs to be modified except for FileContents.
|
|
\**/
|
|
HRESULT CFtpObj::EnumFormatEtc(DWORD dwDirection, IEnumFORMATETC **ppenum)
|
|
{
|
|
HRESULT hres;
|
|
|
|
switch (dwDirection)
|
|
{
|
|
case DATADIR_GET:
|
|
hres = CFtpEfe_Create(DROP_OFFERMAX - DROP_FCont, &g_dropTypes[DROP_FCont],
|
|
&m_stgCache[DROP_FCont], this, ppenum);
|
|
TraceMsg(TF_FTPDRAGDROP, "CFtpObj::EnumFormatEtc(DATADIR_GET) CFtpEfe_Create() returned hres=%#08lx", hres);
|
|
break;
|
|
|
|
case DATADIR_SET:
|
|
hres = CFtpEfe_Create(DROP_OFFERMAX - DROP_OFFERMIN, &g_dropTypes[DROP_OFFERMIN],
|
|
&m_stgCache[DROP_OFFERMIN], NULL, ppenum);
|
|
TraceMsg(TF_FTPDRAGDROP, "CFtpObj::EnumFormatEtc(DATADIR_SET) CFtpEfe_Create() returned hres=%#08lx", hres);
|
|
break;
|
|
|
|
default:
|
|
ASSERT(0);
|
|
hres = E_NOTIMPL;
|
|
break;
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::DAdvise
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::DAdvise(FORMATETC *pfe, DWORD advfl, IAdviseSink *padv, DWORD *pdwConnection)
|
|
{
|
|
return OLE_E_ADVISENOTSUPPORTED;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::DUnadvise
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::DUnadvise(DWORD dwConnection)
|
|
{
|
|
return OLE_E_ADVISENOTSUPPORTED;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: IDataObject::EnumDAdvise
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj::EnumDAdvise(IEnumSTATDATA **ppeadv)
|
|
{
|
|
return OLE_E_ADVISENOTSUPPORTED;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: CFtpObj_Create
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj_Create(CFtpFolder * pff, CFtpPidlList * pflHfpl, REFIID riid, LPVOID * ppvObj)
|
|
{
|
|
HRESULT hres;
|
|
CFtpObj * pfo;
|
|
|
|
*ppvObj = NULL;
|
|
|
|
hres = CFtpObj_Create(pff, pflHfpl, &pfo);
|
|
if (EVAL(SUCCEEDED(hres)))
|
|
{
|
|
pfo->QueryInterface(riid, ppvObj);
|
|
pfo->Release();
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: CFtpObj_Create
|
|
|
|
DESCRIPTION:
|
|
\**/
|
|
HRESULT CFtpObj_Create(CFtpFolder * pff, CFtpPidlList * pflHfpl, CFtpObj ** ppfo)
|
|
{
|
|
HRESULT hres = S_OK;
|
|
|
|
if (EVAL(pflHfpl->GetCount()))
|
|
{
|
|
*ppfo = new CFtpObj();
|
|
|
|
if (EVAL(*ppfo))
|
|
{
|
|
CFtpObj * pfo = *ppfo;
|
|
pfo->m_pfd = pff->GetFtpDir();
|
|
|
|
if (EVAL(pfo->m_pfd))
|
|
{
|
|
pfo->m_pff = pff;
|
|
if (pff)
|
|
pff->AddRef();
|
|
|
|
IUnknown_Set(&pfo->m_pflHfpl, pflHfpl);
|
|
|
|
if (pfo->m_pflHfpl->GetCount() == 1)
|
|
{
|
|
pfo->m_stgCache[DROP_URL].tymed = TYMED_HGLOBAL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hres = E_FAIL;
|
|
(*ppfo)->Release();
|
|
*ppfo = NULL;
|
|
}
|
|
}
|
|
else
|
|
hres = E_OUTOFMEMORY;
|
|
|
|
}
|
|
else
|
|
{
|
|
*ppfo = NULL;
|
|
hres = E_INVALIDARG; /* Trying to get UI object of nil? */
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: CFtpObj_Create
|
|
|
|
DESCRIPTION:
|
|
This will be called by the Class Factory when the IDataObject gets
|
|
persisted and then wants to be recreated in a new process. (Happens
|
|
after the original thread/process calls OleFlushClipboard.
|
|
\**/
|
|
HRESULT CFtpObj_Create(REFIID riid, void ** ppvObj)
|
|
{
|
|
HRESULT hr = E_OUTOFMEMORY;
|
|
CFtpObj * pfo = new CFtpObj();
|
|
|
|
*ppvObj = NULL;
|
|
if (pfo)
|
|
{
|
|
hr = pfo->QueryInterface(riid, ppvObj);
|
|
pfo->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
#define SETDATA_GROWSIZE 3
|
|
|
|
/**\
|
|
Constructor
|
|
\**/
|
|
CFtpObj::CFtpObj() : m_cRef(1)
|
|
{
|
|
DllAddRef();
|
|
|
|
// This needs to be allocated in Zero Inited Memory.
|
|
// Assert that all Member Variables are inited to Zero.
|
|
ASSERT(!m_pff);
|
|
ASSERT(!m_pfd);
|
|
ASSERT(!m_pflHfpl);
|
|
ASSERT(!m_fDidAsynchStart);
|
|
|
|
// NT #245306: If the user drags files from an FTP window (Thread 1)
|
|
// to a shell window (Thread 2), the shell window will do
|
|
// the drop on a background thread (thread 3). Since the
|
|
// UI thread is no longer blocked, the user can now close
|
|
// the window. The problem is that OLE is using Thread 2
|
|
// for marshalling. In order to solve this problem, we
|
|
// ref count the thread for items that rely on it.
|
|
// This include FTP, normal Download, and other things
|
|
// in the future.
|
|
// SHIncrementThreadModelessCount();
|
|
|
|
m_nStartIndex = -1; // -1 means we don't know the start.
|
|
m_fFGDRendered = FALSE;
|
|
m_fCheckSecurity = TRUE; // We need to keep checking.
|
|
|
|
m_hdsaSetData = DSA_Create(sizeof(FORMATETC_STGMEDIUM), SETDATA_GROWSIZE);
|
|
|
|
for (int nIndex = 0; nIndex < ARRAYSIZE(c_stgInit); nIndex++)
|
|
{
|
|
ASSERT(nIndex < ARRAYSIZE(m_stgCache));
|
|
m_stgCache[nIndex] = c_stgInit[nIndex];
|
|
}
|
|
|
|
_RefThread();
|
|
// The receiver may use us in the background, so make sure that our thread
|
|
// doesn't go away.
|
|
LEAK_ADDREF(LEAK_CFtpObj);
|
|
}
|
|
|
|
|
|
/**\
|
|
Destructor
|
|
\**/
|
|
CFtpObj::~CFtpObj()
|
|
{
|
|
int ife;
|
|
|
|
_CloseProgressDialog();
|
|
for (ife = DROP_OFFERMIN; ife < DROP_OFFERMAX; ife++)
|
|
{
|
|
ReleaseStgMedium(&m_stgCache[ife]);
|
|
}
|
|
|
|
if (m_ppd)
|
|
m_ppd->StopProgressDialog();
|
|
|
|
IUnknown_Set((IUnknown **)&m_ppd, NULL);
|
|
IUnknown_Set(&m_pff, NULL);
|
|
IUnknown_Set(&m_pfd, NULL);
|
|
IUnknown_Set(&m_pflHfpl, NULL);
|
|
|
|
DSA_DestroyCallback(m_hdsaSetData, &_DSA_FreeCB, NULL);
|
|
|
|
// NT #245306: If the user drags files from an FTP window (Thread 1)
|
|
// to a shell window (Thread 2), the shell window will do
|
|
// the drop on a background thread (thread 3). Since the
|
|
// UI thread is no longer blocked, the user can now close
|
|
// the window. The problem is that OLE is using Thread 2
|
|
// for marshalling. In order to solve this problem, we
|
|
// ref count the thread for items that rely on it.
|
|
// This include FTP, normal Download, and other things
|
|
// in the future.
|
|
ATOMICRELEASE(m_punkThreadRef);
|
|
|
|
DllRelease();
|
|
LEAK_DELREF(LEAK_CFtpObj);
|
|
}
|
|
|
|
|
|
|
|
// ** IUnknown Interface **
|
|
|
|
|
|
ULONG CFtpObj::AddRef()
|
|
{
|
|
m_cRef++;
|
|
return m_cRef;
|
|
}
|
|
|
|
ULONG CFtpObj::Release()
|
|
{
|
|
ASSERT(m_cRef > 0);
|
|
m_cRef--;
|
|
|
|
if (m_cRef > 0)
|
|
return m_cRef;
|
|
|
|
delete this;
|
|
return 0;
|
|
}
|
|
|
|
|
|
HRESULT CFtpObj::QueryInterface(REFIID riid, void **ppvObj)
|
|
{
|
|
static const QITAB qit[] = {
|
|
QITABENT(CFtpObj, IDataObject),
|
|
QITABENT(CFtpObj, IInternetSecurityMgrSite),
|
|
QITABENT(CFtpObj, IPersist),
|
|
QITABENT(CFtpObj, IPersistStream),
|
|
QITABENT(CFtpObj, IAsyncOperation),
|
|
{ 0 },
|
|
};
|
|
|
|
return QISearch(this, qit, riid, ppvObj);
|
|
}
|