在应用程序中集成外壳的上下文菜单(下)

类别:VC语言 点击:0 评论:0 推荐:

(二)  响应鼠标单击消息,获取IContextMenu接口并弹出菜单;

通过C盘的IShellFolder接口的GetUIObjectOf方法我们可以得到该节点的一个或多个指定子节点的IContextMenu接口,该方法原型如下:

HRESULT GetUIObjectOf(

    HWND hwndOwner,

    UINT cidl,     //指定apidl指向的数组中包函的pidl的个数

    LPCITEMIDLIST *apidl,//指向cidl个PIDL,需要注意的是,这些pidl都必须是相对的

    REFIID riid,   //我们要的是上下文菜单接口,这里指定为IID_IContextMenu_

    UINT *rgfReserved,

    VOID **ppv

   );

HRESULT        hr;

ContextMenu   *pcm = NULL;

hr = m_psfFolderC->GetUIObjectOf(GetSafeHwnd(),aryListBoxSel.GetSize(),

(LPCITEMIDLIST*)pPidls, IID_IContextMenu, NULL, (void**)&pcm);

 

得到IContextMenu后我们得提供一个弹出式菜单的句柄,并把他传给IContextMenu::QueryContextMenu,

如果该方法执行成功的话,会在我们的菜单里加入相应的菜单项。

CMenu Menu;

Menu.CreatePopupMenu();

Hr = pcm->QueryContextMenu(Menu.m_hMenu, 0, 1, 0x7fff, CMF_NORMAL | CMF_EXPLORE);

有了菜单项,我们就可以弹出该菜单了,我们用TPM_RETURNCMD标志指定TrackPopupMenu必须返回用户所选菜单项的ID,以便稍后通过IContextMenu::InvokeCommand来调用真正的Shell动作:

idCmd = Menu.TrackPopupMenu(TPM_LEFTALIGN | TPM_RETURNCMD | TPM_RIGHTBUTTON,

                                                                                                pt.x,

                                                                                                pt.y,

                                                                                                AfxGetMainWnd());

if(idCmd)

{

         CMINVOKECOMMANDINFO  cmi;

         cmi.cbSize = sizeof(CMINVOKECOMMANDINFO);

         cmi.fMask = 0;

         cmi.hwnd = GetSafeHwnd();

         cmi.lpVerb = (LPCSTR)(INT_PTR)(idCmd - 1);

         cmi.lpParameters = NULL;

         cmi.lpDirectory = NULL;

         cmi.nShow = SW_SHOWNORMAL;

         cmi.dwHotKey = 0;

         cmi.hIcon = NULL;

         hr = pcm->InvokeCommand(&cmi);

}

//注:事实上还有两个跟上下文菜单相关的接口,分别为IContextMenu2和IContextMenu3,他们是用来实现扩展菜单项的自画行为的,这里为简单起见,并没有调用他们,在通常使用中我们必须有查询这两个接口并在我们的窗口收到WM_DRAWITEM/ WM_MENUCHAR/ WM_MEASUREITEM/ WM_INITMENUPOPUP等菜单相关的消息时调用这两个接口的相关方法,否则那些自画菜单项和子级菜单项都不会正常显示。具体请看本文附带源码。

 

(三)  不弹出菜单直接调用菜单项相应的命令?

大家还记得怎么显示一个文件或文件夹的属性对话框吗?

Yes,用ShellExecuteEx并指定SHELLEXECUTEINFO的lpVerb域为properties就可,但是这种方法只能查看一个文件的属性,怎么同时查看多个的?

要知道ShellExecuteEx查看文件属性最终也是靠IContextMenu帮忙的,所以答案还是在IContextMenu上,我们只要在调用GetUIObjectOf时把想查看的文件或文件件的PIDL做为参数传进去,然后直接调用InvokeCommand方法就OK啦。

if (m_psfFolderC != NULL)

{

         HRESULT        hr;

         IContextMenu   *pcm = NULL;

         hr = m_psfFolderC->GetUIObjectOf(GetSafeHwnd(),aryListBoxSel.GetSize(), (LPCITEMIDLIST*)pPidls,IID_IContextMenu, NULL, (void**)&pcm);

         if(SUCCEEDED(hr) && pcm != NULL)

         {

                   CMINVOKECOMMANDINFO  cmi;

                   cmi.cbSize = sizeof(CMINVOKECOMMANDINFO);

                   cmi.fMask = 0;

                   cmi.hwnd = GetSafeHwnd();

                   cmi.lpVerb = "properties";//在弹出菜单的时候,这里是用户选择的cmdid,但这时候我们没弹出菜单,所以只能以verb来调用了,关于verb的更多信息请参考MSDN

                   cmi.lpParameters = NULL;

                   cmi.lpDirectory = NULL;

                   cmi.nShow = SW_SHOWNORMAL;

                   cmi.dwHotKey = 0;

                   cmi.hIcon = NULL;

                   hr = pcm->InvokeCommand(&cmi);

                   pcm->Release();

         }

}

 

 

后记和参考

         其实现在网上介绍shell编程的文章不少(当然,大部分是英文的,国内能找到的多是托盘编程L),在www.codeproject.com上就有不少好文章,但由于他们侧重于reusable上,用C++把乱七八糟的都封装了,反而使我们没法很容易的学习外壳扩展和调用的机制,本文以最直白的方法向你展示了怎么在你的程序中调用Shell。

         当然,小弟水平有限,文中有错误难免,非常希望得到您的指正。

 

       本文演示源码下载: http://iunknown.com.cn/csdn/kShellContextMenu.rar

 

         参考:

 Windows外壳名字空间的浏览 by 姜伟华
     (少数的介绍shell的中文文章之一,推荐)

Use Shell ContextMenu in your applications
     (该文作者提供了一个类,在你的程序中调用shell ContextMenu只需两三行代码,很方便J)

Platform sample: EnumDesk

MSDN Library->Platform SDK Document->User Interface Services->Windows Shell

本文地址:http://com.8s8s.com/it/it932.htm