title author date CreateTime categories
WPF 使用 Direct2D1 画图入门
lindexi
2018-08-10 19:16:53 +0800
2018-04-18 20:46:45 +0800
WPF D2D DirectX 渲染

本文来告诉大家如何在 WPF 使用 D2D 画图。

本文是一个系列

什么是 D2D

实际上现在很多小伙伴对于渲染性能就是听到 DirectX 才会去搜索这个博客。我在博客园看到很少的博客讲到这个。即使有也很少会说如何使用 WPF 的。

那么 D2D 是一个提高性能的方法,具体是怎么做?基于 Direct3D 可以使用硬件加速的功能,即使在没有显卡,进行软渲染的性能也是比 GDI 快,但是渲染静态的还是建议使用 GDI。

现在的 WPF 底层使用的渲染是 Dx9 渲染 或使用 Dx11 Dx12 优化 fl9 渲染,所以性能实际上和直接使用 D2D 是差不多,但是 WPF 没有充分使用DX,所以如果自己写的性能会比较高。

因为 WPF 渲染使用的是 Dx9 或虽然使用了 Dx11 Dx12 但是优化是 fl9 ,所以在现在很多设备无法使用全部的性能。

Direct2D运行需求

这是我从大神的博客看到,如果需要运行 Direct2D 那么就需要在 win7 之后才可以。所以在现在几乎可以直接运行,很少有人会使用 win7 以下的设备。

安装

下面是我从 Nuget 安装,这个是很老的库,不太建议大家使用。

Nuget 搜索 WindowsAPICodePackDirectX 就可以安装,如果不知道安装哪个,请点击WindowsAPICodePackDirectX

这个库只是 x64 的库,所以想要运行还需要设置环境。

环境

如果直接使用这个库是无法运行,下面的代码只是作为大家快速入门,不能用于产品。安装这个库可以用在 x64 的进程,但是不能用在 x86 进程。

而且这个库不能直接在 dot net framework 4.5 的环境运行,需要创建 App.config 文件添加下面代码。需要注意,请修改创建项目使用 dot net framework 4.5 而不是更高的版本。

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
<startup useLegacyV2RuntimeActivationPolicy="true">
<supportedRuntime version="v4.0" sku = ".NETFramework,Version=v4.5"/>
<supportedRuntime version="v2.0.50727"/>
</startup>
</configuration>

那么如何让软件使用 x64 进程?尝试右击项目点击属性,在生成页面就可以看到平台目标,选择 x64 就会编译 x64 的软件。

如果对于平台目标感觉有兴趣,请看WPF 编译为 AnyCPU 和 x86 有什么区别

创建工厂

首先打开 MainPage 的代码,添加下面代码

using D2D = Microsoft.WindowsAPICodePack.DirectX.Direct2D1;

这样下面就不需要写那么多代码,因为所有使用Microsoft.WindowsAPICodePack.DirectX.Direct2D1的都可以使用 D2D 来找到,这样下面的代码大家直接复制就可以运行。

在使用 Direct2D1 的第一步就是创建工厂。

虽然工厂有很多重载,不过这里不会告诉大家,因为只是快速入门,如果需要知道参数的意思就请自己多看文章。

       public MainWindow()
{
InitializeComponent(); Loaded += (s, e) =>
{
var d2DFactory = D2D.D2DFactory.CreateFactory(D2D.D2DFactoryType.Multithreaded);
};
}

把代码写在 Loaded 是因为下面需要拿到窗口。

获得窗口

从上面代码大家也许会说为什么需要在 Load 才写,因为需要拿到窗口,在 Load 之后拿才不会是空。窗口创建虽然不是只在 Loaded 拿才可以,不过为了代码比较简单,于是写在 Loaded ,这时拿到一般就是可以使用。

使用下面代码就可以拿到窗口

                var windowHandle = new WindowInteropHelper(this).Handle;

渲染目标

最主要就是创建 RenderTarget ,请使用下面代码

var renderTarget = d2DFactory.CreateHwndRenderTarget(new D2D.RenderTargetProperties(),
new D2D.HwndRenderTargetProperties(windowHandle,
new D2D.SizeU((uint) ActualWidth, (uint) ActualHeight),
D2D.PresentOptions.RetainContents));

实际上上面的代码不可以用在实际项目,因为写入的大小,如果窗口修改了大小,那么显示的就一般不是期望。

在渲染的时候还需要使用 RenderTarget ,先创建一个字段保存

    public MainWindow()
{
InitializeComponent(); Loaded += (s, e) =>
{
var d2DFactory = D2D.D2DFactory.CreateFactory(D2D.D2DFactoryType.Multithreaded); var windowHandle = new WindowInteropHelper(this).Handle;
var renderTarget = d2DFactory.CreateHwndRenderTarget(new D2D.RenderTargetProperties(),
new D2D.HwndRenderTargetProperties(windowHandle,
new D2D.SizeU((uint) ActualWidth, (uint) ActualHeight),
D2D.PresentOptions.RetainContents)); _renderTarget = renderTarget;
};
} private D2D.RenderTarget _renderTarget;

写线段

上面说了主要就是拿 RenderTarget ,因为拿到 RenderTarget 就和拿到 DrawContext 一样,自己尝试点一下 RenderTarget 就可以看到很多画图的方法,在里面画图的性能很高。

那么尝试对RenderTarget写入线段

因为需要知道在什么时候才进行渲染,所以先添加下面代码。在 CompositionTarget 拿到渲染就是一个耗性能的过程,但是为了让 DX 渲染和 WPF 时间一样,所以需要在这个事件进行渲染。主要不要让这个方法执行时间比较长,除了画出来就不要做其他的。

            CompositionTarget.Rendering += OnRendering;

想要画出一条线,需要知道线的两个点,和线的颜色,宽度。但是在 RenderTarget 传入线的样式需要使用下面的方法。注意传入的值是 ColorF 而且三个值都是[0,1],所以对普通的颜色传入需要计算。

创建笔刷的方法

            var brush = _renderTarget.CreateSolidColorBrush(new D2D.ColorF(0, 1, 0));

进行渲染

            if (_renderTarget == null)
{
return;
} _renderTarget.BeginDraw();
var brush = _renderTarget.CreateSolidColorBrush(new D2D.ColorF(0, 1, 0)); _renderTarget.DrawLine(new D2D.Point2F(10, 10), new D2D.Point2F(100, 100), brush, 10); _renderTarget.EndDraw();

渲染需要先 BeginDraw 然后画出,最后调用 EndDraw 画出来。注意,如果运行看不到画出的,那么请先看一下是不是调了多次 BeginDraw 没有匹配 EndDraw 。

尝试运行就可以看到下面界面

这时看一下 CPU ,几乎不需要。

下面来做很小修改,写出一个会动的图,下面的代码放在最后。

Direct2D教程I——简介及首个例子 - 万仓一黍 - 博客园

所有代码:WPF Direct2D 入门-CSDN下载

       public MainWindow()
{
InitializeComponent(); CompositionTarget.Rendering += OnRendering; Loaded += (s, e) =>
{
var d2DFactory = D2D.D2DFactory.CreateFactory(D2D.D2DFactoryType.Multithreaded); var windowHandle = new WindowInteropHelper(this).Handle;
var renderTarget = d2DFactory.CreateHwndRenderTarget(new D2D.RenderTargetProperties(),
new D2D.HwndRenderTargetProperties(windowHandle,
new D2D.SizeU((uint) ActualWidth, (uint) ActualHeight),
D2D.PresentOptions.RetainContents)); _redBrush = renderTarget.CreateSolidColorBrush(new D2D.ColorF(1, 0, 0, 1)); _greenBrush = renderTarget.CreateSolidColorBrush(new D2D.ColorF(0, 1, 0, 1)); _blueBrush = renderTarget.CreateSolidColorBrush(new D2D.ColorF(0, 0, 1, 1)); _renderTarget = renderTarget;
};
} private D2D.RenderTarget _renderTarget; private D2D.SolidColorBrush _redBrush; private D2D.SolidColorBrush _greenBrush; private D2D.SolidColorBrush _blueBrush; private void OnRendering(object sender, EventArgs e)
{
if (_renderTarget == null)
{
return;
} D2D.SolidColorBrush brush = null; switch (ran.Next(3))
{
case 0:
brush = _redBrush;
break;
case 1:
brush = _greenBrush;
break;
case 2:
brush = _blueBrush;
break;
} _renderTarget.BeginDraw();
_renderTarget.DrawRectangle(new D2D.RectF(_x, _y, _x + 10, _y + 10), brush, 1);
_renderTarget.EndDraw(); _x = _x + _dx;
_y = _y + _dy;
if (_x >= ActualWidth - 100 || _x <= 0)
{
_dx = -_dx;
} if (_y >= ActualHeight - 100 || _y <= 0)
{
_dy = -_dy;
}
} private float _dx = 1;
private float _dy = 1;
private float _x;
private float _y; private Random ran = new Random();

更多博客

为何使用 DirectComposition

C++ 的 Direct2D 请看 Direct2D

2018-8-10-WPF-使用-Direct2D1-画图入门的更多相关文章

  1. WPF 使用 Direct2D1 画图入门

    本文来告诉大家如何在 WPF 使用 D2D 画图. 本文是一个系列 WPF 使用 Direct2D1 画图入门 WPF 使用 Direct2D1 画图 绘制基本图形 WPF 使用 SharpDX WP ...

  2. WPF 使用 Direct2D1 画图 绘制基本图形

    本文来告诉大家如何在 Direct2D1 绘制基本图形,包括线段.矩形.椭圆 本文是一个系列 WPF 使用 Direct2D1 画图入门 WPF 使用 Direct2D1 画图 绘制基本图形 本文的组 ...

  3. WPF 像素着色器入门:使用 Shazzam Shader Editor 编写 HLSL 像素着色器代码

    原文:WPF 像素着色器入门:使用 Shazzam Shader Editor 编写 HLSL 像素着色器代码 HLSL,High Level Shader Language,高级着色器语言,是 Di ...

  4. 不用搭环境的10分钟AngularJS指令简易入门01(含例子)

    不用搭环境的10分钟AngularJS指令简易入门01(含例子) `#不用搭环境系列AngularJS教程01,前端新手也可以轻松入坑~阅读本文大概需要10分钟~` AngularJS的指令是一大特色 ...

  5. 申请Office 365一年免费的开发者账号攻略(2018年10月份版本)

    要进行Office 365开发,当然需要有完整的Office 365环境才可以.为了便于广大开发人员快速地启动这项工作,微软官方给所有开发人员提供了免费的一年开发者账号   那么如何申请Office ...

  6. IntelliJ IDEA 最新激活码(截止到2018年10月14日)

    IntelliJ IDEA 注册码: EB101IWSWD-eyJsaWNlbnNlSWQiOiJFQjEwMUlXU1dEIiwibGljZW5zZWVOYW1lIjoibGFuIHl1IiwiYX ...

  7. 新手C#SQL Server使用记录2018.08.10

    主键(PrimaryKey):主键就是每个数据行(记录)的唯一标识,不会有重复值的列(字段)才能当做主键.一个表可以没有主键,但是这样会很难处理表,因此一般情况表都要设置主键. 主键有两张选用策略,分 ...

  8. Prism(WPF) 拐着尝试入门

    原文:Prism(WPF) 拐着尝试入门 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/s261676224/article/details/852 ...

  9. HTML5 Canvas 画图入门

    HTML5 Canvas 画图入门 HTML5 Canvas 画图入门,仅供学习參考 <!DOCTYPE html> <html> <head> <meta ...

  10. 01 mybatis框架整体概况(2018.7.10)-

    01 mybatis框架整体概况(2018.7.10)- F:\廖雪峰 JavaEE 企业级分布式高级架构师课程\廖雪峰JavaEE一期\第一课(2018.7.10) maven用的是3.39的版本 ...

随机推荐

  1. JavaScript实现防抖与节流

    1. 引言 有这么一种场景:某个页面表单按钮设置了点击提交事件,有时因为网络不好,点击后后台服务端很久才返回信息,然而用户因等待许久已经多次点击导致多次发送数据,实际上服务器只需要一次发送的数据即可 ...

  2. 记录--使用率比较低的10个Web API

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 avaScript中有些API可能使用率比较低,下面我们逐一介绍它们的用法和使用场景. 至于标题,主要是想让你进来看看,兄弟们别打我! B ...

  3. 《.NET内存管理宝典 》(Pro .NET Memory Management) 阅读指南 - 第4章

    本章勘误: 暂无,等待细心的你告诉我哦. 本章注解: 暂无 本章释疑: 暂无,等待你的提问 致谢: MVP 林德熙 MVP 吕毅 sPhinX 相关链接 试读记录

  4. 替换Spring容器中已经存在的Bean

    一.背景 我们在开发的过程中,经常会引入别人写的jar包实现某些功能.而别人的jar包一般都自动注入Spring容器中,假设别人都是通过@Bean或@Component注入的,并且没有加入@Condi ...

  5. 基于SCCB协议的FPGA实现

    SCCB协议 1.协议内容 SCCB协议常用于vo系列的摄像头的寄存器配置中,是有IIC协议演变而来.本来,本人接触这个协议也是想配置摄像头用于摄像模块.但是,由于配置寄存器实在是太多,而且需要找的资 ...

  6. #线段树,树状数组#CodeChef Merciless Chef

    MLCHEF 分析 首先按照dfs序将子树转换为区间,其实就是区间减和区间维护最小值判断是否大于0 因为大于0一定最多只有 \(n\) 个,所以直接将一个数记录被删除并设为正无穷. 代码 #inclu ...

  7. #状压dp,贪心#CF1316E Team Building

    题目 为了组织一支排球队,你需要为队伍里的\(p\)个不同的位置,从\(n\)个人中选出\(p\)个人, 且每个位置上都恰好有一个人.另外还需要从剩下的人中选出恰好\(k\)个人作为观众. 对于第\( ...

  8. HMS Core上新啦!

    HMS Core上新啦!分析服务营销分析报告全新上线:运动健康服务支持目标场景事件订阅:音频编辑服务提供专业的三维声音频编辑与渲染能力,更多HMS Core能力可点击网页链接了解. 了解更多详情> ...

  9. 鸿蒙HarmonyOS实战-ArkUI组件(Video)

    一.Video 视频组件是用于应用程序中嵌入视频的一种方法.它可以让用户在网站上观看视频并与其进行交互.通常,视频组件将一个视频文件嵌入应用程序中,并提供一组控件,这些控件允许用户播放.暂停.跳过.音 ...

  10. Prometheus 性能调优-水平分片

    简介 之前笔者有连续 2 篇文章: Prometheus 性能调优 - 什么是高基数问题以及如何解决? 如何精简 Prometheus 的指标和存储占用 陆续介绍了一些 Prometheus 的性能调 ...