Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

282 lines
7.6 KiB

#include "precomp.h" // pch file
#include "sendto.h"
#pragma hdrstop
// class that implements the send to desktop (as shortcut)
const GUID CLSID_DesktopShortcut = { 0x9E56BE61L, 0xC50F, 0x11CF, 0x9A, 0x2C, 0x00, 0xA0, 0xC9, 0x0A, 0x90, 0xCE };
class CDesktopShortcut : public CSendTo
{
private:
LPIDA _GetHIDA(IDataObject *pdtobj, STGMEDIUM *pmedium);
LPCITEMIDLIST _GetIDListPtr(LPIDA pida, UINT i);
void _ReleaseStgMedium(void *pv, STGMEDIUM *pmedium);
HRESULT _BindToObject(IShellFolder *psf, REFIID riid, LPCITEMIDLIST pidl, void **ppvOut);
HRESULT _InvokeVerbOnItems(HWND hwnd, LPCTSTR pszVerb, UINT uFlags, IShellFolder *psf, UINT cidl, LPCITEMIDLIST *apidl, LPCTSTR pszDirectory);
HRESULT _InvokeVerbOnDataObj(HWND hwnd, LPCTSTR pszVerb, UINT uFlags, IDataObject *pdtobj, LPCTSTR pszDirectory);
protected:
HRESULT v_DropHandler(IDataObject *pdtobj, DWORD grfKeyState, DWORD dwEffect);
public:
CDesktopShortcut();
};
// construct the sendto object with the appropriate CLSID.
CDesktopShortcut::CDesktopShortcut() :
CSendTo(CLSID_DesktopShortcut)
{
}
// helper methods
LPIDA CDesktopShortcut::_GetHIDA(IDataObject *pdtobj, STGMEDIUM *pmedium)
{
FORMATETC fmte = {g_cfHIDA, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL};
if (pmedium)
{
pmedium->pUnkForRelease = NULL;
pmedium->hGlobal = NULL;
}
if (!pmedium)
{
if (SUCCEEDED(pdtobj->QueryGetData(&fmte)))
return (LPIDA)TRUE;
else
return (LPIDA)FALSE;
}
else if (SUCCEEDED(pdtobj->GetData(&fmte, pmedium)))
{
return (LPIDA)GlobalLock(pmedium->hGlobal);
}
return NULL;
}
LPCITEMIDLIST CDesktopShortcut::_GetIDListPtr(LPIDA pida, UINT i)
{
if (NULL == pida)
{
return NULL;
}
if (i == (UINT)-1 || i < pida->cidl)
{
return (LPCITEMIDLIST)(((LPBYTE)pida)+(pida)->aoffset[i+1]);
}
return NULL;
}
// release a storage medium (doing a Global unlock as required).
void CDesktopShortcut::_ReleaseStgMedium(void *pv, STGMEDIUM *pmedium)
{
if (pmedium->hGlobal && (pmedium->tymed == TYMED_HGLOBAL))
{
GlobalUnlock(pmedium->hGlobal);
}
ReleaseStgMedium(pmedium);
}
// dupe of shell\lib SHBindToObject() to avoid link dependancies... (OLEAUT32 gets pulled in by
// stuff that does VARIANT goo in that lib)
HRESULT CDesktopShortcut::_BindToObject(IShellFolder *psf, REFIID riid, LPCITEMIDLIST pidl, void **ppvOut)
{
HRESULT hr;
IShellFolder *psfRelease;
*ppvOut = NULL;
if (!psf)
{
hr = SHGetDesktopFolder(&psf);
psfRelease = psf;
}
else
{
psfRelease = NULL;
hr = S_OK;
}
if (FAILED(hr))
{
// leave error code in hr
}
else if (!pidl || ILIsEmpty(pidl))
{
hr = psf->QueryInterface(riid, ppvOut);
}
else
{
hr = psf->BindToObject(pidl, NULL, riid, ppvOut);
}
if (psfRelease)
psfRelease->Release();
if (SUCCEEDED(hr) && (*ppvOut == NULL))
{
hr = E_FAIL;
}
return hr;
}
// invoke a verb on an array of items in the folder.
HRESULT CDesktopShortcut::_InvokeVerbOnItems(HWND hwnd, LPCTSTR pszVerb, UINT uFlags, IShellFolder *psf, UINT cidl, LPCITEMIDLIST *apidl, LPCTSTR pszDirectory)
{
IContextMenu *pcm;
HRESULT hr = psf->GetUIObjectOf(hwnd, cidl, apidl, IID_IContextMenu, NULL, (void **)&pcm);
if (SUCCEEDED(hr))
{
CHAR szVerbA[128];
WCHAR szVerbW[128];
CHAR szDirA[MAX_PATH];
WCHAR szDirW[MAX_PATH];
CMINVOKECOMMANDINFOEX ici =
{
SIZEOF(CMINVOKECOMMANDINFOEX),
uFlags | CMIC_MASK_UNICODE | CMIC_MASK_FLAG_NO_UI,
hwnd,
NULL,
NULL,
NULL,
SW_NORMAL,
};
SHTCharToAnsi(pszVerb, szVerbA, ARRAYSIZE(szVerbA));
SHTCharToUnicode(pszVerb, szVerbW, ARRAYSIZE(szVerbW));
if (pszDirectory)
{
SHTCharToAnsi(pszDirectory, szDirA, ARRAYSIZE(szDirA));
SHTCharToUnicode(pszDirectory, szDirW, ARRAYSIZE(szDirW));
ici.lpDirectory = szDirA;
ici.lpDirectoryW = szDirW;
}
ici.lpVerb = szVerbA;
ici.lpVerbW = szVerbW;
hr = pcm->InvokeCommand((CMINVOKECOMMANDINFO*)&ici);
pcm->Release();
}
return hr;
}
// invoke a verb on the data object item
HRESULT CDesktopShortcut::_InvokeVerbOnDataObj(HWND hwnd, LPCTSTR pszVerb, UINT uFlags, IDataObject *pdtobj, LPCTSTR pszDirectory)
{
HRESULT hr;
STGMEDIUM medium;
LPIDA pida = _GetHIDA(pdtobj, &medium);
if (pida)
{
LPCITEMIDLIST pidlParent = _GetIDListPtr(pida, (UINT)-1);
IShellFolder *psf;
hr = _BindToObject(NULL, IID_IShellFolder, pidlParent, (void **)&psf);
if (SUCCEEDED(hr))
{
LPCITEMIDLIST *ppidl = (LPCITEMIDLIST *)LocalAlloc(LPTR, pida->cidl * sizeof(LPCITEMIDLIST));
if (ppidl)
{
UINT i;
for (i = 0; i < pida->cidl; i++)
{
ppidl[i] = _GetIDListPtr(pida, i);
}
hr = _InvokeVerbOnItems(hwnd, pszVerb, uFlags, psf, pida->cidl, ppidl, pszDirectory);
LocalFree((LPVOID)ppidl);
}
psf->Release();
}
_ReleaseStgMedium(pida, &medium);
}
else
hr = E_FAIL;
return hr;
}
// handle the drop on the object, so for each item in the HIDA invoke the create
// link verb on them.
HRESULT CDesktopShortcut::v_DropHandler(IDataObject *pdtobj, DWORD grfKeyState, DWORD dwEffect)
{
TCHAR szDesktop[MAX_PATH];
if (_SHGetSpecialFolderPath(NULL, szDesktop, CSIDL_DESKTOPDIRECTORY, FALSE))
{
if (g_cfHIDA == 0)
{
g_cfHIDA = (CLIPFORMAT) RegisterClipboardFormat(CFSTR_SHELLIDLIST);
}
return _InvokeVerbOnDataObj (NULL, TEXT("link"), 0, pdtobj, szDesktop);
}
return E_OUTOFMEMORY;
}
// create an instance of desktop link object
STDAPI DesktopShortcut_CreateInstance(IUnknown *punkOuter, IUnknown **ppunk, LPCOBJECTINFO poi)
{
*ppunk = NULL; // assume failure
if ( punkOuter )
return CLASS_E_NOAGGREGATION;
CDesktopShortcut *pds = new CDesktopShortcut;
if ( !pds )
return E_OUTOFMEMORY;
HRESULT hr = pds->QueryInterface(IID_PPV_ARG(IUnknown, ppunk));
pds->Release();
return hr;
}
// handler registration of the desktop link verb
#define DESKLINK_EXTENSION TEXT("DeskLink")
STDAPI DesktopShortcut_RegUnReg(BOOL bReg, HKEY hkCLSID, LPCTSTR pszCLSID, LPCTSTR pszModule)
{
TCHAR szFile[MAX_PATH];
if (bReg)
{
HKEY hk;
// get rid of old name "Desktop as Shortcut" link from IE4
if (SUCCEEDED(GetDropTargetPath(szFile, IDS_DESKTOPLINK_FILENAME, DESKLINK_EXTENSION)))
DeleteFile(szFile);
if (RegCreateKey(hkCLSID, DEFAULTICON, &hk) == ERROR_SUCCESS)
{
TCHAR szExplorer[MAX_PATH];
TCHAR szIcon[MAX_PATH+10];
GetWindowsDirectory(szExplorer, ARRAYSIZE(szExplorer));
wnsprintf(szIcon, ARRAYSIZE(szIcon), TEXT("%s\\explorer.exe,-103"), szExplorer); // ICO_DESKTOP res ID
RegSetValueEx(hk, NULL, 0, REG_SZ, (LPBYTE)szIcon, (lstrlen(szIcon) + 1) * SIZEOF(TCHAR));
RegCloseKey(hk);
}
CommonRegister(hkCLSID, pszCLSID, DESKLINK_EXTENSION, IDS_DESKTOPLINK_FILENAME_NEW);
}
else
{
if (SUCCEEDED(GetDropTargetPath(szFile, IDS_DESKTOPLINK_FILENAME, DESKLINK_EXTENSION)))
{
DeleteFile(szFile);
}
}
return S_OK;
}