如何在VC中显示动态的GIF
作者:薛碧
下载本文示例源代码
如果是使用VB,也许这个话题是多余的,因为VB有一个图象控件可以非常容易地实现各种格式的图象显示功能,然而对于VC却没有一个象样的控件可以达到这种效果,怎么办?经过一段日子的研究,发现只需要实现两步工作,就可以在VC中实现如同VB中一样的gif动态效果。
本文将介绍的两部分是IPicture接口的使用和gif的储存格式,好象一听到储存格式,读者就不想再看下去了!其实不然,这里只须用到其最基本的一部分,请读者耐心地往下看。
一.IPicture接口
IPicture接口是一个com类,其成员函数可参见微软的MSDN,这里只需用到以下几个函数:
get_Width |
返回当前图像的宽度 |
get_Height |
返回当前图像的高度 |
Render |
在指定的位置、设备上下文上绘制指定的图像 |
IPicture的使用不需要CoCreateInstance函数,而只需要使用OleLoadPicture,鉴于此接口在许多文章杂志上均有介绍,这里略去(因为不是本文的重点)。
二.Gif储存格式
gif储存格式是一个非常复杂的内容,如果要讲透彻可以写很多篇文章,庆幸的是要实现本文的主题只需要知道其中的一个图象储存结构就可以了,这里定义该图象结构为gifImage:
typedef struct gifImage{
WORD logX;
WORD logY;
WORD width;
WORD height;
struct flag{
BYTE d:3;
BYTE c:1;
BYTE b:3;
BYTE a:1;
}Flag;
}GifImage,*PGifImage;
在该结构中,
logX为图象相对于逻辑屏幕左上角的x坐标,常为0;
logY为图象相对于逻辑屏幕左上角的y坐标,常为0;
width为图象的宽度;
height为图象的高度;
Flag为一个标志,a指是否存在局域性调色板,b为存储方式(与本主题无关),c为RGB值是否经过排序(无关),d为调色板的大小,为3*2^(d+1);
最后想提一下,因为每副图象都以0x2c开头,并且0x2c前面必为0,故在储存格式中要找到图象的起始位置,只需查找0x2c并且前面一个值为0(具体请看下面代码),其次,一副图象可能储存为多个数据块,每个数据快都是以数据长度为起始的,这一点很重要。即其图象储存为:
0x2c |
图象开头 |
gifImage |
图象头结构 |
BYTE Number |
Number为一个跟gif压缩有关的数字,可以不理踩。 |
第一副图象的大小 |
|
...... |
图象存储内容 |
第二副图象大小 |
|
...... |
|
...... |
|
...... |
|
0x00 |
数据块结束 |
好,介绍完了主要的两大部分也该进入今天的主题了。由于使用IPicture接口可以非常轻松地显示gif的第一副图画,那么我们就是利用这一点,利用gif的图象格式,把第二,三。。。图画逐一拷贝到第一副图画的位置,再使用IPicture进行读取,便可以形成一副十分连续的gif动画了,接下来就让我来给大家展示以下:
HINSTANCE handle = ::AfxGetResourceHandle();//首先获得资源句柄
HRSRC hrsrc= ::FindResource(handle,MAKEINTRESOURCE(IDR_IMAGE2),"IMAGE");
DWORD word = ::SizeofResource(handle,hrsrc);
BYTE* lpBy = (BYTE*)LoadResource(handle,hrsrc);//获得图象的首地址
BYTE* pByte[20]; //用来储存gif每幅图象的地址
DWORD nu[20]; //用来储存每幅图象的大小
int num = 0; //用来计算有几副图象
DWORD firstLocation = 0; //第一副图象的位置,用来替换
for(DWORD j=0;j<word;j++)
{
if(lpBy[j]==0x2c) //图象开头
{
if(lpBy[j-1]==0x00) //确认是否图象开头
{
if(num==0)
{
firstLocation = j; //得到第一副图象位置
}
PGifImage nowImage = (PGifImage)&lpBy[j+1];
if(nowImage->Flag.a==0) //a为0时指图象不存在局部调色板
{
DWORD number = 1+sizeof(GifImage);
while(lpBy[j+number]!=0)
{
number = number+(DWORD)lpBy[j+number]+1;
} //算得图象大小
number++; //把最后一个0x00加上
pByte[num] = new BYTE[number];
for(DWORD n=0;n<number;n++)
{
*(BYTE*)(pByte[num]+n) = lpBy[j+n];
} //将图象储存起来。
nu[num] = number;
j = j+number-1; //跳过图象
num++;
}
else
{ //当a为1时需要加上局部调色板的大小,其他与a为0时一样
int number = 1+sizeof(GifImage)+3*(int)floor(pow(2,nowImage->Flag.d));
while(lpBy[j+number]!=0)
{
number = number+(DWORD)lpBy[j+number]+1;
} //算得图象大小
number++; //把最后一个0x00加上
pByte[num] = new BYTE[number];
for(DWORD n=0;n<number;n++)
{
*(BYTE*)(pByte[num]+n) = lpBy[j+n];
} //将图象储存起来。
nu[num] = number;
j = j+number-1; //跳过图象
num++;
}
}
}
}
int working= 1;
while(working)
{
for(int m=0;m<num;m++)
{
CBrush brush(RGB(255,255,255));
pdc->FillRect(CRect(0,0,500,500),&brush);
DWORD DDD;
VirtualProtect(lpBy,word,PAGE_READWRITE,&DDD);
//修改页面的保护属性,以进行写操作
for(DWORD n=0;n<nu[m];n++)
{
lpBy[firstLocation+n] = *(BYTE*)(pByte[m]+n);
}
VirtualProtect(lpBy,word,DDD,NULL);
//因为IPicture必须把图象存成流的形式才能工作,所以有下面一段函数
CMemFile file(lpBy,word);
CArchive ar(&file,CArchive::load|CArchive::bNoFlushOnDelete);
CArchiveStream arcstream(&ar);
CComQIPtr<IPicture> m_picture;
HRESULT hr = OleLoadPicture((LPSTREAM)&arcstream,0,false,
IID_IPicture,(void**)&m_picture);
long a,b;
m_picture->get_Width(&a);
m_picture->get_Height(&b);
CSize sz(a,b);
pdc->HIMETRICtoDP(&sz); //时OLE格式的大小转化为正常大小
CRect rect;
m_picture->Render(*pdc,0,0,sz.cx,sz.cy,0,b,a,-b,&rect);
Sleep(100); //停止一段时间。
}
}
结尾语:本程序最好放在一个线程中进行工作,对于最后Sleep的毫秒数就读者喜好进行修改,其实gif储存格式中有一个图象间隔的毫秒数,但是,笔者认为在此没有必要,还是随读者的喜好较好。由于笔者时间较紧,没有把程序整理成原代码供读者下载,深表歉意,并且本文中出现的小错误,希望读者批评指正。
评论