|
由于工作的需求要实现像winrara那样的shell扩展菜单,带图标的,针对某个类型的文件执行相应的操作, 查了不少资料,终于解决了,效果如下图中红色矩形部分。
最先想到的当然是最方便的,直接添加注册表,只需在HCLR中需要处理的类型(即右键选中的文件类型)的注册表项中的shell子键中加入新项, 修改默认值为显示名称,显示效果如上图的<用“记事本”打开>菜单项。(*)代表任意类型的文件。
然后再新项中加入command子键,默认值设置为运行程序的参数,如"notepad.exe %1”,其中%1代表右键选中的文件。
这样当点击菜单后就可以执行你指定的程序("notepad")并传入选中的文件名作为命令参数,这样就可以处理这个文件了。
其实这样足以满足需求了,但效果上还差了一点,很明显少了个图标。既然winrar都能实现,肯定是还有其它实现的方法了。
于是就是处理图标了,然而最终没能找到好的方法,但如果是win7的话,还是有的,那就是在command中加入icon项,值为
你的目标程序,然而如果是xp就行不通了。
本想简单的解决,但没想到还是得用上复杂的方法,仅仅为了一个图标 - -;。
复杂的方法只能是通过扩展shell接口来实现,首先肯定得牵涉到COM,所以关于COM可以上网查下资料,再这里就不多说(其实
让我讲我也不一定讲得清楚),这里我就写下得注意的地方就是了。
1.创建工程这里我用的VS2005,创建一个ATL 项目,属性不需要改,点默认的就可以了。
2.右击项目,添加,类,添加一个新ATL简单对象。
3.编辑代码,关于IShellExtInit和IContextMenu接口可以查看MSDN,上面写的很详细。
// CCContextMenuExt
class ATL_NO_VTABLE CCContextMenuExt : public CComObjectRootEx<CComSingleThreadModel>, public CComCoClass<CCContextMenuExt, &CLSID_CContextMenuExt>, public IDispatchImpl<ICContextMenuExt, &IID_ICContextMenuExt, &LIBID_CtxMenuExtLib, /**//*wMajor =*/ 1, /**//*wMinor =*/ 0>, public IShellExtInit, public IContextMenu { public: CCContextMenuExt() { }
DECLARE_REGISTRY_RESOURCEID(IDR_CCONTEXTMENUEXT)
BEGIN_COM_MAP(CCContextMenuExt) COM_INTERFACE_ENTRY(ICContextMenuExt) COM_INTERFACE_ENTRY(IDispatch) COM_INTERFACE_ENTRY(IShellExtInit) COM_INTERFACE_ENTRY(IContextMenu) END_COM_MAP()
DECLARE_PROTECT_FINAL_CONSTRUCT()
HRESULT FinalConstruct() { m_hBitmap = LoadBitmap(_hInstance, MAKEINTRESOURCE(IDB_MENU)); return S_OK; }
void FinalRelease() { if (m_hBitmap) { DeleteObject(m_hBitmap); } }
public: enum { IDM_CTXMENU = 0, };
public:
HRESULT STDMETHODCALLTYPE Initialize( /**//* [in] */ LPCITEMIDLIST pidlFolder, /**//* [in] */ IDataObject *pdtobj, /**//* [in] */ HKEY hkeyProgID) { HRESULT hr; UINT nFileCount; UINT nLen;
FORMATETC fmt = { CF_HDROP, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL };
STGMEDIUM sm = { TYMED_HGLOBAL };
hr = pdtobj->GetData(&fmt, &sm); if (FAILED(hr)) { return hr; }
nFileCount = DragQueryFile((HDROP)sm.hGlobal, 0xFFFFFFFF, NULL, 0);
if (nFileCount >= 1) { nLen = DragQueryFile((HDROP)sm.hGlobal, 0, m_pszFileName, sizeof(m_pszFileName)); if (nLen >0 && nLen <MAX_PATH) { m_pszFileName[nLen] = _T('\0'); hr = S_OK; } else { hr = E_INVALIDARG; } } else { hr = E_INVALIDARG; }
ReleaseStgMedium(&sm);
return hr; }
STDMETHOD(QueryContextMenu)(THIS_ HMENU hmenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags) { MENUITEMINFO mii;
if (uFlags & CMF_DEFAULTONLY) { return MAKE_HRESULT(SEVERITY_SUCCESS, FACILITY_NULL, 0); }
memset((void*)&mii, 0, sizeof(mii)); mii.cbSize = sizeof(mii); mii.fMask = MIIM_STRING | MIIM_CHECKMARKS | MIIM_ID | MIIM_STATE; mii.cch = lstrlen(SZ_MENUTEXT); mii.dwTypeData = SZ_MENUTEXT;
/**//* 这里用hbmpChecked而不用hbmpItem的原因 - -自己动手试试就知道了。 */ mii.hbmpItem mii.hbmpChecked = m_hBitmap; mii.hbmpUnchecked = m_hBitmap; mii.fState = MFS_ENABLED; mii.wID = idCmdFirst + indexMenu;
if (!InsertMenuItem(hmenu, indexMenu, TRUE, &mii)) { return E_FAIL; }
lstrcpynA(m_pszVerb, "protected_run", 32); lstrcpynW(m_pwszVerb, L"protected_run", 32);
return MAKE_HRESULT(SEVERITY_SUCCESS, FACILITY_NULL, IDM_CTXMENU + 1); }
STDMETHOD(InvokeCommand)(THIS_ LPCMINVOKECOMMANDINFO lpici) { BOOL fEx = FALSE; BOOL fUnicode = FALSE;
if(lpici->cbSize == sizeof(CMINVOKECOMMANDINFOEX)) { fEx = TRUE; if((lpici->fMask & CMIC_MASK_UNICODE)) { fUnicode = TRUE; } }
if( !fUnicode && HIWORD(lpici->lpVerb)) { if(StrCmpIA(lpici->lpVerb, m_pszVerb)) { return E_FAIL; } }
else if( fUnicode && HIWORD(((CMINVOKECOMMANDINFOEX *) lpici)->lpVerbW)) { if(StrCmpIW(((CMINVOKECOMMANDINFOEX *)lpici)->lpVerbW, m_pwszVerb)) { return E_FAIL; } }
else if(LOWORD(lpici->lpVerb) != IDM_CTXMENU) { return E_FAIL; }
else { //在此处理点击事件. MessageBox(NULL, m_pszFileName, _T(""), MB_OK);
return S_OK; }
return E_FAIL;
}
STDMETHOD(GetCommandString)(THIS_ UINT_PTR idCmd, UINT uType, UINT * pwReserved, LPSTR pszName, UINT cchMax) { HRESULT hr = E_INVALIDARG; static CHAR szHelpTextA[] = "windows扩展菜单!"; static WCHAR szHelpTextW[] = L"windows扩展菜单!";
if(idCmd != IDM_CTXMENU) { return hr; }
switch(uType) { case GCS_HELPTEXTA: lstrcpynA((CHAR*)pszName, szHelpTextA, cchMax); break;
case GCS_HELPTEXTW: lstrcpynW((WCHAR*)pszName, szHelpTextW, cchMax);; break;
case GCS_VERBA: lstrcpynA((CHAR*)pszName, m_pszVerb, cchMax); break;
case GCS_VERBW: lstrcpynW((WCHAR*)pszName, m_pwszVerb, cchMax); break;
default: hr = S_OK; break; } return hr; }
private: TCHAR m_pszFileName[MAX_PATH]; HBITMAP m_hBitmap; CHAR m_pszVerb[32]; WCHAR m_pwszVerb[32];
};
4.修改服务注册、取消注册函数,这里只需在需要处理的文件类型的shllex下的ContextMenuHandlers下创建项,并设置接口ID。
// DllRegisterServer - 将项添加到系统注册表 STDAPI DllRegisterServer(void) { // 注册对象、类型库和类型库中的所有接口 HRESULT hr; HKEY hKey;
static char pszGUID[] = "{C2397F2E-4BA3-4B9D-858A-F775761C023B}";
hr = _AtlModule.DllRegisterServer(); if (FAILED(hr)) { return hr; }
if (RegCreateKeyA(HKEY_CLASSES_ROOT, "*\\shellex\\ContextMenuHandlers\\CtxMenu", &hKey) != ERROR_SUCCESS) { return E_FAIL; }
if (RegSetValueA(hKey, NULL, REG_SZ, pszGUID, (DWORD)strlen(pszGUID)) != ERROR_SUCCESS) { RegCloseKey(hKey); return E_FAIL; }
return hr; }
// DllUnregisterServer - 将项从系统注册表中移除 STDAPI DllUnregisterServer(void) { RegDeleteKeyA(HKEY_CLASSES_ROOT, "*\\shellex\\ContextMenuHandlers\\CtxMenu");
return _AtlModule.DllUnregisterServer(); }
5.编译运行,VS会自动替你注册,当然也可以用regsvr32 自己注册。
下载地址:http://www.cppblog.com/Files/shly/CtxMenuExt.rar
注意:注册后的DLL无法删除,有种方法可以就是regsvr32 /u 取消注册dll, 然后重启explorer.exe。
|