前言
在游戏开发中实时阴影是比较常见的需求,我们最常见的方法是实时光照,但是这个会带来性能的问题,如果场景中模型比较多,例如我最近在做的3D足球游戏,场景中22个球员,如果采用实时光照DC会增加好几百,会造成渲染的压力,就有必要采用关照贴图的方案,比关闭掉实时光,但这种方案就降低了DC,减轻了渲染压力,这就要求阴影必须采用其他方案,我这里介绍Shader来现在的方案,貌似是王者荣耀采用的一种方案,专门用一个pass来渲染球员的阴影。
效果图
会看到模型有实时阴影效果。
原理
下面用2D视角来分析,阴影产生正如下图:
由上图可知,阴影的a点与遮挡物的b点在光照方向的垂直面上其实映射的是同一个点d。这很容易让我们联想到MVP当中的投影变换。其实是一样的。我们将于光照方向当做照相机的视线,而与光照方向垂直的面,就是投影变换最后的盒子的正面。这意味着我们可以制作一个正交照相机,使得它与光照方向一致,再将地面模型的各个顶点投影到该照相机上,这时候照相机的纹理UV坐标范围是0~1,地面投影到了照相机,也将其坐标映射到0~1之间,由此与照相机的纹理一一对应,于是将照相机照出来的纹理叠加在地面上,就形成了阴影。
程序实现
shader
1 | Shader "Custom/PlayerShadow" |
第二个pass专门用来渲染阴影,power参数是颜色加强,因为关闭了实时灯光,模型会显得暗淡,这个参数跟阴影效果没太大关系。
CSharp
1 | using System.Collections; |
主要是将光照向量和当前模型的坐标传递给Shader,以便Shader实时计算出当前模型的阴影。
案例Demo
下方留言获取最新案例