Windows2000/private/shell/ext/nsc/nsc.c
2020-09-30 17:12:32 +02:00

1473 lines
39 KiB
C

#define NOSHELLDEBUG // don't take shell versions of this
// todo:
// "New Folder" capability
// dlb click on folders should expand/collapse
// delayed keyboard selection so keyboard navigation does not generate a sel change
// drag and drop image drawing (not just OLE cursors)
// name space change notification. hook up notify handler
// default keyboard accelerators (F2 = Rename, etc)
// Partial expanded nodes in the tree "Tree Down" (TVIS_EXPANDPARTIAL) for net cases
// Programbility:
// notifies - sel changed, node expanded, verb executed, etc.
// cmds - do verb, get item, etc.
#include <windows.h>
#include <shlobj.h>
#include <shlobjp.h> // for SHChangeNotifyReigster
#include "idlist.h"
#include "nsc.h"
#include "dropsrc.h"
#include "common.h"
#define ID_CONTROL 100
#define WM_CHANGENOTIFY (WM_USER + 11)
#define SHCNE_FOLDERS \
(SHCNE_MKDIR | SHCNE_RMDIR | \
SHCNE_MEDIAINSERTED | SHCNE_MEDIAREMOVED | \
SHCNE_DRIVEREMOVED | SHCNE_DRIVEADD | \
SHCNE_RENAMEFOLDER | \
SHCNE_UPDATEDIR | \
SHCNE_UPDATEITEM | \
SHCNE_SERVERDISCONNECT | \
SHCNE_UPDATEIMAGE | \
SHCNE_DRIVEADDGUI)
#define SHCNE_ITEMS (SHCNE_CREATE | SHCNE_DELETE | SHCNE_RENAMEITEM | SHCNE_ASSOCCHANGED)
void _RegisterNotify(NSC *pns, BOOL fReRegister)
{
if (pns->nChangeNotifyID)
{
SHChangeNotifyDeregister(pns->nChangeNotifyID);
pns->nChangeNotifyID = 0;
}
if (fReRegister)
{
SHChangeNotifyEntry fsne;
fsne.pidl = pns->pidlRoot;
fsne.fRecursive = TRUE;
pns->nChangeNotifyID = SHChangeNotifyRegister(pns->hwnd,
// SHCNRF_NewDelivery |
SHCNRF_ShellLevel | SHCNRF_InterruptLevel,
pns->style & NSS_SHOWNONFOLDERS ? (SHCNE_FOLDERS | SHCNE_ITEMS) : (SHCNE_FOLDERS),
WM_CHANGENOTIFY, 1, &fsne);
}
}
LRESULT _OnNCCreate(HWND hwnd, LPCREATESTRUCT pcs)
{
NSC *pns = LocalAlloc(LPTR, sizeof(NSC));
if (pns)
{
pns->hwnd = hwnd;
pns->hwndParent = pcs->hwndParent;
pns->style = pcs->style;
pns->id = (UINT)pcs->hMenu;
// remove border styles from our window that we propogated to
// our child control
SetWindowLong(hwnd, GWL_STYLE, pcs->style & ~WS_BORDER);
SetWindowLong(hwnd, GWL_EXSTYLE, pcs->dwExStyle & ~WS_EX_CLIENTEDGE);
pns->hwndTree = CreateWindowEx(pcs->dwExStyle, WC_TREEVIEW, NULL,
(pcs->style & 0xFFFF0000) | (WS_CHILD | TVS_HASBUTTONS | TVS_EDITLABELS | TVS_SHOWSELALWAYS), // TVS_HASLINES |
pcs->x, pcs->y, pcs->cx, pcs->cy,
hwnd, (HMENU)ID_CONTROL, pcs->hInstance, NULL);
if (pns->hwndTree)
{
SHFILEINFO sfi;
HIMAGELIST himl = (HIMAGELIST)SHGetFileInfo("C:\\", 0, &sfi, sizeof(SHFILEINFO), SHGFI_SYSICONINDEX | SHGFI_SMALLICON);
SetWindowLong(hwnd, 0, (LONG)pns);
TreeView_SetImageList(pns->hwndTree, himl, TVSIL_NORMAL);
if (pns->style & NSS_DROPTARGET)
CTreeDropTarget_Register(pns);
return TRUE; // success
}
LocalFree(pns);
}
DebugMsg(DM_ERROR, "Failing NameSpaceControl create");
return FALSE; // fail the create
}
void _ReleaseCachedShellFolder(NSC *pns)
{
if (pns->psfCache)
{
Release(pns->psfCache);
pns->psfCache = NULL;
pns->htiCache = NULL;
}
}
void _ReleaseRootFolder(NSC *pns)
{
if (pns->psfRoot)
{
Release(pns->psfRoot);
pns->psfRoot = NULL;
if (pns->pidlRoot)
{
ILFree(pns->pidlRoot);
pns->pidlRoot = NULL;
}
}
}
void _OnNCDestroy(NSC *pns)
{
_ReleaseRootFolder(pns);
Assert(pns->pidlRoot == NULL);
// ILFree(pns->pidlRoot);
// pns->pidlRoot = NULL;
_ReleaseCachedShellFolder(pns);
if (pns->style & NSS_DROPTARGET)
CTreeDropTarget_Revoke(pns);
_RegisterNotify(pns, FALSE);
LocalFree(pns);
}
// builds a fully qualified IDLIST from a given tree node by walking up the tree
// be sure to free this when you are done!
LPITEMIDLIST _GetFullIDList(HWND hwndTree, HTREEITEM hti)
{
LPITEMIDLIST pidl;
TV_ITEM tvi;
Assert(hti);
// now lets get the information about the item
tvi.mask = TVIF_PARAM | TVIF_HANDLE;
tvi.hItem = hti;
if (!TreeView_GetItem(hwndTree, &tvi))
{
DebugMsg(DM_ERROR, "bogus tree item passed");
return NULL;
}
pidl = ILClone((LPITEMIDLIST)tvi.lParam);
// Now walk up parents.
while ((tvi.hItem = TreeView_GetParent(hwndTree, tvi.hItem)) && pidl)
{
LPITEMIDLIST pidlT;
if (!TreeView_GetItem(hwndTree, &tvi))
return pidl; // will assume I screwed up...
pidlT = ILCombine((LPITEMIDLIST)tvi.lParam, pidl);
ILFree(pidl);
pidl = pidlT;
}
return pidl;
}
/*
pitem->flags;
pitem->hitem;
pitem->psf;
pitem->pidl;
pitem->dwAttributes;
*/
BOOL _GetItem(NSC *pns, NSC_ITEMINFO *pitem)
{
// BUGBUG: validate pitem->hitem
if (pitem->flags & NSIF_HITEM)
{
}
if (pitem->flags & NSIF_FOLDER)
{
Assert(!(pitem->flags & NSIF_PARENTFOLDER)); // should be exclusive
}
else if (pitem->flags & NSIF_PARENTFOLDER)
{
}
if (pitem->flags & (NSIF_IDLIST | NSIF_FULLIDLIST))
{
pitem->pidl = NULL;
if (pitem->flags & NSIF_FULLIDLIST)
pitem->pidl = _GetFullIDList(pns->hwndTree, (HTREEITEM)pitem->hitem);
}
else
{
TV_ITEM tvi;
tvi.mask = TVIF_PARAM | TVIF_HANDLE;
tvi.hItem = (HTREEITEM)pitem->hitem;
if (TreeView_GetItem(pns->hwndTree, &tvi))
{
pitem->pidl = (LPITEMIDLIST)tvi.lParam;
}
}
if (pitem->flags & NSIF_ATTRIBUTES)
{
}
return TRUE;
}
BOOL _SetItemNotify(NSC *pns, NSI_FLAGS flags)
{
// NS_NOTIFY nsn;
return TRUE;
}
// Some helper functions for processing the dialog
HTREEITEM _AddItemToTree(HWND hwndTree, HTREEITEM htiParent, LPCITEMIDLIST pidl, int cChildren)
{
TV_INSERTSTRUCT tii;
// Initialize item to add with callback for everything
tii.item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_PARAM | TVIF_CHILDREN;
tii.hParent = htiParent;
tii.hInsertAfter = TVI_FIRST;
tii.item.iImage = I_IMAGECALLBACK;
tii.item.iSelectedImage = I_IMAGECALLBACK;
tii.item.pszText = LPSTR_TEXTCALLBACK; //
tii.item.cChildren = cChildren; // Assume it has children
tii.item.lParam = (LPARAM)pidl;
return TreeView_InsertItem(hwndTree, &tii);
}
void _ExpandTree(NSC *pns, HTREEITEM htiRoot, int iDepth)
{
HTREEITEM hti;
if (iDepth == 0)
return;
TreeView_Expand(pns->hwndTree, htiRoot, TVE_EXPAND);
if (iDepth == 1)
return; // avoid useless loop
// recurse to children, expanding them
for (hti = TreeView_GetChild(pns->hwndTree, htiRoot); hti; hti = TreeView_GetNextSibling(pns->hwndTree, hti))
{
_ExpandTree(pns, hti, iDepth - 1);
}
}
// set the root of the name space control.
// in:
// pidlRoot NULL means the desktop
// HIWORD 0 -> LOWORD == ID of special folder (CSIDL_* values)
BOOL _OnSetRoot(NSC *pns, NSC_SETROOT *psr)
{
_ReleaseRootFolder(pns);
if (psr->psf)
{
pns->psfRoot = psr->psf;
}
else if (FAILED(SHGetDesktopFolder(&pns->psfRoot)))
{
DebugMsg(DM_ERROR, "Failed to get desktop folder");
return FALSE;
}
AddRef(pns->psfRoot); // we hang on to this
// HIWORD/LOWORD stuff is to support pidl IDs instead of full pidl here
if (HIWORD(psr->pidlRoot))
pns->pidlRoot = ILClone(psr->pidlRoot);
else
{
Assert(psr->psf == NULL); // special folders are only valid for desktop shell folder
SHGetSpecialFolderLocation(NULL, LOWORD(psr->pidlRoot) ? LOWORD(psr->pidlRoot) : CSIDL_DESKTOP, &pns->pidlRoot);
}
if (pns->pidlRoot)
{
HTREEITEM htiRoot = _AddItemToTree(pns->hwndTree, TVI_ROOT, pns->pidlRoot, 1);
if (htiRoot)
{
_ExpandTree(pns, htiRoot, psr->iExpandDepth);
TreeView_SelectItem(pns->hwndTree, htiRoot);
_RegisterNotify(pns, TRUE);
return TRUE;
}
}
DebugMsg(DM_ERROR, "set root failed");
_ReleaseRootFolder(pns);
return FALSE;
}
// cache the shell folder for a given tree item
// in:
// hti tree node to cache shell folder for. this my be
// NULL indicating the root item.
BOOL _CacheShellFolder(NSC *pns, HTREEITEM hti)
{
// in the cache?
if ((hti != pns->htiCache) || (pns->psfCache == NULL))
{
// cache miss, do the work
LPITEMIDLIST pidl;
_ReleaseCachedShellFolder(pns);
if (hti)
pidl = _GetFullIDList(pns->hwndTree, hti);
else
{
// root item...
pidl = ILClone(pns->pidlRoot);
if (pidl && pidl->mkid.cb != 0)
{
ILRemoveLastID(pidl);
}
}
if (pidl)
{
// special case for root of evil...
if (pidl->mkid.cb == 0)
{
pns->psfCache = pns->psfRoot;
AddRef(pns->psfCache); // to match ref count
}
else
{
pns->psfRoot->lpVtbl->BindToObject(pns->psfRoot, pidl, NULL, &IID_IShellFolder, &pns->psfCache);
}
ILFree(pidl);
if (pns->psfCache)
{
pns->htiCache = hti; // this is for the cache match
return TRUE;
}
else
DebugMsg(DM_ERROR, "failed to get cached shell folder");
}
else
DebugMsg(DM_ERROR, "failed to get create PIDL for cached shell folder");
return FALSE;
}
return TRUE;
}
// pidlItem is typically a relative pidl, except in the case of the root where
// it can be a fully qualified pidl
LPITEMIDLIST _CacheParentShellFolder(NSC *pns, HTREEITEM hti, LPITEMIDLIST pidl)
{
Assert(hti);
if (_CacheShellFolder(pns, TreeView_GetParent(pns->hwndTree, hti)))
{
if (pidl == NULL)
{
TV_ITEM tvi;
tvi.mask = TVIF_PARAM | TVIF_HANDLE;
tvi.hItem = hti;
if (!TreeView_GetItem(pns->hwndTree, &tvi))
return NULL;
pidl = (LPITEMIDLIST)tvi.lParam;
}
return ILFindLastID(pidl);
}
return NULL;
}
int CALLBACK _TreeCompare(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
{
IShellFolder *psf = (IShellFolder *)lParamSort;
HRESULT hres = psf->lpVtbl->CompareIDs(psf, 0, (LPITEMIDLIST)lParam1, (LPITEMIDLIST)lParam2);
Assert(SUCCEEDED(hres));
return (short)SCODE_CODE(hres);
}
void _Sort(NSC *pns, HTREEITEM hti, IShellFolder *psf)
{
TV_SORTCB scb;
scb.hParent = hti;
scb.lpfnCompare = _TreeCompare;
scb.lParam = (LPARAM)psf;
TreeView_SortChildrenCB(pns->hwndTree, &scb, FALSE);
}
// filter function... let clients filter what gets added here
BOOL _ShouldAdd(NSC *pns, LPCITEMIDLIST pidl)
{
#if 0
// We need to special case here in the netcase where we onlyu
// browse down to workgroups...
// Here is where I also need to special case to not go below
// workgroups when the appropriate option is set.
bType = SIL_GetType(pidl);
if ((pns->ulFlags & BIF_DONTGOBELOWDOMAIN) && (bType & SHID_NET))
{
switch (bType & (SHID_NET | SHID_INGROUPMASK))
{
case SHID_NET_SERVER:
ILFree(pidl); // Dont want to add this one
continue; // Try the next one
case SHID_NET_DOMAIN:
cChildren = 0; // Force to not have children;
}
}
else if ((pns->ulFlags & BIF_BROWSEFORCOMPUTER) && (bType & SHID_NET))
{
if ((bType & (SHID_NET | SHID_INGROUPMASK)) == SHID_NET_SERVER)
cChildren = 0; // Don't expand below it...
}
else if (fPrinterTest)
{
// Special case when we are only allowing printers.
// for now I will simply key on the fact that it is non-FS.
ULONG ulAttr = SFGAO_FILESYSTEM;
psf->lpVtbl->GetAttributesOf(psf, 1, &pidl, &ulAttr);
if ((ulAttr & SFGAO_FILESYSTEM)== 0)
{
cChildren = 0; // Force to not have children;
}
else
{
ILFree(pidl); // Dont want to add this one
continue; // Try the next one
}
}
#endif
// send notify up to partent to let them filter
return TRUE;
}
BOOL _OnItemExpanding(NSC *pns, NM_TREEVIEW *pnm)
{
IShellFolder *psf;
IEnumIDList *penum; // Enumerator in use.
DWORD grfFlags = SHCONTF_FOLDERS;
int cAdded = 0;
if ((pnm->action != TVE_EXPAND) || (pnm->itemNew.state & TVIS_EXPANDEDONCE))
return FALSE;
Assert(pnm->itemNew.hItem);
if (!_CacheShellFolder(pns, pnm->itemNew.hItem))
return FALSE;
psf = pns->psfCache;
AddRef(psf); // hang on as adding items may change the cached psfCache
#if 0
// Need to do a couple of special cases here to allow us to
// browse for a network printer. In this case if we are at server
// level we then need to change what we search for non folders when
// we are the level of a server.
if (pns->ulFlags & BIF_BROWSEFORPRINTER)
{
grfFlags = SHCONTF_FOLDERS | SHCONTF_NETPRINTERSRCH;
pidl = ILFindLastID(pidlToExpand);
bType = SIL_GetType(pidl);
fPrinterTest = ((bType & (SHID_NET|SHID_INGROUPMASK))==SHID_NET_SERVER);
if (fPrinterTest)
grfFlags |= SHCONTF_NONFOLDERS;
}
else
#endif
if (pns->style & NSS_SHOWNONFOLDERS)
grfFlags |= SHCONTF_NONFOLDERS;
if (pns->style & NSS_SHOWHIDDEN)
grfFlags |= SHCONTF_INCLUDEHIDDEN;
// passing NULL hwnd makes the enum not put up UI. this is what we want
// in auto-expand cases
if (SUCCEEDED(psf->lpVtbl->EnumObjects(psf,
pns->fAutoExpanding ? NULL : pns->hwnd, grfFlags, &penum)))
{
UINT celt;
LPITEMIDLIST pidl;
while (penum->lpVtbl->Next(penum, 1, &pidl, &celt) == S_OK && celt == 1)
{
int cChildren = I_CHILDRENCALLBACK; // Do call back for children
if (_ShouldAdd(pns, pidl))
{
_AddItemToTree(pns->hwndTree, pnm->itemNew.hItem, pidl, cChildren);
cAdded++;
}
else
{
ILFree(pidl);
}
}
ReleaseLast(penum);
_Sort(pns, pnm->itemNew.hItem, psf);
}
Release(psf);
// If we did not add anything we should update this item to let
// the user know something happened.
if (cAdded == 0)
{
TV_ITEM tvi;
tvi.mask = TVIF_CHILDREN | TVIF_HANDLE; // only change the number of children
tvi.hItem = pnm->itemNew.hItem;
tvi.cChildren = 0;
TreeView_SetItem(pns->hwndTree, &tvi);
}
return TRUE;
}
void _OnDeleteItem(NSC *pns, NM_TREEVIEW *pnm)
{
ILFree((LPITEMIDLIST)pnm->itemOld.lParam);
}
void _GetIconIndex(NSC *pns, LPITEMIDLIST pidl, ULONG ulAttrs, TVITEM *pitem)
{
IShellIcon *psi;
if (SUCCEEDED(QueryInterface(pns->psfCache, &IID_IShellIcon, &psi)))
{
if (psi->lpVtbl->GetIconOf(psi, pidl, 0, &pitem->iImage) == S_OK)
{
if (!(ulAttrs & SFGAO_FOLDER) || FAILED(psi->lpVtbl->GetIconOf(psi, pidl, GIL_OPENICON, &pitem->iSelectedImage)))
{
pitem->iSelectedImage = pitem->iImage;
}
Release(psi);
return;
}
Release(psi);
}
{
// slow way...
LPITEMIDLIST pidlFull = _GetFullIDList(pns->hwndTree, pitem->hItem);
if (pidlFull)
{
SHFILEINFO sfi;
SHGetFileInfo((LPCSTR)pidlFull, 0, &sfi, sizeof(SHFILEINFO), SHGFI_PIDL | SHGFI_SYSICONINDEX); // | SHGFI_SMALLICON
pitem->iImage = sfi.iIcon;
if (!(ulAttrs & SFGAO_FOLDER))
pitem->iSelectedImage = pitem->iImage;
else
{
SHGetFileInfo((LPCSTR)pidlFull, 0, &sfi, sizeof(SHFILEINFO), SHGFI_PIDL | SHGFI_OPENICON | SHGFI_SYSICONINDEX);
pitem->iSelectedImage = sfi.iIcon;
}
ILFree(pidlFull);
}
}
}
int _GetChildren(NSC *pns, IShellFolder *psf, LPCITEMIDLIST pidl, ULONG ulAttrs)
{
int cChildren = 0; // assume none
if (ulAttrs & SFGAO_FOLDER)
{
if (pns->style & NSS_SHOWNONFOLDERS)
{
// there is no SFGAO_ bit that includes non folders so we need to enum
IShellFolder *psfItem;
if (SUCCEEDED(psf->lpVtbl->BindToObject(psf, pidl, NULL, &IID_IShellFolder, &psfItem)))
{
// if we are showing non folders we have to do an enum to peek down at items below
IEnumIDList *penum;
DWORD grfFlags = SHCONTF_FOLDERS | SHCONTF_NONFOLDERS;
if (pns->style & NSS_SHOWHIDDEN)
grfFlags |= SHCONTF_INCLUDEHIDDEN;
if (SUCCEEDED(psfItem->lpVtbl->EnumObjects(psfItem, NULL, grfFlags, &penum)))
{
UINT celt;
LPITEMIDLIST pidlTemp;
if (penum->lpVtbl->Next(penum, 1, &pidlTemp, &celt) == S_OK && celt == 1)
{
ILFree(pidlTemp);
cChildren = 1;
}
Release(penum);
}
Release(psfItem);
}
}
else
{
// if just folders we can peek at the attributes
ULONG ulAttrs = SFGAO_HASSUBFOLDER;
psf->lpVtbl->GetAttributesOf(psf, 1, &pidl, &ulAttrs);
cChildren = (ulAttrs & SFGAO_HASSUBFOLDER) ? 1 : 0;
}
}
return cChildren;
}
void _OnGetDisplayInfo(NSC *pns, TV_DISPINFO *pnm)
{
LPITEMIDLIST pidl = _CacheParentShellFolder(pns, pnm->item.hItem, (LPITEMIDLIST)pnm->item.lParam);
Assert(pidl);
if (pidl == NULL)
return;
Assert(pns->psfCache);
Assert(pnm->item.mask & (TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_TEXT | TVIF_CHILDREN));
if (pnm->item.mask & TVIF_TEXT)
{
STRRET str;
pns->psfCache->lpVtbl->GetDisplayNameOf(pns->psfCache, pidl, SHGDN_INFOLDER, &str);
StrRetToStrN(pnm->item.pszText, pnm->item.cchTextMax, &str, pidl);
}
// make sure we set the attributes for those flags that need them
if (pnm->item.mask & (TVIF_CHILDREN | TVIF_IMAGE | TVIF_SELECTEDIMAGE))
{
ULONG ulAttrs = SFGAO_FOLDER;
pns->psfCache->lpVtbl->GetAttributesOf(pns->psfCache, 1, &pidl, &ulAttrs);
// Also see if this guy has any child folders
if (pnm->item.mask & TVIF_CHILDREN)
{
pnm->item.cChildren = _GetChildren(pns, pns->psfCache, pidl, ulAttrs);
}
if (pnm->item.mask & (TVIF_IMAGE | TVIF_SELECTEDIMAGE))
{
// We now need to map the item into the right image index.
_GetIconIndex(pns, pidl, ulAttrs, &pnm->item);
}
}
// force the treeview to store this so we don't get called back again
pnm->item.mask |= TVIF_DI_SETITEM;
}
// send up the sel changed to let clients enable/disable buttons, etc.
void _OnSelChanged(NSC *pns, LPNM_TREEVIEW pnm)
{
#if 0
LPITEMIDLIST pidl;
ULONG ulAttrs = SFGAO_FILESYSTEM;
BYTE bType;
// We only need to do anything if we only want to return File system
// level objects.
if ((pns->ulFlags & (BIF_RETURNONLYFSDIRS | BIF_RETURNFSANCESTORS | BIF_BROWSEFORPRINTER | BIF_BROWSEFORCOMPUTER)) == 0)
goto NotifySelChange;
// We need to get the attributes of this object...
if (_CacheParentShellFolder(pns, pnm->itemNew.hItem, (LPITEMIDLIST)pnm->itemNew.lParam))
{
BOOL fEnable = TRUE;
bType = SIL_GetType(pidl);
if ((pns->ulFlags & (BIF_RETURNFSANCESTORS | BIF_RETURNONLYFSDIRS)) != 0)
{
int i;
// if this is the root pidl, then do a get attribs on 0
// so that we'll get the attributes on the root, rather than
// random returned values returned by FSFolder
if (ILIsEmpty(pidl)) {
i = 0;
} else
i = 1;
pns->psfCache->lpVtbl->GetAttributesOf(pns->psfCache,
i, &pidl, &ulAttrs);
fEnable = (((ulAttrs & SFGAO_FILESYSTEM) && (pns->ulFlags & BIF_RETURNONLYFSDIRS)) ||
((ulAttrs & SFGAO_FILESYSANCESTOR) && (pns->ulFlags & BIF_RETURNFSANCESTORS))) ||
((bType & (SHID_NET | SHID_INGROUPMASK)) == SHID_NET_SERVER);
}
else if ((pns->ulFlags & BIF_BROWSEFORCOMPUTER) != 0)
{
fEnable = ((bType & (SHID_NET | SHID_INGROUPMASK)) == SHID_NET_SERVER);
}
else if ((pns->ulFlags & BIF_BROWSEFORPRINTER) != 0)
{
// Printers are of type Share and usage Print...
fEnable = ((bType & (SHID_NET | SHID_INGROUPMASK)) == SHID_NET_SHARE);
}
EnableWindow(GetDlgItem(pns->hwnd, IDOK), fEnable);
}
NotifySelChange:
if (pns->lpfn)
{
pidl = _GetFullIDList(pns->hwndTree, pnm->itemNew.hItem);
BFSFCallback(pns, BFFM_SELCHANGED, (LPARAM)pidl);
ILFree(pidl);
}
#endif
}
const char c_szCut[] = "cut";
const char c_szRename[] = "rename";
LRESULT _ContextMenu(NSC *pns, short x, short y)
{
HTREEITEM hti;
POINT ptPopup; // in screen coordinate
if (x == -1 && y == -1)
{
// Keyboard-driven: Get the popup position from the selected item.
hti = TreeView_GetSelection(pns->hwndTree);
if (hti)
{
RECT rc;
// Note that TV_GetItemRect returns it in client coordinate!
TreeView_GetItemRect(pns->hwndTree, hti, &rc, TRUE);
ptPopup.x = (rc.left + rc.right) / 2;
ptPopup.y = (rc.top + rc.bottom) / 2;
MapWindowPoints(pns->hwndTree, HWND_DESKTOP, &ptPopup, 1);
}
}
else
{
TV_HITTESTINFO tvht;
// Mouse-driven: Pick the treeitem from the position.
ptPopup.x = x;
ptPopup.y = y;
tvht.pt = ptPopup;
ScreenToClient(pns->hwndTree, &tvht.pt);
hti = TreeView_HitTest(pns->hwndTree, &tvht);
}
if (hti)
{
LPCITEMIDLIST pidl = _CacheParentShellFolder(pns, hti, NULL);
if (pidl)
{
IContextMenu *pcm;
TreeView_SelectDropTarget(pns->hwndTree, hti);
if (SUCCEEDED(pns->psfCache->lpVtbl->GetUIObjectOf(pns->psfCache, pns->hwnd, 1, &pidl, &IID_IContextMenu, NULL, &pcm)))
{
HMENU hmenu = CreatePopupMenu();
if (hmenu)
{
UINT idCmd;
pns->pcm = pcm; // for IContextMenu2 code
pcm->lpVtbl->QueryContextMenu(pcm, hmenu, 0, 1, 0x7fff,
CMF_EXPLORE | CMF_CANRENAME);
// use pns->hwnd so menu msgs go there and I can forward them
// using IContextMenu2 so "Sent To" works
idCmd = TrackPopupMenu(hmenu,
TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTALIGN,
ptPopup.x, ptPopup.y, 0, pns->hwnd, NULL);
if (idCmd)
{
char szCommandString[64];
BOOL fHandled = FALSE;
BOOL fCutting = FALSE;
// We need to special case the rename command
if (SUCCEEDED(pcm->lpVtbl->GetCommandString(pcm, idCmd - 1,
0, NULL, szCommandString, sizeof(szCommandString))))
{
if (lstrcmpi(szCommandString, c_szRename)==0)
{
TreeView_EditLabel(pns->hwndTree, hti);
fHandled = TRUE;
}
else if (!lstrcmpi(szCommandString, c_szCut))
{
fCutting = TRUE;
}
}
if (!fHandled)
{
CMINVOKECOMMANDINFO ici = {
sizeof(CMINVOKECOMMANDINFO),
0L,
pns->hwndTree,
MAKEINTRESOURCE(idCmd - 1),
NULL, NULL,
SW_NORMAL,
};
HRESULT hres = pcm->lpVtbl->InvokeCommand(pcm, &ici);
if (fCutting && SUCCEEDED(hres))
{
TV_ITEM tvi;
tvi.mask = TVIF_STATE;
tvi.stateMask = TVIS_CUT;
tvi.state = TVIS_CUT;
tvi.hItem = hti;
TreeView_SetItem(pns->hwndTree, &tvi);
// pns->hwndNextViewer = SetClipboardViewer(pns->hwndTree);
// pns->htiCut = hti;
}
}
}
DestroyMenu(hmenu);
pns->pcm = NULL;
}
ReleaseLast(pcm);
}
TreeView_SelectDropTarget(pns->hwndTree, NULL);
}
}
return 0;
}
LRESULT _OnBeginLabelEdit(NSC *pns, TV_DISPINFO *ptvdi)
{
BOOL fCantRename = TRUE;
LPCITEMIDLIST pidl = _CacheParentShellFolder(pns, ptvdi->item.hItem, NULL);
if (pidl)
{
DWORD dwAttribs = SFGAO_CANRENAME;
pns->psfCache->lpVtbl->GetAttributesOf(pns->psfCache, 1, &pidl, &dwAttribs);
if (dwAttribs & SFGAO_CANRENAME)
fCantRename = FALSE;
}
if (fCantRename)
MessageBeep(0);
return fCantRename;
}
LRESULT _OnEndLabelEdit(NSC *pns, TV_DISPINFO *ptvdi)
{
LPCITEMIDLIST pidl;
// See if the user cancelled
if (ptvdi->item.pszText == NULL)
return TRUE; // Nothing to do here.
Assert(ptvdi->item.hItem);
pidl = _CacheParentShellFolder(pns, ptvdi->item.hItem, NULL);
if (pidl)
{
UINT cch = lstrlen(ptvdi->item.pszText)+1;
LPOLESTR pwsz = (LPOLESTR)LocalAlloc(LPTR, cch * sizeof(WCHAR));
if (pwsz)
{
StrToOleStrN(pwsz, cch, ptvdi->item.pszText, -1);
if (SUCCEEDED(pns->psfCache->lpVtbl->SetNameOf(pns->psfCache, pns->hwnd, pidl, pwsz, 0, NULL)))
{
// SHChangeNotifyHandleEvents();
// NOTES: pidl is no longer valid here.
// Set the handle to NULL in the notification to let
// the system know that the pointer is probably not
// valid anymore.
ptvdi->item.hItem = NULL;
}
else
{
SendMessage(pns->hwndTree, TVM_EDITLABEL, (WPARAM)ptvdi->item.pszText, (LPARAM)ptvdi->item.hItem);
}
LocalFree((HLOCAL)pwsz);
}
}
return 0; // We always return 0, "we handled it".
}
void _OnBeginDrag(NSC *pns, NM_TREEVIEW *pnmhdr)
{
LPCITEMIDLIST pidl = _CacheParentShellFolder(pns, pnmhdr->itemNew.hItem, NULL);
if (pidl)
{
DWORD dwEffect = DROPEFFECT_MOVE | DROPEFFECT_COPY | DROPEFFECT_LINK;
pns->psfCache->lpVtbl->GetAttributesOf(pns->psfCache, 1, &pidl, &dwEffect);
dwEffect &= DROPEFFECT_MOVE | DROPEFFECT_COPY | DROPEFFECT_LINK;
if (dwEffect)
{
IDataObject *pdtobj;
HRESULT hres = pns->psfCache->lpVtbl->GetUIObjectOf(pns->psfCache, pns->hwnd, 1, &pidl, &IID_IDataObject, NULL, &pdtobj);
if (SUCCEEDED(hres))
{
hres = OleInitialize(NULL);
if (SUCCEEDED(hres))
{
IDropSource *pdsrc;
if (SUCCEEDED(CDropSource_CreateInstance(&pdsrc)))
{
DWORD dwRet;
pns->htiDragging = pnmhdr->itemNew.hItem;
DoDragDrop(pdtobj, pdsrc, dwEffect, &dwRet);
pns->htiDragging = NULL;
DebugMsg(DM_TRACE, "DoDragDrop returns dwRet: %d", dwRet);
Release(pdsrc);
}
OleUninitialize();
}
#if 0
HIMAGELIST himlDrag = TreeView_CreateDragImage(pns->hwndTree, pnmhdr->itemNew.hItem);
if (himlDrag)
{
if (DAD_SetDragImage(himlDrag, NULL))
{
SHDoDragDrop(hwndOwner, pdtobj, NULL, dwEffect, &dwEffect);
DAD_SetDragImage((HIMAGELIST)-1, NULL);
}
else
{
DebugMsg(DM_TRACE, "sh ER - Tree_OnBeginDrag DAD_SetDragImage failed");
Assert(0);
}
ImageList_Destroy(himlDrag);
}
#endif
ReleaseLast(pdtobj);
}
}
}
}
void _InvokeContextMenu(IShellFolder *psf, LPCITEMIDLIST pidl, HWND hwnd, LPCSTR pszVerb)
{
IContextMenu *pcm;
if (SUCCEEDED(psf->lpVtbl->GetUIObjectOf(psf, hwnd, 1, &pidl, &IID_IContextMenu, NULL, &pcm)))
{
HMENU hmenu = CreatePopupMenu();
if (hmenu)
{
pcm->lpVtbl->QueryContextMenu(pcm, hmenu, 0, 1, 255, pszVerb ? 0 : CMF_DEFAULTONLY);
if (pszVerb == NULL)
pszVerb = MAKEINTRESOURCE(GetMenuDefaultItem(hmenu, MF_BYCOMMAND, 0) - 1);
if (pszVerb)
{
CMINVOKECOMMANDINFOEX ici = {
sizeof(CMINVOKECOMMANDINFOEX),
0L,
hwnd,
pszVerb,
NULL, NULL,
SW_NORMAL,
};
pcm->lpVtbl->InvokeCommand(pcm, (LPCMINVOKECOMMANDINFO)&ici);
}
DestroyMenu(hmenu);
}
Release(pcm);
}
}
void _DoVerb(NSC *pns, HTREEITEM hti, LPCSTR pszVerb)
{
hti = hti ? hti : TreeView_GetSelection(pns->hwndTree);
if (hti)
{
LPCITEMIDLIST pidl = _CacheParentShellFolder(pns, hti, NULL);
if (pidl)
{
_InvokeContextMenu(pns->psfCache, pidl, pns->hwnd, pszVerb);
}
}
}
BOOL _DoDlbClick(NSC *pns)
{
HTREEITEM hti = TreeView_GetSelection(pns->hwndTree);
if (hti)
{
LPCITEMIDLIST pidl = _CacheParentShellFolder(pns, hti, NULL);
if (pidl)
{
ULONG ulAttrs = SFGAO_FOLDER;
pns->psfCache->lpVtbl->GetAttributesOf(pns->psfCache, 1, &pidl, &ulAttrs);
if (ulAttrs & SFGAO_FOLDER)
return FALSE; // do default action (expand/collapse)
_InvokeContextMenu(pns->psfCache, pidl, pns->hwnd, NULL);
}
}
return TRUE;
}
HTREEITEM _GetNodeFromIDList(LPITEMIDLIST pidl)
{
return NULL;
}
BOOL TryQuickRename(LPITEMIDLIST pidl, LPITEMIDLIST pidlExtra)
{
#if 0
LPOneTreeNode lpnSrc;
BOOL fRet = FALSE;
// This can happen when a folder is moved from a "rooted" Explorer outside
// of the root
if (!pidl || !pidlExtra)
return FALSE;
// this one was deleted
lpnSrc = _GetNodeFromIDList(pidl, 0);
if (!lpnSrc)
return FALSE;
if (lpnSrc == s_lpnRoot)
{
OTInvalidateRoot();
return TRUE;
}
else
{
// this one was created
LPITEMIDLIST pidlClone = ILClone(pidlExtra);
if (pidlClone)
{
LPOneTreeNode lpnDestParent;
ILRemoveLastID(pidlClone);
// if the parent isn't created yet, let's not bother
lpnDestParent = _GetNodeFromIDList(pidlClone, 0);
ILFree(pidlClone);
if (lpnDestParent)
{
LPITEMIDLIST pidlLast = OTGetRealFolderIDL(lpnDestParent, ILFindLastID(pidlExtra));
if (pidlLast)
{
LPSHELLFOLDER psf = OTBindToFolder(lpnDestParent);
if (psf)
{
OTAddRef(lpnSrc); // addref because AdoptKid doesn't and OTAbandonKid releases
// remove lpnSrc from its parent's list.
OTAbandonKid(lpnSrc->lpnParent, lpnSrc);
// invalidate the new node's parent to get any children flags right
OTInvalidateNode(lpnDestParent);
// free any cached folders
OTSweepFolders(lpnSrc);
SFCFreeNode(lpnSrc);
OTFreeNodeData(lpnSrc);
lpnSrc->pidl = pidlLast;
lpnSrc->lpnParent = lpnDestParent;
OTUpdateNodeName(psf, lpnSrc);
AdoptKid(lpnDestParent, lpnSrc);
fRet = TRUE;
IUnknown_Release(psf);
}
else
{
ILFree(pidlLast);
}
}
}
}
}
return fRet;
#else
return FALSE;
#endif
}
void _DoChangeNotify(NSC *pns, LONG lEvent, LPITEMIDLIST pidl, LPITEMIDLIST pidlExtra)
{
#if 0
switch(lEvent)
{
case SHCNE_RENAMEFOLDER:
// first try to just swap the nodes if it's true rename (not a move)
if (!TryQuickRename(pidl, pidlExtra))
{
// Rename is special. We need to invalidate both
// the pidl and the pidlExtra. so we call ourselves
_DoHandleChangeNotify(pns, 0, pidlExtra, NULL);
}
break;
case SHCNE_RMDIR:
if (ILIsEmpty(pidl)) {
// we've deleted the desktop dir.
lpNode = s_lpnRoot;
OTInvalidateRoot();
break;
}
// Sitemaps are "inserted" items. We need the ability to remove them.
// Unless the "fInserted" flag is reset, we can not remove them.
if(lpNode = _GetNodeFromIDList(pidl, 0))
{
if(lpNode->fInserted)
lpNode->fInserted = FALSE;
lpNode = OTGetParent(lpNode);
break;
}
case 0:
case SHCNE_MKDIR:
case SHCNE_DRIVEADD:
case SHCNE_DRIVEREMOVED:
if (pidl)
{
LPITEMIDLIST pidlClone = ILClone(pidl);
if (pidlClone)
{
ILRemoveLastID(pidlClone);
lpNode = _GetNodeFromIDList(pidlClone, 0);
ILFree(pidlClone);
}
}
break;
case SHCNE_MEDIAINSERTED:
case SHCNE_MEDIAREMOVED:
lpNode = _GetNodeFromIDList(pidl, 0);
if (lpNode)
lpNode = lpNode->lpnParent;
break;
case SHCNE_DRIVEADDGUI:
case SHCNE_UPDATEITEM:
case SHCNE_NETSHARE:
case SHCNE_NETUNSHARE:
case SHCNE_UPDATEDIR:
lpNode = _GetNodeFromIDList(pidl, 0);
break;
case SHCNE_SERVERDISCONNECT:
// nuke all our kids and mark ourselves invalid
lpNode = _GetNodeFromIDList(pidl, 0);
if (lpNode && NodeHasKids(lpNode))
{
int i;
for (i = GetKidCount(lpNode) -1; i >= 0; i--) {
OTRelease(GetNthKid(lpNode, i));
}
DPA_Destroy(lpNode->hdpaKids);
lpNode->hdpaKids = KIDSUNKNOWN;
OTInvalidateNode(lpNode);
SFCFreeNode(lpNode);
} else {
lpNode = NULL;
}
break;
case SHCNE_ASSOCCHANGED:
break;
case SHCNE_UPDATEIMAGE:
if (pidl) {
InvalidateImageIndices();
DoInvalidateAll(s_lpnRoot, *(int UNALIGNED *)((BYTE*)pidl + 2));
}
break;
}
#endif
}
#if 0
void _OnChangeNotify(NSC *pns, WPARAM wParam, LPARAM lParam)
{
LPITEMIDLIST *ppidl;
LONG lEvent;
LPSHChangeNotificationLock pshcnl;
pshcnl = SHChangeNotification_Lock((HANDLE)wParam, (DWORD)lParam, &ppidl, &lEvent);
if (pshcnl)
{
_DoChangeNotify(pns, lEvent, ppidl[0], ppidl[1]);
SHChangeNotification_Unlock(pshcnl);
}
}
#endif
LRESULT CALLBACK NameSpaceWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
NSC *pns = (NSC *)GetWindowLong(hwnd, 0);
switch (uMsg) {
case WM_NCCREATE:
Assert(pns == NULL);
return _OnNCCreate(hwnd, (LPCREATESTRUCT)lParam);
case WM_NCDESTROY:
if (pns)
_OnNCDestroy(pns);
break;
case WM_SIZE:
if (pns->hwndTree)
MoveWindow(pns->hwndTree, 0, 0, LOWORD(lParam), HIWORD(lParam), TRUE);
break;
case WM_STYLECHANGED:
if (pns && wParam == GWL_STYLE)
pns->style = ((LPSTYLESTRUCT)lParam)->styleNew;
break;
case WM_SETFOCUS:
if (pns && pns->hwndTree)
SetFocus(pns->hwndTree);
break;
case WM_SETFONT:
case WM_GETFONT:
if (pns && pns->hwndTree)
return SendMessage(pns->hwndTree, uMsg, wParam, lParam);
break;
case WM_NOTIFY:
Assert(((NMHDR *)lParam)->idFrom == ID_CONTROL);
switch (((NMHDR *)lParam)->code) {
// we track this through WM_CONTEXTMENU
// case NM_RCLICK:
case NM_RETURN:
case NM_DBLCLK:
return _DoDlbClick(pns);
case TVN_GETDISPINFO:
_OnGetDisplayInfo(pns, (TV_DISPINFO *)lParam);
break;
case TVN_ITEMEXPANDING:
SetCursor(LoadCursor(NULL, IDC_WAIT));
_OnItemExpanding(pns, (LPNM_TREEVIEW)lParam);
break;
case TVN_ITEMEXPANDED:
SetCursor(LoadCursor(NULL, IDC_ARROW));
break;
case TVN_DELETEITEM:
_OnDeleteItem(pns, (LPNM_TREEVIEW)lParam);
break;
case TVN_SELCHANGED:
_OnSelChanged(pns, (LPNM_TREEVIEW)lParam);
break;
case TVN_BEGINLABELEDIT:
return _OnBeginLabelEdit(pns, (TV_DISPINFO *)lParam);
case TVN_ENDLABELEDIT:
return _OnEndLabelEdit(pns, (TV_DISPINFO *)lParam);
case TVN_BEGINDRAG:
case TVN_BEGINRDRAG:
_OnBeginDrag(pns, (NM_TREEVIEW *)lParam);
break;
}
break;
case WM_CONTEXTMENU:
_ContextMenu(pns, (short)LOWORD(lParam), (short)HIWORD(lParam));
break;
case WM_INITMENUPOPUP:
case WM_DRAWITEM:
case WM_MEASUREITEM:
if (pns->pcm)
{
IContextMenu2 *pcm2;
if (SUCCEEDED(QueryInterface(pns->pcm, &IID_IContextMenu2, &pcm2)))
{
pcm2->lpVtbl->HandleMenuMsg(pcm2, uMsg, wParam, lParam);
Release(pcm2);
}
}
break;
case WM_CHANGENOTIFY:
#define ppidl ((LPITEMIDLIST *)wParam)
_DoChangeNotify(pns, (LONG)lParam, ppidl[0], ppidl[1]);
break;
case NSM_SETROOT:
return _OnSetRoot(pns, (NSC_SETROOT *)lParam);
case NSM_GETIDLIST:
if (wParam)
return (LRESULT)_GetFullIDList(pns->hwndTree, (HTREEITEM)lParam);
else
{
TV_ITEM tvi;
// now lets get the information about the item
tvi.mask = TVIF_PARAM | TVIF_HANDLE;
tvi.hItem = (HTREEITEM)lParam;
if (!TreeView_GetItem(pns->hwndTree, &tvi))
return 0;
return (LRESULT)tvi.lParam; // relative PIDL
}
break;
case NSM_DOVERB:
_DoVerb(pns, NULL, (LPCSTR)lParam);
return TRUE;
default:
return DefWindowProc(hwnd, uMsg, wParam, lParam);
}
return 0;
}
const char c_szNameSpaceClass[] = NAME_SPACE_CLASS;
BOOL NameSpace_RegisterClass(HINSTANCE hinst)
{
WNDCLASS wc;
InitCommonControls();
if (!GetClassInfo(hinst, c_szNameSpaceClass, &wc))
{
wc.lpfnWndProc = NameSpaceWndProc;
wc.hCursor = NULL;
wc.hIcon = NULL;
wc.lpszMenuName = NULL;
wc.hInstance = hinst;
wc.lpszClassName = c_szNameSpaceClass;
wc.hbrBackground = NULL;
wc.style = 0;
wc.cbWndExtra = sizeof(NSC *);
wc.cbClsExtra = 0;
return RegisterClass(&wc);
}
return TRUE;
}