无法正确地将hdc保存到位图

时间:2012-09-08 18:21:04

标签: winapi bitmap hdc

我尝试将内存DC保存到位图文件中。但是我无法获得有关以下尺寸的正确值:

infoHeader.biSizeImage

fileHeader.bfSize


fileHeader.bfOffBits

WriteFile(hFile, &fileHeader, sizeof(BITMAPFILEHEADER), &dwBytesWritten, NULL);

WriteFile(hFile, &infoHeader, sizeof(BITMAPINFOHEADER), &dwBytesWritten, NULL);

WriteFile(hFile, pPixels, info.bmiHeader.biSizeImage, &dwBytesWritten, NULL);

我可以获得一个文件,但我无法打开它,因为它抱怨它已损坏或不再受支持。

关于保存到bmp有很多来源,但是其中许多都有不同的大小值。 有些还包括调色板信息。我很困惑。

有人可以说明如何填写正确的信息吗?

static void SaveAsBmp(TCHAR *fileName)
{
HDC hdcView = GetDC(hwndView);
HDC memDC = CreateCompatibleDC(hdcView);

RECT rcView;
GetClientRect(hwndView, &rcView);

int rcView_dx = rcView.right - rcView.left;
int rcView_dy = rcView.bottom - rcView.top;

HBITMAP hMemBmp = CreateCompatibleBitmap(hdcView, rcView_dx, rcView_dy);
HBITMAP hOldBmp = (HBITMAP)SelectObject(memDC, hMemBmp);

BitBlt(memDC, 0, 0, rcView_dx, rcView_dy, hdcView, 0, 0, SRCCOPY);

//----------

BITMAP bmp;
GetObject(hMemBmp, sizeof(BITMAP), &bmp);

//----------

WORD wBits = (WORD)(bmp.bmBitsPixel * bmp.bmPlanes);

if(wBits <=  1)                                                   
    wBits = 1;             
else  if(wBits <=  4)                               
    wBits  = 4;             
else if(wBits <=  8)                               
    wBits  = 8;             
else if (wBits <= 16)
    wBits = 16;
else if (wBits <= 24)
    wBits = 24;
else wBits = 32; 

//----------

BITMAPINFOHEADER infoHeader;
BITMAPINFO info;
info.bmiHeader = infoHeader;

infoHeader.biSize = sizeof(BITMAPINFOHEADER);    
infoHeader.biWidth = bmp.bmWidth; 
infoHeader.biHeight = bmp.bmHeight;    
infoHeader.biPlanes = bmp.bmPlanes;    
infoHeader.biBitCount = bmp.bmBitsPixel;     
infoHeader.biCompression = BI_RGB;   

infoHeader.biSizeImage = 2 * ((bmp.bmWidth * bmp.bmBitsPixel + 15) / 16)  * bmp.bmPlanes * bmp.bmHeight;
infoHeader.biXPelsPerMeter = 0;  
infoHeader.biYPelsPerMeter = 0;
infoHeader.biClrUsed = 0;
infoHeader.biClrImportant = 0;

RGBQUAD *pPixels = new RGBQUAD[bmp.bmWidth * bmp.bmWidth]; 
GetDIBits(memDC, hMemBmp, 0, bmp.bmWidth, pPixels, &info, DIB_RGB_COLORS);

BITMAPFILEHEADER fileHeader;
fileHeader.bfType = 0x4d42;
fileHeader.bfSize = (DWORD)(sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + info.bmiHeader.biSizeImage);
fileHeader.bfReserved1 = 0;
fileHeader.bfReserved2 = 0;
fileHeader.bfOffBits = (DWORD)(sizeof(BITMAPFILEHEADER) + info.bmiHeader.biSize);

HANDLE hFile = CreateFile(fileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);

DWORD dwBytesWritten;

WriteFile(hFile, &fileHeader, sizeof(BITMAPFILEHEADER), &dwBytesWritten, NULL);
WriteFile(hFile, &infoHeader, sizeof(BITMAPINFOHEADER), &dwBytesWritten, NULL);
WriteFile(hFile, pPixels, info.bmiHeader.biSizeImage, &dwBytesWritten, NULL);

CloseHandle(hFile);

delete pPixels;

SelectObject(memDC, hOldBmp);
DeleteObject(hMemBmp);
DeleteDC(memDC);

1 个答案:

答案 0 :(得分:0)

  

infoHeader.biSizeImage = 2 *((bmp.bmWidth * bmp.bmBitsPixel + 15)/ 16)* bmp.bmPlanes * bmp.bmHeight;

看起来不正确,它应该是bmp.bmBitsPixel >> 3,假设它是24或32.也就是说,它是每像素的位数,并且你正在使用它,好像它是每像素的字节数。

一般来说代码看起来不错,您应该使用调试器检查变量以查找不匹配(或将其粘贴到此处,尤其是BITMAPINFOHEADER)。

另请注意,您可能采用了不同的方式:不使用CreateCompatibleBitmap,而是使用您感兴趣的格式CreateDIBSection,例如24 bpp RGB,会立即收到指向位的原始指针。然后将数据分成这个位图,您不再需要GetObject调用,因为您已经拥有了所需的一切。并且,GetObject可能会返回DIBSECTION格式,并已准备好使用BITMAPINFOHEADER。它完全相同,只是更简单。