[基础] Unity Shader:法向量与光照

在现实世界中,太阳照射在物体的那一面较亮,而物体背朝太阳的那一面较暗,这就是我们要讨论的问题:如何计算一个物体各个表面的阴暗程度。

1. 法向量

法向量(normal vector) 也被称为法线(normal),它是垂直于平面的向量,在游戏中,模型的一个顶点往往会携带额外的信息,顶点法线就是其中之一。

2. 辐照度

在实时渲染中,我们通常把光源当成一个没有体积的点,用 来表示它的方向,那么,我们如何量化光呢?在光学里,我们使用辐照度(irradiance) 来量化光。在计算光照模型时,我们需要知道一个物体表面的辐照度,而物体表面往往是和 不垂直的,那么如何计算这样的表面的辐照度呢?

如上图所示,我们可以使用光源方向 和表面法线 之间的夹角 的余弦值来得到。

3. 辐照度的计算

两个向量之间的一个点积公式如下:

由此公式我们可以得出:

若我们将 规范化,也就是模长变为1,则公式又变成了:

由此可得出,辐照度可通过规范化光源向量 和规范化法向量 的点积来得到。

4. 应用于Shader

接下来,我们将理论真正地应用于Shader中,首先,我们要先规范化法向量和光源向量:

cpp 复制代码
float3 N = normalize(appData.normal); // 规范化法向量
float3 L = normalize(_WorldSpaceLightPos0); // 规范化光源向量

然后通过点积求得辐照度,为了防止点积结果为负值,我们需要将其限制在[0, 1]范围内,Cg提供了 saturate 函数恰恰可以实现这样的功能:

cpp 复制代码
float E = saturate(dot(N, L)); // 辐照度

得到辐照度后,我们就可以用物体颜色乘上辐照度了:

cpp 复制代码
outData.color = _Color.rgb * E; // 物体颜色 × 辐照度

当然,我们也可以让物体表面颜色也受到光照色的影响,不过在此之前要先引入 Lighting.cginc 文件:

cpp 复制代码
outData.color = (_Color * 0.5 + _LightColor0 * 0.5) * E; // 物体颜色混合光照色 × 辐照度

当然,我们也可以在片元着色器最后返回颜色时加上环境色 UNITY_LIGHTMODEL_AMBIENT,这样物体表面也将会受到环境色的影响:

cpp 复制代码
return inputData.color + UNITY_LIGHTMODEL_AMBIENT;

最终效果如下:

相关推荐
NRatel15 小时前
亚马逊S3的使用简记(游戏资源发布更新)
游戏·unity·amazon s3
SmalBox21 小时前
【渲染流水线】[几何阶段]-[屏幕映射]以UnityURP为例
unity·渲染
SmalBox2 天前
【渲染流水线】[几何阶段]-[归一化NDC]以UnityURP为例
unity·渲染
SmalBox3 天前
【渲染流水线】[几何阶段]-[图元装配]以UnityURP为例
unity·渲染
霜绛3 天前
Unity:GUI笔记(一)——文本、按钮、多选框和单选框、输入框和拖动条、图片绘制和框绘制
笔记·学习·unity·游戏引擎
谷宇.3 天前
【Unity3D实例-功能-移动】角色行走和奔跑的相互切换
游戏·unity·c#·unity3d·游戏开发·游戏编程
17岁的勇气3 天前
Unity Shader unity文档学习笔记(十九):粘土效果,任意网格转化成一个球(顶点动画,曲面着色器)
笔记·学习·unity·图形渲染·顶点着色器·曲面着色器
benben0444 天前
《Unity Shader入门精要》学习笔记一
unity·shader
YF云飞4 天前
Unity图片优化与比例控制全攻略
游戏·unity·游戏引擎·游戏程序·个人开发
SmalBox4 天前
【渲染流水线】[几何阶段]-[几何着色]以UnityURP为例
unity·渲染