使用win32 API或GDI+保存到位图

saving to bitmap with win32 API or GDI+

本文关键字:保存 位图 GDI+ win32 API 使用      更新时间:2023-10-16

问题很简单,但我找不到好的&我在互联网上的问题的干净解决方案。我得到的是我窗户上的一些画。现在,我可以使用BitBlt函数将那些从窗口设备上下文保存到图像设备上下文,也可以从那里保存到位图句柄:

HDC bitmapDC = CreateCompatibleDC(dc);
HBITMAP bitmap = CreateCompatibleBitmap(bitmapDC, 200, 200);
SelectObject(bitmapDC,bitmap);
BitBlt(bitmapDC, 0, 0, 200, 200, dc, 200, 200, SRCCOPY);

但从那以后,我迷失了方向。我看了一下GDI+Bitmap类,它有save函数,我发现了如何实现检索图片编码CLSID的代码。然而,我不知道我是否正确地使用了对该类的加载。HBITMAP有一个重载的构造函数,但它也要求一些调色板,我将其设置为NULL:

Bitmap image(bitmap,NULL);

我试图保存png文件,但它导致了黑色方块,没有我期望的图纸。如果你愿意,我的绘画程序的完整代码:

void GetCLSID(const WCHAR* format, CLSID* pClsid){
    UINT  num = 0;          // number of image encoders
    UINT  size = 0;         // size of the image encoder array in bytes
    ImageCodecInfo* pImageCodecInfo = NULL;
    GetImageEncodersSize(&num, &size);
    pImageCodecInfo = (ImageCodecInfo*)(malloc(size));
    GetImageEncoders(num, size, pImageCodecInfo);
    for(UINT j = 0; j < num; ++j)
    {
      if( wcscmp(pImageCodecInfo[j].MimeType, format) == 0 )
      {
         *pClsid = pImageCodecInfo[j].Clsid;
         free(pImageCodecInfo);
      }    
    }
}
void OnPaint(HDC dc){
    RECT rect; rect.bottom = 0; rect.top = 20; rect.left = 0; rect.right = 100;
    HBRUSH blueBrush = CreateSolidBrush(RGB(0,0,200));
    FillRect(dc, &rect, blueBrush);
    Graphics graphics(dc);
    Pen      pen(Color(255, 0, 0, 255));
    graphics.DrawLine(&pen, 0, 0, 200, 100);
    SolidBrush greenBrush(Color(0,200,0));
    Rect ellipseRect(20,20,20,20);
    graphics.FillEllipse(&greenBrush, ellipseRect);
    SolidBrush redBrush(Color(200,0,0));
    Rect boxRectangle(0,40,20,100);
    graphics.FillRectangle(&redBrush, boxRectangle);
    pen.SetColor(Color(200,0,200));
    pen.SetWidth(20);
    graphics.DrawBezier(&pen, 100, 20, 130, 40, 200, 10, 230, 20);
    HDC bitmapDC = CreateCompatibleDC(dc);
    HBITMAP bitmap = CreateCompatibleBitmap(bitmapDC, 200, 200);
    SelectObject(bitmapDC,bitmap);
    BitBlt(bitmapDC, 0, 0, 500, 500, dc, 500, 500, SRCCOPY);
    Bitmap image(bitmap,NULL);
    CLSID clsID;
    GetCLSID(L"image/png", &clsID);
    image.Save(L"pic.png", &clsID);
}

我甚至无法想象简单的储蓄会成为这样的问题,所以我很乐意得到任何帮助,谢谢!

我在这里给出的代码可以满足您的需求:如何将子窗口的客户端区域保存为位图文件?

它在C中非常冗长。它在C++中要好得多,因为CImage