自定义阴影Pass

作者:追风剑情 发布于:2017-12-2 13:09 分类:Shader

自定义阴影Shader

// Upgrade NOTE: replaced '_LightMatrix0' with 'unity_WorldToLight'

Shader "Custom/LightShader" {
	Properties {
		_MainTex ("Albedo (RGB)", 2D) = "white" {}
		_Color ("Color", Color) = (1,1,1,1)
		_Specular ("Specular", Color) = (1,1,1,1)
		_Gloss ("Specular Power", Range(0, 30)) = 1

		_Magnitude("Magnitude", float) = 1
		_Frequency("Frequency", float) = 1
		_InvWaveLength("InvWaveLength", float) = 1
		_Speed("Speed", float) = 1
	}
	SubShader {
		Tags { "RenderType"="Opaque" "DisableBatching"="True"}
		LOD 200
		Pass {
			Tags {"LightMode"="ForwardBase"}
 
			CGPROGRAM
			//multi_compile_fwdbase可以保证我们在Shader中使用光照衰减等光照变量可以被正确赋值。
			#pragma multi_compile_fwdbase
			#pragma vertex vert
			#pragma fragment frag
			#include "UnityCG.cginc"
			#include "Lighting.cginc"
			//计算阴影所用到的宏都在Autolight.cginc文件中
			#include "Autolight.cginc"
 
			sampler2D _MainTex;
			float4 _MainTex_ST;
			fixed4 _Color;
			fixed4 _Specular;
			float _Gloss;
 
			struct a2v {
				float4 vertex : POSITION;
				float3 normal : NORMAL;
				//Unity会将模型的第一组纹理坐标存储到该变量中
				float4 texcoord : TEXCOORD0;//阴影宏中要用到texcoord
			};
 
			struct v2f {
				//注意: 结构体中定义的某些成员变量名称要与所用宏中写的名字一致
				float4 pos : SV_POSITION;//阴影宏中要用到pos
				float3 worldNormal : TEXCOORD0;
				float3 worldPosition : TEXCOORD1;
				float2 uv : TEXCOORD2;
				//这个宏的作用很简单,就是声明一个用于对阴影纹理采样的坐标。
				//需要注意的是,这个宏的参数需要是下一个可用的插值寄存器的索引值
				SHADOW_COORDS(3)
			};
 
			v2f vert(a2v v) {
				v2f o;
				o.pos = UnityObjectToClipPos(v.vertex);
				//o.worldNormal = mul(v.normal, (float3x3)unity_WorldToObject);
				//Unity5中可用UnityObjectToWorldNormal()函数得到o.worldNormal
				o.worldNormal = UnityObjectToWorldNormal(v.normal);
				o.worldPosition = mul(unity_ObjectToWorld, v.vertex).xyz;
				o.uv = v.texcoord.xy*_MainTex_ST.xy+_MainTex_ST.zw;
				//或者调用Unity内建的函数计算o.uv
				//o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);

				//在顶点着色器返回之前添加这个宏.
				//这个宏用于在顶点着色器中计算上一步中声明的阴影纹理坐标
				TRANSFER_SHADOW(o);
				return o;
			}
 
			fixed4 frag(v2f i) : SV_Target {
 
				fixed3 worldNormal = normalize(i.worldNormal);
				fixed3 worldLightDir = normalize(UnityWorldSpaceLightDir(i.worldPosition));
				fixed3 viewDir = normalize(UnityWorldSpaceViewDir(i.worldPosition));
				fixed3 halfDir = normalize(worldLightDir + viewDir);
 
				//这里我们希望环境光、自发光只被处理一次,所以在ForwardBase中处理
				//环境光
				fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz;
				//如果场景中包含多个平行光,Unity会选择最亮的平行光传递给Base Pass进行逐像素处理,
				//其他平行光会按照逐顶点或在Additional Pass中按逐像素的方式处理。
				//如果场景中没平行光,那么Bass Pass会当成全黑的光源处理。
 
				//每一个光源有5个属性: 位置、方向、颜色、强度、衰减
				//对于Base Pass来说,它处理的逐像素光源类型一定是平行光。
				//我们可以使用_WorldSpaceLightPos0来得到这个平行光的方向(位置对平行光来说没有意义),
				//使用_LightColor0来和到它的颜色和强度(_LightColor0已经是颜色和强度相乘后的结果),
				//由于平行光可以认为是没有衰减的,因此这里我们直接令衰减值为1.0

				//用纹理颜色作为漫反射颜色
				fixed3 albedo = tex2D(_MainTex, i.uv).rgb * _Color;
				//漫反射光
				fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(worldNormal, worldLightDir));
				//高光
				fixed3 specular = _LightColor0.rgb * _Specular.rgb * pow(max(0, dot(worldNormal, halfDir)), _Gloss);

				//利用UNITY_LIGHT_ATTENUATION宏来同时计算光照衰减和阴影
				//这个宏内部会帮我们声明atten变量
				UNITY_LIGHT_ATTENUATION(atten, i, i.worldPosition)
				//此时的atten值已经是光照衰减值与阴影值相乘后的值
				return fixed4(ambient + (diffuse + specular) * atten, 1.0);
			}
			ENDCG
		}
 
		Pass {
			//通常来说,Additional Pass的光照处理和Base Pass的处理方式是一样的,
			//因此我们只需要把Base Pass的顶点和片元着色器代码粘贴到Additional Pass中,
			//然后再稍微修改一下即可。这些修改往往是为了去掉Base Pass中环境光、自发光、
			//逐顶点光照、SH光照部分,并添加一些对不同光源类型的支持。因此,在Additional Pass
			//的片元着色器中,我们没有再计算场景中的环境光。由于Additional Pass处理的光源类型
			//可能是平行光、点光源、聚光灯,因此在计算光源的5个属性(位置、方向、颜色、强度、衰减)时,
			//颜色和强度我们仍然可以使用_LightColor0来得到,但对于位置、方向和衰减属性,我们就需要
			//根据光源类型分别计算。
			//为场景中其他逐像素光源定义Additional Pass
			//Additional Pass的阴影处理和Base Pass是一样的。
			Tags { "LightMode"="ForwardAdd" }
 
			//我们希望Additional Pass计算得到的光照结果可以在帧缓存中与之前
			//的光照结果进行叠加,如果没有使用Blend命令的话,Additional Pass会直接
			//覆盖掉之前的光照结果。
			//也可以使用其他的混合方式,如Blend SrcAlpha One等
			Blend One One
 
			CGPROGRAM
			//multi_compile_fwdadd指令可以保证我们在Additional Pass中访问到正确的光照变量
			//#pragma multi_compile_fwdadd
			//如果希望在Additional Pass中添加阴影效果,就需要使用multi_compile_fwdadd_fullshadows编译指令代替multi_compile_fwdadd
			#pragma multi_compile_fwdadd_fullshadows
			#pragma vertex vert
			#pragma fragment frag
			#include "UnityCG.cginc"
			#include "Lighting.cginc"
			#include "Autolight.cginc"
 
			sampler2D _MainTex;
			float4 _MainTex_ST;
			fixed4 _Color;
			fixed4 _Specular;
			float _Gloss;
 
			struct a2v {
				float4 vertex : POSITION;
				float3 normal : NORMAL;
				//Unity会将模型的第一组纹理坐标存储到该变量中
				float4 texcoord : TEXCOORD0;
			};
 
			struct v2f {
				float4 pos : SV_POSITION;
				float3 worldNormal : TEXCOORD0;
				float3 worldPosition : TEXCOORD1;
				float2 uv : TEXCOORD2;
				SHADOW_COORDS(3)
			};
 
			v2f vert(a2v v) {
				v2f o;
				o.pos = UnityObjectToClipPos(v.vertex);
				//o.worldNormal = mul(v.normal, (float3x3)unity_WorldToObject);
				o.worldNormal = UnityObjectToWorldNormal(v.normal);
				o.worldPosition = mul(unity_ObjectToWorld, v.vertex).xyz;
				//o.uv = v.texcoord.xy*_MainTex_ST.xy+_MainTex_ST.zw;
				//或者调用Unity内建的函数计算o.uv
				o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);
				TRANSFER_SHADOW(o);
				return o;
			}
 
			fixed4 frag(v2f i) : SV_Target {
 
				fixed3 worldNormal = normalize(i.worldNormal);
				fixed3 viewDir = normalize(UnityWorldSpaceViewDir(i.worldPosition));
 
				//-----处理不同光源的方向-----//
				#ifdef USING_DIRECTIONAL_LIGHT
					//平行光
					//_WorldSpaceLightPos0.xyz表示的是光源方向
					fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz);
				#else
					//点光源 或 聚光灯
					//_WorldSpaceLightPos0.xyz表示的是光源在世界空间中的位置
					fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz - i.worldPosition.xyz);
				#endif
 
				fixed3 halfDir = normalize(worldLightDir + viewDir);
 
				//用纹理颜色作为漫反射颜色
				fixed3 albedo = tex2D(_MainTex, i.uv).rgb * _Color;
				//漫反射光
				fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(worldNormal, worldLightDir));
				//高光
				fixed3 specular = _LightColor0.rgb * _Specular.rgb * pow(max(0, dot(worldNormal, halfDir)), _Gloss);

				//利用UNITY_LIGHT_ATTENUATION宏来同时计算光照衰减和阴影
				//这个宏内部会帮我们声明atten变量
				UNITY_LIGHT_ATTENUATION(atten, i, i.worldPosition)
				//此时的atten值已经是光照衰减值与阴影值相乘后的值

				return fixed4((diffuse + specular) * atten, 1.0);
			}
			ENDCG
		}


		Pass {
			//当进行顶点动画时,如果仍使用内置的ShadowCaster Pass来渲染阴影,可能会得到错误的阴影效果,
			//因为Unity内置的ShadowCaster Pass是按原来的顶点位置来计算阴影(即,阴影不会有动画效果)。
			//所有内置的Transparent/*下的Shader是没有定义ShadowCaster Pass的.
			//顶点动画需要在模型空间下进行计算,所以需要在SubShader的DisableBatching标签来强制取消对该Shader的批处理。
			//然而,取消批处理会带来一定的性能下降,增加了Draw Call,因此我们应该尽量避免使用模型空间下的一些绝对位置和方向
			//来进行计算。

			//在广告牌例子中,为了避免显式使用模型空间的中心来作为锚点,我们可以利用顶点颜色来存储每个顶点到锚点的距离值,
			//这种做法在商业游戏中很常见。

			//阴影内置宏在UnityCG.cginc文件中被定义
			//如果FallBack Off, Unity将无法从上层找到投射阴影的Pass
			//我们可以自己实现投射阴影的Pass
			Name "ShadowCaster"
			Tags { "LightMode" = "ShadowCaster" }

			CGPROGRAM
			#pragma vertex vert
			#pragma fragment frag
			#pragma multi_compile_shadowcaster
			#include "UnityCG.cginc"

			float _Magnitude;
			float _Frequency;
			float _InvWaveLength;
			float _Speed;

			struct a2v {
				//阴影宏要用到vertex、normal
				float4 vertex : POSITION;//顶点位置
				float3 normal : NORMAL;//顶点法线
				float4 texcoord : TEXCOORD0;
			};

			struct v2f {
				//V2F_SHADOW_CASTER宏帮我们声明阴影投射需要的变量
				V2F_SHADOW_CASTER;
			};

			//也可以直接使用appdata_base结构体
			v2f vert( a2v v )//阴影宏里用到了名称为v的输入结构体
			{
				v2f o;

				//------自定义顶点动画
				float4 offset;
				offset.yzw = float3(0.0, 0.0, 0.0);
				//让阴影振动
				offset.x = sin(_Frequency * _Time.y + v.vertex.x * _InvWaveLength + v.vertex.y * _InvWaveLength +
				v.vertex.z * _InvWaveLength) * _Magnitude;

				v.vertex = v.vertex + offset;
				//------end

				//接下来的工作由阴影宏来完成
				//阴影宏为我们处理了所有阴影相关的计算,我们只需要关心自定义部分
				//Unity5
				TRANSFER_SHADOW_CASTER_NORMALOFFSET(o)
				//Unity4
				//TRANSFER_SHADOW_CASTER(O)

				return o;
			}

			float4 frag( v2f i ) : SV_Target
			{
				//直接利用SHADOW_CASTER_FRAGMENT宏完成阴影投射,并把结果输出到深度图和阴影映射纹理中
				SHADOW_CASTER_FRAGMENT(i)
			}
			ENDCG
		}
	}
	//Transparent/Cutout/VertexLit里的ShadowCaster Pass
	//计算了透明度测试,会把裁剪后的物体深度信息写入深度图和阴影映射纹理中。
	//但需要注意的是,由于Transparent/Cutout/VertexLit中计算透明度测试时,使用了
	//名为_Cutooff的属性来进行透明度测试,因此,这要求我们的Shader中也必须提供名为_Cutoff的属性。
	//否则,同样无法得到正确的阴影结果。
	//默认情况下把物体渲染到深度图和阴影映射纹理中仅考虑物体的正面。若需要考虑背面,可以将
	//模型的Mesh Renderer组件中的Cast Shadows属性设置为Two Sided,强制Unity在计算阴影映射纹理时
	//计算所有面的深度信息。

	//与透明度测试的物体相比,想要为使用透明度混合的物体添加阴影是一件比较复杂的事情。事实上,
	//所有内置的透明度混合的Unity Shader,如Transparent/VertexLit等,都没有包含阴影投射的Pass。
	//这意味着,这些半透明物体不会参与深度图和阴影映射纹理的计算,也就是说,它们不会向其他物体
	//投射阴影,同样它们也不会接收来自其他物体的阴影。Unity会这样处理半透明物体是有它的原因的。
	//由于透明度混合需要关闭深度写入,由此带来的问题也影响了阴影的生成。总体来说,要想为这些半透明
	//物体产生正确的阴影,需要在每个光源空间下仍然严格按照从后往前的顺序进行渲染,这会让阴影处理变得
	//非常复杂,而且也会影响性能。因此,在Unity中,所有内置的半透明Shader是不会产生任何阴影效果的。当然,
	//我们可以使用一些dirty trick来强制为半透明物体生成阴影,这可以通过把它们的Fallback设置为VertexLit、
	//Diffuse这些不透明物体使用的Unity Shader,这样Unity就会在它的Fallback找到一个阴影投射的Pass。然后
	//我们可以通过物体的Mesh Render组件上的Cast Shadows和Receive Shadows选项来控制是否需要向其他物体投射或
	//接收阴影。
	FallBack Off//"Transparent/Cutout/VertexLit"
}

效果

gaollg0.GIF

标签: Shader

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号