当前位置:网站首页>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。在我的设备上面的代码是可以运行,所以暂时不修改。
多谢暗影吉他手发现问题
边栏推荐
- Babbitt | Metaverse daily must-read: Weibo animation will recruit all kinds of virtual idols around the world and provide support for them...
- LVS负载均衡群集之原理叙述
- 数仓相关,总结
- 数据库SqlServer迁移PostgreSql实践
- Kubernetes入门到精通- Operator 模式入门
- #yyds干货盘点# 面试必刷TOP101:链表相加(二)
- Day018 继承
- gbase8s创建RANGE分片表
- Homework 8.3 Thread Synchronization Mutex Condition Variables
- EasyCVR如何通过接口调用设备录像的倍速回放?
猜你喜欢
报道称任天堂在2023年3月前不会推出任何新硬件产品
方法的重写
Documentary on Security Reinforcement of Network Range Monitoring System (1)—SSL/TLS Encrypted Transmission of Log Data
curl命令的那些事
Flask框架实现注册加密功能详解【Flask企业课学习】
BigDecimal 使用注意!!“别踩坑”
MMDetection 使用示例:从入门到出门
通配符SSL证书不支持多域名吗?
2019 Haidian District Youth Programming Challenge Activity Elementary Group Rematch Test Questions Detailed Answers
【STM32】入门(五):串口TTL、RS232、RS485
随机推荐
Scala104-Spark.sql的内置日期时间函数
Win10只读文件夹怎么删除
用Excel绘制统计图
Thrift installation configuration
Understanding of margin collapse and coincidence
Short-term reliability and economic evaluation of resilient microgrids under incentive-based demand response programs (Matlab code implementation)
网站设计师:Nicepage 4.15 Crack By Xacker
CIFAR发布《AI伦理的文化:研讨会报告》【附下载】
猜数字游戏
2019 Haidian District Youth Programming Challenge Activity Elementary Group Rematch Test Questions Detailed Answers
EuROC dataset format and related codes
股票开户广发证券,网上开户安全吗?
情绪的波动起伏
当前最快的实例分割模型:YOLACT 和 YOLACT++
gbase8s创建RANGE分片表
运力升级助力算力流转,中国数字经济的加速时刻
工业元宇宙对工业带来的改变
基于 eBPF 的 Kubernetes 可观测实践
mood swings
2018年南海区小学生程序设计竞赛详细答案