執行
許多人都知道 ShellExecute ,用于執行一個外部指令。但對于 IShellFoloder 對象來說,它的執行指令,都在它的 ContextMenu 裡面了。記得前幾節說過如何直接調用 ContextMenu 裡的項,是以,執行一個 IShellFoloder,也無非是調用它的 ContextMenu 裡預設的項而已:
//存放 PIDL 的數組
IntPtr[] pidls = new IntPtr[1];
pidls[0] = pidl;
//得到 IContextMenu 接口
IntPtr iContextMenuPtr = IntPtr.Zero;
iContextMenuPtr = IParent.GetUIObjectOf(IntPtr.Zero, (uint)pidls.Length,
pidls, ref Guids.IID_IContextMenu, out iContextMenuPtr);
IContextMenu iContextMenu = (IContextMenu)Marshal.GetObjectForIUnknown(iContextMenuPtr);
//提供一個彈出式菜單的句柄
IntPtr contextMenu = API.CreatePopupMenu();
iContextMenu.QueryContextMenu(contextMenu, 0,
API.CMD_FIRST, API.CMD_LAST, CMF.NORMAL | CMF.EXPLORE);
//擷取預設的指令項
int defaultCommand = API.GetMenuDefaultItem(contextMenu, false, 0);
CMINVOKECOMMANDINFOEX invoke = new CMINVOKECOMMANDINFOEX();
invoke.cbSize = Marshal.SizeOf(typeof(CMINVOKECOMMANDINFOEX));
invoke.lpVerb = (IntPtr)(defaultCommand - API.CMD_FIRST);
invoke.lpDirectory = string.Empty;
invoke.fMask = 0;
invoke.ptInvoke = new POINT(MousePosition.X, MousePosition.Y);
invoke.nShow = 1;
iContextMenu.InvokeCommand(ref invoke);
GetMenuDefaultItem 的原型:
[DllImport("user32", SetLastError = true, CharSet = CharSet.Auto)]
public static extern int GetMenuDefaultItem(IntPtr hMenu, bool fByPos, uint gmdiFlags);
一個 IShellFolder 的預設菜單一般都是“打開”,但有些卻不是。是以 lpVerb 不應該直接使用 "open"。
資料總管
經過把前幾節中的例子修改,大緻得到一個資料總管的原型,但它還有很多問題:
1,不會釋放資源
2,無法顯示快捷方式、共享等圖示标志
3,ContextMenu 某些地方沒有處理,例如發送到...
4,拖拉沒有實作
5,沒有實時監控更改
是以,要做一個完整的資料總管,是非常麻煩的事情,你可以參考 C# FileBrowser ,它已經做得非常好了。
以後會講述一些在資料總管實作 Shell 操作的内容,希望大家多多支援^_^。