整个 WPF 就是一个UI框架,一个 UI 框架最重要的是 交互 和 显示 部分,而书写这个功能将会完全贯穿 WPF 整个框架的功能。本文非入门级博客,本文包含了大量链接博客,阅读本文你将会了解从用户手指触摸屏幕到最终屏幕打印出笔迹的应用程序执行的步骤

本文实际内容不多,但是如果加上链接的博客,那么总内容将会非常多,还请小伙伴仔细阅读本文链接的博客

从软件的角度上,可以将触摸屏看成是一个软件制作的驱动组件,因此就可以规避复杂的硬件带来的问题。当前的触摸是有 HID 标准的,任何走标准 HID 设备的硬件设备,只要实现得对,咱上层软件是不需要关注硬件的细节的。更多有关协议部分请看 Windows 的 Pen 协议

规避了硬件设备,此时咱就不需要画精力去了解硬件设备的收集触摸点的机制,以及封装数据和系统的解包是如何做的

在 WPF 的触摸在系统最底层使用的是 RealTime Stylus 机制实现,这个机制能达到比 WM_Touch 触摸消息快非常多倍的接收速度,基本可以认为硬件设备发送到系统瞬间就到应用程序上,中间过程仅有发生几次锁和读取内存数据的时间。从 RealTime Stylus 到 WPF 框架经过 PenIMC 模块,请看 WPF 触摸底层 PenImc 是如何工作的

而 WPF 尽管可以在 Stylus Input 线程使用 PenThreadWorker 通过 RealTime Stylus 机制快速获取触摸点,但是 WPF 为了让业务逻辑更好实现,此时将会在在主线程触发 Touch 或 Stylus 事件。因此如果监听 Touch 等这些事件,那么将需要等待线程切换和等待主线程忙碌。因此高性能的笔迹实现推荐通过 StylusPlugIn 的方法,在触摸线程获取触摸点,详细请看 WPF 高速书写 StylusPlugIn 原理

从 WPF 使用 PenIMC 在 WISPTIS 服务获取 RealTime Stylus 到 StylusPlugIn 收到消息或在 Touch 等事件收到消息,请看 WPF 触摸到事件

这就是需要涉及整个 WPF 的命中测试以及触摸输入机制,这也就是从手指触摸到屏幕到 WPF 框架将信息给业务层的步骤。还请小伙伴阅读本文的链接博客,本文接下来来和小伙伴聊聊下半部分的逻辑

在业务层收到了触摸的信息,如何转换为笔迹对象?首先笔迹的本质绘制就是将输入的离散的点,绘制成为 Geometry 几何加入到 WPF 的渲染中

在 WPF 中提供了 Stroke 类用于协助以上计算,通过给 Stroke 输入离散的点,可以通过调用 Stroke 的 Draw 方法,向某个 DrawingContext 绘制出 Geometry 的笔迹内容。这部分的逻辑很简单,请看 WPF 最简逻辑实现多指顺滑的笔迹书写

在绘制到某个 Visual 里面之后,需要将 Visual 加入到 WPF 的视觉树中,在 WPF 的渲染机制里面,将会依据视觉树上的元素的更改刷新视觉树的渲染内容。这部分细节请看 WPF 渲染原理

而此时离屏幕渲染依然还有一段路线,在 WPF 通过 MIL 层,给出 Geometry 的绘制原语之后,将会和 WPF 界面的其他元素,如按钮文字等等在渲染线程合成为 DirectX 渲染图元,交给 DirectX 底层执行绘制。而其实在进行输出渲染图元这个步骤就算是进入了 DirectX 渲染管线部分,后续渲染管线的工作请看 细说图形学渲染管线

但事实上不是 WPF 将绘制原语准备好之后,就会调用 绘制调用 Draw Call 指令,显卡就会进行工作,为了让整体效率最高,系统层或者说 DirectX 将会打包多个 Draw call 指令,一次交给 GPU 去渲染

而经过了渲染管线之后是否就能在屏幕上实际显示?其实不然,还需要经过 DWM 桌面窗口管理器的调度,将多个窗口的画面合成之后在交给显卡的缓冲区,等待屏幕刷新

这就是整个的步骤

从这个步骤了解上,可以理解 高性能笔迹原理 的内容

那开发者端能控制的部分包括哪些?首先是获取触摸的逻辑,可以选择从上层的 Touch 或 Stylus 事件获取触摸消息,也可以选择从底层的 StylusPlugIn 获取,当然也可以选择禁用实时触摸通过 WM_Touch 消息获取触摸。不过采用 WM_Touch 就需要使用 WPF 模拟触摸设备 的方法

如果硬件触摸框是可以定制的,那么也可以通过 USB 读取 HID 的方式拿到触摸框原始信息

其次就是如何绘制笔迹的方式,此时可以利用 WPF 框架提供的笔迹绘制算法,在完全开源的 WPF 框架里面,可以看到有一个大文件夹很多代码用来实现一个看起来比较顺滑的笔迹。小伙伴也可以去抄 WPF 的源代码自己魔改

然后笔迹的绘制方式基本上可以选 Geometry 或 Image 的方式,加入到视觉树中,或者重绘已有位图的方式

接着在进入 DX 渲染管线部分,可以使用 WPF 的 Effect 机制,通过 HLSL 对画面显示进行优化,这部分属于像素着色器的知识

更多触摸请看 WPF 触摸相关


本文会经常更新,请阅读原文: https://blog.lindexi.com/post/WPF-%E5%BA%95%E5%B1%82-%E4%BB%8E%E6%89%8B%E6%8C%87%E8%A7%A6%E6%91%B8%E5%B1%8F%E5%B9%95%E5%88%B0%E7%AC%94%E8%BF%B9%E5%9C%A8%E5%B1%8F%E5%B9%95%E6%98%BE%E7%A4%BA%E4%B8%AD%E9%97%B4%E7%9A%84%E6%AD%A5%E9%AA%A4.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者收藏我的博客导航

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

微软最具价值专家


无盈利,不卖课,做纯粹的技术博客

以下是广告时间

推荐关注 Edi.Wang 的公众号

欢迎进入 Eleven 老师组建的 .NET 社区

以上广告全是友情推广,无盈利