zoukankan      html  css  js  c++  java
  • 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(" "); 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

  • 相关阅读:
    [报错]编译报错:clang: error: linker command failed with exit code 1及duplicate symbol xxxx in错误解决方法之一
    修改UISearBar的文字颜色,placehoder颜色及输入框颜色
    designated initializer和secondary initializer是什么?
    设置UIButton的文字居右显示 去掉点击默认置灰效果
    设置UITextField的placeholder的颜色
    Xcode8和iOS10的适配问题
    [转载]做一个 App 前需要考虑的几件事
    git 常用命令行整理
    Xcode中使用GitHub详解
    截取字符串
  • 原文地址:https://www.cnblogs.com/chechen/p/6427958.html
Copyright © 2011-2022 走看看