文章目录
如何用Shader Graph创建水面倒影效果
在游戏开发中,我们有些时候会需要一面镜子或者一个水面的效果,这些效果通常是基于shader来实现,但对于很多初学者来讲,别说直接去写shader相关代码,连看懂别人写好的代码都非常困难,那么,有没有一种方法可以让我们不用写代码就实现这种效果呢?答案是有的,Unity官方提供了内置插件Shader Graph,它可以让我们零代码实现各种各样我们需要的效果~~(但是需要付出相对应的性能牺牲)~~,下面的水面倒影效果也是基于Shader Graph来进行实现。
1. 创建Render Texture
创建方式为右键->Create->Render Textrue
,将名字更改为想要的名字,这里我取名为Water Render Textrue
。
在创建完成后,我们可以看见目前的Render Texture是黑色的,因为我们暂时没有对他进行任何的设置。
2. 创建并设置观察相机
这里的相机是非常关键的组件,它是用于获取我们对需要倒影的对象的观察结果并输出到之前创建的Render Texture上。
在Hierarchy
窗口右键->Camera
进行创建:
创建完成后,可以按照个人需求对其进行命名,这里我就不对其进行修改。
创建完成后,首先要删除掉自带的Audio Listener
组件,否则运行时将会报错(详情见音频监听器 - Unity 手册),然后需要保证观察相机与我们的Main Camera
设置相同,这里可以直接到Main Camera
中点击Camera
组件左边的三个点并选择Copy Component
选项,他将复制这个组建的所有信息:
随后到我们的观察相机中,同样点击右侧三个点,选择Paste Component Values
,它会将刚刚复制的组件中的设置粘贴到当前组件中。
(这是可能有些小伙伴会发现选中观察相机后右下角并没有显示场景中的任何物体,这是因为创建时相机坐标出现了问题,建议同样也可以复制Main Camera
的坐标信息到观察相机,之后再进行调整)。
3. 设置相机输出到Water Render Texture
(在第一部创建的Texture)中,并简单进行配置
在观察相机中,找到Output
一栏,找到Ouput Texture
,将Water Render Texture
拖入进去(或点击右侧圆点进行选择):
我们马上就可以在窗口中看见Water Render Texture
不再是黑色,而是变成了场景中观察相机范围内的物体的图像,并且相机观察范围也发生了改变(变为在Water Render Texture
中的默认大小):
这个大小可能并不是我们所需要的大小,这时我们可以点击Water Render Texture
,修改右侧的Size
属性直到大小合适:
然后在对观察相机进行修改,同样也是修改他的size
属性到一个合适的范围:
4. Shader Graph启动!
基础的准备工作完成后,终于迎来了重头戏——Shader Graph!我们先选择一个合适的位置创建文件夹,来存储我们后续编写的Shader Graph文件,接下来我们将一步一步实现水面效果。
4.1 创建Shader Graph
文件
在Project窗口右键,选择Create->Shader Graph->URP->Unlit Shader Graph
创建Shader Graph文件,并给它取名为合适的值,双击可以打开如下所示的编辑界面,这就是我们后续编写Shader Graph的工作界面,其中已经包含有节点Fragment
,它将是我们渲染结果的输出节点;右下方的Main Preview
则是最终效果的预览图,可以右键他更改为需要展示的效果。
4.2 先将原图渲染出来
在Shader Graph中进行设置
在正式编写前,我们可以先将观测相机的画面输出到场景中来看看目前的观察效果,通过观察效果可以再次对相机观察范围等进行微调。
右键空界面,选择Create Node
,在输入框中输入Texture
,选择Sample Texture 2D
将节点添加到场景中:
在左侧的Texture
属性中选择刚刚创建好的Water Render Texture
,可以看见下面那已经变成了与Texture
相同的样子:
再将RGBA(4)
输出到Fragment
节点中的Base Color(3)
,可以看见右下方Main Preview
已经变成相同的样子:
为了方便后续更改使用,我们也可以单独创建一个Texture 2D
属性节点,方法为在左下角与这个文件同名的窗口中点击+
号,选择Texture 2D
,命名为合适的名称,再拖入主窗口连接到Texture
上,选中该节点设置默认值为Water Render Texture
,:
创建材质球输出渲染结果并将材质球添加到水面上
在Project
窗口右键,依次点击Create->Material
创建材质球,为材质球取上一个名字(这里我命名为Water Material
:
直接将Shader Graph
文件拖到材质球上,可以看见材质球图标变成了Shader Graph
中Main Preview
的样子(如果没有改变重新进入该文件夹刷新即可):
最后将材质球拖放到显示水面的位置或者他的Sprite
组件中的Material
一栏:
最终效果:
4.3 实现水面倒影的上下颠倒
实现上下颠倒的效果需要添加一个新的节点,名为Tiling And Offset
,他可以设置图片平铺范围和显示偏移,最终输出UV坐标(用于控制图片在显示器上显示位置的已经进行归一化了的坐标)。
所以第一步,先添加一个Tiling And Offset
节点(添加节点方式在前面写过就不再赘述),在节点中可以看见三个属性值,分别是UV
、Tiling
和Offset
,其中,Tiling
控制图片的平铺方向和范围,Offset
则控制图片显示偏移值
其中我们主要对Tiling
和Offset
进行修改,为了更直观看见修改效果,我们可以直接将他的输出连接到Sample Texture 2D
中的UV(2)
。
在水面中的倒影主要实现上下颠倒,左右不进行改变,所以首先要改变Tiling
中的y值为-1让他反向平铺,在更改Offset
中的y值为1,让其向上偏移显示出反向平铺后的图像:
在场景中也直接进行了自动更改:
4.4 实现水面波纹
一般而言,睡眠不会绝对静止,有风吹过等等原因都会有波纹产生,所以为了更加真实,为其添加动态波纹效果。
通过添加噪点模拟水面波纹偏移
为了实现波纹效果,可以使用噪点来改变Offset
的偏移值,添加一个噪声节点,输入noise
,选择Sample Noise
。
但是普通噪声节点无法动态改变,为了实现动态功能,我们需要一个时间节点Time
和Gradient Noise
节点:
将Time
和Gradient Noise
进行加法运算,输出到Sample Noise
节点中,可以看见Sample Noise
节点下方开始运动,但是速度过快:
因此我们将时间先乘以一个浮点型影响因子,给他命名为Speed
,默认值设置为0.05f,可以看见变换明显变缓了许多:
而对于变换的大小,我们也可以添加一个名为Wave Scale
的浮点数属性,便于后续设置变化大小,这里默认值设置为3,此时的效果为:
将偏移运用到图像上
如果直接将Sample Noise
输出到Sample Texture
中的Offset
,会发现图片偏移到了一个过于离谱的程度,这是因为同时对x轴和y轴进行了更改,导致图片扭曲过大,同时还因为边界没有任何像素导致出现较大的黑块,实际上需要修改的仅为x轴即可,让他只在x轴有部分偏移,这样可以避免出现大量的黑块。
直接输出时的效果:
如果只需要修改x轴,我们需要一个二维向量,设置他的y轴永远为1,x轴随噪声进行改变,此时可以看见图片偏移变得稍显合理,但是偏移量还是非常大:
这是因为x轴偏移过多造成的图像偏移范围变大,只需要更改一下偏移的范围即可,具体做法即为用一个相对较小的数值与原本偏移的值进行乘法运算,修正偏移量,这里我使用的值为0.05:
此时他的效果就仿佛像微风吹拂水面一样:
4.5 实现更加“真实”的水
通常而言,海水会偏向于蓝色,要实现这个效果我们可以用对应的颜色与这张图片进行叠加,例如这里我使用颜色比较深的蓝绿色与设置好的图片进行叠加:
此时场景中水面便会有蓝绿色元素存在: