winapi - 在启用 GDI 缩放的 Windows 10 上保存位图
问题描述
我有一个带有工具栏的 MFC 应用程序(使用 CMFCToolbar)。我使用文件和资源中的位图即时创建工具栏位图。DIB 有不同的颜色格式。
- 所以我创建了一个与屏幕 DC 兼容的空位图工具栏图像。
- 然后我打开所有位图并将内容blit 到工具栏位图(GDI 为我进行色彩空间转换和拉伸)。
- 然后我将位图保存到 24 位 DIB 文件中。
- 然后我创建工具栏对象并加载图像。
这已经工作了很长时间,现在正在工作,除了一个案例:
最近我们不得不为 Windows 10 1703 及更高版本启用 GDI 缩放。在具有高分辨率显示和 200% 缩放的系统(如 Surface)上,会出现以下效果:
所有工具栏图标都变形了。
我也找到了原因:
保存合成图像时,我只得到图像的左上角四分之一。与没有 GDI 缩放的正常分辨率显示相比,位图的宽度和高度没有变化(比如 1024x15)。但该位图仅包含左上角的像素(参见下面的示例)。
所以我假设设备上下文告诉 Windows 200% 的缩放。当从源到目标时,图像会自动放大,但位图的尺寸不会改变。
如何保存未缩放的位图?
-或者-
如何正确保存缩放的位图?从哪里获取丢失的像素?从哪里获得合适的尺寸?(HBITMAP 仅指未缩放的尺寸)。
例子:
没有 GDI 缩放,正确:
200% 缩放,相同的尺寸,但只有正确图像的左上角四分之一:
解决方案
总结及解决方案:
假设我们创建了一个与屏幕格式 (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 缩放时正常工作。
推荐阅读
- spring-boot - 在 SpringBootTest 中禁用使用 spring.config.import 激活的配置服务客户端
- linux - 检查 .o 文件中的调试符号
- python - 在python中实现函数分数微分方程的欧拉方法
- python - 从熊猫数据框中同一列的可用数据中转换缺失的列数据?
- php - php - 动态标题
- arrays - 将指针与数组一起使用
- php - 如何使用 php 验证学校电子邮件域
- c# - C#简单链表中的“this”
- java - 如何将屏幕截图和自定义消息嵌入到使用 cucumber 生成的 Masterthought 报告中?
- flutter - 非空迁移后 Flutter StreamBuilder