赞助

Windows Shell编程之如何编写为文件对象弹出信息框的Shell扩展

有关COM编程资料

转载:http://www.cnblogs.com/lzjsky/archive/2010/11/22/1884702.html

活动桌面引入一项新特性, 当你在某些特定对象上旋停鼠标时,工具提示将显示它们的描述

例如对TXT格式文件:

系统默认的:                                                            扩展之后的:

                               

 

第一步:新建一个ATL项目,输入工程名:TxtInfo,具体如下图:

 

第二步:点击next

 

第三步:应用类型选择动态库链接(DLL),同时勾选支持MFC,最后点击Finish。

第四步:新建一个ATL简单对象(英文版的VS为ATLSimple Object)

 

单击 Add,在第二页面中, 在Short Name编辑框中输入TxtInofShlExt,点击 Finish.

 

第五步:开始我们需要添加IPersistFileCTxtInfoShlExt实现的接口列表中.打开 TxtInfoShlExt.h, 并添加如下代码:

 

 1 #include <comdef.h>
 2 #include <shlobj.h>
 3 class ATL_NO_VTABLE CTxtInfoShlExt : 
 4 public CComObjectRootEx, 
 5 public CComCoClass,
 6 public IDispatchImpl,
 7 public IPersistFile
 8 { 
 9 BEGIN_COM_MAP(CTxtInfoShlExt) 
10 COM_INTERFACE_ENTRY(ITxtInfoShlExt)
11 COM_INTERFACE_ENTRY(IDispatch)
12 COM_INTERFACE_ENTRY(IPersistFile)
13 END_COM_MAP()

我们需要一个保存浏览器给出的文件名的变量:
protected:
// ITxtInfoShlExt

CString m_sFilename;

注意我们可以在任何地方使用 MFC 对象.如果你看一下 IpersistFile 的文档, 你会看到很多方法.
幸运的是, 对于Shell扩展, 我们只用实现Load(), 而忽略其它方法. 以下是 IPersistFile 方法的原型: 

public: 
// IPersistFile 
STDMETHOD(GetClassID)(LPCLSID) { return E_NOTIMPL; }
STDMETHOD(IsDirty)() { return E_NOTIMPL; } 
STDMETHOD(Load)(LPCOLESTR, DWORD); 
STDMETHOD(Save)(LPCOLESTR, BOOL) { return E_NOTIMPL; } STDMETHOD(SaveCompleted)(LPCOLESTR) { return E_NOTIMPL; } STDMETHOD(GetCurFile)(LPOLESTR*) { return E_NOTIMPL; } 

除开 Load() 外的方法都只返回 E_NOTIMPL 以表明我们没有实现它们.
更妙的是, Load() 方法也相当简单. 我们只需保存浏览器传给我们的文件名. 也就是当前鼠标在其上盘旋的文件.

HRESULT CTxtInfoShlExt::Load ( LPCOLESTR wszFilename, DWORD dwMode )
{ 
AFX_MANAGE_STATE(AfxGetStaticModuleState()); // init MFC 
// 让CString 自动转化文件名为 ANSI 字符.
m_sFilename = wszFilename; 
return S_OK;
} 

请注意函数的第一行. 要让MFC正确地工作该行代码是必要的.
由于我们的 DLL 要被非MFC程序所调用, 任一个使用MFC的输出函数必须手动初始化 MFC.
如果你不写这行代码, 则许多MFC函数 (大多是与资源处理有关的函数) 将失败或出错.

文件名被保存在 m_sFilename 以备后用.
注意我利用了 CString 的赋值操作符的特性来转化字符串为ANSI格式 - 如果该 DLL以ANSI方式建立.

创建工具提示的文本
在浏览器调用了我们的 Load() 方法之后, 它接着调用 QueryInterface() 获取另一个接口: IQueryInfo. IQueryInfo 是个相当简单的接口,只有两个接口 (而其中也只有一个被真正使用). 打开 TxtInfoShlExt.h ,添加如下标红的代码:

class ATL_NO_VTABLE CTxtInfoShlExt : 
public CComObjectRootEx, 
public CComCoClass,
public IDispatchImpl, 
public IPersistFile,
public IQueryInfo
{
BEGIN_COM_MAP(CTxtInfoShlExt)
COM_INTERFACE_ENTRY(ITxtInfoShlExt)
COM_INTERFACE_ENTRY(IDispatch)
COM_INTERFACE_ENTRY(IPersistFile) 
COM_INTERFACE_ENTRY(IQueryInfo)
END_COM_MAP()

然后添加 IQueryInfo 方法的实现:

// IQueryInfo 
STDMETHOD(GetInfoFlags)(DWORD*) { return E_NOTIMPL; } 
STDMETHOD(GetInfoTip)(DWORD, LPWSTR*); 

GetInfoFlags() 方法当前并不使用, 所以我们只返回 E_NOTIMPL.
GetInfoTip() 让我们返回工具提示文本 字符串. 首先是开头繁琐的代码:

 

HRESULT CTxtInfoShlExt::GetInfoTip ( DWORD dwFlags, LPWSTR* ppwszTip )
{
AFX_MANAGE_STATE(AfxGetStaticModuleState()); // init MFC

LPMALLOC pMalloc;
CStdioFile file;
DWORD dwFileSize;
CString sFirstLine;
BOOL bReadLine;
CString sTooltip;
USES_CONVERSION;

接着,AFX_MANAGE_STATE首先被调用以初始化 MFC.
这是每个函数都该做的第一件事, 甚至应该在变量声明之前,因为MFC构造函数可能调用其它的 MFC 函数.

dwFlags 当前并不被使用. ppwszTip 是个 LPWSTR (Unicode 字符串指针) 变量的指针,我们要将其赋值为我们所分配的字符
串缓冲区的指针.(指向指针的指针)

首先, 我们试着打开文件读取. 由于我们在Load()中保存了文件名,现在就可以使用了.
if ( !file.Open ( m_sFilename , CFile::modeRead | CFile::shareDenyWrite )) return E_FAIL;

现在, 我们需要使用Shell的内存分配器分配一个缓冲, 我们通过SHGetMalloc()函数获取一个IMalloc接口:
if ( FAILED( SHGetMalloc ( &pMalloc ))) return E_FAIL;

关于Imalloc 稍后我有更多的要说. 下一步是取得文件大小并读取第一行:
// 取得文件大小.
dwFileSize = file.GetLength();

// 读取第一行.
bReadLine = file.ReadString ( sFirstLine );
bReadLine总是为真, 除非文件不可获取或长度为0.下一步是创建工具提示的第一部分:文件大小.
sTooltip.Format ( _T("File size: %lu"), dwFileSize );

现在, 我们读取第一行并添加到工具提示中.
if ( bReadLine )
{
sTooltip += _T("\n"); sTooltip += sFirstLine;
}

现在我们完成了工具提示, 我们要分配一个缓冲.在这我们将使用 Imalloc 接口.
由 SHGetMalloc() 返回的指针是一个Shell的Imalloc接口指针的拷贝.
我们用这个接口分配的任何内存都位于Shell的进程空间内, 所以Shell可以使用它.
更重要的是, Shell可以释放它. 所以我们所作的就是分配缓冲区,然后忘掉它.
Shell将在完成操作时释放该内存.

要认识到的一件事是我们返回给Shell的字符串必须是 Unicode 格式的.
这就是为什么下面的Alloc()中的计算要乘以 sizeof(wchar_t);
只分配lstrlen(sToolTip)长的内存只够一半所需的内存.

*ppwszTip=(LPWSTR)pMalloc->Alloc ((1 + lstrlen(sTooltip))*sizeof(wchar_t));
if (NULL == *ppwszTip)
{
pMalloc->Release();
return E_OUTOFMEMORY;
}
// 使用 Unicode 字符串拷贝函数将工具提示文本拷入缓冲区.
wcscpy ( *ppwszTip, T2COLE((LPCTSTR) sTooltip) );
//最后我们释放先前获取得 IMalloc 接口.
pMalloc->Release();
return S_OK;
}

 

用命令行注册COM :regsvr32 dll的绝对路径

注:必须以管理员运行cmd

 卸载COM: regsvr32 /u dll的绝对路径

 

如果你的操作系统是64位的 编译的工程必须x64

如果你的操作系统是32位的 编译工程是Win32

第六步:编辑TxtInfoShlExt.rgs文件

          方式一:

                  

 

方式二:

 源码:为文件对象弹出信息框的Shell扩展

 参考源码:实例程序

参考资料:Windows Shell扩展编程完全指南.rar

posted @ 2017-02-22 11:07  车臣  阅读(1822)  评论(0编辑  收藏  举报