此代码根据我在哪台计算机上运行而获得不同的缩放比例。
Metafile image;
IntPtr dib;
var memoryHdc = Win32Utils.CreateMemoryHdc(IntPtr.Zero, 1, 1, out dib);
try
{
image = new Metafile(memoryHdc, EmfType.EmfOnly);
using (var g = Graphics.FromImage(image))
{
Render(g, html, left, top, maxWidth, cssData, stylesheetLoad, imageLoad);
}
}
finally
{
Win32Utils.ReleaseMemoryHdc(memoryHdc, dib);
}
进入 Render 方法,Metafile 对象的 PixelFormat 为 DontCare,因此没有有效的垂直或水平分辨率。
从 Render 方法中出来,它的值Format32bppRgb
和增加,以适应渲染的图像PhysicalDimension.Width
。PhysicalDimension.Height
如何使缩放独立于本地设置?
这是实现CreateMemoryHdc
(我没有写,它来自一个OSS库)。
public static IntPtr CreateMemoryHdc(IntPtr hdc, int width, int height, out IntPtr dib)
{
// Create a memory DC so we can work off-screen
IntPtr memoryHdc = CreateCompatibleDC(hdc);
SetBkMode(memoryHdc, 1);
// Create a device-independent bitmap and select it into our DC
var info = new BitMapInfo();
info.biSize = Marshal.SizeOf(info);
info.biWidth = width;
info.biHeight = -height;
info.biPlanes = 1;
info.biBitCount = 32;
info.biCompression = 0; // BI_RGB
IntPtr ppvBits;
dib = CreateDIBSection(hdc, ref info, 0, out ppvBits, IntPtr.Zero, 0);
SelectObject(memoryHdc, dib);
return memoryHdc;
}
如您所见,传递给 DC 构造函数的宽度、高度和位深度是恒定的。创建元文件会产生不同的物理尺寸。执行此操作后
image = new Metafile(memoryHdc, EmfType.EmfOnly);
图元文件PhysicalDimension.Height
在我的工作站上(和宽度)为 26.43,在我部署的服务器上为 31.25,因此缩放的差异已经很明显,因此可能不是渲染中的任何结果。
这可能是相关的。BitMapInfo
在 OSS 库中定义,如下所示:
internal struct BitMapInfo
{
public int biSize;
public int biWidth;
public int biHeight;
public short biPlanes;
public short biBitCount;
public int biCompression;
public int biSizeImage;
public int biXPelsPerMeter;
public int biYPelsPerMeter;
public int biClrUsed;
public int biClrImportant;
public byte bmiColors_rgbBlue;
public byte bmiColors_rgbGreen;
public byte bmiColors_rgbRed;
public byte bmiColors_rgbReserved;
}
所以可能设置biXPelsPerMeter
并biYPelsPerMeter
会有所帮助。上面的代码没有设置它们并且可能允许平台值。
不幸的是,设置这些值似乎没有任何区别。msdn 说
biXPelsPerMeter
位图的目标设备的水平分辨率,以每米像素为单位。应用程序可以使用此值从资源组中选择与当前设备的特征最匹配的位图。
因此,从资源加载位图时会使用这些设置。这里没有帮助。
知道此代码不在应用程序中运行可能会有所帮助。它将 HTML 呈现为用于打印的元文件,并且它存在于 Web API 网络服务中。
没有用户界面,所以我不确定如何解释它是否是 DPI Aware 的问题。证据表明它受到 DPI 的影响,因此这个问题可能是相关的。