WPF怎样使用RenderTargetBitmap
发布网友
发布时间:2022-04-22 10:30
我来回答
共1个回答
热心网友
时间:2023-10-10 19:12
RenderTargetBitmap把视觉树中的一部分光栅化以位图的形式保存,你可以利用下面的这个类以位图的形式呈现视觉元素,当然得考虑系统DPI设置及视觉元素的转换。(感谢Adam Smith在反转换这方面的建议)
public class VisualUtility
{
public static BitmapSource CreateBitmapFromVisual(Double width,
Double height,
Visual visualToRender,
Boolean undoTransformation)
{
if (visualToRender == null)
{
return null;
}
// The PixelsPerInch()方法用于读取屏幕上DPI的设置
//如果你想以特定的分辨率创建一个位图,你可以直接把某一dpiX或dpiY传参给RenderTargetBitmap构造器。
RenderTargetBitmap bmp = new RenderTargetBitmap((Int32)Math.Ceiling(width),
(Int32)Math.Ceiling(height),
(Double)DeviceHelper.PixelsPerInch(Orientation.Horizontal),
(Double)DeviceHelper.PixelsPerInch(Orientation.Vertical),
PixelFormats.Pbgra32);
//如果我们想反转换,我们可以使用VisualBrush
if (undoTransformation)
{
DrawingVisual dv = new DrawingVisual();
using (DrawingContext dc = dv.RenderOpen())
{
VisualBrush vb = new VisualBrush(visualToRender);
dc.DrawRectangle(vb, null, new Rect(new Point(), new Size(width, height)));
}
bmp.Render(dv);
}
else
{
bmp.Render(visualToRender);
}
return bmp;
}
}
internal class DeviceHelper
{
public static Int32 PixelsPerInch(Orientation orientation)
{
Int32 capIndex = (orientation == Orientation.Horizontal) ? 0x58 : 90;
using (DCSafeHandle handle = UnsafeNativeMethods.CreateDC("DISPLAY"))
{
return (handle.IsInvalid ? 0x60 : UnsafeNativeMethods.GetDeviceCaps(handle, capIndex));
}
}
}
internal sealed class DCSafeHandle : SafeHandleZeroOrMinusOneIsInvalid
{
private DCSafeHandle() : base(true) { }
protected override Boolean ReleaseHandle()
{
return UnsafeNativeMethods.DeleteDC(base.handle);
}
}
[SuppressUnmanagedCodeSecurity]
internal static class UnsafeNativeMethods
{
[DllImport("gdi32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
public static extern Boolean DeleteDC(IntPtr hDC);
[DllImport("gdi32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
public static extern Int32 GetDeviceCaps(DCSafeHandle hDC, Int32 nIndex);
[DllImport("gdi32.dll", EntryPoint = "CreateDC", CharSet = CharSet.Auto)]
public static extern DCSafeHandle IntCreateDC(String lpszDriver,
String lpszDeviceName, String lpszOutput, IntPtr devMode);
public static DCSafeHandle CreateDC(String lpszDriver)
{
return UnsafeNativeMethods.IntCreateDC(lpszDriver, null, null, IntPtr.Zero);
}
}
之所以你需要反转换是因为如果你需要光栅化成RenderTargetBitmap的视觉目标元素已经是转换过的话(比如旋转,比例缩放,或是平移之类),这些效果会对最终产生位图有影响,这可能不是你想要的结果。“反转换”参数不仅使你能够实现反转换,而且可以得到原始的未经过转换的视觉元素呈现。
在利用RenderTargetBitmap时的一些*你也应该清楚:首先,RenderTargetBitmap不会利用硬件加速,位图是完全在内存中产生的,并且整个过程也是在UI线程中实现的。其次,字体的显示会显示锯齿效果,而不是清晰的呈现线条样的效果。