[WinUI 3] 如何利用D3D11在SwapChainPanel控件上绘制OpenGL(Uwp通用)
预览



技术实现
看过我上篇在 WPF 中实现 OpenGL 与 D3D 渲染的同学应该知道,我是依靠 WGL 中 WGL_NV_DX_interop 扩展与 D3D Surface 关联并在使用该 Surface 实现渲染。
所以我们这次实现也是如此,但与 WPF 不同的是 WinUI 支持 D3D11 并在控件中提供 SwapChainPanel 作为载体。(WPF 中为 D3DImage)
代码部分使用 Silk.Net 和 OpenTK 实现。
有一些关键部分,接下来我会单独讲下。
WGL_NV_DX_interop
WGL_NV_DX_interop是NVIDIA公司提出的一套扩展API(支持OpenGL 2.1及以上版本),该扩展允许 OpenGL 可以直接访问 DirectX 缓冲区与 Surface,并作为 OpenGL 共享纹理或渲染缓冲区对象使用。
扩展支持版本
WGL_NV_DX_interop - Direct3D 9
WGL_NV_DX_interop2 - Direct3D 10、11
D3D11与SwapChainPanel
SwapChainPanel 是用于支持高性能图形和游戏的 Windows 运行时类型,你可以在其中直接管理交换链。
在此情况下,你可以创建自己的 DirectX 交换链并管理所呈现内容的显示。
需要注意的是,为了确保良好的性能 SwapChainPanel 存在一些限制。
- 一个程序中 SwapChainPanel 实例不能超过四个。
- 关联的 DirectX 交换链应要明确设置宽高。
- DirectX 交换链的缩放模式必须设置为 DXGI_SCALING_STRETCH。
- 只能通过 DXGI 中 CreateSwapChainForComposition 函数进行交换链创建。
代码实现 (只存在关键代码,完整代码在文章最下方 GitHub 链接)
D3D
- 创建 DXGI 工厂,用于后续创建 SwapChain(交换链)
IDXGIFactory2* factory;
// 获取该类型 Guid,这跟Com组件有关,本文不多阐述。
// GetTypeInfo 为扩展函数,存在 WinRT 命名空间。
Guid guid = typeof(IDXGIFactory2).GetTypeInfo().GUID;
DXGI.GetApi().CreateDXGIFactory2(0, &guid, (void**)&factory);
- 创建设备
ID3D11Device* device;
ID3D11DeviceContext* devCtx;
// 创建设备
D3D11.GetApi().CreateDevice(null, D3DDriverType.Hardware, 0, 0, null, 0, D3D11.SdkVersion, &device, null, &devCtx);
- 创建 SwapChain (使用 IDXGIFactory2 创建)
IDXGISwapChain1* swapChain;
// SwapChain 配置信息。
SwapChainDesc1 swapChainDesc = new()
{
Width = 宽度,
Height = 高度,
Format = Format.FormatB8G8R8A8Unorm,
Stereo = 0,
SampleDesc = new SampleDesc()
{
Count = 1,
Quality = 0
},
BufferUsage = DXGI.UsageRenderTargetOutput,
BufferCount = 2,
Scaling = Scaling.Stretch,
SwapEffect = SwapEffect.FlipSequential,
Flags = 0,
};
factory->CreateSwapChainForComposition((IUnknown*)device, &swapChainDesc, null, &swapChain);
- 获取 SwapChain 缓冲区(我们需要用它来创建 OpenGL 的渲染缓冲区)
ID3D11Texture2D* colorbuffer;
Guid guid = typeof(ID3D11Texture2D).GetTypeInfo().GUID;
swapChain->GetBuffer(0, &guid, (void**)&colorbuffer);
OpenGL
- 创建 GL 设备与帧缓冲区
nint glDeviceHandle = Wgl.DXOpenDeviceNV((IntPtr)device);
// 创建一个帧缓冲区,后续渲染前需要先进行绑定。
int gLFramebufferHandle = GL.GenFramebuffer();
- 关联 SwapChain 缓冲区
// 创建一个渲染缓冲区。
int gLColorRenderbufferHandle = GL.GenRenderbuffer();
// 关联 SwapChain 缓冲区到指定 Renderbuffer 上。
nint dxInteropColorHandle = Wgl.DXRegisterObjectNV(device, (nint)colorbuffer, (uint)gLColorRenderbufferHandle, (uint)RenderbufferTarget.Renderbuffer, WGL_NV_DX_interop.AccessReadWrite);
- 渲染
// 锁定缓冲区进行操作
Wgl.DXLockObjectsNV(glDeviceHandle, 1, new[] { dxInteropColorHandle });
// 绑定帧缓冲区
GL.BindFramebuffer(FramebufferTarget.Framebuffer, gLFramebufferHandle);
// 重置视口
GL.Viewport(0, 0, 宽度, 高度);
// GL 绘制代码
// 结束绘制
GL.BindFramebuffer(FramebufferTarget.Framebuffer, 0);
Wgl.DXUnlockObjectsNV(glDeviceHandle, 1, new[] { dxInteropColorHandle });
// 注意:一定要取消注册并删除缓冲区,不然 swapChain 读取不到数据。
Wgl.DXUnregisterObjectNV(glDeviceHandle, dxInteropColorHandle);
GL.DeleteRenderbuffer(gLColorRenderbufferHandle);
swapChain->Present(1, 0);
关联 WinUI
- 创建 ISwapChainPanelNative 接口类
[ComImport]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
[Guid("63aad0b8-7c24-40ff-85a8-640d944cc325")]
public interface ISwapChainPanelNative
{
[PreserveSig] HResult SetSwapChain([In] IntPtr swapChain);
[PreserveSig] ulong Release();
}
- 与 SwapChain 关联
SwapChainPanel swapChainPanel = new SwapChainPanel();
swapChainPanel.As<ISwapChainPanelNative>().SetSwapChain((IntPtr)swapChain)
基本上,一次的渲染流程到这里就结束了。
但有很多问题需要解决,比如修改控件宽高,重新注册缓冲区等等。。。
完整代码我会放到 GitHub 上,需要的同学可以下载看看,里面包含 WPF、WinUI 的 3D Demo,并解决了刚才说的一些问题。
WPF、WinUI 使用 Silk.Net 绘制示例(OpenGL、DirectX)
[WinUI 3] 如何利用D3D11在SwapChainPanel控件上绘制OpenGL(Uwp通用)的更多相关文章
- 如何在双向绑定的Image控件上绘制自定义标记(wpf)
我们的需求是什么? 答:需要在图片上增加一些自定义标记,例如:2个图片对比时,对相同区域进行高亮. 先上效果图: 设计思路 1.概述 1.通过TargeUpdated事件,重新绘制图片进行替换. 2. ...
- c#在pictureBox控件上绘制多个矩形框及删除绘制的矩形框
在pictureBox上每次只绘制一个矩形框,绘制下一个矩形框时上次绘制的矩形框取消,代码如链接:https://www.cnblogs.com/luxiao/p/5625196.html 在绘制矩形 ...
- 利用foreach对页面控件的遍历 及三目运算符的使用
1.利用foreach对页面控件的遍历 及三目运算符的使用 利用div将一组CheckBox放在一起用于遍历 <body> <form id="form1" ru ...
- 详解如何利用FarPoint Spread表格控件来构造Winform的Excel表格界面输入
我们先来简单了解一下WinForm和FarPoint,WinForm是·Net开发平台中对Windows Form的一种称谓.而FarPoint是一款模拟EXCEL的控件.它可以根据用户的要求实现很大 ...
- 在GridControl控件上绑定图片的几种操作方式
我们知道,基于DevExpress的开发Winform的项目界面的时候,GridControl控件是经常用来绑定数据的,一般以常规的字符内容为主,有时候也会有图片的显示需要,那么如果显示图片,我们应该 ...
- 对话框上动态控件的创建、在Picture Control控件上显示图片
1 MFC对话框之上的动态控件的创建 对话框上的控件是MFC类的一个具体对象. 当在对话框之上使用静态控件时,可以根据类向导来为每个控件添加消息.响应函数以及变量. 当需要在对话框中动态的创建某个控 ...
- Windows 程序设计(4) MFC-02 基本控件-上
1. Button 按钮控件 1.1.按钮控件的基本使用 新建对话框工程,拖拽按钮控件,添加点击事件响应函数! a.双击模版进行添加: b.事件方式进行添加: button的常见事件类型 void C ...
- 在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke
今天关闭一个窗体,报出这样的一个错误"在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke.",这个不用多想,肯定是那个地方没有释放掉.既然碰到这个问题, ...
- [MFC] MFC 打开HTML资源(用ID版,也可加载到自己的web控件上)
@ ^ @:如果是加载到web控件上,就把注释掉的解除注释(改为web控件点后面的函数),把下一句注释 BOOL Button::LoadFromResource(UINT nRes){//打开网页加 ...
- Cocos2dx 把 glview 渲染到 Qt 控件上(Mac 环境)
本文原链接:http://www.cnblogs.com/zouzf/p/4423256.html 环境:Mac 10.9.2 Xcode5.1.1 Qt5.3 cocos2dx-2.2.4 ...
随机推荐
- [2001年NOIP普及组] 求先序排列
给出一棵二叉树的中序与后序排列.求出它的先序排列.(约定树结点用不同的大写字母表示,长度<=8). 输入 第一行输入一个字符串表示二叉树的中序排列,第二行输入一个字符串表示二叉树的后序排列. 输 ...
- 一种基于Modbus的工业通信网关设计
近年来,随着工业自动化领域的发展,工业现场对网络的可靠性及成本有极高的要求.传统基于串口的工业网关可以满足工业现场的应用,但却要付出高额成本.一种基于 ModBus 设计的工业通信网关就走进人们的眼中 ...
- 法拉第未来任命新CFO!贾跃亭激动发声
近段时间以来,贾跃亭旗下的的法拉第未来(Faraday Future,简称 FF)可谓是动作频频. 一天前,有媒体报道称,FF 任命 Zvi Glasman 为其首席财务官.其将负责公司财务.投资者关 ...
- 在VUE里实现一个简单的中国地图
如何在vue里面实现一个简单的中国地图,并且实现一些简单的个性化修改. 下面是最终实现的效果图.透明的地图加一个背景图. 1.在你的项目里安装echarts的依赖 npm install echart ...
- c++学习6 指针变量
一 指针变量的定义 *是用来修饰指针变量的,通常情况下我们定义的手法都是"类型名"+"*"+"指针变量名称". 有一种简单无脑的" ...
- zxb2022习题班16
(1) 原则:合同中包含多项履约义务的,企业应当按照各单项履约义务所承诺的商品的单独售价的比例,将交易价格分摊至各单项履约义务. 方法:按照A和B商品单独售价的相对比例, 2x22年4月16日 借:合 ...
- 用python提取txt文件中的特定信息并写入Excel
这个是用 excel里面的 去掉空格最后导出的一个list: 原本是有空格的 后面是抵消了中间的空格. 然后 这里侧重说一下什么是split()函数 语法:str.split(str="&q ...
- base64与中文字符串互转
实现代码如下 // 字符串转base64 getEncode64(str){ return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g ...
- Springboot项目记录1配置环境
一.电脑商城项目: 项目功能:登录,注册,热销商品,用户管理(密码,个人信息,头像,收货地址).购物车(展示.增加.删除).订单模块. 二.开发顺序: 注册.登录.用户管理.购物车.商品.订单模块 三 ...
- pytorch 简简单单求个值
能用张量处理就用张量,不要使用for in 跑循环,一个是容易出错,一个是比较浪费时间,应用广播机制的话去做很容易的 1.5 使用mean处理平均值 2. 在处理梯度的时候无法更改自身,因此使用的办法 ...