热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

shader反射水面_【博物纳新】水面涟漪反射效果开源库测评

【博物纳新】是UWA重磅推出的全新栏目,旨在为开发者推荐新颖、易用、有趣的开源项目,帮助大家在项目研发之余发现世界上的热门项目、前沿技术或者令人惊叹的视

【博物纳新】是UWA重磅推出的全新栏目,旨在为开发者推荐新颖、易用、有趣的开源项目,帮助大家在项目研发之余发现世界上的热门项目、前沿技术或者令人惊叹的视觉效果,并探索将其应用到自己项目的可行性。很多时候,我们并不知道自己想要什么,直到某一天我们遇到了它。

大千世界,无奇不有!更多精彩内容尽在开源库:lab.uwa4d.com

导读

WaveProvider是一个组件和着色器库,可以使用Unity轻松表达波形,利用该组件可以实现受物理影响的水面动态涟漪效果,

效果展示

实现原理

1. 添加用于绘制倒影图像的反射相机

考虑到主摄像机的位置可能会动态地改变,所以反射摄像机也需要使用脚本控制使其动态移动到合适位置(与主相机相对水面对称)以绘制正确的反射图像。

2. 根据投影面编写一个水面投影反射的着色器

投影时,需要统一投影侧的空间并确定投影面(项目中的这个投影面就是水面)。使用反射相机的变换矩阵变换,并渲染每个对象。当使用主摄像机在水面上绘制对象时,使用反射相机的变换矩阵来转换对象顶点。

下面的代码段是水面投影反射着色器的一部分:

1 struct v2f

2 {

3 float2 uv : TEXCOORD0;

4 float4 vertex : SV_POSITION;

5 float4 ref : TEXCOORD1;

6 };

7

8 sampler2D _MainTex;

9 float4 _MainTex_ST;

10 sampler2D _RefTex; //反射绘制的纹理

11 float4x4 _RefVP; //反向VP矩阵输入

12 float4x4 _RefW; //反射对象的局部变换矩阵

13

14 v2f vert (appdata v)

15 {

16 v2f o;

17 o.vertex = UnityObjectToClipPos(v.vertex);

18 o.ref = mul(_RefVP, mul(_RefW, v.vertex));

19 o.uv = TRANSFORM_TEX(v.uv, _MainTex);

20 return o;

21 }

22

23 fixed4 frag (v2f i) : SV_Target

24 {

25 return tex2D(_RefTex, i.ref.xy / i.ref.w * 0.5 + 0.5);

26 }

其中_ RefTex是反射绘制的纹理,_ RefVP是反向VP矩阵输入,_RefW是反射对象的局部变换矩阵。顶点着色器使用_RefW将顶点转换为世界坐标,再通过_RefVP将世界坐标转换为反射相机的投影空间,片元着色器将其转换为普通坐标转换+纹理坐标,并用于主摄像机渲染。

然后,我们得到了如下效果:

这样,即使相机移动或反射物体移动,旋转或缩放,也可以在没有违和感的情况下显示正确的投影图像。

3. 在投影着色器的基础上编写一个根据法线扭曲的着色器

接下来,我们添加扭曲投影的代码,这里通过引用法线贴图的方式实现。

1float2 offset = bump * _BumpAmt * _RefTex_TexelSize.xy;

2i.ref.xy = offset * i.ref.z + i.ref.xy

其中_BumpAmt是可调节的影响系数,_BumpAmt越大,则涟漪越强烈,但同时会影响到水面的反射效果,因此建议将参数控制在300以下,并按自己的需求进行调整。

_BumpAmt参数对效果的影响示意图

4. 根据脚的运动输入波动方程

利用波动方程,使波浪的高度可以作为纹理。(此处不对波动方程进行详细介绍)

左边区域是波纹输入纹理的调试界面,接下来,将存储在此纹理中的波的高度(被视为高度图)解析为法线并扭曲反射纹理,实际上,当从高度图计算法线贴图时,也可以直接求解偏导数,从而输出正常值作为着色器的凹凸。感兴趣的朋友可以参考该作者的另一篇相关文章:http://esprog.hatenablog.com/entry/2016/10/24/183733

同时根据角色脚的运动输入更改波动方程

作者在人物的脚点上附上了Collider组件和一个改造了InkPainter的CollisionPainter的脚本,可以理解为在人物的脚点上各加了一个笔刷,具体实现可以参考InkPainter插件里的Brush类。

PS:InkPainter插件链接附在文末,喜欢Splatoon的,或者想实现类似Splatoon效果的朋友,该插件可以提供不错的参考价值。

性能测试

使用UWA GOT Online对这个效果进行了性能测试。为了更真实地反映这个项目的渲染开销,我们使用了非多线程渲染的版本。

同时准备了4个版本的场景作比较:

版本A:即原版,3个角色加水面涟漪反射效果;

版本B:3个角色,去除涟漪反射效果,同时将地板Shader更换成简单的Diffuse;

版本C:9个角色加水面涟漪反射效果;

版本D:9个角色,去除涟漪反射效果,同时将地板Shader更换成简单的Diffuse。

测试4个版本的场景后,通过UWA GOT Online解析数据,最终得到对比数据如下:

我们观察到,有无涟漪反射的效果,对场景的帧率有一定的影响,但随着角色数量的上升,场景帧率发生了明显的下降,通过逻辑代码的分析我们发现开销主要集中在渲染模块上,接下来我们具体看一下Camera.Render的耗时均值对比:

下面我们以版本A为例具体看一下渲染耗时曲线:

可以观察到渲染耗时在8ms左右波动,其中不透明渲染在5ms左右,半透明渲染在1ms左右。然后再来对比一下四个版本的不透明和半透明渲染的平均耗时:

这里我们可以看到,开启效果后,不透明渲染和半透明渲染开销均有上升,且影响较大,开启效果会额外增加40%左右的渲染开销。

考虑到项目中的角色材质比较复杂,会对测试结果造成一定影响,对红米Note3这类低性能机型而言,能跑到这个数据,该效果的性能还是可以接受的,但建议开发者在使用时控制反射场景的复杂度,同时可以考虑优化水波反射的Shader以达到很好的优化效果。

注意事项

1)文章前两图素材来源于《LOST SPHEAR》,图三来自推主オノッチ利用该组件制作出来的效果,twitter链接:https://twitter.com/onotchi_/status/958188421045866496(仅做效果参考,并非项目内容);

2)该项目需要搭配InkPainter(nv1.2.1或更高版本)使用。

该项目来自UWA开源库,您可以下载、收藏或评论Waveform Provider项目。开源库传送门:https://lab.uwa4d.com/



推荐阅读
  • Android 构建基础流程详解
    Android 构建基础流程详解 ... [详细]
  • 使用 ListView 浏览安卓系统中的回收站文件 ... [详细]
  • 卓盟科技:动态资源加载技术的兼容性优化与升级 | Android 开发者案例分享
    随着游戏内容日益复杂,资源加载过程已不仅仅是简单的进度显示,而是连接玩家与开发者的桥梁。玩家对快速加载的需求越来越高,这意味着开发者需要不断优化和提升动态资源加载技术的兼容性和性能。卓盟科技通过一系列的技术创新,不仅提高了加载速度,还确保了不同设备和系统的兼容性,为用户提供更加流畅的游戏体验。 ... [详细]
  • 本文深入探讨了CGLIB BeanCopier在Bean对象复制中的应用及其优化技巧。相较于Spring的BeanUtils和Apache的BeanUtils,CGLIB BeanCopier在性能上具有显著优势。通过详细分析其内部机制和使用场景,本文提供了多种优化方法,帮助开发者在实际项目中更高效地利用这一工具。此外,文章还讨论了CGLIB BeanCopier在复杂对象结构和大规模数据处理中的表现,为读者提供了实用的参考和建议。 ... [详细]
  • 如何高效启动大数据应用之旅?
    在前一篇文章中,我探讨了大数据的定义及其与数据挖掘的区别。本文将重点介绍如何高效启动大数据应用项目,涵盖关键步骤和最佳实践,帮助读者快速踏上大数据之旅。 ... [详细]
  • 本文将带你快速了解 SpringMVC 框架的基本使用方法,通过实现一个简单的 Controller 并在浏览器中访问,展示 SpringMVC 的强大与简便。 ... [详细]
  • Spring – Bean Life Cycle
    Spring – Bean Life Cycle ... [详细]
  • 解决Only fullscreen opaque activities can request orientation错误的方法
    本文介绍了在使用PictureSelectorLight第三方框架时遇到的Only fullscreen opaque activities can request orientation错误,并提供了一种有效的解决方案。 ... [详细]
  • 在多线程并发环境中,普通变量的操作往往是线程不安全的。本文通过一个简单的例子,展示了如何使用 AtomicInteger 类及其核心的 CAS 无锁算法来保证线程安全。 ... [详细]
  • 微信公众号推送模板40036问题
    返回码错误码描述说明40001invalidcredential不合法的调用凭证40002invalidgrant_type不合法的grant_type40003invalidop ... [详细]
  • javascript分页类支持页码格式
    前端时间因为项目需要,要对一个产品下所有的附属图片进行分页显示,没考虑ajax一张张请求,所以干脆一次性全部把图片out,然 ... [详细]
  • 本文详细介绍了MySQL数据库的基础语法与核心操作,涵盖从基础概念到具体应用的多个方面。首先,文章从基础知识入手,逐步深入到创建和修改数据表的操作。接着,详细讲解了如何进行数据的插入、更新与删除。在查询部分,不仅介绍了DISTINCT和LIMIT的使用方法,还探讨了排序、过滤和通配符的应用。此外,文章还涵盖了计算字段以及多种函数的使用,包括文本处理、日期和时间处理及数值处理等。通过这些内容,读者可以全面掌握MySQL数据库的核心操作技巧。 ... [详细]
  • 在尝试对 QQmlPropertyMap 类进行测试驱动开发时,发现其派生类中无法正常调用槽函数或 Q_INVOKABLE 方法。这可能是由于 QQmlPropertyMap 的内部实现机制导致的,需要进一步研究以找到解决方案。 ... [详细]
  • Web开发框架概览:Java与JavaScript技术及框架综述
    Web开发涉及服务器端和客户端的协同工作。在服务器端,Java是一种优秀的编程语言,适用于构建各种功能模块,如通过Servlet实现特定服务。客户端则主要依赖HTML进行内容展示,同时借助JavaScript增强交互性和动态效果。此外,现代Web开发还广泛使用各种框架和库,如Spring Boot、React和Vue.js,以提高开发效率和应用性能。 ... [详细]
  • 《Intel IA-32 架构软件开发人员手册详尽指南》提供了详尽的 IA-32 架构技术文档,涵盖指令集、系统编程和硬件接口等内容,为软件开发人员提供全面的技术支持和参考。该手册不仅包括详细的架构说明,还提供了丰富的编程示例和最佳实践,帮助开发人员更好地理解和应用 IA-32 架构。 ... [详细]
author-avatar
小景森的童年
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有