我尝试将内存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);
答案 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
。它完全相同,只是更简单。