mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
6f91b6c0fe
Some style-improvements I wanted to apply before a planned backport. E.g. The biggest source-file CDefView.cpp in [SHELL32] can be shrinked considerably by stripping some outdated comments from 10 and more years ago, when this class was still non-OOP and named ShellView. Less excessive line-lengths also here and there. In case it is not obvious when reviewing: In the beginning of CDevView.cpp in the declaration-part I just reduced the indentation-level by 4 spaces for a big block. Some shell classes were touched during 0.4.15-dev'ing for fixes like the 'unkown'-typo in 0.4.15-dev-6323-g3854a5d8e9
for example or missing \n were added to dbgprints like in 0.4.15-dev-4492-g03422451b3
Touched again here, because I want to port those improvements back, but don't intend to increase the size in older releases. The commit looks big, but does consist solely of trivialities. There is no functional change in this commit, aside the logging ofc.
757 lines
24 KiB
C++
757 lines
24 KiB
C++
|
|
/*
|
|
* file system folder drop target
|
|
*
|
|
* Copyright 1997 Marcus Meissner
|
|
* Copyright 1998, 1999, 2002 Juergen Schmied
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include <precomp.h>
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL (shell);
|
|
|
|
|
|
/****************************************************************************
|
|
* CFSDropTarget::_CopyItems
|
|
*
|
|
* copies or moves items to this folder
|
|
*/
|
|
HRESULT CFSDropTarget::_CopyItems(IShellFolder * pSFFrom, UINT cidl,
|
|
LPCITEMIDLIST * apidl, BOOL bCopy)
|
|
{
|
|
HRESULT ret;
|
|
WCHAR wszDstPath[MAX_PATH + 1] = {0};
|
|
PWCHAR pwszSrcPathsList = (PWCHAR) HeapAlloc(GetProcessHeap(), 0, MAX_PATH * sizeof(WCHAR) * cidl + 1);
|
|
if (!pwszSrcPathsList)
|
|
return E_OUTOFMEMORY;
|
|
|
|
PWCHAR pwszListPos = pwszSrcPathsList;
|
|
STRRET strretFrom;
|
|
SHFILEOPSTRUCTW fop;
|
|
BOOL bRenameOnCollision = FALSE;
|
|
|
|
/* Build a double null terminated list of C strings from source paths */
|
|
for (UINT i = 0; i < cidl; i++)
|
|
{
|
|
ret = pSFFrom->GetDisplayNameOf(apidl[i], SHGDN_FORPARSING, &strretFrom);
|
|
if (FAILED(ret))
|
|
goto cleanup;
|
|
|
|
ret = StrRetToBufW(&strretFrom, NULL, pwszListPos, MAX_PATH);
|
|
if (FAILED(ret))
|
|
goto cleanup;
|
|
|
|
pwszListPos += lstrlenW(pwszListPos) + 1;
|
|
}
|
|
/* Append the final null. */
|
|
*pwszListPos = L'\0';
|
|
|
|
/* Build a double null terminated target (this path) */
|
|
ret = StringCchCopyW(wszDstPath, MAX_PATH, m_sPathTarget);
|
|
if (FAILED(ret))
|
|
goto cleanup;
|
|
|
|
wszDstPath[lstrlenW(wszDstPath) + 1] = UNICODE_NULL;
|
|
|
|
/* Set bRenameOnCollision to TRUE if necesssary */
|
|
if (bCopy)
|
|
{
|
|
WCHAR szPath1[MAX_PATH], szPath2[MAX_PATH];
|
|
GetFullPathNameW(pwszSrcPathsList, _countof(szPath1), szPath1, NULL);
|
|
GetFullPathNameW(wszDstPath, _countof(szPath2), szPath2, NULL);
|
|
PathRemoveFileSpecW(szPath1);
|
|
if (_wcsicmp(szPath1, szPath2) == 0)
|
|
bRenameOnCollision = TRUE;
|
|
}
|
|
|
|
ZeroMemory(&fop, sizeof(fop));
|
|
fop.hwnd = m_hwndSite;
|
|
fop.wFunc = bCopy ? FO_COPY : FO_MOVE;
|
|
fop.pFrom = pwszSrcPathsList;
|
|
fop.pTo = wszDstPath;
|
|
fop.fFlags = FOF_ALLOWUNDO | FOF_NOCONFIRMMKDIR;
|
|
if (bRenameOnCollision)
|
|
fop.fFlags |= FOF_RENAMEONCOLLISION;
|
|
|
|
ret = S_OK;
|
|
|
|
if (SHFileOperationW(&fop))
|
|
{
|
|
ERR("SHFileOperationW failed\n");
|
|
ret = E_FAIL;
|
|
}
|
|
|
|
cleanup:
|
|
HeapFree(GetProcessHeap(), 0, pwszSrcPathsList);
|
|
return ret;
|
|
}
|
|
|
|
CFSDropTarget::CFSDropTarget():
|
|
m_cfShellIDList(0),
|
|
m_fAcceptFmt(FALSE),
|
|
m_sPathTarget(NULL),
|
|
m_hwndSite(NULL),
|
|
m_grfKeyState(0)
|
|
{
|
|
}
|
|
|
|
HRESULT CFSDropTarget::Initialize(LPWSTR PathTarget)
|
|
{
|
|
if (!PathTarget)
|
|
return E_UNEXPECTED;
|
|
|
|
m_cfShellIDList = RegisterClipboardFormatW(CFSTR_SHELLIDLIST);
|
|
if (!m_cfShellIDList)
|
|
return E_FAIL;
|
|
|
|
m_sPathTarget = (WCHAR *)SHAlloc((wcslen(PathTarget) + 1) * sizeof(WCHAR));
|
|
if (!m_sPathTarget)
|
|
return E_OUTOFMEMORY;
|
|
|
|
wcscpy(m_sPathTarget, PathTarget);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
CFSDropTarget::~CFSDropTarget()
|
|
{
|
|
SHFree(m_sPathTarget);
|
|
}
|
|
|
|
BOOL
|
|
CFSDropTarget::_GetUniqueFileName(LPCWSTR pwszBasePath, LPCWSTR pwszExt, LPWSTR pwszTarget, BOOL bShortcut)
|
|
{
|
|
WCHAR wszLink[40];
|
|
|
|
if (!bShortcut)
|
|
{
|
|
if (!LoadStringW(shell32_hInstance, IDS_LNK_FILE, wszLink, _countof(wszLink)))
|
|
wszLink[0] = L'\0';
|
|
}
|
|
|
|
if (!bShortcut)
|
|
swprintf(pwszTarget, L"%s%s%s", wszLink, pwszBasePath, pwszExt);
|
|
else
|
|
swprintf(pwszTarget, L"%s%s", pwszBasePath, pwszExt);
|
|
|
|
for (UINT i = 2; PathFileExistsW(pwszTarget); ++i)
|
|
{
|
|
if (!bShortcut)
|
|
swprintf(pwszTarget, L"%s%s (%u)%s", wszLink, pwszBasePath, i, pwszExt);
|
|
else
|
|
swprintf(pwszTarget, L"%s (%u)%s", pwszBasePath, i, pwszExt);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* IDropTarget implementation
|
|
*/
|
|
BOOL CFSDropTarget::_QueryDrop(DWORD dwKeyState, LPDWORD pdwEffect)
|
|
{
|
|
/* TODO Windows does different drop effects if dragging across drives.
|
|
i.e., it will copy instead of move if the directories are on different disks. */
|
|
|
|
DWORD dwEffect = m_dwDefaultEffect;
|
|
|
|
*pdwEffect = DROPEFFECT_NONE;
|
|
|
|
if (m_fAcceptFmt) { /* Does our interpretation of the keystate ... */
|
|
*pdwEffect = KeyStateToDropEffect (dwKeyState);
|
|
|
|
if (*pdwEffect == DROPEFFECT_NONE)
|
|
*pdwEffect = dwEffect;
|
|
|
|
/* ... matches the desired effect ? */
|
|
if (dwEffect & *pdwEffect) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
HRESULT CFSDropTarget::_GetEffectFromMenu(IDataObject *pDataObject, POINTL pt, DWORD *pdwEffect, DWORD dwAvailableEffects)
|
|
{
|
|
HMENU hmenu = LoadMenuW(shell32_hInstance, MAKEINTRESOURCEW(IDM_DRAGFILE));
|
|
if (!hmenu)
|
|
return E_OUTOFMEMORY;
|
|
|
|
HMENU hpopupmenu = GetSubMenu(hmenu, 0);
|
|
|
|
if ((dwAvailableEffects & DROPEFFECT_COPY) == 0)
|
|
DeleteMenu(hpopupmenu, IDM_COPYHERE, MF_BYCOMMAND);
|
|
else if ((dwAvailableEffects & DROPEFFECT_MOVE) == 0)
|
|
DeleteMenu(hpopupmenu, IDM_MOVEHERE, MF_BYCOMMAND);
|
|
else if ((dwAvailableEffects & DROPEFFECT_LINK) == 0)
|
|
DeleteMenu(hpopupmenu, IDM_LINKHERE, MF_BYCOMMAND);
|
|
|
|
if ((*pdwEffect & DROPEFFECT_COPY))
|
|
SetMenuDefaultItem(hpopupmenu, IDM_COPYHERE, FALSE);
|
|
else if ((*pdwEffect & DROPEFFECT_MOVE))
|
|
SetMenuDefaultItem(hpopupmenu, IDM_MOVEHERE, FALSE);
|
|
else if ((*pdwEffect & DROPEFFECT_LINK))
|
|
SetMenuDefaultItem(hpopupmenu, IDM_LINKHERE, FALSE);
|
|
|
|
/* FIXME: We need to support shell extensions here */
|
|
|
|
/* We shouldn't use the site window here because the menu should work even when we don't have a site */
|
|
HWND hwndDummy = CreateWindowEx(0,
|
|
WC_STATIC,
|
|
NULL,
|
|
WS_OVERLAPPED | WS_DISABLED | WS_CLIPSIBLINGS | WS_BORDER | SS_LEFT,
|
|
pt.x,
|
|
pt.y,
|
|
1,
|
|
1,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
|
|
UINT uCommand = TrackPopupMenu(hpopupmenu,
|
|
TPM_LEFTALIGN | TPM_RETURNCMD | TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_NONOTIFY,
|
|
pt.x, pt.y, 0, hwndDummy, NULL);
|
|
|
|
DestroyWindow(hwndDummy);
|
|
|
|
if (uCommand == 0)
|
|
return S_FALSE;
|
|
else if (uCommand == IDM_COPYHERE)
|
|
*pdwEffect = DROPEFFECT_COPY;
|
|
else if (uCommand == IDM_MOVEHERE)
|
|
*pdwEffect = DROPEFFECT_MOVE;
|
|
else if (uCommand == IDM_LINKHERE)
|
|
*pdwEffect = DROPEFFECT_LINK;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CFSDropTarget::_RepositionItems(IShellFolderView *psfv, IDataObject *pdtobj, POINTL pt)
|
|
{
|
|
CComPtr<IFolderView> pfv;
|
|
POINT ptDrag;
|
|
HRESULT hr = psfv->QueryInterface(IID_PPV_ARG(IFolderView, &pfv));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
hr = psfv->GetDragPoint(&ptDrag);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
PIDLIST_ABSOLUTE pidlFolder;
|
|
PUITEMID_CHILD *apidl;
|
|
UINT cidl;
|
|
hr = SH_GetApidlFromDataObject(pdtobj, &pidlFolder, &apidl, &cidl);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
CComHeapPtr<POINT> apt;
|
|
if (!apt.Allocate(cidl))
|
|
{
|
|
SHFree(pidlFolder);
|
|
_ILFreeaPidl(apidl, cidl);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
for (UINT i = 0; i<cidl; i++)
|
|
{
|
|
pfv->GetItemPosition(apidl[i], &apt[i]);
|
|
apt[i].x += pt.x - ptDrag.x;
|
|
apt[i].y += pt.y - ptDrag.y;
|
|
}
|
|
|
|
pfv->SelectAndPositionItems(cidl, apidl, apt, SVSI_SELECT);
|
|
|
|
SHFree(pidlFolder);
|
|
_ILFreeaPidl(apidl, cidl);
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CFSDropTarget::DragEnter(IDataObject *pDataObject,
|
|
DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
TRACE("(%p)->(DataObject=%p)\n", this, pDataObject);
|
|
|
|
if (*pdwEffect == DROPEFFECT_NONE)
|
|
return S_OK;
|
|
|
|
FORMATETC fmt;
|
|
FORMATETC fmt2;
|
|
m_fAcceptFmt = FALSE;
|
|
|
|
InitFormatEtc (fmt, m_cfShellIDList, TYMED_HGLOBAL);
|
|
InitFormatEtc (fmt2, CF_HDROP, TYMED_HGLOBAL);
|
|
|
|
if (SUCCEEDED(pDataObject->QueryGetData(&fmt)))
|
|
m_fAcceptFmt = TRUE;
|
|
else if (SUCCEEDED(pDataObject->QueryGetData(&fmt2)))
|
|
m_fAcceptFmt = TRUE;
|
|
|
|
m_grfKeyState = dwKeyState;
|
|
|
|
#define D_NONE DROPEFFECT_NONE
|
|
#define D_COPY DROPEFFECT_COPY
|
|
#define D_MOVE DROPEFFECT_MOVE
|
|
#define D_LINK DROPEFFECT_LINK
|
|
m_dwDefaultEffect = *pdwEffect;
|
|
switch (*pdwEffect & (D_COPY | D_MOVE | D_LINK))
|
|
{
|
|
case D_COPY | D_MOVE:
|
|
if (dwKeyState & MK_CONTROL)
|
|
m_dwDefaultEffect = D_COPY;
|
|
else
|
|
m_dwDefaultEffect = D_MOVE;
|
|
break;
|
|
case D_COPY | D_MOVE | D_LINK:
|
|
if ((dwKeyState & (MK_SHIFT | MK_CONTROL)) == (MK_SHIFT | MK_CONTROL))
|
|
m_dwDefaultEffect = D_LINK;
|
|
else if ((dwKeyState & (MK_SHIFT | MK_CONTROL)) == MK_CONTROL)
|
|
m_dwDefaultEffect = D_COPY;
|
|
else
|
|
m_dwDefaultEffect = D_MOVE;
|
|
break;
|
|
case D_COPY | D_LINK:
|
|
if ((dwKeyState & (MK_SHIFT | MK_CONTROL)) == (MK_SHIFT | MK_CONTROL))
|
|
m_dwDefaultEffect = D_LINK;
|
|
else
|
|
m_dwDefaultEffect = D_COPY;
|
|
break;
|
|
case D_MOVE | D_LINK:
|
|
if ((dwKeyState & (MK_SHIFT | MK_CONTROL)) == (MK_SHIFT | MK_CONTROL))
|
|
m_dwDefaultEffect = D_LINK;
|
|
else
|
|
m_dwDefaultEffect = D_MOVE;
|
|
break;
|
|
}
|
|
|
|
STGMEDIUM medium;
|
|
if (SUCCEEDED(pDataObject->GetData(&fmt2, &medium)))
|
|
{
|
|
WCHAR wstrFirstFile[MAX_PATH];
|
|
if (DragQueryFileW((HDROP)medium.hGlobal, 0, wstrFirstFile, _countof(wstrFirstFile)))
|
|
{
|
|
/* Check if the drive letter is different */
|
|
if (wstrFirstFile[0] != m_sPathTarget[0])
|
|
{
|
|
m_dwDefaultEffect = DROPEFFECT_COPY;
|
|
}
|
|
}
|
|
ReleaseStgMedium(&medium);
|
|
}
|
|
|
|
if (!m_fAcceptFmt)
|
|
*pdwEffect = DROPEFFECT_NONE;
|
|
else
|
|
*pdwEffect = m_dwDefaultEffect;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CFSDropTarget::DragOver(DWORD dwKeyState, POINTL pt,
|
|
DWORD *pdwEffect)
|
|
{
|
|
TRACE("(%p)\n", this);
|
|
|
|
if (!pdwEffect)
|
|
return E_INVALIDARG;
|
|
|
|
m_grfKeyState = dwKeyState;
|
|
|
|
_QueryDrop(dwKeyState, pdwEffect);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CFSDropTarget::DragLeave()
|
|
{
|
|
TRACE("(%p)\n", this);
|
|
|
|
m_fAcceptFmt = FALSE;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI CFSDropTarget::Drop(IDataObject *pDataObject,
|
|
DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
TRACE("(%p) object dropped, effect %u\n", this, *pdwEffect);
|
|
|
|
if (!pdwEffect)
|
|
return E_INVALIDARG;
|
|
|
|
IUnknown_GetWindow(m_site, &m_hwndSite);
|
|
|
|
DWORD dwAvailableEffects = *pdwEffect;
|
|
|
|
_QueryDrop(dwKeyState, pdwEffect);
|
|
|
|
TRACE("pdwEffect: 0x%x, m_dwDefaultEffect: 0x%x, dwAvailableEffects: 0x%x\n", *pdwEffect, m_dwDefaultEffect, dwAvailableEffects);
|
|
|
|
if (m_grfKeyState & MK_RBUTTON)
|
|
{
|
|
HRESULT hr = _GetEffectFromMenu(pDataObject, pt, pdwEffect, dwAvailableEffects);
|
|
if (FAILED_UNEXPECTEDLY(hr) || hr == S_FALSE)
|
|
return hr;
|
|
}
|
|
|
|
if (*pdwEffect == DROPEFFECT_MOVE && m_site)
|
|
{
|
|
CComPtr<IShellFolderView> psfv;
|
|
HRESULT hr = IUnknown_QueryService(m_site, SID_IFolderView, IID_PPV_ARG(IShellFolderView, &psfv));
|
|
if (SUCCEEDED(hr) && psfv->IsDropOnSource(this) == S_OK)
|
|
{
|
|
_RepositionItems(psfv, pDataObject, pt);
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
BOOL fIsOpAsync = FALSE;
|
|
CComPtr<IAsyncOperation> pAsyncOperation;
|
|
|
|
if (SUCCEEDED(pDataObject->QueryInterface(IID_PPV_ARG(IAsyncOperation, &pAsyncOperation))))
|
|
{
|
|
if (SUCCEEDED(pAsyncOperation->GetAsyncMode(&fIsOpAsync)) && fIsOpAsync)
|
|
{
|
|
_DoDropData *data = static_cast<_DoDropData*>(HeapAlloc(GetProcessHeap(), 0, sizeof(_DoDropData)));
|
|
data->This = this;
|
|
// Need to maintain this class in case the window is closed or the class exists temporarily (when dropping onto a folder).
|
|
pDataObject->AddRef();
|
|
pAsyncOperation->StartOperation(NULL);
|
|
CoMarshalInterThreadInterfaceInStream(IID_IDataObject, pDataObject, &data->pStream);
|
|
this->AddRef();
|
|
data->dwKeyState = dwKeyState;
|
|
data->pt = pt;
|
|
// Need to dereference as pdweffect gets freed.
|
|
data->pdwEffect = *pdwEffect;
|
|
SHCreateThread(CFSDropTarget::_DoDropThreadProc, data, NULL, NULL);
|
|
return S_OK;
|
|
}
|
|
}
|
|
return this->_DoDrop(pDataObject, dwKeyState, pt, pdwEffect);
|
|
}
|
|
|
|
HRESULT
|
|
WINAPI
|
|
CFSDropTarget::SetSite(IUnknown *pUnkSite)
|
|
{
|
|
m_site = pUnkSite;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT
|
|
WINAPI
|
|
CFSDropTarget::GetSite(REFIID riid, void **ppvSite)
|
|
{
|
|
if (!m_site)
|
|
return E_FAIL;
|
|
|
|
return m_site->QueryInterface(riid, ppvSite);
|
|
}
|
|
|
|
HRESULT CFSDropTarget::_DoDrop(IDataObject *pDataObject,
|
|
DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
TRACE("(%p) performing drop, effect %u\n", this, *pdwEffect);
|
|
FORMATETC fmt;
|
|
FORMATETC fmt2;
|
|
STGMEDIUM medium;
|
|
|
|
InitFormatEtc (fmt, m_cfShellIDList, TYMED_HGLOBAL);
|
|
InitFormatEtc (fmt2, CF_HDROP, TYMED_HGLOBAL);
|
|
|
|
HRESULT hr;
|
|
bool bCopy = TRUE;
|
|
bool bLinking = FALSE;
|
|
|
|
/* Figure out what drop operation we're doing */
|
|
if (pdwEffect)
|
|
{
|
|
TRACE("Current drop effect flag %i\n", *pdwEffect);
|
|
if ((*pdwEffect & DROPEFFECT_MOVE) == DROPEFFECT_MOVE)
|
|
bCopy = FALSE;
|
|
if ((*pdwEffect & DROPEFFECT_LINK) == DROPEFFECT_LINK)
|
|
bLinking = TRUE;
|
|
}
|
|
|
|
if (SUCCEEDED(pDataObject->QueryGetData(&fmt)))
|
|
{
|
|
hr = pDataObject->GetData(&fmt, &medium);
|
|
TRACE("CFSTR_SHELLIDLIST\n");
|
|
if (FAILED(hr))
|
|
{
|
|
ERR("CFSTR_SHELLIDLIST failed\n");
|
|
}
|
|
/* lock the handle */
|
|
LPIDA lpcida = (LPIDA)GlobalLock(medium.hGlobal);
|
|
if (!lpcida)
|
|
{
|
|
ReleaseStgMedium(&medium);
|
|
return E_FAIL;
|
|
}
|
|
|
|
/* convert the data into pidl */
|
|
LPITEMIDLIST pidl;
|
|
LPITEMIDLIST *apidl = _ILCopyCidaToaPidl(&pidl, lpcida);
|
|
if (!apidl)
|
|
{
|
|
ReleaseStgMedium(&medium);
|
|
return E_FAIL;
|
|
}
|
|
|
|
CComPtr<IShellFolder> psfDesktop;
|
|
CComPtr<IShellFolder> psfFrom = NULL;
|
|
|
|
/* Grab the desktop shell folder */
|
|
hr = SHGetDesktopFolder(&psfDesktop);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR("SHGetDesktopFolder failed\n");
|
|
SHFree(pidl);
|
|
_ILFreeaPidl(apidl, lpcida->cidl);
|
|
ReleaseStgMedium(&medium);
|
|
return E_FAIL;
|
|
}
|
|
|
|
/* Find source folder, this is where the clipboard data was copied from */
|
|
if (_ILIsDesktop(pidl))
|
|
{
|
|
/* use desktop shell folder */
|
|
psfFrom = psfDesktop;
|
|
}
|
|
else
|
|
{
|
|
hr = psfDesktop->BindToObject(pidl, NULL, IID_PPV_ARG(IShellFolder, &psfFrom));
|
|
if (FAILED(hr))
|
|
{
|
|
ERR("no IShellFolder\n");
|
|
SHFree(pidl);
|
|
_ILFreeaPidl(apidl, lpcida->cidl);
|
|
ReleaseStgMedium(&medium);
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
if (bLinking)
|
|
{
|
|
WCHAR wszPath[MAX_PATH];
|
|
WCHAR wszTarget[MAX_PATH];
|
|
|
|
TRACE("target path = %s\n", debugstr_w(m_sPathTarget));
|
|
|
|
/* We need to create a link for each pidl in the copied items, so step through the pidls from the clipboard */
|
|
for (UINT i = 0; i < lpcida->cidl; i++)
|
|
{
|
|
// Find out which file we're linking.
|
|
STRRET strFile;
|
|
hr = psfFrom->GetDisplayNameOf(apidl[i], SHGDN_FORPARSING, &strFile);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
hr = StrRetToBufW(&strFile, apidl[i], wszPath, _countof(wszPath));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
TRACE("source path = %s\n", debugstr_w(wszPath));
|
|
|
|
WCHAR wszDisplayName[MAX_PATH];
|
|
LPWSTR pwszFileName = PathFindFileNameW(wszPath);
|
|
if (PathIsRootW(wszPath)) // Drive?
|
|
{
|
|
hr = psfFrom->GetDisplayNameOf(apidl[i], SHGDN_NORMAL, &strFile);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
hr = StrRetToBufW(&strFile, apidl[i], wszDisplayName, _countof(wszDisplayName));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
// Delete a ':' in wszDisplayName.
|
|
LPWSTR pch0 = wcschr(wszDisplayName, L':');
|
|
if (pch0)
|
|
{
|
|
do
|
|
{
|
|
*pch0 = *(pch0 + 1);
|
|
++pch0;
|
|
} while (*pch0);
|
|
}
|
|
|
|
pwszFileName = wszDisplayName; // Use wszDisplayName
|
|
}
|
|
else if (wszPath[0] == L':' && wszPath[1] == L':') // ::{GUID}?
|
|
{
|
|
CLSID clsid;
|
|
hr = ::CLSIDFromString(&wszPath[2], &clsid);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
LPITEMIDLIST pidl = ILCreateFromPathW(wszPath);
|
|
if (pidl)
|
|
{
|
|
SHFILEINFOW fi = { NULL };
|
|
SHGetFileInfoW((LPCWSTR)pidl, 0, &fi, sizeof(fi),
|
|
SHGFI_DISPLAYNAME | SHGFI_PIDL);
|
|
if (fi.szDisplayName[0])
|
|
{
|
|
lstrcpynW(wszDisplayName, fi.szDisplayName, _countof(wszDisplayName));
|
|
pwszFileName = wszDisplayName; // Use wszDisplayName
|
|
}
|
|
ILFree(pidl);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Creating a buffer to hold the combined path.
|
|
WCHAR wszCombined[MAX_PATH];
|
|
PathCombineW(wszCombined, m_sPathTarget, pwszFileName);
|
|
|
|
// Check to see if the source is a link
|
|
BOOL fSourceIsLink = FALSE;
|
|
if (!wcsicmp(PathFindExtensionW(wszPath), L".lnk"))
|
|
{
|
|
fSourceIsLink = TRUE;
|
|
PathRemoveExtensionW(wszCombined);
|
|
}
|
|
|
|
// Create a pathname to save the new link.
|
|
_GetUniqueFileName(wszCombined, L".lnk", wszTarget, TRUE);
|
|
|
|
CComPtr<IPersistFile> ppf;
|
|
if (fSourceIsLink)
|
|
{
|
|
hr = IShellLink_ConstructFromPath(wszPath, IID_PPV_ARG(IPersistFile, &ppf));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
CComPtr<IShellLinkW> pLink;
|
|
hr = CShellLink::_CreatorClass::CreateInstance(NULL, IID_PPV_ARG(IShellLinkW, &pLink));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
WCHAR szDirPath[MAX_PATH], *pwszFile;
|
|
GetFullPathName(wszPath, MAX_PATH, szDirPath, &pwszFile);
|
|
if (pwszFile)
|
|
pwszFile[0] = 0;
|
|
|
|
hr = pLink->SetPath(wszPath);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
hr = pLink->SetWorkingDirectory(szDirPath);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
hr = pLink->QueryInterface(IID_PPV_ARG(IPersistFile, &ppf));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
}
|
|
|
|
hr = ppf->Save(wszTarget, !fSourceIsLink);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
break;
|
|
|
|
SHChangeNotify(SHCNE_CREATE, SHCNF_PATHW, wszTarget, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = _CopyItems(psfFrom, lpcida->cidl, (LPCITEMIDLIST*)apidl, bCopy);
|
|
}
|
|
|
|
SHFree(pidl);
|
|
_ILFreeaPidl(apidl, lpcida->cidl);
|
|
ReleaseStgMedium(&medium);
|
|
}
|
|
else if (SUCCEEDED(pDataObject->QueryGetData(&fmt2)))
|
|
{
|
|
FORMATETC fmt2;
|
|
InitFormatEtc (fmt2, CF_HDROP, TYMED_HGLOBAL);
|
|
if (SUCCEEDED(pDataObject->GetData(&fmt2, &medium)) /* && SUCCEEDED(pDataObject->GetData(&fmt2, &medium))*/)
|
|
{
|
|
WCHAR wszTargetPath[MAX_PATH + 1];
|
|
LPWSTR pszSrcList;
|
|
|
|
wcscpy(wszTargetPath, m_sPathTarget);
|
|
//Double NULL terminate.
|
|
wszTargetPath[wcslen(wszTargetPath) + 1] = '\0';
|
|
|
|
LPDROPFILES lpdf = (LPDROPFILES) GlobalLock(medium.hGlobal);
|
|
if (!lpdf)
|
|
{
|
|
ERR("Error locking global\n");
|
|
return E_FAIL;
|
|
}
|
|
pszSrcList = (LPWSTR) (((byte*) lpdf) + lpdf->pFiles);
|
|
ERR("Source file (just the first) = %s, target path = %s, bCopy: %d\n", debugstr_w(pszSrcList), debugstr_w(wszTargetPath), bCopy);
|
|
|
|
SHFILEOPSTRUCTW op;
|
|
ZeroMemory(&op, sizeof(op));
|
|
op.pFrom = pszSrcList;
|
|
op.pTo = wszTargetPath;
|
|
op.hwnd = m_hwndSite;
|
|
op.wFunc = bCopy ? FO_COPY : FO_MOVE;
|
|
op.fFlags = FOF_ALLOWUNDO | FOF_NOCONFIRMMKDIR;
|
|
int res = SHFileOperationW(&op);
|
|
if (res)
|
|
{
|
|
ERR("SHFileOperationW failed with 0x%x\n", res);
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
ERR("Error calling GetData\n");
|
|
hr = E_FAIL;
|
|
}
|
|
else
|
|
{
|
|
ERR("No viable drop format\n");
|
|
hr = E_FAIL;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
DWORD WINAPI CFSDropTarget::_DoDropThreadProc(LPVOID lpParameter)
|
|
{
|
|
CoInitialize(NULL);
|
|
_DoDropData *data = static_cast<_DoDropData*>(lpParameter);
|
|
CComPtr<IDataObject> pDataObject;
|
|
HRESULT hr = CoGetInterfaceAndReleaseStream (data->pStream, IID_PPV_ARG(IDataObject, &pDataObject));
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CComPtr<IAsyncOperation> pAsyncOperation;
|
|
hr = data->This->_DoDrop(pDataObject, data->dwKeyState, data->pt, &data->pdwEffect);
|
|
if (SUCCEEDED(pDataObject->QueryInterface(IID_PPV_ARG(IAsyncOperation, &pAsyncOperation))))
|
|
{
|
|
pAsyncOperation->EndOperation(hr, NULL, data->pdwEffect);
|
|
}
|
|
}
|
|
//Release the CFSFolder and data object holds in the copying thread.
|
|
data->This->Release();
|
|
//Release the parameter from the heap.
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
CoUninitialize();
|
|
return 0;
|
|
}
|
|
|
|
HRESULT CFSDropTarget_CreateInstance(LPWSTR sPathTarget, REFIID riid, LPVOID * ppvOut)
|
|
{
|
|
return ShellObjectCreatorInit<CFSDropTarget>(sPathTarget, riid, ppvOut);
|
|
}
|