首页 > 解决方案 > 在启用 GDI 缩放的 Windows 10 上保存位图

问题描述

我有一个带有工具栏的 MFC 应用程序(使用 CMFCToolbar)。我使用文件和资源中的位图即时创建工具栏位图。DIB 有不同的颜色格式。

这已经工作了很长时间,现在正在工作,除了一个案例:

最近我们不得不为 Windows 10 1703 及更高版本启用 GDI 缩放。在具有高分辨率显示和 200% 缩放的系统(如 Surface)上,会出现以下效果:

所有工具栏图标都变形了。

我也找到了原因:

保存合成图像时,我只得到图像的左上角四分之一。与没有 GDI 缩放的正常分辨率显示相比,位图的宽度和高度没有变化(比如 1024x15)。但该位图仅包含左上角的像素(参见下面的示例)。

所以我假设设备上下文告诉 Windows 200% 的缩放。当从源到目标时,图像会自动放大,但位图的尺寸不会改变。

如何保存未缩放的位图?

-或者-

如何正确保存缩放的位图?从哪里获取丢失的像素?从哪里获得合适的尺寸?(HBITMAP 仅指未缩放的尺寸)。

例子:

没有 GDI 缩放,正确:

没有缩放

200% 缩放,相同的尺寸,但只有正确图像的左上角四分之一:

200 缩放

标签: winapimfcwindows-10gdibitmapimage

解决方案


总结及解决方案:

假设我们创建了一个与屏幕格式 (DDB) 兼容的内存位图:

CBitmap toolBitmap;
toolBitmap.CreateCompatibleBitmap (pDC, 1000, 20);

稍后我们将一些东西放入内存位图中(在这里无关紧要)。现在我们要保存位图(作为 DIB 写入文件)。

虽然我们知道尺寸(这里:1000x20),但我们不应该使用它们。因为在 Window 10 和进程上激活了 GDI 缩放并且使用了带缩放的高分辨率显示 - 尺寸可能在内部发生了变化。因此位图不再是 1000x20。

这个失败了:

BITMAP bmHdr;
toolBitmap.GetObject(sizeof(BITMAP), &bmHdr);

位图标头包含原始尺寸 (1000x20)。使用它们保存到文件会导致图像不完整。只存储左上部分。

这个可行 - 我们可以检索缩放尺寸:

BITMAPINFO bi = {};
bi.dwSize = sizeof(bi);
int result = GetDIBits(pDC->GetSafeHdc(), (HBITMAP)toolBitmap.GetSafeHandle(), 0, 0, NULL, &bi, DIB_RGB_COLORS);

现在我们可以继续新的维度了。

我最终使用了 GDI+ 函数,它还保存了完整的(缩放的)位图:

Gdiplus::Bitmap bm((HBITMAP)toolBitmap.GetSafeHandle(), NULL);
Gdiplus::Status status = bm.Save(pwszFileName, &clsidEncoder, NULL);

我认为有大量旧的 MFC 和 GDI 代码无法在 Windows 10 上激活 GDI 缩放时正常工作。


推荐阅读