如何从现有的 HBITMAP(CreateCompatibleBitmap) 创建新的 HBITMAP(DIB)?

发布于 2024-11-06 07:44:38 字数 4220 浏览 0 评论 0原文

我通过 CreateCompatibleBitmap() 创建了一个位图来捕获屏幕。 然后我想从这个位图创建一个新的 DIB(在内存中)。

代码应该如下所示:

HBITMAP ConvertToDIB(HBITMAP ddb)
{ ... }

我已经有一些代码可以将位图写入文件中。但是,我想要内存中的 HBITMAP。

//---------------------------------------------------------------------------
void WriteBitmapToFile(HBITMAP bitmap, wchar_t* filename, HDC hDC)
{
    BITMAP bmp;
    PBITMAPINFO pbmi;
    WORD cClrBits;
    HANDLE hf; // file handle
    BITMAPFILEHEADER hdr; // bitmap file-header
    PBITMAPINFOHEADER pbih; // bitmap info-header
    LPBYTE lpBits; // memory pointer
    DWORD dwTotal; // total count of bytes
    DWORD cb; // incremental count of bytes
    BYTE *hp; // byte pointer
    DWORD dwTmp;

    // create the bitmapinfo header information
    if (!::GetObjectW(bitmap, sizeof(BITMAP), &bmp))
    {
        return;
    }

    // Convert the color format to a count of bits.
    cClrBits = (WORD)(bmp.bmPlanes * bmp.bmBitsPixel);
    if (cClrBits == 1)
    cClrBits = 1;
    else if (cClrBits <= 4)
    cClrBits = 4;
    else if (cClrBits <= 8)
    cClrBits = 8;
    else if (cClrBits <= 16)
    cClrBits = 16;
    else if (cClrBits <= 24)
    cClrBits = 24;
    else cClrBits = 32;



    // Allocate memory for the BITMAPINFO structure.
    if (cClrBits != 24)
    pbmi = (PBITMAPINFO) LocalAlloc(LPTR,
    sizeof(BITMAPINFOHEADER) + sizeof(RGBQUAD) * (1<< cClrBits));
    else
    pbmi = (PBITMAPINFO) LocalAlloc(LPTR, sizeof(BITMAPINFOHEADER));

    // Initialize the fields in the BITMAPINFO structure.

    pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
    pbmi->bmiHeader.biWidth = bmp.bmWidth;
    pbmi->bmiHeader.biHeight = bmp.bmHeight;
    pbmi->bmiHeader.biPlanes = bmp.bmPlanes;
    pbmi->bmiHeader.biBitCount = bmp.bmBitsPixel;
    if (cClrBits < 24)
    pbmi->bmiHeader.biClrUsed = (1<<cClrBits);

    // If the bitmap is not compressed, set the BI_RGB flag.
    pbmi->bmiHeader.biCompression = BI_RGB;

    // Compute the number of bytes in the array of color
    // indices and store the result in biSizeImage.
    pbmi->bmiHeader.biSizeImage = (pbmi->bmiHeader.biWidth + 7) /8 * pbmi->bmiHeader.biHeight * cClrBits;
    // Set biClrImportant to 0, indicating that all of the
    // device colors are important.
    pbmi->bmiHeader.biClrImportant = 0;

    // now open file and save the data
    pbih = (PBITMAPINFOHEADER) pbmi;
    lpBits = (LPBYTE) GlobalAlloc(GMEM_FIXED, pbih->biSizeImage);

    if (!lpBits)
    {
        return;
    }

    // Retrieve the color table (RGBQUAD array) and the bits
    if (!GetDIBits(hDC, HBITMAP(bitmap), 0, (WORD) pbih->biHeight, lpBits, pbmi,
    DIB_RGB_COLORS))
    {
        return;
    }

    // Create the .BMP file.
    hf = CreateFileW(filename, GENERIC_READ | GENERIC_WRITE, (DWORD) 0,
    NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
    (HANDLE) NULL);
    if (hf == INVALID_HANDLE_VALUE){
    return;
    }
    hdr.bfType = 0x4d42; // 0x42 = "B" 0x4d = "M"
    // Compute the size of the entire file.
    hdr.bfSize = (DWORD) (sizeof(BITMAPFILEHEADER) +
    pbih->biSize + pbih->biClrUsed
    * sizeof(RGBQUAD) + pbih->biSizeImage);
    hdr.bfReserved1 = 0;
    hdr.bfReserved2 = 0;

    // Compute the offset to the array of color indices.
    hdr.bfOffBits = (DWORD) sizeof(BITMAPFILEHEADER) +
    pbih->biSize + pbih->biClrUsed
    * sizeof (RGBQUAD);

    // Copy the BITMAPFILEHEADER into the .BMP file.
    if (!WriteFile(hf, (LPVOID) &hdr, sizeof(BITMAPFILEHEADER),
    (LPDWORD) &dwTmp, NULL)) {
    return;
    }

    // Copy the BITMAPINFOHEADER and RGBQUAD array into the file.
    if (!WriteFile(hf, (LPVOID) pbih, sizeof(BITMAPINFOHEADER)
    + pbih->biClrUsed * sizeof (RGBQUAD),
    (LPDWORD) &dwTmp, ( NULL))){
    return;
    }


    // Copy the array of color indices into the .BMP file.
    dwTotal = cb = pbih->biSizeImage;
    hp = lpBits;
    if (!WriteFile(hf, (LPSTR) hp, (int) cb, (LPDWORD) &dwTmp,NULL)){
    return;
    }

    // Close the .BMP file.
    if (!CloseHandle(hf)){
    return;
    }

    // Free memory.
    GlobalFree((HGLOBAL)lpBits);
}

此致, 扎克

I created a bitmap by CreateCompatibleBitmap() for capturing the screen.
Then I want to create a new DIB (in memory) from this bitmap.

The code should look be:

HBITMAP ConvertToDIB(HBITMAP ddb)
{ ... }

I already have some code to write a bitmap into a file. However, I want a HBITMAP in memory.

//---------------------------------------------------------------------------
void WriteBitmapToFile(HBITMAP bitmap, wchar_t* filename, HDC hDC)
{
    BITMAP bmp;
    PBITMAPINFO pbmi;
    WORD cClrBits;
    HANDLE hf; // file handle
    BITMAPFILEHEADER hdr; // bitmap file-header
    PBITMAPINFOHEADER pbih; // bitmap info-header
    LPBYTE lpBits; // memory pointer
    DWORD dwTotal; // total count of bytes
    DWORD cb; // incremental count of bytes
    BYTE *hp; // byte pointer
    DWORD dwTmp;

    // create the bitmapinfo header information
    if (!::GetObjectW(bitmap, sizeof(BITMAP), &bmp))
    {
        return;
    }

    // Convert the color format to a count of bits.
    cClrBits = (WORD)(bmp.bmPlanes * bmp.bmBitsPixel);
    if (cClrBits == 1)
    cClrBits = 1;
    else if (cClrBits <= 4)
    cClrBits = 4;
    else if (cClrBits <= 8)
    cClrBits = 8;
    else if (cClrBits <= 16)
    cClrBits = 16;
    else if (cClrBits <= 24)
    cClrBits = 24;
    else cClrBits = 32;



    // Allocate memory for the BITMAPINFO structure.
    if (cClrBits != 24)
    pbmi = (PBITMAPINFO) LocalAlloc(LPTR,
    sizeof(BITMAPINFOHEADER) + sizeof(RGBQUAD) * (1<< cClrBits));
    else
    pbmi = (PBITMAPINFO) LocalAlloc(LPTR, sizeof(BITMAPINFOHEADER));

    // Initialize the fields in the BITMAPINFO structure.

    pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
    pbmi->bmiHeader.biWidth = bmp.bmWidth;
    pbmi->bmiHeader.biHeight = bmp.bmHeight;
    pbmi->bmiHeader.biPlanes = bmp.bmPlanes;
    pbmi->bmiHeader.biBitCount = bmp.bmBitsPixel;
    if (cClrBits < 24)
    pbmi->bmiHeader.biClrUsed = (1<<cClrBits);

    // If the bitmap is not compressed, set the BI_RGB flag.
    pbmi->bmiHeader.biCompression = BI_RGB;

    // Compute the number of bytes in the array of color
    // indices and store the result in biSizeImage.
    pbmi->bmiHeader.biSizeImage = (pbmi->bmiHeader.biWidth + 7) /8 * pbmi->bmiHeader.biHeight * cClrBits;
    // Set biClrImportant to 0, indicating that all of the
    // device colors are important.
    pbmi->bmiHeader.biClrImportant = 0;

    // now open file and save the data
    pbih = (PBITMAPINFOHEADER) pbmi;
    lpBits = (LPBYTE) GlobalAlloc(GMEM_FIXED, pbih->biSizeImage);

    if (!lpBits)
    {
        return;
    }

    // Retrieve the color table (RGBQUAD array) and the bits
    if (!GetDIBits(hDC, HBITMAP(bitmap), 0, (WORD) pbih->biHeight, lpBits, pbmi,
    DIB_RGB_COLORS))
    {
        return;
    }

    // Create the .BMP file.
    hf = CreateFileW(filename, GENERIC_READ | GENERIC_WRITE, (DWORD) 0,
    NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
    (HANDLE) NULL);
    if (hf == INVALID_HANDLE_VALUE){
    return;
    }
    hdr.bfType = 0x4d42; // 0x42 = "B" 0x4d = "M"
    // Compute the size of the entire file.
    hdr.bfSize = (DWORD) (sizeof(BITMAPFILEHEADER) +
    pbih->biSize + pbih->biClrUsed
    * sizeof(RGBQUAD) + pbih->biSizeImage);
    hdr.bfReserved1 = 0;
    hdr.bfReserved2 = 0;

    // Compute the offset to the array of color indices.
    hdr.bfOffBits = (DWORD) sizeof(BITMAPFILEHEADER) +
    pbih->biSize + pbih->biClrUsed
    * sizeof (RGBQUAD);

    // Copy the BITMAPFILEHEADER into the .BMP file.
    if (!WriteFile(hf, (LPVOID) &hdr, sizeof(BITMAPFILEHEADER),
    (LPDWORD) &dwTmp, NULL)) {
    return;
    }

    // Copy the BITMAPINFOHEADER and RGBQUAD array into the file.
    if (!WriteFile(hf, (LPVOID) pbih, sizeof(BITMAPINFOHEADER)
    + pbih->biClrUsed * sizeof (RGBQUAD),
    (LPDWORD) &dwTmp, ( NULL))){
    return;
    }


    // Copy the array of color indices into the .BMP file.
    dwTotal = cb = pbih->biSizeImage;
    hp = lpBits;
    if (!WriteFile(hf, (LPSTR) hp, (int) cb, (LPDWORD) &dwTmp,NULL)){
    return;
    }

    // Close the .BMP file.
    if (!CloseHandle(hf)){
    return;
    }

    // Free memory.
    GlobalFree((HGLOBAL)lpBits);
}

Best regards,
Zach

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

送君千里 2024-11-13 07:44:38

如果您只是想将 HBITMAP 转换为 DIB(设备独立位图),可以使用名为 GetDIBits 的 Win32 GDI 函数。

int GetDIBits(
  __in     HDC hdc,
  __in     HBITMAP hbmp,
  __in     UINT uStartScan,
  __in     UINT cScanLines,
  __out    LPVOID lpvBits,
  __inout  LPBITMAPINFO lpbi,
  __in     UINT uUsage
);

以下是 MSDN 上 GetDIBits 函数的链接:
http://msdn.microsoft.com/en -us/library/dd144879(v=vs.85).aspx

请注意,在调用 GetDIBits 之前必须为 lpvBits 分配内存。但是应该分配多少内存是另一个问题:
应该使用多少内存分配给使用 GetDIBits 函数从 HBITMAP 接收的 DIB 数据?

If you simply want to convert HBITMAP to DIB (device independent bitmap), there's a Win32 GDI function called GetDIBits.

int GetDIBits(
  __in     HDC hdc,
  __in     HBITMAP hbmp,
  __in     UINT uStartScan,
  __in     UINT cScanLines,
  __out    LPVOID lpvBits,
  __inout  LPBITMAPINFO lpbi,
  __in     UINT uUsage
);

Here's the link to GetDIBits function on MSDN:
http://msdn.microsoft.com/en-us/library/dd144879(v=vs.85).aspx

Note that you have to allocate memory for the lpvBits before calling GetDIBits. But how much memory should be allocated is another question here:
How much memory should be allocated for the DIB data received from HBITMAP using GetDIBits function?

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文