About me

大家好,我是Asher,Epic Games的Developer Relations Techical Artist,平时我的工作包括帮助各种大小游戏工作室解决技术/美术问题及实现相关引擎功能,有时间也会用引擎制作内容作为演讲和教程的示例。流程化的炫酷效果一向是我很感兴趣的领域,还在beta中的Niagara示例内容并不多,希望这篇解析可以给大家一些启发。

Summary

本篇特效的实验性质比较强,最初是为了探索BP-Material-Niagara协同工作的可能性,因为在Unreal Circle上反响很好所以决定写一篇更详尽的分析,希望可以使UE4开发者拓展对引擎框架的认识。

本文涉及到的内容比较广,关于Niagara最基础的一些部分比如怎么新建、怎样添加使用模块就不详细说明了,没用过的同学可以先看一下Epic官方文档的介绍:
Niagara核心概念

和贾越同学的系列教学直播:
https://www.bilibili.com/video/av73602807

本次工程效果:

Niagara procedural wave splashhttps://www.zhihu.com/video/1196146896954314752

部分工程文件下载:http://asher.gg/?p=2592

为了实现海浪拍到岩石上激起千层浪的感觉,我们可以看到有几个表现上的关键点

  • 在岩石附近生成水花
  • 水花激起的时机和位置与海水的流动匹配
  • 出射速度受到海水运动和石头表面的撞击情况影响
  • 激起后水花的体积感

海面的模拟 – Gerstner Waves

游戏里模拟海水的途径是一门艰深的课题,本文就不多涉及了。这里为了制作需要,介绍一下相对直观实用的Gerstner Waves。

在流体动力学中,Gerstner Waves是周期表面重力波的欧拉方程方程的精确解。它计算机图形技术之前很久就出现了,用来描述不可压缩流体的表面波动。

单一Gerstner Wave材质应用在一个平面上的效果

数学时间

我们可以看到每一个点不止有Z轴方向的运动,也有XY轴的运动。并且这不是一个简单的sin波而是带有‘浪尖’的波形,这些都是Gerstner Wave的重要特征。

因为篇幅原因这里就不具体说公式了,本节末尾提供了我做好的材质节点可以直接下载取用。详细算法可以参考:

Chapter 1. Effective Water Simulation from Physical Modelshttps://developer.download.nvidia.com/books/HTML/gpugems/gpugems_ch01.html

简单说来,定义一个波形的属性有:
[高度, 前进方向, 波长, 陡峭度]

我们定义相位φ
φ = 方向 ⋅ 位置 / 波长 + 频率 * 时间

其中
频率 = sqrt(重力 * 2π / 波长)

那么以
z轴偏移 = 高度 * sin(φ)
xy轴偏移 = 陡峭度 * 高度 * 方向 * cos(φ)

的形式去移动海面的顶点,顶点会沿着椭圆绕圈

一系列的顶点组合在一起就会形成具有浪尖的波形。

引擎内实现

在引擎里的实现可以用HLSL写在Custom节点里,你也可以选择用节点连。执行效率上,节点连出来的公式和HLSL代码没有区别。

材质函数 MF_GerstnerWave

核心部分:

 float3 d = float3(sin(DirectionInDegrees * 0.0174533), cos(DirectionInDegrees * 0.0174533), 0);  // 为了控制方便,输入并不是一个向量而是角度,这里转换一下float w = 6.2831854 / L;float theta = sqrt(6.2831854 * Gravity / L);  // 由波长算出频率float q = Stiffness / (Amplitude * w);   // q是控制浪尖陡峭度的系数float phase = dot((w * d).xy, WorldPosition.xy) - theta * Time; // 喂给cos和sin的相位float3 offset = float3(q * Amplitude * d.xy * cos(phase), Amplitude * sin(phase));  // 照抄上面公式

这部分我放在了http://BlueprintUE.com里,点下面链接直接把节点复制粘贴到UE材质编辑器里就可以得到图上的节点:https://blueprintue.com/blueprint/tt5p124n/

波形叠加

1层Gerstner wave
2层Gerstner wave
4层Gerstner wave + 随机角度

显然,一条波并不能说非常炫酷,Gerstner Wave的典型做法是把多条波叠加在一起。上面提到每条波可以控制的变量有:[高度, 前进方向, 波长, 陡峭度]

要合并多个波形,首先考虑的是波长,因为能组合的波长数量有限(8个已经很多了),我们无法完全参照真实世界的海洋数据,只能尽可能利用能付出的资源。由于波长相似的wave在叠加时看起来更有错综复杂的动态表现,我们可以在输入参数里首先给一个波长的中值LengthMedian,然后让不同wave的波长围绕这个中值变化。这样只要保留所有wave的波长与LenghMedian的比例,就可以通过改变LengthMedian来调整整个海洋的波浪大小:

在定义一组waves的结构FGerstnerWavesParameters中,美术可以在海洋的蓝图Actor上直接填写LengthMedian。

接下来,美术可以手动填写各个wave和LengthMedian的比例,也可以像我一样偷懒,填一个LengthMultiplierRange然后随机选取范围内的比例,在场景里拖动Seed直到随出一个好看的组合。

类似的方法同样适用于高度, 前进方向, 陡峭度。要注意的是,wave的高度和波长存在正相关关系,简单的做法是定义一个常数比例,让每个wave高度和和波长的比例保持一致。而在定义前进方向时会类似的给出一个DirectionRange定义前进方向的范围。

Niagara GPU粒子的生成

匹配水面位移

上面说了这么多,海面Shader内的位移数据其实并不能被外界直接读取,为了让Niagara能获取到海面的位移数据从而实现位置的同步,我们需要额外做一些工作:

方块:Niagara GPU particles

因为Gerstner Wave算法是确定性的,即给定同样的 [ 高度 | 前进方向 | 波长 | 陡峭度 ] 这样一组参数,不管在哪算出来的波形都是一样一样的。所以我们只要把与海面同样的参数传入Niagara System,就可以让Niagara粒子去‘追踪‘海面的粒子。(如上图)

传参

Normal难度:

上面说到,海面的波形是由好几个不同的Gerstner Wave组成,这里我用了8个,那么需要匹配海面的形状,其实在生成浪花时我们不用考虑所有8个waves,只考虑最大的一或两级wave,视觉上就很难看出不完全吻合的细节差别了。

Hard难度:

这一块可选阅读,跳过的话对下面内容的理解没有影响。

可是如果我觉得这样做身体不适,非想传所有数据达到完美同步呢? 也不是没有办法。4个参数 x 8组 = 一共32个float变量,手动命名并且在蓝图里拉面条,再在Niagara模块上一个一个拖上去也不是不行.. 如果你想做得灵巧些,我发掘了一个输入数组的hack:

因为Niagara还在beta阶段,一些功能还在持续改进中,这个hack可能以后也不需要。我介绍一下的另一个原因是觉得对增加对Niagara的理解有所帮助。

开始我想把数据通过DrawToRenderTarget写到一张贴图上,让Niagara读,但因为操作太不友好并且不支持CPU粒子放弃了。后来发现Niagara里的Curve型变量是同时支持CPU和GPU的,读取也很方便。我在蓝图里把需要的变量写入到一个Curve asset里,Niagara内reimport更新就可以,很方便。

不过Curve key的格式是有讲究的,Niagara CPU sim在读curve时,直接就读对应位置的key value。但GPU sim上会首先把curve编码成一个1x128像素的Lookup table,然后再读对应位置的数据,这就导致如果key的time位置不是100%对到LUT的像素位置上,encode后会出现偏差,这个偏差在我们现在的应用中是无法容许的。

好在encode的逻辑非常直接,只是取第一个key和最后一个key的time看范围,normalize到[0.0, 1.0]之间:

所以要达到key和LUT的像素对齐,最简单的方式是让time = [0, 1, 2… 127],这样不需要额外操作,在GPU sim上读取即可。

最后一个key的time是127.0

这样我们就实现了可以在一个curve里记录128个float值。剩下要做的就是再BP里根据一定规则把8个wave的一系列参数编码,再从Niagara模块里通过相同规则解码。这里4个参数*8个wave=一共32个值,规则是简单的按照一定间隔记录它们。

每组间隔6.0,一组4个参数
在Niagara module中通过类似方式解码

定义GPU粒子的碰撞行为

Distance Field Collision

系统提供的collision模块非常健全,包括CPU trace碰撞,GPU depth / distance field碰撞等分支可以在下拉菜单中选择使用。我们这里使用GPU distance field碰撞,depth碰撞比较适合比较粗犷的效果,比如下雨下雪,稍微有点问题也看不出来,但如果水浪使用depth碰撞,岩石背面屏幕看不到的地方就会出错。

我们想有一些比较细腻的控制逻辑,比如这里水花撞到石头上,如果用默认的collision碰撞,反弹是朝着下图绿色的反射向量方向飞出去的,而流体撞到刚体的行为并不是这样完美的反弹,我们更想让它偏向下图紫色的角度。

下面三个gif分别展示不同反射角度的视觉表现:

反弹方向为绿色反射角
反弹方向为紫色切线角
反弹方向反射和切线之间的随机角度

定义这样的碰撞行为需要自己写模块,幸运的是这种逻辑都可以用Niagara的节点实现。引擎提供的Collision模块本身是一个宝藏,里面有各种碰撞的实现方式和模块编辑的best practice。在Collision - CollisionQueryAndResponse模块中我们探索一下可以发现:

点进去最底层的模块是:

即给出world position获得global distance field的数值和gradient,和材质里的DistanceToNearestSurface / DistanceFieldGradient结果是一样的,都是GPU内的query。

知道了这些,我们就可以很方便的自己写一个简单的collision判断:

并且在后面接上上面说到的反射角和切线角的逻辑--如果发现水浪粒子进行了第一次和岩石碰撞,那么就沿着我们想要的角度给一个初速度:

粒子数量优化

同时因为有了distance field信息,可以在particle spawn时判断水浪粒子是否在岩石附近,如果不在就直接删除。

Smear

水花的材质想做好也是一项涉及很广的任务,我这里就是用了一个比较简单的云的贴图,稍微加工了下。值得一提的是通过particle的速度可以在材质里实现速度拉伸效果,对表达浪花飞溅的夸张形态非常有帮助:

Smear = 0
Smear =3
Smear = 15

实现方式也很简单,因为粒子sprite是朝向camera的,我们只要以local position和移动速度的点积缩放sprite即可:

叨叨完了

因为这个项目涉及的知识和技巧很多,很多地方只能大概说下思路。开头提供了部分实现的工程下载,有兴趣的同学可以下载看看。有问题可以留言讨论。

下期文章计划解析Volumetric Fog的一些细节表现方法,也就是去年(已经是去年了)11月Gears 5在Unreal Dev Day活动上展示的一些云雾表现,演讲发出后很多开发者都表现了强烈的兴趣。所以上个月杭州Unreal Circle我也做了些研究并做了一些解析,可以先看这里:https://www.bilibili.com/video/av81443196

期待下期再见!

Asher Zhu
Tech Artist, Epic Games

ue4 怎么传递变量到另一个蓝图_UE4中用Niagara实现procedural浪花相关推荐

  1. ue4 怎么传递变量到另一个蓝图_资深建模教你放置UE4蓝图节点,所以你就不要偷懒啦,认真点学...

    蓝图是UE4的一大特色,蓝图节点作为UE4学习的一个重要知识点,一直被很多人所重视.那你知道UE4是怎么放置蓝图节点的呢? 放置蓝图节点 在"图形模式"下,有几种方法可以放置节点. ...

  2. ue4 怎么传递变量到另一个蓝图_【UE4】UI注意事项

    文章内容导图: 以下仅是自己在实际操作过程中记录的一些笔记,可能不是很全,关于UI这块以后会根据具体情况以及自己的疑惑点不断更新,不断完整的. (另:自己把相关知识总结一遍形成一个框架,相当于建一座图 ...

  3. ue4 怎么传递变量到另一个蓝图_[UE4蓝图]虚幻4中实现简易天气系统(三)—— 受风力影响的Cascade雨水粒子...

    上一篇: 架狙只打脚:[UE4蓝图]虚幻4中实现简易天气系统(二)-- 随机风力​zhuanlan.zhihu.com Cascade就是现在UE4中正在使用的ParticleSystem. 制作雨水 ...

  4. (转)用ASP.NET向Javascript传递变量 方法1:

    (转)用ASP.NET向Javascript传递变量 方法1: 用一个隐藏控件,把变量的值给隐藏控件,再用Javascript去找隐藏控件的值 window.document.getElementBy ...

  5. 在hadoop中传递变量

    在hadoop中传递变量 @(HADOOP)[hadoop] 在主类中定义的变量,如定义了一个outputname,需要将其写入conf分发至其它nodemanager: Configuration ...

  6. 线程组之间的JMeter传递变量

    下面,我们将看看如何在线程组之间共享和传递变量. 在开发高级JMeter脚本时,很可能您将拥有多个线程组.每个线程组将执行不同的请求. 一个很好的例子是我们需要使用Bearer Tokens对用户进行 ...

  7. mongoose如何发送html页面,Mongoose/Express/Nodejs尝试从服务器到html传递变量

    我试图从我的server.js文件传递一个变量为HTML,但该变量不会显示在我的EJS文件中.我必须错过一些东西,因为它正在另一条路线上工作(使用另一个EJS文件),但我看不到我想要传递的表单或变量. ...

  8. 在两个ASP.NET页面之间传递变量【转】

    ASP.NET提供了事件驱动编程模型,使开发者简化了应用程序的总体设计,但是这个也造成了它固有的一些问题,例如,在传统的ASP里,我们可以通过使用POST方法很容易地实现页面间传递变量,同样的事情,在 ...

  9. QT中使用全局变量在多个源程序中传递变量

    使用QT5.5开发一个程序,有时需要多个源文件,包括若干个头文件和若干个定义文件.因此如何在多个源程序间开发传递变量就成了一个关键问题.一般来说在多个源程序间传递变量大概有两种方法: 一.是将全局变量 ...

  10. jsp:param能不能传递变量_变量、作用域与内存

    不要老叹息过去,它是不再回来的;要明智地改善现在.要以不忧不惧的坚决意志投入扑朔迷离的未来. 文章目录 变量.作用域与内存 前言 相比于其他语言,JavaScript 中的变量可谓独树一帜.正如 EC ...

最新文章

  1. 计算机书籍-Scratch少儿编程
  2. 近期活动盘点:2019清华大数据系统软件峰会(9.15)
  3. 深度学习100例-卷积神经网络(VGG-16)猫狗识别 | 第21天
  4. Java获得泛型类中T的实例
  5. python sqllite远程_Python实现Sqlite将字段当做索引进行查询的方法
  6. php数组交集方法,PHP获得数组交集与差集的方法
  7. 城市大轰炸++(洛谷P1847题题解,Java语言描述)
  8. 创建模板_在 GNOME 中创建文档模板 | Linux 中国
  9. 战略、业务流程和知识管理
  10. 安装mysql 错误重新安装
  11. WPF DataGrid 数据绑定
  12. 数据库——添加外键约束
  13. 安卓手机 ADB 操作指令
  14. android模拟power键,android 发送模拟按键
  15. 自动化记账程序1.0
  16. 用SHOPEX增强工具把淘宝数据包批量上传到自己的ShopEX独立网店
  17. chrome 全屏无法退出
  18. 扫描二维码,提示请在指定客户端打开连接
  19. 联机五子棋小程序:C++ MFC创建游戏界面
  20. 《浪潮之巅》第一、二、三、六次印刷勘误表

热门文章

  1. 汉字转拼音首字母大写
  2. 计算机丢失lua51dll怎么修复,lua51.dll
  3. 数据推荐 | 自然场景OCR文字识别数据集一览
  4. java我的世界光影推荐_最棒的7款我的世界光影水反效果包
  5. vue项目引入sass
  6. PostgreSQL内核扩展之 - ElasticSearch同步插件
  7. 小电流接地系统配电线路弧光高阻接地故障电压特征分析ATP-EMTP仿真建模
  8. python上传文件
  9. 项目:识别Twitter用户性别
  10. k2pb1官改和梅林_K2P B1免拆机刷官改和梅林阉割版固件(方法超简单)