当前位置:网站首页>win10 uwp win2d 离屏渲染
win10 uwp win2d 离屏渲染
2022-08-04 18:37:00 【林德熙】
离屏渲染(Offscreen drawing)是一个不错的科技,在系统有空的时候,提前先画出部分界面。这样在需要直接渲染的时候就可以直接拿出来而不需要等待进行渲染的时候才画出来。
实际上之前我已经写过一篇博客关于 CanvasRenderTarget ,离屏渲染需要使用到 CanvasRenderTarget 。
先来写一个简单的界面,在写之前,请安装 Win2d 。
<Page
x:Class="SairkaDeebowhar.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:SairkaDeebowhar"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:win2d="using:Microsoft.Graphics.Canvas.UI.Xaml"
mc:Ignorable="d"
Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
<Grid>
<win2d:CanvasControl x:Name="CanvasControl" Draw="CanvasControl_OnDraw"></win2d:CanvasControl>
<Button Content="确定" HorizontalAlignment="Center" VerticalAlignment="Center" Click="Button_OnClick"></Button>
</Grid>
</Page>在点击确定的时候开始进行离屏渲染,因为如果需要让CanvasControl开始渲染需要调用Invalidate,但是调用这个函数不是立刻就渲染,而且在下一个无法控制的时间进行渲染。
所以在 按钮点击的时候可以进行离屏渲染,这时CanvasControl在渲染的时候直接拿按钮点击下去画的就可以。
离屏渲染也叫离线渲染,为了让用户看到炫酷的界面,要求渲染的时间小于 16 毫秒,也就是一秒钟至少60刷新。
渲染的时间是很有限的,如果在渲染中需要耦合计算,那么会让渲染的性能降低。
例如下面的代码,在按钮点击下去的时候,经过很长时间的计算才能知道写入我的名字,如果把代码放在CanvasControl画的时候计算,那么会影响性能。
建议的方法是在按钮点击的时候,在按钮点击那里做计算,这时CanvasControl还可以画其他的东西。
因为我写的是呆磨,很简单,没有让大家看到CanvasControl在空闲的时候做的事情。
private void Button_OnClick(object sender, RoutedEventArgs e)
{
CanvasDevice device = CanvasDevice.GetSharedDevice();
CanvasRenderTarget offscreen = new CanvasRenderTarget(device, width: 100, height: 100, dpi: 96);
using (CanvasDrawingSession ds = offscreen.CreateDrawingSession())
{
//经过很多时间的计算,才知道需要写我的名字
ds.DrawText("lindexi", new Vector2(10, 10), Colors.Black);
}
_offscreen = offscreen;
// 调用这个函数,会在无法控制的时间调用 CanvasControl_OnDraw ,这时可以发出已经画出来的
CanvasControl.Invalidate();
}
private CanvasRenderTarget _offscreen;在 CanvasControl_OnDraw 就判断 _offscreen 不是空就显示
private void CanvasControl_OnDraw(CanvasControl sender, CanvasDrawEventArgs args)
{
if (_offscreen != null)
{
args.DrawingSession.DrawImage(_offscreen,
new Rect(50, 50, _offscreen.Bounds.Width, _offscreen.Bounds.Height));
}
}为什么需要判断_offscreen不是空?因为在xaml加载的时候,就会触发CanvasControl_OnDraw,如果传入DrawImage是一个空,那么会出现参数异常。
通过离屏渲染的图可以使用特效,请看代码
private void Button_OnClick(object sender, RoutedEventArgs e)
{
CanvasDevice device = CanvasDevice.GetSharedDevice();
CanvasRenderTarget offscreen = new CanvasRenderTarget(device, width: 100, height: 100, dpi: 96);
using (CanvasDrawingSession ds = offscreen.CreateDrawingSession())
{
//经过很多时间的计算,才知道需要写我的名字
ds.DrawText("lindexi", new Vector2(10, 10), Colors.Black);
}
GaussianBlurEffect effect = new GaussianBlurEffect()
{
Source = offscreen,
BlurAmount = 1.0f
};
_offscreen = offscreen;
_effect = effect;
// 调用这个函数,会在无法控制的时间调用 CanvasControl_OnDraw ,这时可以发出已经画出来的
CanvasControl.Invalidate();
}
private CanvasRenderTarget _offscreen;
private GaussianBlurEffect _effect;这时尝试在Win2d显示效果
private void CanvasControl_OnDraw(CanvasControl sender, CanvasDrawEventArgs args)
{
if (_offscreen != null)
{
args.DrawingSession.DrawImage(_offscreen,
new Rect(50, 50, _offscreen.Bounds.Width, _offscreen.Bounds.Height));
}
if (_effect != null)
{
args.DrawingSession.DrawImage(_effect);
}
}需要注意,这时的特效的计算是在CanvasControl_OnDraw这时渲染才计算特效。
离线渲染的意思是可以在其他线程渲染,虽然渲染都是在GPU渲染,都是渲染是包括告诉如何渲染和把图片画出来,可以看到如何渲染就可能需要在CPU做很多计算。上面的代码,我经过很多计算才知道需要写 lindexi ,因为本渣叫金鱼,金鱼的意思就是记忆力很差,经常不知道自己叫什么。计算出名字需要的时间是很长的。
如果在按钮按下需要很长时间的计算,那么性能也是很差,这时建议在其他线程做。
private async void Button_OnClick(object sender, RoutedEventArgs e)
{
await Task.Run(() =>
{
// 下面代码可能在 CanvasControl_OnDraw 画出 CanvasRenderTarget 会出现 0x88990012 异常,解决方法请看文章最后
CanvasDevice device = CanvasDevice.GetSharedDevice();
CanvasRenderTarget offscreen = new CanvasRenderTarget(device, width: 100, height: 100, dpi: 96);
using (CanvasDrawingSession ds = offscreen.CreateDrawingSession())
{
//经过很多时间的计算,才知道需要写我的名字
ds.DrawText("lindexi", new Vector2(10, 10), Colors.Black);
}
GaussianBlurEffect effect = new GaussianBlurEffect()
{
Source = offscreen,
BlurAmount = 1.0f
};
_offscreen = offscreen;
_effect = effect;
});
// 调用这个函数,会在无法控制的时间调用 CanvasControl_OnDraw ,这时可以发出已经画出来的
CanvasControl.Invalidate();
}这就是离屏渲染的使用方法,在其他线程做耗时的渲染,在显示的时候可以快速画出
win2d CanvasRenderTarget vs CanvasBitmap
注意,暗影吉他手告诉我,在 Button_OnClick 的第一句话CanvasDevice device = CanvasDevice.GetSharedDevice();,使用 CanvasDevice.GetSharedDevice() 是错误写法。在CanvasControl_OnDraw里面使用用这个device创建的 CanvasRenderTarget 会弹 0x88990012 异常(Objects used together must be created from the same factory instance)。应该在CreateResources里面得到device。在我的设备上面的代码是可以运行,所以暂时不修改。
多谢暗影吉他手发现问题
边栏推荐
- 合宙Cat1 4G模块Air724UG配置RNDIS网卡或PPP拨号,通过RNDIS网卡使开发板上网(以RV1126/1109开发板为例)
- 路由技术
- 【软件工程之美 - 专栏笔记】37 | 遇到线上故障,你和高手的差距在哪里?
- Kubernetes入门到精通- Operator 模式入门
- 阿里云国际版使用ROS搭建WordPress教程
- 火灾报警联网FC18中CAN光端机常见问题解答和使用指导
- Global electronics demand slows: Samsung's Vietnam plant significantly reduces capacity
- "No title"
- 测试工程师如何突破职业瓶颈?
- Those things about the curl command
猜你喜欢

Hezhou Cat1 4G module Air724UG is configured with RNDIS network card or PPP dial-up, and the development board is connected to the Internet through the RNDIS network card (taking the RV1126/1109 devel

ACP-Cloud Computing By Wakin自用笔记(2)CPU和内存虚拟化

在线生成接口文档

EasyCVR本地接入国标设备映射公网后,本地设备出现无法播放与级联的解决方法

基于激励的需求响应计划下弹性微电网的短期可靠性和经济性评估(Matlab代码实现)

在表格数据集上训练变分自编码器 (VAE)示例

How does EasyCVR call the double-speed playback of device recording through the interface?
[Distributed Advanced] Let's fill in those pits in Redis distributed locks.

袋鼠云思枢:数驹DTengine,助力企业构建高效的流批一体数据湖计算平台

当前最快的实例分割模型:YOLACT 和 YOLACT++
随机推荐
EasyCVR calls the cloud recording API and returns an error and no recording file is generated. What is the reason?
ECCV 2022 | FPN错位对齐,实现高效半监督目标检测(PseCo)
powershell和cmd对比
Google AppSheet: 无需编程构建零代码应用
flink-cdc支持并行读取一张mysql表的binlog不?
PHP代码审计8—SSRF 漏洞
The upgrade of capacity helps the flow of computing power, the acceleration moment of China's digital economy
DHCP&OSPF combined experimental demonstration (Huawei routing and switching equipment configuration)
自己经常使用的三种调试:Pycharm、Vscode、pdb调试
浅谈web网站架构演变过程
数仓相关,总结
什么是网站监控,网站监控软件有什么用?
【杰神说说】物联大师2.0版本预告
防火墙基础之防火墙做出口设备安全防护
MySQL安装教程(详细)
智能视频监控平台EasyCVR如何使用接口批量导出iframe地址?
作业8.3 线程同步互斥机制条件变量
如何进行自动化测试?
测试/开发程序员男都秃头?女都满脸痘痘?过好我们“短暂“的一生......
DOM Clobbering的原理及应用