//原著:joes Tomas
//译者:重庆大学光电工程学院 贾旭滨
//欢迎批评指教,谢谢!
位图文件是分成4部分的。第一部分是位图文件头,它包括位图文件名,位图的大小和位图数据离文件头的偏移量。接下去的是位图信息头,它包括了位图的许多信息,比如位图的宽度,高度和位图使用的颜色数。再后面是颜色表,它可能包含了2个或更多的RGBQUAD结构。最后面是位图图象的数据。
一.位图结构如下:
---- 一、BMP文件结构
---- 1. BMP文件组成
---- BMP文件由文件头、位图信息头、颜色信息和图形数据四部分组成。
---- 2. BMP文件头
---- BMP文件头数据结构含有BMP文件的类型、文件大小和位图起始位置等信息。
---- 其结构定义如下:
typedef struct tagBITMAPFILEHEADER
{
WORDbfType; // 位图文件的类型,必须为BM
DWORD bfSize; // 位图文件的大小,以字节为单位
WORDbfReserved1; // 位图文件保留字,必须为0
WORDbfReserved2; // 位图文件保留字,必须为0
DWORD bfOffBits; // 位图数据的起始位置,以相对于位图
// 文件头的偏移量表示,以字节为单位
} BITMAPFILEHEADER;
---- 3. 位图信息头
BMP位图信息头数据用于说明位图的尺寸等信息。
typedef struct tagBITMAPINFOHEADER{
DWORD biSize; // 本结构所占用字节数
LONGbiWidth; // 位图的宽度,以像素为单位
LONGbiHeight; // 位图的高度,以像素为单位
WORD biPlanes; // 目标设备的级别,必须为1
WORD biBitCount// 每个像素所需的位数,必须是1(双色),
// 4(16色),8(256色)或24(真彩色)之一
DWORD biCompression; // 位图压缩类型,必须是 0(不压缩),
// 1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一
DWORD biSizeImage; // 位图的大小,以字节为单位
LONGbiXPelsPerMeter; // 位图水平分辨率,每米像素数
LONGbiYPelsPerMeter; // 位图垂直分辨率,每米像素数
DWORD biClrUsed;// 位图实际使用的颜色表中的颜色数
DWORD biClrImportant;// 位图显示过程中重要的颜色数
} BITMAPINFOHEADER;
---- 4. 颜色表
颜色表用于说明位图中的颜色,它有若干个表项,每一个表项是一个RGBQUAD类型的结构,定义一种颜色。RGBQUAD结构的定义如下:
typedef struct tagRGBQUAD {
BYTErgbBlue;// 蓝色的亮度(值范围为0-255)
BYTErgbGreen; // 绿色的亮度(值范围为0-255)
BYTErgbRed; // 红色的亮度(值范围为0-255)
BYTErgbReserved;// 保留,必须为0
} RGBQUAD;
颜色表中RGBQUAD结构数据的个数有biBitCount来确定:
当biBitCount=1,4,8时,分别有2,16,256个表项;
当biBitCount=24时,没有颜色表项。
位图信息头和颜色表组成位图信息,BITMAPINFO结构定义如下:
typedef struct tagBITMAPINFO {
BITMAPINFOHEADER bmiHeader; // 位图信息头
RGBQUAD bmiColors[1]; // 颜色表
} BITMAPINFO;
---- 5. 位图数据
位图数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右,扫描行之间是从下到上。位图的一个像素值所占的字节数:
当biBitCount=1时,8个像素占1个字节;
当biBitCount=4时,2个像素占1个字节;
当biBitCount=8时,1个像素占1个字节;
当biBitCount=24时,1个像素占3个字节;
Windows规定一个扫描行所占的字节数必须是 4的倍数(即以long为单位),不足的以0填充,一个扫描行所占的字节数计算方法: DataSizePerLine= (biWidth* biBitCount+31)/8; 一个扫描行所占的字节数 DataSizePerLine= DataSizePerLine/4*4; // 字节数必须是4的倍数位图数据的大小(不压缩情况下): DataSize= DataSizePerLine* biHeight;
随便说一句,位图是设备无关图象,所以它的文件的扩展名就dib
下面说的函数在win95/winNT都行。在win95上,你可以使用函数loadimage()来加载一个位图文件,但是在winNT上,你也可以使用这个函数,但是其中的lr_loadfromfile标志位却不能用了。
第1步:载入位图
为了正确的画出位图来,我们就需要位图结构中后面3部分的信息了。首先要用到的是位图信息头,从中我们可以知道得分配多少内存给位图,然后就可以把位图读进来了,而且调色板也可以创建起来。
//loadbmp -载入位图文件,并且给它创建调色板
//return -成功的标志
//sbmpfile -位图文件的路径
//phdib -指向分配给位图文件的内存空间的指针,会被释放掉
//ppal -控制逻辑调色板
//
bool loadbmp( lpctstr sbmpfile, hglobal *phdib, cpalette *ppal )
{
cfile file;
if( !file.open( sbmpfile, cfile::moderead) )
return false;
bitmapfileheader bmfheader;
long nfilelen;
nfilelen = file.getlength();
// read file header
if (file.read((lpstr)&bmfheader, sizeof(bmfheader)) != sizeof(bmfheader))
return false;
// file type should be 'bm'
if (bmfheader.bftype != ((word) ('m' << 8) | 'b')) return false;
hglobal hdib="::globalalloc(gmem_fixed," nfilelen);
if (hdib="=" 0) return false;
//read the remainder of the bitmap file.
if (file.readhuge((lpstr)hdib, nfilelen sizeof(bitmapfileheader)) !="nfilelen" sizeof(bitmapfileheader) )
{ ::globalfree(hdib);
return false; }
bitmapinfo &bminfo="*(lpbitmapinfo)hdib" ;
int ncolors="bminfo.bmiheader.biclrused" ? bminfo.bmiheader.biclrused : 1 << bminfo.bmiheader.bibitcount;
// create the palette
if(ncolors <="256" ) { uint nsize="sizeof(logpalette)" + (sizeof(paletteentry) * ncolors); logpalette *plp="(logpalette" *) new byte[nsize]; plp->palversion = 0x300;
plp->palnumentries = ncolors;
for( int i=0; i palpalentry[i].pered = bminfo.bmicolors[i].rgbred;
plp->palpalentry[i].pegreen = bminfo.bmicolors[i].rgbgreen;
plp->palpalentry[i].peblue = bminfo.bmicolors[i].rgbblue;
plp->palpalentry[i].peflags = 0;
}
ppal->createpalette( plp );
delete[] plp;
}
*phdib = hdib;
return true;
}
//第2步:画位图
下面的函数只是个例子来说明怎么样画位图,它使用了函数setdibitstodevice()来达到目的,你应该知道位图文件存储是从最后面一行开始的,所以,你应该特别注意!其实你也可以使用函数stretchdibits()来显示位图,它的功能更强,不但可以伸展和压缩位图,还可以不同的光栅操作来产生位图。
void drawdib( cdc* pdc, hglobal hdib, cpalette *ppal )
{
lpvoid lpdibbits; // pointer to dib bits
bool bsuccess=false; // success/fail flag
bitmapinfo &bminfo = *(lpbitmapinfo)hdib ;
int ncolors = bminfo.bmiheader.biclrused ? bminfo.bmiheader.biclrused :
1 << bminfo.bmiheader.bibitcount; if( bminfo.bmiheader.bibitcount> 8 )
lpdibbits = (lpvoid)((lpdword)(bminfo.bmicolors +
bminfo.bmiheader.biclrused) +
((bminfo.bmiheader.bicompression == bi_bitfields) ? 3 : 0));
else
lpdibbits = (lpvoid)(bminfo.bmicolors + ncolors);
if( ppal && (pdc->getdevicecaps(rastercaps) & rc_palette) )
{
pdc->selectpalette(ppal, false);
pdc->realizepalette();
}
::setdibitstodevice(pdc->m_hdc, // hdc
0, // destx
0, // desty
bminfo.bmiheader.biwidth, // ndestwidth
bminfo.bmiheader.biheight, // ndestheight
0, // srcx
0, // srcy
0, // nstartscan
bminfo.bmiheader.biheight, // nnumscans
lpdibbits, // lpbits
(lpbitmapinfo)hdib, // lpbitsinfo
dib_rgb_colors); // wusage
}