Windows2000/private/shell/ext/ftp/ftpobj.cpp
2020-09-30 17:12:32 +02:00

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);
}