在VC下显示JPEG、GIF格式图像的一种简便方法

在VC下显示JPEG、GIF格式图像的一种简便方法

一、 引言 
  
JPEG图像压缩标准随然是一种有损图像压缩标准,但由于人眼视觉的不敏感,经压缩后的画质基本没有发生变化,很快便以较高的压缩率得到了广泛的认可。GIF格式虽然仅支持256色但它对于颜色较少的图像有着很高的压缩率,甚至超过JPEG标准,也得到了广泛的认同。但作为众多程序员的一个重要的开发工具--Microsoft
Visual C++
6.0的MFC库却仅对没有经过任何压缩的BMP位图文件有着良好的支持,可以读取、显示、存储甚至在内存中创建一块内存位图。由于BMP格式的图像没有经过任何的压缩,不论是作为程序的外部文件,还是作为程序的内部资源都要占据大量的空间,尤其是后者会大大增加可执行文件的长度。可以看出,如果能用经过压缩、具有较好的压缩率的JPEG或GIF格式的图像来取代BMP文件在VC中的应用,无疑还是很有吸引力的。 
二、
设计思路 
   虽然有一些操作、处理JPEG、GIF等其他格式图像的Active
X控件,但总的来说使用起来并不太方便,笔者经过实验摸索,总结出了一种借助于COM接口的OLE方法来实现上述功能的一种简便方法,现介绍如下以飨广大读者: 
下面我们要使用IPicture
的COM接口,有必要对该图像接口做些了解:该接口主要管理图像对象及其属性,图像对象为位图、图标和图元等提供一种与语言无关的抽象。和标准的字体对象一样,系统也提供了对图像对象的标准实现。其主要的接口是IPicture和IPictureDisp,后者是由IDispatch接口派生以便通过自动化对图像的属性进行访问。图像对象也支持外部接口IPropertyNotifySink,以便用户能在图像属性发生改变时作出决定。图像对象也支持IPersistStream接口,所以它能从一个IStream接口的实例对象保存、装载自己,而IStream接口也支持对流对象的数据读写。 
  
我们可以用函数OleLoadPicture从包含有图像数据的流中装载图像。该函数简化了基于流的图像对象的创建过程,可以创建一个新的图像对象并且用流中的内容对它进行初始化。其函数原型为: 
STDAPI
OleLoadPicture( IStream * pStream, //指向包含有图像数据的流的指针LONG lSize, //从流中读取的字节数BOOL fRunmode,
//图像属性对应的初值REFIID riid, //涉及到的接口标识,描述要返回的接口指针的类型VOID ppvObj //
在rrid中用到的接口指针变量的地址); 

三、 具体的实现 
  
在显示图像之前,首先要获取到图像文件的存放路径,这里采用标准的文件打开对话框来选取图像文件,文件名存放在CString型的变量m_sPath中: 
CFileDialog
dlg(TRUE,"jpg","*.jpg", 
OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT, 
"JPEG文件(*.jpg)|*.jpg|GIF文件(*.gif)|*.gif||",NULL); 
if(dlg.DoModal()==IDOK) 

m_sPath=dlg.GetPathName(); 
Invalidate(); 

为简单计,图形显示的代码直接在视类中的OnDraw中编写,首先打开文件并判断文件的可用性,并把文件内容放到流接口IStream的对象pStm中: 
IStream
*pStm; 
CFileStatus fstatus; 
CFile file; 
LONG
cb; 
…… 
if
(file.Open(m_Path,CFile::modeRead)&&file.GetStatus(m_Path,fstatus)&&
((cb = fstatus.m_size) != -1)) 

HGLOBAL hGlobal =
GlobalAlloc(GMEM_MOVEABLE, cb); 
LPVOID pvData = NULL; 
if
(hGlobal != NULL) 

if ((pvData = GlobalLock(hGlobal)) !=
NULL) 

file.ReadHuge(pvData,
cb); 
GlobalUnlock(hGlobal); 
CreateStreamOnHGlobal(hGlobal,
TRUE, &pStm); 




  
然后,就直接调用OleLoadPicture函数从流中装载图像: 
IPicture
*pPic; 
…… 
OleLoadPicture(pStm,fstatus.m_size,TRUE,IID_IPicture,(LPVOID*)&pPic)); 

  
由于该函数有时会导致失败,所以应当用SUCCEEDED宏来做一些适当的保护工作,只有在数据装载成功的前提下才能继续下面的图像显示工作: 
if(SUCCEEDED(OleLoadPicture(pStm,fstatus.m_size,TRUE,IID_IPicture,(LPVOID*)&pPic))) 

OLE_XSIZE_HIMETRIC
hmWidth; 
OLE_YSIZE_HIMETRIC
hmHeight; 
pPic->get_Width(&hmWidth); 
pPic->get_Height(&hmHeight); 
double
fX,fY; 
…… 
fX =
(double)pDC->GetDeviceCaps(HORZRES)*(double)hmWidth/((double)pDC->GetDeviceCaps(HORZSIZE)*100.0); 
fY
=
(double)pDC->GetDeviceCaps(VERTRES)*(double)hmHeight/((double)pDC->GetDeviceCaps(VERTSIZE)*100.0); 
if(FAILED(pPic->Render(*pDC,0,0,(DWORD)fX,(DWORD)fY,0,hmHeight,hmWidth,-hmHeight,NULL))) 
AfxMessageBox("渲染图像失败!"); 
pPic->Release(); 

else 
AfxMessageBox("从流中装载图像失败!"); 

  
其中,显示工作主要是由IPicture接口对象的Render函数来完成的,该函数主要用来将图片的指定部分画到指定的设备环境的指定位置。原型如下: 
HRESULT
Render( HDC hdc, //渲染图像用的设备环境句柄 
long x, //在hdc上的水平坐标 
long y,
//在hdc上的垂直坐标 
long cx, //图像宽度 
long cy,
//图像高度 
OLE_XPOS_HIMETRIC xSrc, //在源图像上的水平偏移 
OLE_YPOS_HIMETRIC
ySrc, //在源图像上的垂直偏移 
OLE_XSIZE_HIMETRIC
cxSrc,//在源图像上水平拷贝的数量 
OLE_YSIZE_HIMETRIC
cySrc,//在源图像上垂直拷贝的数量 
LPCRECT prcWBounds
//指向目标图元设备环境句柄的指针); 

  
小结:到此为止,通过上述代码已经能够在程序的客户区内显示JPEG、GIF等标准的图像了,但对于有多帧图片(即有动画)的GIF格式的图像,目前还只能显示第一帧,如要完整的显示GIF
动画的全过程,还需要外部Active X控件的支持。

在VC下显示JPEG、GIF格式图像的一种简便方法,码迷,mamicode.com

时间: 2024-10-04 23:39:20

在VC下显示JPEG、GIF格式图像的一种简便方法的相关文章

VC下加载多种格式图片的方法总结IPicture, CxImage, CImage(AtlImage), CPictureEx

尽管VC有提供相应的API和类来操作bmp位图.图标和(增强)元文件,但却不支持jpg.gif和png等格式的图片,而这几种格式却是常常要用到的.这里我给大家介绍两种办法来操作这些格式的图片. 1.用API OleLoadPicture来加载JPG.GIF格式的图片(注:不支持PNG格式,另外GIF只能加载第一帧,且不支持透明) OleLoadPicture 函数实际上创建了一个IPicture类型的COM接口对象,然后我们可以通过这个COM接口来操作图片(实际上你也可以用API OleCrea

解决方案下显示的网站名称被追加编号的问题解决方法

问题描述 只看标题 大家是不是觉得有点不知所以?好吧,直接上图,大家请往下看. 如图中所示,我用VS2012编辑器添加一个解决方案[TestSolution],接着我先添加一个网站[Test],然后因为路径选择不对,就把站点文件删除重建,结果就有了上图显示的问题了.大家会不会觉得奇怪,为什么我的物理文件夹名字是 Test,但是解决方案打开就成了 Test(1) 这么个让人不爽的名字了?这个问题说明了一点,网站名称的显示不是按照物理文件来的. 怎样让解决方案下显示的站点名恢复正常?   在这里暂时

判断pdf、word文档、图片等文件类型(格式)、大小的简便方法

很久没发文了,今天有时间就写一下吧. 关于上传文件,通常我们都需要对其进行判断,限制上传的类型,如果是上传图片,我们甚至会把图片转化成base64数据后,再进行上传.普遍的方法是直接写在上传按钮的触发方法里面,但是对于大型的项目而言,这必然是会重复着同一段代码,使得代码臃肿繁重,这样也不利于平台的优化以及后续的维护,于是,我便封装了一个小小的判断上传文件的类型,图片类型的简便方法,这样不紧节省了重复的劳动力,而且还可以更好的优化项目,提高性能,代码如下: /* Type: 该值为类型数组,例如:

关于在xp系统下CTreeCtrl控件不显示图标的一种解决方法

在最近的项目中,遇到 了这样一个比较棘手的问题,最后花了一天的时间才解决,现总结一下,以供大家参考! 问题:原来的项目工程在XP系统下用VC6编写的,一切没有问题.但由于现实需要需要在WIN7/XP系统运行,我们把工程转到了VC2010,在WIN7下没有问题.但在XP却出现了一个莫名的问题:就是CTreeCtrl控件的BMP图标不显示,在显示图标的地方全部不显示(为空),经过反复检测,确认代码没有问题(在WIN7系统下没有出现问题). 其中的显示图标的CTreeCtrl控件代码如下: .h文件

Windows7下默认网关不能保存ip的两种解决方法

Windows7下默认网关不能保存ip解决方法 第一种: 第一步:点击左下角的WIN图标或WIN图标+R也可,输入CMD然后回车,打开DOS模式窗口.第二步:在DOS命令行提示符下直接输入:“netsh winsock reset”,输入的是引号内的,不包含引号,然后回车,提示需要重启,暂时不重启.第三步:在DOS命令行提示符下直接输入:“netsh int ip reset reset.log”,然后回车,提示重新启动计算机完成此操作.第四步:关闭DOS窗口,重新启动计算机,问题解决. 第二种

嵌入式linux------SDL移植(am335x下显示bmp图片)

#include<stdio.h> #include "/usr/local/ffmpeg_arm/include/SDL/SDL.h" char *bmp_name[3] = {"000.bmp","111.bmp","222.bmp"}; int main() { int i=0; //The images SDL_Surface* hello = NULL; SDL_Surface* screen = NUL

VC 下加载 JPG / JPEG / GIF / PNG 图片最简单的方法

VC MFC 提供的 API LoadBitmap / LoadImage 类 CBitmap 等都只能操作 BMP 位图,图标.对于其他常用的 JPG / JPEG / GIF / PNG 格式,它无能为力.VC 下怎样才能加载各种非 BMP 格式的图片呢? 下面介绍一种最简单的办法.用 CImage 类的 Load 函数加载图片,之后用 Detach 取得HBITMAP 句柄.取得图片的HBITMAP 句柄后就可以像操作 BMP 图片一样处理 JPG / JPEG / GIF / PNG 格

jpg格式图片在ie下显示XX的问题

jpg格式图片在chrome等高级浏览器下显示都是正常的,但是在ie下却显示不出来 原因是图片的模式是cmyk,ie貌似不能识别这种模式的图片,编辑一下图片,改成rgb模式就行了

数字图像处理 CImage类的使用与封装(jpg png gif tif bmp等格式图像的加载、数据读写、保存等功能)

引入CImage类的原因 原有的CBitmap 类只能处理BMP格式的图片,非常受限.而CImage可以处理JPGE.GIF.BMP.PNG等多种格式图片,扩展了图片处理功能且能与CBitmap 进行转换( 因为所载入的位图句柄都是HBITMAP,所以可相互转换),因此引入CImage类进行图像处理. CImage类简介 CImage是MFC和ATL共享的新类,它能从外部磁盘中调入一个JPEG.GIF.BMP和PNG格式的图像文件加以显示,而且这些文件格式可以相互转换. CImage提供增强型的