mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 18:15:11 +00:00
1488 lines
38 KiB
C++
1488 lines
38 KiB
C++
/*
|
|
* PROJECT: ReactOS shdocvw
|
|
* LICENSE: LGPL-2.1-or-later (https://spdx.org/licenses/LGPL-2.1-or-later)
|
|
* PURPOSE: NameSpace Control Band
|
|
* COPYRIGHT: Copyright 2024 Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
|
|
*/
|
|
|
|
#include "objects.h"
|
|
#include <shlobj.h>
|
|
#include <commoncontrols.h>
|
|
#include <undocshell.h>
|
|
|
|
#define TIMER_ID_REFRESH 9999
|
|
|
|
#include <wine/debug.h>
|
|
WINE_DEFAULT_DEBUG_CHANNEL(shdocvw);
|
|
|
|
HRESULT
|
|
SHDOCVW_GetPathOfShortcut(
|
|
_In_opt_ HWND hWnd,
|
|
_In_ LPCWSTR pszLnkFile,
|
|
_Out_ LPWSTR pszPath)
|
|
{
|
|
*pszPath = UNICODE_NULL;
|
|
CComPtr<IShellLink> pShellLink;
|
|
HRESULT hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
|
|
IID_PPV_ARG(IShellLink, &pShellLink));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
CComPtr<IPersistFile> pPersistFile;
|
|
hr = pShellLink->QueryInterface(IID_PPV_ARG(IPersistFile, &pPersistFile));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
hr = pPersistFile->Load(pszLnkFile, STGM_READ);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
WIN32_FIND_DATA find;
|
|
hr = pShellLink->GetPath(pszPath, MAX_PATH, &find, 0);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT
|
|
SHDOCVW_CreateShortcut(
|
|
_In_ LPCWSTR pszLnkFileName,
|
|
_In_ PCIDLIST_ABSOLUTE pidlTarget,
|
|
_In_opt_ LPCWSTR pszDescription)
|
|
{
|
|
HRESULT hr;
|
|
|
|
CComPtr<IShellLink> psl;
|
|
hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
|
|
IID_PPV_ARG(IShellLink, &psl));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
psl->SetIDList(pidlTarget);
|
|
|
|
if (pszDescription)
|
|
psl->SetDescription(pszDescription);
|
|
|
|
CComPtr<IPersistFile> ppf;
|
|
hr = psl->QueryInterface(IID_PPV_ARG(IPersistFile, &ppf));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
return ppf->Save(pszLnkFileName, TRUE);
|
|
}
|
|
|
|
CNSCBand::CNSCBand()
|
|
{
|
|
SHDOCVW_LockModule();
|
|
|
|
INITCOMMONCONTROLSEX iccx = { sizeof(iccx), ICC_TREEVIEW_CLASSES | ICC_BAR_CLASSES };
|
|
::InitCommonControlsEx(&iccx);
|
|
}
|
|
|
|
CNSCBand::~CNSCBand()
|
|
{
|
|
if (m_hToolbarImageList)
|
|
{
|
|
ImageList_Destroy(m_hToolbarImageList);
|
|
m_hToolbarImageList = NULL;
|
|
}
|
|
SHDOCVW_UnlockModule();
|
|
}
|
|
|
|
VOID CNSCBand::OnFinalMessage(HWND)
|
|
{
|
|
// The message loop is finished, now we can safely destruct!
|
|
static_cast<IDeskBand *>(this)->Release();
|
|
}
|
|
|
|
// *** helper methods ***
|
|
|
|
CNSCBand::CItemData* CNSCBand::GetItemData(_In_ HTREEITEM hItem)
|
|
{
|
|
if (hItem == TVI_ROOT)
|
|
return NULL;
|
|
|
|
TVITEMW tvItem = { TVIF_PARAM, hItem };
|
|
if (!TreeView_GetItem(m_hwndTreeView, &tvItem))
|
|
return NULL;
|
|
|
|
return reinterpret_cast<CItemData*>(tvItem.lParam);
|
|
}
|
|
|
|
static HRESULT
|
|
SHDOCVW_GetCurrentLocationFromView(_In_ IShellView& View, _In_ PIDLIST_ABSOLUTE *ppidl)
|
|
{
|
|
CComPtr<IFolderView> pfv;
|
|
CComPtr<IShellFolder> psf;
|
|
HRESULT hr = View.QueryInterface(IID_PPV_ARG(IFolderView, &pfv));
|
|
if (SUCCEEDED(hr) && SUCCEEDED(hr = pfv->GetFolder(IID_PPV_ARG(IShellFolder, &psf))))
|
|
hr = SHELL_GetIDListFromObject(psf, ppidl);
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CNSCBand::_GetCurrentLocation(_Out_ PIDLIST_ABSOLUTE *ppidl)
|
|
{
|
|
*ppidl = NULL;
|
|
CComPtr<IShellBrowser> psb;
|
|
HRESULT hr = IUnknown_QueryService(m_pSite, SID_STopLevelBrowser, IID_PPV_ARG(IShellBrowser, &psb));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
CComPtr<IBrowserService> pbs;
|
|
if (SUCCEEDED(hr = psb->QueryInterface(IID_PPV_ARG(IBrowserService, &pbs))))
|
|
if (SUCCEEDED(hr = pbs->GetPidl(ppidl)) && *ppidl)
|
|
return hr;
|
|
|
|
CComPtr<IShellView> psv;
|
|
if (!FAILED_UNEXPECTEDLY(hr = psb->QueryActiveShellView(&psv)))
|
|
if (SUCCEEDED(hr = psv.p ? SHDOCVW_GetCurrentLocationFromView(*psv.p, ppidl) : E_FAIL))
|
|
return hr;
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CNSCBand::_IsCurrentLocation(_In_ PCIDLIST_ABSOLUTE pidl)
|
|
{
|
|
if (!pidl)
|
|
return E_INVALIDARG;
|
|
HRESULT hr = E_FAIL;
|
|
PIDLIST_ABSOLUTE location = NULL;
|
|
hr = _GetCurrentLocation(&location);
|
|
if (SUCCEEDED(hr))
|
|
hr = SHELL_IsEqualAbsoluteID(location, pidl) ? S_OK : S_FALSE;
|
|
ILFree(location);
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CNSCBand::_ExecuteCommand(_In_ CComPtr<IContextMenu>& menu, _In_ UINT nCmd)
|
|
{
|
|
CComPtr<IOleWindow> pBrowserOleWnd;
|
|
HRESULT hr = IUnknown_QueryService(m_pSite, SID_SShellBrowser,
|
|
IID_PPV_ARG(IOleWindow, &pBrowserOleWnd));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
HWND browserWnd;
|
|
hr = pBrowserOleWnd->GetWindow(&browserWnd);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
CMINVOKECOMMANDINFO cmi = { sizeof(cmi) };
|
|
cmi.lpVerb = MAKEINTRESOURCEA(nCmd);
|
|
cmi.hwnd = browserWnd;
|
|
cmi.nShow = SW_SHOW;
|
|
if (::GetKeyState(VK_SHIFT) < 0)
|
|
cmi.fMask |= CMIC_MASK_SHIFT_DOWN;
|
|
if (::GetKeyState(VK_CONTROL) < 0)
|
|
cmi.fMask |= CMIC_MASK_CONTROL_DOWN;
|
|
|
|
return menu->InvokeCommand(&cmi);
|
|
}
|
|
|
|
void CNSCBand::_RegisterChangeNotify()
|
|
{
|
|
#define TARGET_EVENTS ( \
|
|
SHCNE_DRIVEADD | SHCNE_MKDIR | SHCNE_CREATE | SHCNE_DRIVEREMOVED | SHCNE_RMDIR | \
|
|
SHCNE_DELETE | SHCNE_RENAMEFOLDER | SHCNE_RENAMEITEM | SHCNE_UPDATEDIR | \
|
|
SHCNE_UPDATEITEM | SHCNE_ASSOCCHANGED \
|
|
)
|
|
// Register shell notification
|
|
SHChangeNotifyEntry shcne = { m_pidlRoot, TRUE };
|
|
m_shellRegID = SHChangeNotifyRegister(m_hWnd,
|
|
SHCNRF_NewDelivery | SHCNRF_ShellLevel,
|
|
TARGET_EVENTS,
|
|
WM_USER_SHELLEVENT,
|
|
1, &shcne);
|
|
if (!m_shellRegID)
|
|
{
|
|
ERR("Something went wrong, error %08x\n", GetLastError());
|
|
}
|
|
}
|
|
|
|
void CNSCBand::_UnregisterChangeNotify()
|
|
{
|
|
SHChangeNotifyDeregister(m_shellRegID);
|
|
m_shellRegID = 0;
|
|
}
|
|
|
|
void CNSCBand::_DestroyTreeView()
|
|
{
|
|
TRACE("Cleaning up treeview...\n");
|
|
/* Remove all items of the treeview */
|
|
::RevokeDragDrop(m_hwndTreeView);
|
|
TreeView_DeleteAllItems(m_hwndTreeView);
|
|
m_hwndTreeView.DestroyWindow();
|
|
m_pDesktop = NULL;
|
|
m_hRoot = NULL;
|
|
TRACE("Cleanup ok\n");
|
|
}
|
|
|
|
void CNSCBand::_DestroyToolbar()
|
|
{
|
|
m_hwndToolbar.DestroyWindow();
|
|
}
|
|
|
|
HRESULT CNSCBand::_CreateTreeView(HWND hwndParent)
|
|
{
|
|
RefreshFlags(&m_dwTVStyle, &m_dwTVExStyle, &m_dwEnumFlags);
|
|
HWND hwndTV = ::CreateWindowExW(m_dwTVExStyle, WC_TREEVIEWW, NULL, m_dwTVStyle, 0, 0, 0, 0,
|
|
hwndParent, (HMENU)UlongToHandle(IDW_TREEVIEW), instance, NULL);
|
|
ATLASSERT(hwndTV);
|
|
if (!hwndTV)
|
|
return E_FAIL;
|
|
|
|
m_hwndTreeView.Attach(hwndTV);
|
|
::RegisterDragDrop(m_hwndTreeView, dynamic_cast<IDropTarget*>(this));
|
|
|
|
// Init the treeview here
|
|
HRESULT hr = SHGetDesktopFolder(&m_pDesktop);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
m_pidlRoot.Free();
|
|
hr = SHGetFolderLocation(m_hWnd, _GetRootCsidl(), NULL, 0, &m_pidlRoot);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
// Create image list and set
|
|
IImageList *piml;
|
|
hr = SHGetImageList(SHIL_SMALL, IID_PPV_ARG(IImageList, &piml));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
TreeView_SetImageList(m_hwndTreeView, (HIMAGELIST)piml, TVSIL_NORMAL);
|
|
return S_OK;
|
|
}
|
|
|
|
BOOL
|
|
CNSCBand::_IsTreeItemInEnum(
|
|
_In_ HTREEITEM hItem,
|
|
_In_ IEnumIDList *pEnum)
|
|
{
|
|
CItemData* pItemData = GetItemData(hItem);
|
|
if (!pItemData)
|
|
return FALSE;
|
|
|
|
pEnum->Reset();
|
|
|
|
CComHeapPtr<ITEMIDLIST_RELATIVE> pidlTemp;
|
|
while (pEnum->Next(1, &pidlTemp, NULL) == S_OK)
|
|
{
|
|
if (ILIsEqual(pidlTemp, pItemData->relativePidl))
|
|
return TRUE;
|
|
|
|
pidlTemp.Free();
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL
|
|
CNSCBand::_TreeItemHasThisChild(
|
|
_In_ HTREEITEM hItem,
|
|
_In_ PCITEMID_CHILD pidlChild)
|
|
{
|
|
for (hItem = TreeView_GetChild(m_hwndTreeView, hItem); hItem;
|
|
hItem = TreeView_GetNextSibling(m_hwndTreeView, hItem))
|
|
{
|
|
CItemData* pItemData = GetItemData(hItem);
|
|
if (ILIsEqual(pItemData->relativePidl, pidlChild))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
HRESULT
|
|
CNSCBand::_GetItemEnum(
|
|
_Out_ CComPtr<IEnumIDList>& pEnum,
|
|
_In_ HTREEITEM hItem,
|
|
_Out_opt_ IShellFolder **ppFolder)
|
|
{
|
|
CComPtr<IShellFolder> psfDesktop;
|
|
HRESULT hr = SHGetDesktopFolder(&psfDesktop);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
CComPtr<IShellFolder> pFolder;
|
|
if (!ppFolder)
|
|
ppFolder = &pFolder;
|
|
|
|
if (hItem == m_hRoot && hItem)
|
|
{
|
|
*ppFolder = psfDesktop;
|
|
(*ppFolder)->AddRef();
|
|
}
|
|
else
|
|
{
|
|
CItemData* pItemData = GetItemData(hItem);
|
|
if (!pItemData && hItem == TVI_ROOT && !_WantsRootItem())
|
|
hr = psfDesktop->BindToObject(m_pidlRoot, NULL, IID_PPV_ARG(IShellFolder, ppFolder));
|
|
else
|
|
hr = psfDesktop->BindToObject(pItemData->absolutePidl, NULL, IID_PPV_ARG(IShellFolder, ppFolder));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
}
|
|
|
|
return (*ppFolder)->EnumObjects(NULL, _GetEnumFlags(), &pEnum);
|
|
}
|
|
|
|
BOOL CNSCBand::_ItemHasAnyChild(_In_ HTREEITEM hItem)
|
|
{
|
|
CComPtr<IEnumIDList> pEnum;
|
|
HRESULT hr = _GetItemEnum(pEnum, hItem);
|
|
if (FAILED(hr))
|
|
return FALSE;
|
|
|
|
CComHeapPtr<ITEMIDLIST_RELATIVE> pidlTemp;
|
|
hr = pEnum->Next(1, &pidlTemp, NULL);
|
|
return SUCCEEDED(hr);
|
|
}
|
|
|
|
void CNSCBand::_RefreshRecurse(_In_ HTREEITEM hTarget)
|
|
{
|
|
CComPtr<IEnumIDList> pEnum;
|
|
HRESULT hrEnum = _GetItemEnum(pEnum, hTarget);
|
|
|
|
// Delete zombie items
|
|
HTREEITEM hItem, hNextItem;
|
|
for (hItem = TreeView_GetChild(m_hwndTreeView, hTarget); hItem; hItem = hNextItem)
|
|
{
|
|
hNextItem = TreeView_GetNextSibling(m_hwndTreeView, hItem);
|
|
|
|
if (SUCCEEDED(hrEnum) && !_IsTreeItemInEnum(hItem, pEnum))
|
|
TreeView_DeleteItem(m_hwndTreeView, hItem);
|
|
}
|
|
|
|
pEnum = NULL;
|
|
hrEnum = _GetItemEnum(pEnum, hTarget);
|
|
|
|
CItemData* pItemData = ((hTarget == TVI_ROOT) ? NULL : GetItemData(hTarget));
|
|
|
|
// Insert new items and update items
|
|
if (SUCCEEDED(hrEnum))
|
|
{
|
|
CComHeapPtr<ITEMIDLIST_RELATIVE> pidlTemp;
|
|
while (pEnum->Next(1, &pidlTemp, NULL) == S_OK)
|
|
{
|
|
if (!_TreeItemHasThisChild(hTarget, pidlTemp))
|
|
{
|
|
if (pItemData)
|
|
{
|
|
CComHeapPtr<ITEMIDLIST> pidlAbsolute(ILCombine(pItemData->absolutePidl, pidlTemp));
|
|
_InsertItem(hTarget, pidlAbsolute, pidlTemp, TRUE);
|
|
}
|
|
else
|
|
{
|
|
CComHeapPtr<ITEMIDLIST> pidlAbsolute(ILCombine(m_pidlRoot, pidlTemp));
|
|
_InsertItem(hTarget, pidlAbsolute, pidlTemp, TRUE);
|
|
}
|
|
}
|
|
pidlTemp.Free();
|
|
}
|
|
}
|
|
|
|
// Update children and recurse
|
|
for (hItem = TreeView_GetChild(m_hwndTreeView, hTarget); hItem; hItem = hNextItem)
|
|
{
|
|
hNextItem = TreeView_GetNextSibling(m_hwndTreeView, hItem);
|
|
|
|
TV_ITEMW item = { TVIF_HANDLE | TVIF_CHILDREN };
|
|
item.hItem = hItem;
|
|
item.cChildren = _ItemHasAnyChild(hItem);
|
|
TreeView_SetItem(m_hwndTreeView, &item);
|
|
|
|
if (TreeView_GetItemState(m_hwndTreeView, hItem, TVIS_EXPANDEDONCE) & TVIS_EXPANDEDONCE)
|
|
_RefreshRecurse(hItem);
|
|
}
|
|
}
|
|
|
|
void CNSCBand::_Refresh()
|
|
{
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, FALSE, 0);
|
|
_RefreshRecurse(_WantsRootItem() ? m_hRoot : TVI_ROOT);
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, TRUE, 0);
|
|
}
|
|
|
|
LRESULT CNSCBand::OnTimer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
KillTimer(wParam);
|
|
|
|
if (wParam == TIMER_ID_REFRESH)
|
|
_Refresh();
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
CNSCBand::OnChangeNotify(
|
|
_In_opt_ LPCITEMIDLIST pidl0,
|
|
_In_opt_ LPCITEMIDLIST pidl1,
|
|
_In_ LONG lEvent)
|
|
{
|
|
switch (lEvent)
|
|
{
|
|
case SHCNE_DRIVEADD:
|
|
case SHCNE_MKDIR:
|
|
case SHCNE_CREATE:
|
|
case SHCNE_DRIVEREMOVED:
|
|
case SHCNE_RMDIR:
|
|
case SHCNE_DELETE:
|
|
case SHCNE_RENAMEFOLDER:
|
|
case SHCNE_RENAMEITEM:
|
|
case SHCNE_UPDATEDIR:
|
|
case SHCNE_UPDATEITEM:
|
|
case SHCNE_ASSOCCHANGED:
|
|
{
|
|
KillTimer(TIMER_ID_REFRESH);
|
|
SetTimer(TIMER_ID_REFRESH, 500, NULL);
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
TRACE("lEvent: 0x%08lX\n", lEvent);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
HTREEITEM
|
|
CNSCBand::_InsertItem(
|
|
_In_opt_ HTREEITEM hParent,
|
|
_Inout_ IShellFolder *psfParent,
|
|
_In_ LPCITEMIDLIST pElt,
|
|
_In_ LPCITEMIDLIST pEltRelative,
|
|
_In_ BOOL bSort)
|
|
{
|
|
/* Get the attributes of the node */
|
|
SFGAOF attrs = SFGAO_STREAM | SFGAO_HASSUBFOLDER;
|
|
HRESULT hr = psfParent->GetAttributesOf(1, &pEltRelative, &attrs);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return NULL;
|
|
|
|
/* Get the name of the node */
|
|
WCHAR wszDisplayName[MAX_PATH];
|
|
STRRET strret;
|
|
hr = psfParent->GetDisplayNameOf(pEltRelative, SHGDN_INFOLDER, &strret);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return NULL;
|
|
|
|
hr = StrRetToBufW(&strret, pEltRelative, wszDisplayName, MAX_PATH);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return NULL;
|
|
|
|
/* Get the icon of the node */
|
|
INT iIcon = SHMapPIDLToSystemImageListIndex(psfParent, pEltRelative, NULL);
|
|
|
|
CItemData* pChildInfo = new CItemData;
|
|
if (!pChildInfo)
|
|
{
|
|
ERR("Failed to allocate CItemData\n");
|
|
return NULL;
|
|
}
|
|
pChildInfo->absolutePidl.Attach(ILClone(pElt));
|
|
pChildInfo->relativePidl.Attach(ILClone(pEltRelative));
|
|
|
|
// Set up our treeview template
|
|
TV_INSERTSTRUCT tvInsert = { hParent, TVI_LAST };
|
|
tvInsert.item.mask = TVIF_PARAM | TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_CHILDREN;
|
|
tvInsert.item.cchTextMax = MAX_PATH;
|
|
tvInsert.item.pszText = wszDisplayName;
|
|
tvInsert.item.iImage = tvInsert.item.iSelectedImage = iIcon;
|
|
tvInsert.item.lParam = (LPARAM)pChildInfo;
|
|
|
|
if (!(attrs & SFGAO_STREAM) && (attrs & SFGAO_HASSUBFOLDER))
|
|
tvInsert.item.cChildren = 1;
|
|
|
|
HTREEITEM htiCreated = TreeView_InsertItem(m_hwndTreeView, &tvInsert);
|
|
|
|
if (bSort)
|
|
_SortItems(hParent);
|
|
|
|
return htiCreated;
|
|
}
|
|
|
|
/* This is the slow version of the above method */
|
|
HTREEITEM
|
|
CNSCBand::_InsertItem(
|
|
_In_opt_ HTREEITEM hParent,
|
|
_In_ LPCITEMIDLIST pElt,
|
|
_In_ LPCITEMIDLIST pEltRelative,
|
|
_In_ BOOL bSort)
|
|
{
|
|
CComPtr<IShellFolder> psfFolder;
|
|
HRESULT hr = SHBindToParent(pElt, IID_PPV_ARG(IShellFolder, &psfFolder), NULL);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return NULL;
|
|
|
|
return _InsertItem(hParent, psfFolder, pElt, pEltRelative, bSort);
|
|
}
|
|
|
|
BOOL CNSCBand::_InsertSubitems(HTREEITEM hItem, LPCITEMIDLIST entry)
|
|
{
|
|
ULONG fetched = 1, uItemCount = 0;
|
|
|
|
CComPtr<IEnumIDList> pEnum;
|
|
CComPtr<IShellFolder> pFolder;
|
|
HRESULT hr = _GetItemEnum(pEnum, hItem, &pFolder);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return FALSE;
|
|
|
|
/* Don't redraw while we add stuff into the tree */
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, FALSE, 0);
|
|
|
|
LPITEMIDLIST pidlSub;
|
|
while (SUCCEEDED(pEnum->Next(1, &pidlSub, &fetched)) && pidlSub && fetched)
|
|
{
|
|
LPITEMIDLIST pidlSubComplete;
|
|
pidlSubComplete = ILCombine(entry, pidlSub);
|
|
|
|
if (_InsertItem(hItem, pFolder, pidlSubComplete, pidlSub, FALSE))
|
|
++uItemCount;
|
|
|
|
ILFree(pidlSubComplete);
|
|
ILFree(pidlSub);
|
|
}
|
|
|
|
/* Let's do sorting */
|
|
_SortItems(hItem);
|
|
|
|
/* Now we can redraw */
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, TRUE, 0);
|
|
|
|
return (uItemCount > 0);
|
|
}
|
|
|
|
// *** message handlers ***
|
|
|
|
LRESULT CNSCBand::OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (FAILED_UNEXPECTEDLY(_CreateToolbar(m_hWnd)))
|
|
return -1;
|
|
if (FAILED_UNEXPECTEDLY(_CreateTreeView(m_hWnd)))
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnDestroy(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
_DestroyTreeView();
|
|
_DestroyToolbar();
|
|
_UnregisterChangeNotify();
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (!m_hwndTreeView)
|
|
return 0;
|
|
|
|
RECT rc;
|
|
GetClientRect(&rc);
|
|
LONG cx = rc.right, cy = rc.bottom;
|
|
|
|
RECT rcTB;
|
|
LONG cyTB = 0;
|
|
if (m_hwndToolbar)
|
|
{
|
|
m_hwndToolbar.SendMessage(TB_AUTOSIZE, 0, 0);
|
|
m_hwndToolbar.GetWindowRect(&rcTB);
|
|
cyTB = rcTB.bottom - rcTB.top;
|
|
}
|
|
|
|
m_hwndTreeView.MoveWindow(0, cyTB, cx, cy - cyTB);
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnSetFocus(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
m_bFocused = TRUE;
|
|
IUnknown_OnFocusChangeIS(m_pSite, reinterpret_cast<IUnknown*>(this), TRUE);
|
|
bHandled = FALSE;
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnKillFocus(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
IUnknown_OnFocusChangeIS(m_pSite, reinterpret_cast<IUnknown*>(this), FALSE);
|
|
m_bFocused = FALSE;
|
|
return 0;
|
|
}
|
|
|
|
HRESULT CNSCBand::_AddFavorite()
|
|
{
|
|
CComHeapPtr<ITEMIDLIST> pidlCurrent;
|
|
_GetCurrentLocation(&pidlCurrent);
|
|
|
|
WCHAR szCurDir[MAX_PATH];
|
|
if (!ILGetDisplayName(pidlCurrent, szCurDir))
|
|
{
|
|
FIXME("\n");
|
|
return E_FAIL;
|
|
}
|
|
|
|
WCHAR szPath[MAX_PATH], szSuffix[32];
|
|
SHGetSpecialFolderPathW(m_hWnd, szPath, CSIDL_FAVORITES, TRUE);
|
|
PathAppendW(szPath, PathFindFileNameW(szCurDir));
|
|
|
|
const INT ich = lstrlenW(szPath);
|
|
for (INT iTry = 2; iTry <= 9999; ++iTry)
|
|
{
|
|
PathAddExtensionW(szPath, L".lnk");
|
|
if (!PathFileExistsW(szPath))
|
|
break;
|
|
szPath[ich] = UNICODE_NULL;
|
|
wsprintfW(szSuffix, L" (%d)", iTry);
|
|
lstrcatW(szPath, szSuffix);
|
|
}
|
|
|
|
TRACE("%S, %S\n", szCurDir, szPath);
|
|
|
|
return SHDOCVW_CreateShortcut(szPath, pidlCurrent, NULL);
|
|
}
|
|
|
|
LRESULT CNSCBand::OnCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case ID_ADD:
|
|
{
|
|
_AddFavorite();
|
|
break;
|
|
}
|
|
case ID_ORGANIZE:
|
|
{
|
|
SHELLEXECUTEINFOW sei = { sizeof(sei), SEE_MASK_INVOKEIDLIST };
|
|
sei.hwnd = m_hWnd;
|
|
sei.nShow = SW_SHOWNORMAL;
|
|
sei.lpIDList = m_pidlRoot;
|
|
::ShellExecuteExW(&sei);
|
|
break;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
BOOL CNSCBand::OnTreeItemExpanding(_In_ LPNMTREEVIEW pnmtv)
|
|
{
|
|
CItemData *pItemData;
|
|
|
|
if (pnmtv->action == TVE_COLLAPSE)
|
|
{
|
|
if (pnmtv->itemNew.hItem == m_hRoot)
|
|
{
|
|
// Prenvent root from collapsing
|
|
pnmtv->itemNew.mask |= TVIF_STATE;
|
|
pnmtv->itemNew.stateMask |= TVIS_EXPANDED;
|
|
pnmtv->itemNew.state &= ~TVIS_EXPANDED;
|
|
pnmtv->action = TVE_EXPAND;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
if (pnmtv->action == TVE_EXPAND)
|
|
{
|
|
// Grab our directory PIDL
|
|
pItemData = GetItemData(pnmtv->itemNew.hItem);
|
|
// We have it, let's try
|
|
if (pItemData && !pItemData->expanded)
|
|
{
|
|
if (_InsertSubitems(pnmtv->itemNew.hItem, pItemData->absolutePidl))
|
|
{
|
|
pItemData->expanded = TRUE;
|
|
}
|
|
else
|
|
{
|
|
// remove subitem "+" since we failed to add subitems
|
|
TVITEMW tvItem = { TVIF_CHILDREN, pnmtv->itemNew.hItem };
|
|
tvItem.cChildren = 0;
|
|
TreeView_SetItem(m_hwndTreeView, &tvItem);
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL CNSCBand::OnTreeItemDeleted(_In_ LPNMTREEVIEW pnmtv)
|
|
{
|
|
// Navigate to parent when deleting selected item
|
|
HTREEITEM hItem = pnmtv->itemOld.hItem;
|
|
HTREEITEM hParent = TreeView_GetParent(m_hwndTreeView, hItem);
|
|
if (hParent && TreeView_GetSelection(m_hwndTreeView) == hItem)
|
|
TreeView_SelectItem(m_hwndTreeView, hParent);
|
|
|
|
/* Destroy memory associated to our node */
|
|
CItemData* pItemData = GetItemData(hItem);
|
|
if (!pItemData)
|
|
return FALSE;
|
|
|
|
delete pItemData;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void CNSCBand::_OnSelectionChanged(_In_ LPNMTREEVIEW pnmtv)
|
|
{
|
|
HTREEITEM hItem = pnmtv->itemNew.hItem;
|
|
if (!hItem)
|
|
return;
|
|
CItemData* pItemData = GetItemData(hItem);
|
|
if (pItemData)
|
|
OnSelectionChanged(pItemData->absolutePidl);
|
|
}
|
|
|
|
void CNSCBand::OnTreeItemDragging(_In_ LPNMTREEVIEW pnmtv, _In_ BOOL isRightClick)
|
|
{
|
|
CItemData* pItemData = GetItemData(pnmtv->itemNew.hItem);
|
|
if (!pItemData)
|
|
return;
|
|
|
|
HRESULT hr;
|
|
CComPtr<IShellFolder> pSrcFolder;
|
|
LPCITEMIDLIST pLast;
|
|
hr = SHBindToParent(pItemData->absolutePidl, IID_PPV_ARG(IShellFolder, &pSrcFolder), &pLast);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return;
|
|
|
|
SFGAOF attrs = SFGAO_CANCOPY | SFGAO_CANMOVE | SFGAO_CANLINK;
|
|
pSrcFolder->GetAttributesOf(1, &pLast, &attrs);
|
|
|
|
DWORD dwEffect = 0;
|
|
if (attrs & SFGAO_CANCOPY)
|
|
dwEffect |= DROPEFFECT_COPY;
|
|
if (attrs & SFGAO_CANMOVE)
|
|
dwEffect |= DROPEFFECT_MOVE;
|
|
if (attrs & SFGAO_CANLINK)
|
|
dwEffect |= DROPEFFECT_LINK;
|
|
|
|
CComPtr<IDataObject> pObj;
|
|
hr = pSrcFolder->GetUIObjectOf(m_hWnd, 1, &pLast, IID_IDataObject, 0, (LPVOID*)&pObj);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return;
|
|
|
|
DoDragDrop(pObj, this, dwEffect, &dwEffect);
|
|
}
|
|
|
|
LRESULT CNSCBand::OnBeginLabelEdit(_In_ LPNMTVDISPINFO dispInfo)
|
|
{
|
|
// TODO: put this in a function ? (mostly copypasta from CDefView)
|
|
DWORD dwAttr = SFGAO_CANRENAME;
|
|
CComPtr<IShellFolder> pParent;
|
|
LPCITEMIDLIST pChild;
|
|
HRESULT hr;
|
|
|
|
CItemData *info = GetItemData(dispInfo->item.hItem);
|
|
if (!info)
|
|
return FALSE;
|
|
|
|
hr = SHBindToParent(info->absolutePidl, IID_PPV_ARG(IShellFolder, &pParent), &pChild);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return FALSE;
|
|
|
|
hr = pParent->GetAttributesOf(1, &pChild, &dwAttr);
|
|
if (SUCCEEDED(hr) && (dwAttr & SFGAO_CANRENAME))
|
|
{
|
|
m_isEditing = TRUE;
|
|
m_oldSelected = NULL;
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
HRESULT CNSCBand::_UpdateBrowser(LPCITEMIDLIST pidlGoto)
|
|
{
|
|
CComPtr<IShellBrowser> pBrowserService;
|
|
HRESULT hr = IUnknown_QueryService(m_pSite, SID_STopLevelBrowser,
|
|
IID_PPV_ARG(IShellBrowser, &pBrowserService));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
hr = pBrowserService->BrowseObject(pidlGoto, SBSP_SAMEBROWSER | SBSP_ABSOLUTE);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return hr;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnEndLabelEdit(_In_ LPNMTVDISPINFO dispInfo)
|
|
{
|
|
CItemData *info = GetItemData(dispInfo->item.hItem);
|
|
HRESULT hr;
|
|
|
|
m_isEditing = FALSE;
|
|
if (m_oldSelected)
|
|
{
|
|
++m_mtxBlockNavigate;
|
|
TreeView_SelectItem(m_hwndTreeView, m_oldSelected);
|
|
--m_mtxBlockNavigate;
|
|
}
|
|
|
|
if (!dispInfo->item.pszText)
|
|
return FALSE;
|
|
|
|
CComPtr<IShellFolder> pParent;
|
|
LPCITEMIDLIST pidlChild;
|
|
BOOL RenamedCurrent = _IsCurrentLocation(info->absolutePidl) == S_OK;
|
|
|
|
hr = SHBindToParent(info->absolutePidl, IID_PPV_ARG(IShellFolder, &pParent), &pidlChild);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return FALSE;
|
|
|
|
CComHeapPtr<ITEMIDLIST> pidlNew;
|
|
hr = pParent->SetNameOf(m_hWnd, pidlChild, dispInfo->item.pszText, SHGDN_INFOLDER, &pidlNew);
|
|
if (SUCCEEDED(hr) && pidlNew)
|
|
{
|
|
CComPtr<IPersistFolder2> pPersist;
|
|
hr = pParent->QueryInterface(IID_PPV_ARG(IPersistFolder2, &pPersist));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return FALSE;
|
|
|
|
CComHeapPtr<ITEMIDLIST> pidlParent;
|
|
hr = pPersist->GetCurFolder(&pidlParent);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return FALSE;
|
|
|
|
CComHeapPtr<ITEMIDLIST> pidlNewAbs(ILCombine(pidlParent, pidlNew));
|
|
if (RenamedCurrent)
|
|
{
|
|
_UpdateBrowser(pidlNewAbs);
|
|
}
|
|
else
|
|
{
|
|
// Tell everyone if SetNameOf forgot, this causes IShellView to update itself when we rename a child
|
|
SHChangeNotify(SHCNE_RENAMEFOLDER, SHCNF_IDLIST, info->absolutePidl, pidlNewAbs);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
LRESULT CNSCBand::OnNotify(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
NMHDR *pnmhdr = (NMHDR*)lParam;
|
|
switch (pnmhdr->code)
|
|
{
|
|
case TVN_ITEMEXPANDING:
|
|
return OnTreeItemExpanding((LPNMTREEVIEW)lParam);
|
|
//case TVN_SINGLEEXPAND:
|
|
case TVN_SELCHANGED:
|
|
if (pnmhdr->hwndFrom == m_hwndTreeView)
|
|
_OnSelectionChanged((LPNMTREEVIEW)lParam);
|
|
break;
|
|
case TVN_DELETEITEM:
|
|
OnTreeItemDeleted((LPNMTREEVIEW)lParam);
|
|
break;
|
|
case NM_CLICK:
|
|
case NM_RCLICK:
|
|
if (pnmhdr->hwndFrom == m_hwndTreeView)
|
|
{
|
|
TVHITTESTINFO HitTest;
|
|
::GetCursorPos(&HitTest.pt);
|
|
::ScreenToClient(m_hwndTreeView, &HitTest.pt);
|
|
TreeView_HitTest(m_hwndTreeView, &HitTest);
|
|
|
|
if (HitTest.flags & (TVHT_ABOVE | TVHT_BELOW | TVHT_NOWHERE))
|
|
return TRUE; // Prevents click processing
|
|
|
|
if (HitTest.flags & TVHT_ONITEMBUTTON) // [+] / [-]
|
|
break; // Do default processing
|
|
|
|
// Generate selection notification even if same item
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, FALSE, 0);
|
|
TreeView_SelectItem(m_hwndTreeView, NULL);
|
|
TreeView_SelectItem(m_hwndTreeView, HitTest.hItem);
|
|
m_hwndTreeView.SendMessage(WM_SETREDRAW, TRUE, 0);
|
|
|
|
if (pnmhdr->code == NM_CLICK)
|
|
return TRUE; // Prevents click processing
|
|
}
|
|
break;
|
|
case TVN_BEGINDRAG:
|
|
case TVN_BEGINRDRAG:
|
|
OnTreeItemDragging((LPNMTREEVIEW)lParam, pnmhdr->code == TVN_BEGINRDRAG);
|
|
break;
|
|
case TVN_BEGINLABELEDITW:
|
|
return OnBeginLabelEdit((LPNMTVDISPINFO)lParam);
|
|
case TVN_ENDLABELEDITW:
|
|
return OnEndLabelEdit((LPNMTVDISPINFO)lParam);
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// Temporary menu
|
|
struct CMenuTemp
|
|
{
|
|
HMENU m_hMenu = NULL;
|
|
CMenuTemp(HMENU hMenu) : m_hMenu(hMenu)
|
|
{
|
|
}
|
|
~CMenuTemp()
|
|
{
|
|
if (m_hMenu)
|
|
::DestroyMenu(m_hMenu);
|
|
}
|
|
operator HMENU() const
|
|
{
|
|
return m_hMenu;
|
|
}
|
|
};
|
|
|
|
// *** ATL event handlers ***
|
|
LRESULT CNSCBand::OnContextMenu(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
HWND hwndTarget = reinterpret_cast<HWND>(wParam);
|
|
if (hwndTarget && (hwndTarget == m_hwndToolbar || hwndTarget == m_hWnd))
|
|
{
|
|
FIXME("Show 'Close Toolbar' menu\n");
|
|
return 0;
|
|
}
|
|
|
|
HTREEITEM hItem = TreeView_GetSelection(m_hwndTreeView);
|
|
if (!hItem)
|
|
return 0;
|
|
|
|
POINT pt = { (SHORT)LOWORD(lParam), (SHORT)HIWORD(lParam) };
|
|
if ((UINT)lParam == (UINT)-1)
|
|
{
|
|
RECT rc;
|
|
if (TreeView_GetItemRect(m_hwndTreeView, hItem, &rc, TRUE))
|
|
{
|
|
// Center of item rectangle
|
|
pt.x = (rc.left + rc.right) / 2;
|
|
pt.y = (rc.top + rc.bottom) / 2;
|
|
}
|
|
ClientToScreen(&pt);
|
|
}
|
|
|
|
CItemData *info = GetItemData(hItem);
|
|
if (!info)
|
|
{
|
|
ERR("No node data, something has gone wrong\n");
|
|
return 0;
|
|
}
|
|
|
|
CComPtr<IShellFolder> pFolder;
|
|
LPCITEMIDLIST pidlChild;
|
|
HRESULT hr = SHBindToParent(info->absolutePidl, IID_PPV_ARG(IShellFolder, &pFolder), &pidlChild);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return 0;
|
|
|
|
CComPtr<IContextMenu> contextMenu;
|
|
hr = pFolder->GetUIObjectOf(m_hWnd, 1, &pidlChild, IID_NULL_PPV_ARG(IContextMenu, &contextMenu));
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return 0;
|
|
|
|
IUnknown_SetSite(contextMenu, (IDeskBand *)this);
|
|
|
|
UINT cmf = CMF_EXPLORE;
|
|
SFGAOF attr = SFGAO_CANRENAME;
|
|
hr = pFolder->GetAttributesOf(1, &pidlChild, &attr);
|
|
if (SUCCEEDED(hr) && (attr & SFGAO_CANRENAME))
|
|
cmf |= CMF_CANRENAME;
|
|
|
|
CMenuTemp menuTemp(::CreatePopupMenu());
|
|
UINT idCmdFirst = max(FCIDM_SHVIEWFIRST, 1);
|
|
hr = contextMenu->QueryContextMenu(menuTemp, 0, idCmdFirst, FCIDM_SHVIEWLAST, cmf);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return 0;
|
|
|
|
enum { flags = TPM_LEFTALIGN | TPM_RETURNCMD | TPM_LEFTBUTTON | TPM_RIGHTBUTTON };
|
|
UINT uCommand = ::TrackPopupMenu(menuTemp, flags, pt.x, pt.y, 0, m_hWnd, NULL);
|
|
if (uCommand)
|
|
{
|
|
uCommand -= idCmdFirst;
|
|
|
|
// Do DFM_CMD_RENAME in the treeview
|
|
if ((cmf & CMF_CANRENAME) && SHELL_IsVerb(contextMenu, uCommand, L"rename"))
|
|
{
|
|
m_hwndTreeView.SetFocus();
|
|
if (TreeView_EditLabel(m_hwndTreeView, hItem))
|
|
m_oldSelected = hItem;
|
|
return 0;
|
|
}
|
|
|
|
hr = _ExecuteCommand(contextMenu, uCommand);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// WM_USER_SHELLEVENT
|
|
LRESULT CNSCBand::OnShellEvent(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL &bHandled)
|
|
{
|
|
// We use SHCNRF_NewDelivery method
|
|
HANDLE hChange = (HANDLE)wParam;
|
|
DWORD dwProcID = (DWORD)lParam;
|
|
|
|
PIDLIST_ABSOLUTE *ppidl = NULL;
|
|
LONG lEvent;
|
|
HANDLE hLock = SHChangeNotification_Lock(hChange, dwProcID, &ppidl, &lEvent);
|
|
if (!hLock)
|
|
{
|
|
ERR("!hLock\n");
|
|
return 0;
|
|
}
|
|
|
|
OnChangeNotify(ppidl[0], ppidl[1], (lEvent & ~SHCNE_INTERRUPT));
|
|
|
|
SHChangeNotification_Unlock(hLock);
|
|
return 0;
|
|
}
|
|
|
|
// *** IOleWindow ***
|
|
|
|
STDMETHODIMP CNSCBand::GetWindow(HWND *lphwnd)
|
|
{
|
|
if (!lphwnd)
|
|
return E_INVALIDARG;
|
|
*lphwnd = m_hWnd;
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::ContextSensitiveHelp(BOOL fEnterMode)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IDockingWindow ***
|
|
|
|
STDMETHODIMP CNSCBand::CloseDW(DWORD dwReserved)
|
|
{
|
|
// We do nothing, we don't have anything to save yet
|
|
TRACE("CloseDW called\n");
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::ResizeBorderDW(const RECT *prcBorder, IUnknown *punkToolbarSite, BOOL fReserved)
|
|
{
|
|
/* Must return E_NOTIMPL according to MSDN */
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::ShowDW(BOOL fShow)
|
|
{
|
|
m_fVisible = fShow;
|
|
ShowWindow(fShow ? SW_SHOW : SW_HIDE);
|
|
return S_OK;
|
|
}
|
|
|
|
// *** IDeskBand ***
|
|
|
|
STDMETHODIMP CNSCBand::GetBandInfo(DWORD dwBandID, DWORD dwViewMode, DESKBANDINFO *pdbi)
|
|
{
|
|
if (!pdbi)
|
|
return E_INVALIDARG;
|
|
|
|
m_dwBandID = dwBandID;
|
|
|
|
if (pdbi->dwMask & DBIM_MINSIZE)
|
|
{
|
|
pdbi->ptMinSize.x = 200;
|
|
pdbi->ptMinSize.y = 30;
|
|
}
|
|
|
|
if (pdbi->dwMask & DBIM_MAXSIZE)
|
|
pdbi->ptMaxSize.y = -1;
|
|
|
|
if (pdbi->dwMask & DBIM_INTEGRAL)
|
|
pdbi->ptIntegral.y = 1;
|
|
|
|
if (pdbi->dwMask & DBIM_ACTUAL)
|
|
{
|
|
pdbi->ptActual.x = 200;
|
|
pdbi->ptActual.y = 30;
|
|
}
|
|
|
|
if (pdbi->dwMask & DBIM_TITLE)
|
|
{
|
|
_GetTitle(pdbi->wszTitle, _countof(pdbi->wszTitle));
|
|
}
|
|
|
|
if (pdbi->dwMask & DBIM_MODEFLAGS)
|
|
pdbi->dwModeFlags = DBIMF_NORMAL | DBIMF_VARIABLEHEIGHT;
|
|
|
|
if (pdbi->dwMask & DBIM_BKCOLOR)
|
|
pdbi->dwMask &= ~DBIM_BKCOLOR;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// *** IObjectWithSite ***
|
|
|
|
STDMETHODIMP CNSCBand::SetSite(IUnknown *pUnkSite)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (pUnkSite == m_pSite)
|
|
return S_OK;
|
|
|
|
TRACE("SetSite called\n");
|
|
|
|
if (!pUnkSite)
|
|
{
|
|
DestroyWindow();
|
|
m_hWnd = NULL;
|
|
}
|
|
|
|
if (pUnkSite != m_pSite)
|
|
m_pSite = NULL;
|
|
|
|
if (!pUnkSite)
|
|
return S_OK;
|
|
|
|
HWND hwndParent;
|
|
hr = IUnknown_GetWindow(pUnkSite, &hwndParent);
|
|
if (FAILED_UNEXPECTEDLY(hr))
|
|
return E_INVALIDARG;
|
|
|
|
m_pSite = pUnkSite;
|
|
|
|
if (m_hWnd)
|
|
{
|
|
SetParent(hwndParent); // Change its parent
|
|
}
|
|
else
|
|
{
|
|
enum { style = WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN };
|
|
this->Create(hwndParent, NULL, NULL, style, 0, 0U, NULL);
|
|
}
|
|
|
|
_RegisterChangeNotify();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::GetSite(REFIID riid, void **ppvSite)
|
|
{
|
|
if (!ppvSite)
|
|
return E_POINTER;
|
|
*ppvSite = m_pSite;
|
|
return S_OK;
|
|
}
|
|
|
|
// *** IOleCommandTarget ***
|
|
|
|
STDMETHODIMP CNSCBand::QueryStatus(const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds [], OLECMDTEXT *pCmdText)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::Exec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn, VARIANT *pvaOut)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IServiceProvider ***
|
|
|
|
STDMETHODIMP CNSCBand::QueryService(REFGUID guidService, REFIID riid, void **ppvObject)
|
|
{
|
|
return IUnknown_QueryService(m_pSite, guidService, riid, ppvObject);
|
|
}
|
|
|
|
// *** IContextMenu ***
|
|
|
|
STDMETHODIMP CNSCBand::QueryContextMenu(
|
|
HMENU hmenu,
|
|
UINT indexMenu,
|
|
UINT idCmdFirst,
|
|
UINT idCmdLast,
|
|
UINT uFlags)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::InvokeCommand(
|
|
LPCMINVOKECOMMANDINFO lpici)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::GetCommandString(
|
|
UINT_PTR idCmd,
|
|
UINT uType,
|
|
UINT *pwReserved,
|
|
LPSTR pszName,
|
|
UINT cchMax)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IInputObject ***
|
|
|
|
STDMETHODIMP CNSCBand::UIActivateIO(BOOL fActivate, LPMSG lpMsg)
|
|
{
|
|
if (fActivate)
|
|
{
|
|
m_hwndTreeView.SetFocus();
|
|
}
|
|
|
|
if (lpMsg)
|
|
{
|
|
TranslateMessage(lpMsg);
|
|
DispatchMessage(lpMsg);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::HasFocusIO()
|
|
{
|
|
return m_bFocused ? S_OK : S_FALSE;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::TranslateAcceleratorIO(LPMSG lpMsg)
|
|
{
|
|
if (lpMsg->hwnd == m_hWnd ||
|
|
(m_isEditing && IsChild(lpMsg->hwnd)))
|
|
{
|
|
TranslateMessage(lpMsg);
|
|
DispatchMessage(lpMsg);
|
|
return S_OK;
|
|
}
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
// *** IPersist ***
|
|
|
|
STDMETHODIMP CNSCBand::GetClassID(CLSID *pClassID)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IPersistStream ***
|
|
|
|
STDMETHODIMP CNSCBand::IsDirty()
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::Load(IStream *pStm)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::Save(IStream *pStm, BOOL fClearDirty)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::GetSizeMax(ULARGE_INTEGER *pcbSize)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IWinEventHandler ***
|
|
|
|
STDMETHODIMP CNSCBand::OnWinEvent(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT *theResult)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::IsWindowOwner(HWND hWnd)
|
|
{
|
|
return SHIsChildOrSelf(m_hWnd, hWnd);
|
|
}
|
|
|
|
// *** IBandNavigate ***
|
|
|
|
STDMETHODIMP CNSCBand::Select(LPCITEMIDLIST pidl)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** INamespaceProxy ***
|
|
|
|
// Returns the ITEMIDLIST that should be navigated when an item is invoked.
|
|
STDMETHODIMP CNSCBand::GetNavigateTarget(
|
|
_In_ PCIDLIST_ABSOLUTE pidl,
|
|
_Out_ PIDLIST_ABSOLUTE *ppidlTarget,
|
|
_Out_ ULONG *pulAttrib)
|
|
{
|
|
*pulAttrib = 0;
|
|
WCHAR szPath[MAX_PATH];
|
|
if (!SHGetPathFromIDListW(pidl, szPath))
|
|
return E_FAIL;
|
|
|
|
if (lstrcmpiW(PathFindExtensionW(szPath), L".lnk") == 0) // shortcut file?
|
|
{
|
|
WCHAR szTarget[MAX_PATH];
|
|
HRESULT hr = SHDOCVW_GetPathOfShortcut(m_hWnd, szPath, szTarget);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
lstrcpynW(szPath, szTarget, _countof(szPath));
|
|
*pulAttrib |= SFGAO_LINK;
|
|
}
|
|
}
|
|
|
|
if (PathIsDirectoryW(szPath) || PathIsRootW(szPath))
|
|
*pulAttrib |= SFGAO_FOLDER;
|
|
|
|
*ppidlTarget = ILCreateFromPathW(szPath);
|
|
return S_OK;
|
|
}
|
|
|
|
// Handles a user action on an item.
|
|
STDMETHODIMP CNSCBand::Invoke(_In_ PCIDLIST_ABSOLUTE pidl)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// Called when the user has selected an item.
|
|
STDMETHODIMP CNSCBand::OnSelectionChanged(_In_ PCIDLIST_ABSOLUTE pidl)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
// Returns flags used to update the tree control.
|
|
STDMETHODIMP CNSCBand::RefreshFlags(
|
|
_Out_ DWORD *pdwStyle,
|
|
_Out_ DWORD *pdwExStyle,
|
|
_Out_ DWORD *dwEnum)
|
|
{
|
|
*pdwStyle = _GetTVStyle();
|
|
*pdwExStyle = _GetTVExStyle();
|
|
*dwEnum = _GetEnumFlags();
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::CacheItem(
|
|
_In_ PCIDLIST_ABSOLUTE pidl)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
// *** IDropTarget methods ***
|
|
STDMETHODIMP CNSCBand::DragEnter(IDataObject *pObj, DWORD glfKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
ERR("Entering drag\n");
|
|
m_pCurObject = pObj;
|
|
m_oldSelected = TreeView_GetSelection(m_hwndTreeView);
|
|
return DragOver(glfKeyState, pt, pdwEffect);
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::DragOver(DWORD glfKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
TVHITTESTINFO info;
|
|
info.pt.x = pt.x;
|
|
info.pt.y = pt.y;
|
|
info.flags = TVHT_ONITEM;
|
|
info.hItem = NULL;
|
|
ScreenToClient(&info.pt);
|
|
|
|
// Move to the item selected by the treeview (don't change right pane)
|
|
TreeView_HitTest(m_hwndTreeView, &info);
|
|
|
|
HRESULT hr;
|
|
if (!info.hItem)
|
|
{
|
|
m_childTargetNode = NULL;
|
|
m_pDropTarget = NULL;
|
|
*pdwEffect = DROPEFFECT_NONE;
|
|
return S_OK;
|
|
}
|
|
|
|
++m_mtxBlockNavigate;
|
|
TreeView_SelectItem(m_hwndTreeView, info.hItem);
|
|
--m_mtxBlockNavigate;
|
|
|
|
// Delegate to shell folder
|
|
if (m_pDropTarget && info.hItem != m_childTargetNode)
|
|
m_pDropTarget = NULL;
|
|
|
|
if (info.hItem != m_childTargetNode)
|
|
{
|
|
CItemData *pItemData = GetItemData(info.hItem);
|
|
if (!pItemData)
|
|
return E_FAIL;
|
|
|
|
CComPtr<IShellFolder> pFolder;
|
|
if (_ILIsDesktop(pItemData->absolutePidl))
|
|
{
|
|
pFolder = m_pDesktop;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pDesktop->BindToObject(pItemData->absolutePidl, 0, IID_PPV_ARG(IShellFolder, &pFolder));
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
/* Don't allow dnd since we couldn't get our folder object */
|
|
ERR("Can't bind to folder object\n");
|
|
*pdwEffect = DROPEFFECT_NONE;
|
|
return E_FAIL;
|
|
}
|
|
}
|
|
|
|
hr = pFolder->CreateViewObject(m_hWnd, IID_PPV_ARG(IDropTarget, &m_pDropTarget));
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
/* Don't allow dnd since we couldn't get our drop target */
|
|
ERR("Can't get drop target for folder object\n");
|
|
*pdwEffect = DROPEFFECT_NONE;
|
|
return E_FAIL;
|
|
}
|
|
|
|
hr = m_pDropTarget->DragEnter(m_pCurObject, glfKeyState, pt, pdwEffect);
|
|
m_childTargetNode = info.hItem;
|
|
}
|
|
|
|
if (m_pDropTarget)
|
|
hr = m_pDropTarget->DragOver(glfKeyState, pt, pdwEffect);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::DragLeave()
|
|
{
|
|
++m_mtxBlockNavigate;
|
|
TreeView_SelectItem(m_hwndTreeView, m_oldSelected);
|
|
--m_mtxBlockNavigate;
|
|
m_childTargetNode = NULL;
|
|
if (m_pCurObject)
|
|
m_pCurObject = NULL;
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::Drop(IDataObject *pObj, DWORD glfKeyState, POINTL pt, DWORD *pdwEffect)
|
|
{
|
|
if (!m_pDropTarget)
|
|
return E_FAIL;
|
|
m_pDropTarget->Drop(pObj, glfKeyState, pt, pdwEffect);
|
|
DragLeave();
|
|
return S_OK;
|
|
}
|
|
|
|
// *** IDropSource methods ***
|
|
|
|
STDMETHODIMP CNSCBand::QueryContinueDrag(BOOL fEscapePressed, DWORD grfKeyState)
|
|
{
|
|
if (fEscapePressed)
|
|
return DRAGDROP_S_CANCEL;
|
|
if ((grfKeyState & MK_LBUTTON) || (grfKeyState & MK_RBUTTON))
|
|
return S_OK;
|
|
return DRAGDROP_S_DROP;
|
|
}
|
|
|
|
STDMETHODIMP CNSCBand::GiveFeedback(DWORD dwEffect)
|
|
{
|
|
return DRAGDROP_S_USEDEFAULTCURSORS;
|
|
}
|