|
由于工作的需求要实现像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。
|