以缩略图的形式在Windows资源浏览器中显示自定义格式图片

本文主要是介绍以缩略图的形式在Windows资源浏览器中显示自定义格式图片,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

在公司培训有一段时间了,终于接到了部门任务,需求很简单,就是先Windows Explorer中以缩略图的形式显示自定义的图片,部门提供图片解码代码,我和我的同事实现需求,在实现的过程中走了很多的弯路,犯了不少错误,不过最终我们还是完成了,通过这次实践我相信的能力又提高了很多,本文不阐述开发过程中遇到了什么问题,只是说明如何实现这个需求的步骤。

   开发语言是C++,工具VC6,需要有COM的知识以及ATL的使用,当然COM和ATL我基本不会,都是在这次任务中学习的,如果跟我一样没有学过COM的,建议先看看《COM本质论》至少把什么是COM以及简单实现原理搞清楚,否则就算实现了,意义也不大。

  其实实现非常简单,只需要3个接口就可以搞定,分别是IPersistFile, IExtractImage和IExtractImage2,每个接口的作用可以查MSDN得到,描述的很清楚,这里就不说了,简单说下原理,假设你的图片后缀名为.tp,那么当你在资源浏览器中点到.tp的文件,或者使用缩略图显示图片的时候当前文件夹中有.tp的文件,那么系统会请求IPersistFile接口的Load方法,这里可以得到选择或浏览到的文件名(包括路径),然后请求IExtractImage接口的GetLocation方法,最后请求IExtractImage的Extract方法,你只需要返回你图片的HBITMAP就可以了,大小都不需要你自己改变,系统会帮你缩小在显示在资源浏览器上。

  下边是详细的实现步骤:

  1。打开VC6建立一个ATL工程,名字为TPThumbnail,OK and Finish。

  2。选择Insert->New ATL Object。然后选择Simple Object,Next,ShortName写TPeExtract其他的也就会自动填写上了,点OK。

  这里牢骚2句,IExtractImage和IExtractImage2接口在MSDN中说只在VISTA中提供,所以你要想实现这2个接口,必须从VISTA的SDK中把这2个接口的声明弄出来,这点不明白MS怎么想的,为什么不在XP中直接提供。

  3。此时TPeExtract.h是实现IDispatchImpl接口的,这个对我们没什么用处,所以我们把他去掉(CComObjectRootEx和CComCoClass接口不要删除), COM_INTERFACE_ENTRY(IDispatch)和COM_INTERFACE_ENTRY(ITPeExtract)这个也要删除,然后#include "IExtractImage.h" 头文件,并把这2个实现加上去,然后实现这3个接口的方法就可以了。最后的TPeExtract.h如下:

 

//  TPeExtract.h : Declaration of the CTPeExtract

#ifndef __TPEEXTRACT_H_
#define  __TPEEXTRACT_H_

#include 
" resource.h "         //  main symbols
#include  " IExtractImage.h "

/////
//  CTPeExtract
class  ATL_NO_VTABLE CTPeExtract : 
    
public  CComObjectRootEx < CComSingleThreadModel > ,
    
public  CComCoClass < CTPeExtract,  & CLSID_TPeExtract > ,
    
public  IPersistFile,
    
public  IExtractImage2
{
public:
    CTPeExtract();


DECLARE_REGISTRY_RESOURCEID(IDR_TPEEXTRACT)

DECLARE_PROTECT_FINAL_CONSTRUCT()

BEGIN_COM_MAP(CTPeExtract)
    COM_INTERFACE_ENTRY(IPersistFile)
    COM_INTERFACE_ENTRY(IExtractImage)
    COM_INTERFACE_ENTRY(IExtractImage2)
END_COM_MAP()

// IExtractImage
public:
    
    STDMETHOD(GetLocation)(LPWSTR pszPathBuffer,
        DWORD cchMax,
        DWORD 
*pdwPriority,
        
const SIZE *prgSize,
        DWORD dwRecClrDepth,
        DWORD 
*pdwFlags);
    STDMETHOD(Extract)(HBITMAP
*);
    
// IExtractImage2
    STDMETHOD(GetDateStamp)(FILETIME *pDateStamp);
    
    
// IPersistFile
    STDMETHOD(Load)(LPCOLESTR wszFile, DWORD dwMode);
    
    STDMETHOD(GetClassID)(LPCLSID clsid)
    
return E_NOTIMPL;    }
    
    STDMETHOD(IsDirty)(VOID)
    
return E_NOTIMPL; }
    
    STDMETHOD(Save)(LPCOLESTR, BOOL)
    
return E_NOTIMPL; }
    
    STDMETHOD(SaveCompleted)(LPCOLESTR)
    
return E_NOTIMPL; }
    
    STDMETHOD(GetCurFile)(LPOLESTR FAR
*)
    
return E_NOTIMPL; }
}
;

#endif   // __TPEEXTRACT_H_

  当然现在是编译不通过的,因为有些方法没实现,其中象Save, GetCurFile等等在本例中没有用到。所以直接返回E_NOTIMPL就可以了。

  4。然后增加1个成员变量TCHAR m_szFileName[MAX_PATH]; 这个是保存文件名用的。

  5。在构造函数中初始化szFileName。

CTPeExtract::CTPeExtract()
{
    ZeroMemory(m_szFileName, MAX_PATH);
}

  6。现在实现Load方法,这个方法只保存传入的文件名,代码如下:

HRESULT CTPeExtract::Load(LPCOLESTR wszFile, DWORD dwMode)
{
    USES_CONVERSION;
    _tcscpy(m_szFileName, OLE2T((WCHAR
*)wszFile)); 

    
return S_OK;    
}
;

   7。实现GetLocation。

HRESULT CTPeExtract::GetLocation(LPWSTR pszPathBuffer,
                                        DWORD cchMax, DWORD 
* pdwPriority,
                                        
const  SIZE  * prgSize, DWORD dwRecClrDepth,
                                        DWORD 
* pdwFlags)
{
    
if (*pdwFlags & IEIFLAG_ASYNC)    
        
return E_PENDING; 

    
return NOERROR;
}

  有人会问为什么return E_PENDING;这个你看MSDN就知道,让是XP or later的系统,就必须这样做。

  8。重要的函数Extract的实现,这个函数是系统问你要你这个图片的HBITMAP,给他就是了,这里就写你图片的解码,由于是例子所以我直接加载个资源就可以了。

HRESULT CTPeExtract::Extract(HBITMAP *  phBmpThumbnail)

    HBITMAP hBM 
= 0;
    hBM 
= ::LoadBitmap(_Module.GetModuleInstance(),MAKEINTRESOURCE(IDB_BITMAP1)); 
    
if (0 != hBM)
        
*phBmpThumbnail = hBM;
    
return NOERROR; 
}

  9。最后是GetDateStamp函数的实现,这个是IExtractImage2的函数,具体作用看MSND。

HRESULT CTPeExtract::GetDateStamp(FILETIME  * pDateStamp)
{
        
    FILETIME ftCreationTime,ftLastAccessTime,ftLastWriteTime;

    HANDLE hFile 
= CreateFile(m_szFileName,GENERIC_READ,FILE_SHARE_READ,NULL,
        OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
    
if(!hFile)
        
return E_FAIL;
    GetFileTime(hFile,
&ftCreationTime,&ftLastAccessTime,&ftLastWriteTime);
    CloseHandle(hFile);
    
*pDateStamp = ftLastWriteTime;
    
return NOERROR; 
}

这个是编译就可以通过了。  TPThumbnail.dll - 0 error(s), 0 warning(s)

但是,光这样还不够,还需要最重要的一步,就是这个组件注册到合适的地方。这样修改Resource Files中的TPeExtract.rgs文件。修改后如下:

HKCR
{
    TPThumbnail.TPeExtract
.1   =  s 'TPeExtract Class'
    {
        CLSID 
=  s '{E45A903D-53AE-4A14-906E-0260D13FF77F}'
    }
    TPThumbnail.TPeExtract 
=  s 'TPeExtract Class'
    {
        CLSID 
=  s '{E45A903D-53AE-4A14-906E-0260D13FF77F}'
        CurVer 
=  s 'TPThumbnail.TPeExtract .1 '
    }
    NoRemove CLSID
    {
        ForceRemove {E45A903D-53AE-4A14-906E-0260D13FF77F} 
=  s 'TPeExtract Class'
        {
            ProgID 
=  s 'TPThumbnail.TPeExtract .1 '
            VersionIndependentProgID 
=  s 'TPThumbnail.TPeExtract'
            ForceRemove 'Programmable'
            InprocServer32 
=  s '%MODULE%'
            {
                val ThreadingModel 
=  s 'Apartment'
            }
            'TypeLib' 
=  s '{F473925C-FDEE-42AD-9FC2-C843DB7BA2B2}'
        }
    }

    NoRemove .tp
    {
        shellex
        {
            {BB2E617C-
0920 -11d1-9A0B-00C04FC2D6C1}  =  s '{E45A903D-53AE-4A14-906E-0260D13FF77F}'
        }
    }

    NoRemove 
.1
    {
        shellex
        {
            {BB2E617C-
0920 -11d1-9A0B-00C04FC2D6C1}  =  s '{E45A903D-53AE-4A14-906E-0260D13FF77F}'
        }
    }
}

  这里的 .tp 就是关联的文件后缀名,BB2E617C-0920-11d1-9A0B-00C04FC2D6C1是IUnknow接口的CLSID,E45A903D-53AE-4A14-906E-0260D13FF77F是本例的CLSID,如果你在开发中需要修改这个CLSID就可以了。

  编译, PASS~~~

  现在需要注册这个COM组件,方法很简单。开始-运行-cmd。

  注册: regsvr32 编译出来的DLL

  反注册,就是移除: regsvr32 /u 编译出来的DLL

  注册和反注册都会有提示的,这个时候你在任意文件夹下建一个后缀名为.tp的文件就可以看到效果了,再切换到缩略图看看。

  最后再次感谢帮助我的人,还有和我一起开发的同事,他在整个开发中给我提出了很多中肯的意见以及帮助。

  (本代码只是例子,并没有做详尽的测试)

因源码链接失败,在此不做转载!

文章转载至:https://blog.csdn.net/jlu3389/article/details/2207005

这篇关于以缩略图的形式在Windows资源浏览器中显示自定义格式图片的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1067913

相关文章

使用python生成固定格式序号的方法详解

《使用python生成固定格式序号的方法详解》这篇文章主要为大家详细介绍了如何使用python生成固定格式序号,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录生成结果验证完整生成代码扩展说明1. 保存到文本文件2. 转换为jsON格式3. 处理特殊序号格式(如带圈数字)4

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配

基于Python开发Windows自动更新控制工具

《基于Python开发Windows自动更新控制工具》在当今数字化时代,操作系统更新已成为计算机维护的重要组成部分,本文介绍一款基于Python和PyQt5的Windows自动更新控制工具,有需要的可... 目录设计原理与技术实现系统架构概述数学建模工具界面完整代码实现技术深度分析多层级控制理论服务层控制注

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型

C#实现SHP文件读取与地图显示的完整教程

《C#实现SHP文件读取与地图显示的完整教程》在地理信息系统(GIS)开发中,SHP文件是一种常见的矢量数据格式,本文将详细介绍如何使用C#读取SHP文件并实现地图显示功能,包括坐标转换、图形渲染、平... 目录概述功能特点核心代码解析1. 文件读取与初始化2. 坐标转换3. 图形绘制4. 地图交互功能缩放

Oracle数据库在windows系统上重启步骤

《Oracle数据库在windows系统上重启步骤》有时候在服务中重启了oracle之后,数据库并不能正常访问,下面:本文主要介绍Oracle数据库在windows系统上重启的相关资料,文中通过代... oracle数据库在Windows上重启的方法我这里是使用oracle自带的sqlplus工具实现的方

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

SpringBoot AspectJ切面配合自定义注解实现权限校验的示例详解

《SpringBootAspectJ切面配合自定义注解实现权限校验的示例详解》本文章介绍了如何通过创建自定义的权限校验注解,配合AspectJ切面拦截注解实现权限校验,本文结合实例代码给大家介绍的非... 目录1. 创建权限校验注解2. 创建ASPectJ切面拦截注解校验权限3. 用法示例A. 参考文章本文