695 lines
22 KiB
C++
695 lines
22 KiB
C++
|
/*****************************************************************************\
|
||
|
FILE: ftpdir.cpp
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Internal object that manages a single FTP directory
|
||
|
|
||
|
The idea is that each FtpSite maintains a linked list of the
|
||
|
FtpDir's that it owns. Gets and Releases are done through the
|
||
|
FtpSite. Each FtpDir retains a non-refcounted pointer back
|
||
|
to the FtpSite that owns it.
|
||
|
|
||
|
The reason this is necessary is that there might be multiple
|
||
|
IShellFolder's all looking at the same physical directory. Since
|
||
|
enumerations are expensive, we cache the enumeration information
|
||
|
here, so that each IShellFolder client can use the information.
|
||
|
|
||
|
This also lets us hold the motd, so that multiple clients can
|
||
|
query for the motd without constantly pinging the site.
|
||
|
\*****************************************************************************/
|
||
|
|
||
|
|
||
|
#include "priv.h"
|
||
|
#include "ftpdir.h"
|
||
|
#include "ftpsite.h"
|
||
|
#include "ftppidl.h"
|
||
|
#include "ftpurl.h"
|
||
|
#include "ftppidl.h"
|
||
|
#include "statusbr.h"
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: GetDisplayPath
|
||
|
|
||
|
DESCRIPTION:
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::GetDisplayPath(LPWSTR pwzDisplayPath, DWORD cchSize)
|
||
|
{
|
||
|
return GetDisplayPathFromPidl(m_pidlFtpDir, pwzDisplayPath, cchSize, FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
An InternetConnect has just completed. Get the motd and cache it.
|
||
|
|
||
|
hint - the connected handle, possibly 0 if error
|
||
|
\*****************************************************************************/
|
||
|
void CFtpDir::CollectMotd(HINTERNET hint)
|
||
|
{
|
||
|
CFtpGlob * pfg = GetFtpResponse(GetFtpSite()->GetCWireEncoding());
|
||
|
|
||
|
if (m_pfgMotd)
|
||
|
m_pfgMotd->Release();
|
||
|
|
||
|
m_pfgMotd = pfg; // m_pfgMotd will take pfg's ref.
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Shove a value into the cached list.
|
||
|
\*****************************************************************************/
|
||
|
void CFtpDir::SetCache(CFtpPidlList * pflHfpl)
|
||
|
{
|
||
|
IUnknown_Set(&m_pflHfpl, pflHfpl);
|
||
|
|
||
|
// If we are flushing the cache, then flush the Ratings info also.
|
||
|
// This way the user can reenter the parent password if wanted.
|
||
|
if (!pflHfpl && m_pfs)
|
||
|
m_pfs->FlushRatingsInfo();
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Get the value out of the cache.
|
||
|
\*****************************************************************************/
|
||
|
CFtpPidlList * CFtpDir::GetHfpl(void)
|
||
|
{
|
||
|
CFtpPidlList * pfl;
|
||
|
|
||
|
pfl = m_pflHfpl;
|
||
|
if (pfl)
|
||
|
pfl->AddRef();
|
||
|
|
||
|
return pfl;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Get the FTP site associated with a directory.
|
||
|
This doesn't AddRef the return value.
|
||
|
\*****************************************************************************/
|
||
|
CFtpSite * CFtpDir::GetFtpSite(void)
|
||
|
{
|
||
|
return m_pfs;
|
||
|
}
|
||
|
|
||
|
|
||
|
CFtpDir * CFtpDir::GetSubFtpDir(CFtpFolder * pff, LPCITEMIDLIST pidl, BOOL fPublic)
|
||
|
{
|
||
|
CFtpDir * pfd = NULL;
|
||
|
|
||
|
if (EVAL(pidl))
|
||
|
{
|
||
|
LPITEMIDLIST pidlChild = GetSubPidl(pff, pidl, fPublic);
|
||
|
|
||
|
if (EVAL(pidlChild))
|
||
|
{
|
||
|
m_pfs->GetFtpDir(pidlChild, &pfd);
|
||
|
ILFree(pidlChild);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return pfd;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPITEMIDLIST CFtpDir::GetSubPidl(CFtpFolder * pff, LPCITEMIDLIST pidlRelative, BOOL fPublic)
|
||
|
{
|
||
|
LPITEMIDLIST pidlRoot = ((fPublic && pff) ? pff->GetPublicPidlRootIDClone() : NULL);
|
||
|
LPITEMIDLIST pidlPublic = ILCombine(pidlRoot, m_pidl);
|
||
|
LPITEMIDLIST pidlFull = NULL;
|
||
|
|
||
|
if (pidlPublic)
|
||
|
{
|
||
|
pidlFull = ILCombine(pidlPublic, pidlRelative);
|
||
|
ILFree(pidlPublic);
|
||
|
}
|
||
|
|
||
|
ILFree(pidlRoot);
|
||
|
return pidlFull;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CFtpDir::AddItem(LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
if (!m_pflHfpl)
|
||
|
return S_OK;
|
||
|
|
||
|
return m_pflHfpl->InsertSorted(pidl);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Get a HINTERNET for this directory.
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::GetHint(HWND hwnd, CStatusBar * psb, HINTERNET * phint, IUnknown * punkSite, CFtpFolder * pff)
|
||
|
{
|
||
|
HRESULT hr = m_pfs->GetHint(hwnd, m_pidlFtpDir, psb, phint, punkSite, pff);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Give a HINTERNET back to the FtpSite.
|
||
|
\*****************************************************************************/
|
||
|
void CFtpDir::ReleaseHint(HINTERNET hint)
|
||
|
{
|
||
|
ASSERT(!hint || m_pfs); // If we have a hint to release, we need to call ::ReleaseHint()
|
||
|
if (m_pfs)
|
||
|
m_pfs->ReleaseHint(m_pidlFtpDir, hint);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CollectMotd
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Perform an operation with a temporary internet handle which is
|
||
|
already connected to the site and resides in the correct directory.
|
||
|
\*****************************************************************************/
|
||
|
STDMETHODIMP CFtpDir::WithHint(CStatusBar * psb, HWND hwnd, HINTPROC hp, LPCVOID pv, IUnknown * punkSite, CFtpFolder * pff)
|
||
|
{
|
||
|
HRESULT hr = E_FAIL;
|
||
|
|
||
|
// Did the user turn off FTP Folders?
|
||
|
// If so, don't connect. This will fix NT #406423 where the user turned
|
||
|
// of FTP Folders because they have a firewall (CISCO filtering Router)
|
||
|
// that will kill packets in such a way the caller (WinSock/Wininet) needs
|
||
|
// to wait for a timeout. During this timeout, the browser will hang causing
|
||
|
// the user to think it crashed.
|
||
|
if (!SHRegGetBoolUSValue(SZ_REGKEY_FTPFOLDER, SZ_REGKEY_USE_OLD_UI, FALSE, FALSE))
|
||
|
{
|
||
|
HINTERNET hint;
|
||
|
HINTPROCINFO hpi;
|
||
|
|
||
|
ASSERTNONCRITICAL; // Cannot do psb (CStatusBar *) with the crst
|
||
|
ASSERT(m_pfs);
|
||
|
hpi.pfd = this;
|
||
|
hpi.hwnd = hwnd;
|
||
|
hpi.psb = psb;
|
||
|
|
||
|
hr = GetHint(hwnd, psb, &hint, punkSite, pff);
|
||
|
if (SUCCEEDED(hr)) // Ok if fails
|
||
|
{
|
||
|
BOOL fReleaseHint = TRUE;
|
||
|
|
||
|
if (hp)
|
||
|
hr = hp(hint, &hpi, (LPVOID)pv, &fReleaseHint);
|
||
|
|
||
|
if (fReleaseHint)
|
||
|
ReleaseHint(hint);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: _SetNameOfCB
|
||
|
|
||
|
DESCRIPTION:
|
||
|
If we were able to rename the file, return the output pidl.
|
||
|
Also tell anybody who cares that this LPITEMIDLIST needs to be refreshed.
|
||
|
|
||
|
The "A" emphasizes that the filename is received in ANSI.
|
||
|
|
||
|
_UNDOCUMENTED_: The documentation on SetNameOf's treatment of
|
||
|
the source pidl is random. It seems to suggest that the source
|
||
|
pidl is ILFree'd by SetNameOf, but it isn't.
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::_SetNameOfCB(HINTERNET hint, HINTPROCINFO * phpi, LPVOID pv, BOOL * pfReleaseHint)
|
||
|
{
|
||
|
LPSETNAMEOFINFO psnoi = (LPSETNAMEOFINFO) pv;
|
||
|
|
||
|
if (phpi->psb)
|
||
|
{
|
||
|
TCHAR szStatus[MAX_PATH];
|
||
|
|
||
|
FtpPidl_GetLastItemDisplayName(psnoi->pidlOld, szStatus, ARRAYSIZE(szStatus));
|
||
|
phpi->psb->SetStatusMessage(IDS_RENAMING, szStatus);
|
||
|
}
|
||
|
|
||
|
// Remember, FTP filenames are always in the ANSI character set
|
||
|
return FtpRenameFilePidlWrap(hint, TRUE, psnoi->pidlOld, psnoi->pidlNew);
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL CFtpDir::_DoesItemExist(HWND hwnd, CFtpFolder * pff, LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
FTP_FIND_DATA wfd;
|
||
|
HRESULT hr = GetFindData(hwnd, FtpPidl_GetLastItemWireName(pidl), &wfd, pff);
|
||
|
|
||
|
return ((S_OK == hr) ? TRUE : FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL CFtpDir::_ConfirmReplaceWithRename(HWND hwnd)
|
||
|
{
|
||
|
TCHAR szTitle[MAX_PATH];
|
||
|
TCHAR szMessage[MAX_PATH];
|
||
|
|
||
|
EVAL(LoadString(HINST_THISDLL, IDS_FTPERR_TITLE, szTitle, ARRAYSIZE(szTitle)));
|
||
|
EVAL(LoadString(HINST_THISDLL, IDS_FTPERR_RENAME_REPLACE, szMessage, ARRAYSIZE(szMessage)));
|
||
|
|
||
|
return ((IDYES == MessageBox(hwnd, szMessage, szTitle, (MB_ICONQUESTION | MB_YESNO))) ? TRUE : FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CFtpDir::SetNameOf(CFtpFolder * pff, HWND hwndOwner, LPCITEMIDLIST pidl,
|
||
|
LPCWSTR pwzName, DWORD dwReserved, LPITEMIDLIST *ppidlOut)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
SETNAMEOFINFO snoi;
|
||
|
CWireEncoding cWireEncoding;
|
||
|
|
||
|
ASSERT(pff);
|
||
|
|
||
|
if (!pwzName)
|
||
|
return E_FAIL;
|
||
|
|
||
|
snoi.pidlOld = pidl;
|
||
|
cWireEncoding.ChangeFtpItemIDName(NULL, pidl, pwzName, IsUTF8Supported(), (LPITEMIDLIST *) &snoi.pidlNew);
|
||
|
|
||
|
if (snoi.pidlNew)
|
||
|
{
|
||
|
#ifdef FEATURE_REPLACE_IN_RENAME
|
||
|
// FEATURE: Currently, if the user renames file <a> to file <b> and a file <b>
|
||
|
// already exists, we fail with the error message, "file already exists...".
|
||
|
// If we had an excess of time, I would turn on this feature. It would
|
||
|
// ask the user if they want to replace the file. I would then need to add the
|
||
|
// code that deletes that file and checks for delete errors.
|
||
|
|
||
|
// Does it already exist? We don't care if we don't have an hwnd because
|
||
|
// we can't ask the user to replace so we will just go ahead.
|
||
|
if (hwndOwner && _DoesItemExist(hwndOwner, pff, snoi.pidlNew))
|
||
|
{
|
||
|
// Yes, so let's make sure it's OK with the user to replace it.
|
||
|
hr = (_ConfirmReplaceWithRename(hwndOwner) ? S_OK : HRESULT_FROM_WIN32(ERROR_CANCELLED));
|
||
|
todo; // Add the delete call here.
|
||
|
}
|
||
|
#endif FEATURE_REPLACE_IN_RENAME
|
||
|
|
||
|
if (S_OK == hr)
|
||
|
{
|
||
|
hr = WithHint(NULL, hwndOwner, _SetNameOfCB, (LPVOID) &snoi, NULL, pff);
|
||
|
if (SUCCEEDED(hr)) // Will fail if use didn't have permission to rename
|
||
|
{
|
||
|
// WARNING: The date/time stamp on the server may be different than what we give to SHChangeNotify()
|
||
|
// but this is probably reasonable for perf reasons.
|
||
|
FtpChangeNotify(hwndOwner, FtpPidl_DirChoose(pidl, SHCNE_RENAMEFOLDER, SHCNE_RENAMEITEM), pff, this, pidl, snoi.pidlNew, TRUE);
|
||
|
|
||
|
if (ppidlOut)
|
||
|
*ppidlOut = ILClone(snoi.pidlNew);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ILFree((LPITEMIDLIST) snoi.pidlNew);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPCITEMIDLIST CFtpDir::GetPidlFromWireName(LPCWIRESTR pwWireName)
|
||
|
{
|
||
|
LPITEMIDLIST pidlToFind = NULL;
|
||
|
LPITEMIDLIST pidlTemp;
|
||
|
WCHAR wzDisplayName[MAX_PATH];
|
||
|
|
||
|
// This isn't valid because the code page could be wrong, but we don't care
|
||
|
// because it's not used in the search for the pidl, the pwWireName is.
|
||
|
SHAnsiToUnicode(pwWireName, wzDisplayName, ARRAYSIZE(wzDisplayName));
|
||
|
if (m_pflHfpl && SUCCEEDED(FtpItemID_CreateFake(wzDisplayName, pwWireName, FALSE, FALSE, FALSE, &pidlTemp)))
|
||
|
{
|
||
|
// PERF: log 2 (sizeof(m_pflHfpl))
|
||
|
pidlToFind = m_pflHfpl->FindPidl(pidlTemp, FALSE);
|
||
|
// We will try again and this time allow for the case to not match
|
||
|
if (!pidlToFind)
|
||
|
pidlToFind = m_pflHfpl->FindPidl(pidlTemp, TRUE);
|
||
|
ILFree(pidlTemp);
|
||
|
}
|
||
|
|
||
|
return pidlToFind;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPCITEMIDLIST CFtpDir::GetPidlFromDisplayName(LPCWSTR pwzDisplayName)
|
||
|
{
|
||
|
WIRECHAR wWireName[MAX_PATH];
|
||
|
CWireEncoding * pwe = GetFtpSite()->GetCWireEncoding();
|
||
|
|
||
|
EVAL(SUCCEEDED(pwe->UnicodeToWireBytes(NULL, pwzDisplayName, (IsUTF8Supported() ? WIREENC_USE_UTF8 : WIREENC_NONE), wWireName, ARRAYSIZE(wWireName))));
|
||
|
return GetPidlFromWireName(wWireName);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: IsRoot
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Returns FALSE if we are at the "FTP Folder" root level, not
|
||
|
inside an actual FTP site.y
|
||
|
\*****************************************************************************/
|
||
|
BOOL CFtpDir::IsRoot(void)
|
||
|
{
|
||
|
return ILIsEmpty(m_pidl);
|
||
|
}
|
||
|
|
||
|
|
||
|
typedef struct tagGETFINDDATAINFO
|
||
|
{
|
||
|
LPCWIRESTR pwWireName;
|
||
|
LPFTP_FIND_DATA pwfd;
|
||
|
} GETFINDDATAINFO, * LPGETFINDDATAINFO;
|
||
|
|
||
|
HRESULT CFtpDir::_GetFindData(HINTERNET hint, HINTPROCINFO * phpi, LPVOID pv, BOOL * pfReleaseHint)
|
||
|
{
|
||
|
LPGETFINDDATAINFO pgfdi = (LPGETFINDDATAINFO) pv;
|
||
|
HRESULT hr = S_FALSE;
|
||
|
|
||
|
// Remember, FTP filenames are always in the ANSI character set
|
||
|
// PERF: Status
|
||
|
hr = FtpDoesFileExist(hint, TRUE, pgfdi->pwWireName, pgfdi->pwfd, INTERNET_NO_CALLBACK);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (!StrCmpIA(pgfdi->pwfd->cFileName, pgfdi->pwWireName))
|
||
|
hr = S_OK; // The are equal.
|
||
|
else if (!StrCmpA(pgfdi->pwfd->cFileName, SZ_DOTA))
|
||
|
{
|
||
|
// Coincidence of coincidences: If we found a ".",
|
||
|
// then the wfd already contains the description of
|
||
|
// the directory! In other words, the wfd contains
|
||
|
// the correct information after all, save for the name.
|
||
|
// Aren't we lucky.
|
||
|
//
|
||
|
// And if it isn't dot, then it's some directory with
|
||
|
// unknown attributes (so we'll use whatever's lying around).
|
||
|
// Just make sure it's a directory.
|
||
|
pgfdi->pwfd->dwFileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
|
||
|
StrCpyNA(pgfdi->pwfd->cFileName, pgfdi->pwWireName, ARRAYSIZE(pgfdi->pwfd->cFileName));
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
#ifndef DEBUG
|
||
|
// Don't display an error msg because some callers will call when they
|
||
|
// know the file may not exist. This is the case for ConfirmCopy().
|
||
|
hr = S_FALSE;
|
||
|
#endif // DEBUG
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: GetFindData
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Get the WIN32_FIND_DATA for a file, given by name.
|
||
|
|
||
|
This is done as part of drag/drop to allow for an overwrite prompt.
|
||
|
|
||
|
This is all a gross hack because the STAT command
|
||
|
isn't supported by WinINet (as FtpGetFileAttributes).
|
||
|
|
||
|
Not that it'd help, because ftp.microsoft.com is OUT OF SPEC
|
||
|
with regard to the STAT command. (The first line of the output
|
||
|
isn't terminated correctly, causing the client to hang.)
|
||
|
|
||
|
Furthermore, UNIX ftp servers implement STAT incorrectly, too,
|
||
|
rendering STAT no more useful than LIST.
|
||
|
|
||
|
HACKHACK -- There is a bug in WinINet where doing a FindFirst
|
||
|
on a name which happens to be a directory returns the contents
|
||
|
of the directory instead of the attributes of the directory itself.
|
||
|
(This is actually a failing of most FTP implementation, because
|
||
|
they just use /bin/ls for directory listings.)
|
||
|
|
||
|
So we compare the name that comes back against the name we ask
|
||
|
for. If they are different, then it's a folder. We'll compare
|
||
|
in a case-insensitive manner because we don't know whether the
|
||
|
server is case-sensitive or not.
|
||
|
|
||
|
Note that we can get faked out if a directory contains a file
|
||
|
which has the same name as the directory. There is nothing we
|
||
|
can do about that. Fortunately, UNIX servers always return "."
|
||
|
as the first file in a subdirectory, so 99% of the time, we'll
|
||
|
do the right thing.
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::GetFindData(HWND hwnd, LPCWIRESTR pwWireName, LPFTP_FIND_DATA pwfd, CFtpFolder * pff)
|
||
|
{
|
||
|
GETFINDDATAINFO gfdi = {pwWireName, pwfd};
|
||
|
HRESULT hr = WithHint(NULL, hwnd, _GetFindData, &gfdi, NULL, pff);
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: GetFindDataForDisplayPath
|
||
|
|
||
|
DESCRIPTION:
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::GetFindDataForDisplayPath(HWND hwnd, LPCWSTR pwzDisplayPath, LPFTP_FIND_DATA pwfd, CFtpFolder * pff)
|
||
|
{
|
||
|
CWireEncoding * pwe = GetFtpSite()->GetCWireEncoding();
|
||
|
WIRECHAR wWirePath[MAX_PATH];
|
||
|
|
||
|
EVAL(SUCCEEDED(pwe->UnicodeToWireBytes(NULL, pwzDisplayPath, (IsUTF8Supported() ? WIREENC_USE_UTF8 : WIREENC_NONE), wWirePath, ARRAYSIZE(wWirePath))));
|
||
|
return GetFindData(hwnd, wWirePath, pwfd, pff);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: GetNameOf
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Common worker that handles SHGDN_FORPARSING style GetDisplayNameOf's.
|
||
|
|
||
|
Note! that since we do not support junctions (duh), we can
|
||
|
safely walk down the pidl generating goop as we go, secure
|
||
|
in the knowledge that we are in charge of every subpidl.
|
||
|
|
||
|
_CHARSET_: Since FTP filenames are always in the ANSI character
|
||
|
set, by RFC 1738, we can return ANSI display names without loss
|
||
|
of fidelity. In a general folder implementation, we should be
|
||
|
using cStr to return display names, so that the UNICODE
|
||
|
version of the shell extension can handle UNICODE names.
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::GetNameOf(LPCITEMIDLIST pidl, DWORD shgno, LPSTRRET pstr)
|
||
|
{
|
||
|
LPITEMIDLIST pidlFull = ILCombine(m_pidl, pidl);
|
||
|
HRESULT hr = E_FAIL;
|
||
|
|
||
|
if (pidlFull)
|
||
|
{
|
||
|
hr = StrRetFromFtpPidl(pstr, shgno, pidlFull);
|
||
|
ILFree(pidlFull);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: ChangeFolderName
|
||
|
|
||
|
DESCRIPTION:
|
||
|
A rename happened on this folder so update the szDir and m_pidl
|
||
|
|
||
|
PARAMETERS:
|
||
|
pidlFtpPath
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir::ChangeFolderName(LPCITEMIDLIST pidlFtpPath)
|
||
|
{
|
||
|
HRESULT hr = E_FAIL;
|
||
|
LPITEMIDLIST pidlNewFtpPath = NULL;
|
||
|
|
||
|
EVAL(SUCCEEDED(m_pfs->FlushSubDirs(m_pidlFtpDir)));
|
||
|
hr = FtpPidl_ReplacePath(m_pidl, pidlFtpPath, &pidlNewFtpPath);
|
||
|
_SetFtpDir(m_pfs, this, pidlFtpPath);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
Pidl_Set(&m_pidl, pidlNewFtpPath);
|
||
|
ILFree(pidlNewFtpPath);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: _CompareDirs
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Check if the indicated pfd is already rooted at the indicated pidl.
|
||
|
\*****************************************************************************/
|
||
|
int CALLBACK _CompareDirs(LPVOID pvPidl, LPVOID pvFtpDir, LPARAM lParam)
|
||
|
{
|
||
|
LPCITEMIDLIST pidl = (LPCITEMIDLIST) pvPidl;
|
||
|
CFtpDir * pfd = (CFtpDir *) pvFtpDir;
|
||
|
|
||
|
return FtpItemID_CompareIDsInt(COL_NAME, pfd->m_pidl, pidl, FCMP_NORMAL);
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CFtpDir::_SetFtpDir(CFtpSite * pfs, CFtpDir * pfd, LPCITEMIDLIST pidl)
|
||
|
{
|
||
|
if (FtpID_IsServerItemID(pidl))
|
||
|
pidl = _ILNext(pidl);
|
||
|
|
||
|
// We don't want pfd->m_pidlFtpDir to include the virtual root.
|
||
|
if (pfd->GetFtpSite()->HasVirtualRoot())
|
||
|
{
|
||
|
LPITEMIDLIST pidlIterate = (LPITEMIDLIST) pidl;
|
||
|
LPITEMIDLIST pidlVRootIterate = (LPITEMIDLIST) pfd->GetFtpSite()->GetVirtualRootReference();
|
||
|
|
||
|
ASSERT(!FtpID_IsServerItemID(pidl) && !FtpID_IsServerItemID(pidlVRootIterate));
|
||
|
// Let's see if pidl starts off with
|
||
|
while (!ILIsEmpty(pidlVRootIterate) && !ILIsEmpty(pidlIterate) &&
|
||
|
FtpItemID_IsEqual(pidlVRootIterate, pidlIterate))
|
||
|
{
|
||
|
pidlVRootIterate = _ILNext(pidlVRootIterate);
|
||
|
pidlIterate = _ILNext(pidlIterate);
|
||
|
}
|
||
|
|
||
|
if (ILIsEmpty(pidlVRootIterate))
|
||
|
pidl = (LPCITEMIDLIST)pidlIterate;
|
||
|
|
||
|
}
|
||
|
|
||
|
Pidl_Set(&pfd->m_pidlFtpDir, pidl);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************\
|
||
|
FUNCTION: CFtpDir_Create
|
||
|
|
||
|
DESCRIPTION:
|
||
|
Create a brand new FtpDir structure.
|
||
|
\*****************************************************************************/
|
||
|
HRESULT CFtpDir_Create(CFtpSite * pfs, LPCITEMIDLIST pidl, CFtpDir ** ppfd)
|
||
|
{
|
||
|
CFtpDir * pfd = new CFtpDir();
|
||
|
HRESULT hr = E_OUTOFMEMORY;
|
||
|
|
||
|
ASSERT(pfs);
|
||
|
if (pfd)
|
||
|
{
|
||
|
// WARNING: No ref held because it's a back pointer.
|
||
|
// This requires that the parent (CFtpSite) always
|
||
|
// out live this object.
|
||
|
pfd->m_pfs = pfs;
|
||
|
|
||
|
Pidl_Set(&pfd->m_pidl, pidl);
|
||
|
if (pfd->m_pidl)
|
||
|
hr = pfd->_SetFtpDir(pfs, pfd, pidl);
|
||
|
else
|
||
|
IUnknown_Set(&pfd, NULL);
|
||
|
}
|
||
|
|
||
|
*ppfd = pfd;
|
||
|
ASSERT(*ppfd ? SUCCEEDED(hr) : FAILED(hr));
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
/****************************************************\
|
||
|
Constructor
|
||
|
\****************************************************/
|
||
|
CFtpDir::CFtpDir() : m_cRef(1)
|
||
|
{
|
||
|
DllAddRef();
|
||
|
|
||
|
// This needs to be allocated in Zero Inited Memory.
|
||
|
// Assert that all Member Variables are inited to Zero.
|
||
|
ASSERT(!m_pfs);
|
||
|
ASSERT(!m_pflHfpl);
|
||
|
ASSERT(!m_pfgMotd);
|
||
|
ASSERT(!m_pidl);
|
||
|
|
||
|
LEAK_ADDREF(LEAK_CFtpDir);
|
||
|
}
|
||
|
|
||
|
|
||
|
/****************************************************\
|
||
|
Destructor
|
||
|
\****************************************************/
|
||
|
CFtpDir::~CFtpDir()
|
||
|
{
|
||
|
// WARNING: m_pfs is a back pointer that doesn't have a ref.
|
||
|
// m_pfs)
|
||
|
|
||
|
IUnknown_Set(&m_pflHfpl, NULL);
|
||
|
IUnknown_Set(&m_pfgMotd, NULL);
|
||
|
|
||
|
if (m_pidl) // Win95's Shell32.dll crashes with ILFree(NULL)
|
||
|
ILFree(m_pidl);
|
||
|
|
||
|
DllRelease();
|
||
|
LEAK_DELREF(LEAK_CFtpDir);
|
||
|
}
|
||
|
|
||
|
|
||
|
//===========================
|
||
|
// *** IUnknown Interface ***
|
||
|
//===========================
|
||
|
|
||
|
ULONG CFtpDir::AddRef()
|
||
|
{
|
||
|
m_cRef++;
|
||
|
return m_cRef;
|
||
|
}
|
||
|
|
||
|
ULONG CFtpDir::Release()
|
||
|
{
|
||
|
ASSERT(m_cRef > 0);
|
||
|
m_cRef--;
|
||
|
|
||
|
if (m_cRef > 0)
|
||
|
return m_cRef;
|
||
|
|
||
|
delete this;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
HRESULT CFtpDir::QueryInterface(REFIID riid, void **ppvObj)
|
||
|
{
|
||
|
if (IsEqualIID(riid, IID_IUnknown))
|
||
|
{
|
||
|
*ppvObj = SAFECAST(this, IUnknown*);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
TraceMsg(TF_FTPQI, "CFtpDir::QueryInterface() failed.");
|
||
|
*ppvObj = NULL;
|
||
|
return E_NOINTERFACE;
|
||
|
}
|
||
|
|
||
|
AddRef();
|
||
|
return S_OK;
|
||
|
}
|