ShaderLab学习小结(五)接收平行光阴影

运行环境:
Win10 x64
Unity 5.5.4

在“ShaderLab学习小结(四)简单产生阴影”基础上,在顶点片断着色器中编写程序使材质能接收阴影
将小结(四)中添加的plane的材质换成和球体一样,即用相同的shader,效果如下:

阴影消失。
球的shader代码没有改动,也就是说球应该是产生阴影了,但是底下的plane也换成球的材质,接收不到阴影。
那就要修改shader让它能接收阴影。
Shader代码:

Shader "Custom/DifSpecPoint" {
    Properties {
        _Spec ("Spec", Color) = (1,1,1,1)
        _Shin ("Shin", range(1,32)) = 2
    }
    SubShader {
        pass {
            tags{ "lightmode" = "forwardbase" }
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "unitycg.cginc"
            #include "lighting.cginc"
            #include "autolight.cginc"  //1.
            #pragma multi_compile_fwdbase //2.
            fixed4 _Spec;
            float _Shin;
            struct v2f{
                float4 pos:POSITION;
                float3 normal:NORMAL;
                float4 vertex:TEXCOORD2;
                SHADOW_COORDS(0)       //3.
            };
            v2f vert(appdata_base v)
            {
                v2f o;
                o.pos = mul(UNITY_MATRIX_MVP, v.vertex);
                o.normal = normalize(v.normal);
                o.vertex = v.vertex;
                TRANSFER_SHADOW(o)  //4.
                return o;
            }
            fixed4 frag(v2f IN):COLOR
            {
              float3 wpos = mul(unity_ObjectToWorld, IN.vertex).xyz;
                UNITY_LIGHT_ATTENUATION(atten, IN, wpos)  //5.
                //diffuse
                float3 N = UnityObjectToWorldNormal(IN.normal);
                float3 L = normalize(_WorldSpaceLightPos0).xyz;
                float ndotl = saturate(dot(N, L));
                fixed4 col = _LightColor0*ndotl;
                //specular
                float3 V = normalize(WorldSpaceViewDir(IN.vertex));
                float3 R = 2 * dot(N, L)*N - L; //phong
                float3 H = normalize(V + L);    //blinphong
                float specScale = pow(saturate(dot(R, V)), _Shin);  //phong
                specScale = pow(saturate(dot(H, N)), _Shin);        //blinphong
                col += _Spec*specScale;
                //pointlight

                float3 pL = Shade4PointLights(unity_4LightPosX0, unity_4LightPosY0, unity_4LightPosZ0,
                    unity_LightColor[0].rgb, unity_LightColor[1].rgb, unity_LightColor[2].rgb, unity_LightColor[3].rgb,
                    unity_4LightAtten0,
                    wpos, N);
                col.rgb += pL;

                //shadow
                col.rgb *= atten;    //6.

                col += UNITY_LIGHTMODEL_AMBIENT;
                return col;
            }
            ENDCG
        }
    }
        fallback "Diffuse"
}

一、代码中添加了六处代码,如上以//1. //2.标注

1.

#include "autolight.cginc"  //1.

引用autolight.cginc

2.

#pragma multi_compile_fwdbase //2.

添加这一行,见unity手册“Making multiple shader program variants"中相关描述: compiles all variants needed by ForwardBase (forward rendering base) pass type. The variants deal with different lightmap types and main directional light having shadows on or off.这会对应主平行光产生的阴影。

以下3~5见autolight.cginc

3.

 SHADOW_COORDS(0)       //3.

4.

TRANSFER_SHADOW(o)  //4.

5.

UNITY_LIGHT_ATTENUATION(atten, IN, wpos)  //5.

6.将5中计算出的atten与颜色的rgb相乘

//shadow
                col.rgb *= atten;    //6.

二、

效果如下图:

可见plane可以接受主平行光照射球体产生的阴影了

1、

从UNITY_LIGHT_ATTENUATION倒推,查看autolight.cginc,在5.0helper中

#ifdef DIRECTIONAL
    #define UNITY_LIGHT_ATTENUATION(destName, input, worldPos)  fixed destName = SHADOW_ATTENUATION(input);
#endif

其中SHADOW_ATTENUATION(input)的定义,查找可知对应不同的函数,但参数一样

a._ShadowCoord

再查

_ShadowCoord

在autolight.cginc开头Shadow helpers中

#define SHADOW_COORDS(idx1) unityShadowCoord4 _ShadowCoord : TEXCOORD##idx1;

定义了,那么在v2f结构体中就要写入

 SHADOW_COORDS(0)       //3.

注意括号里的0,这个是代表TEXCOORD0,如果0被占了,则依次向后找,比如改成1、2什么的

2.

TRANSFER_SHADOW(o)  //4.

个人理解是把计算的阴影要从vertex程序传至fragment程序中,然后fragment中的UNITY_LIGHT_ATTENUATION
才能起作用(不信可以把TRANSFER_SHADOW(o)注释掉,不报错,但阴影没了)
而TRANSFER_SHADOW在autolight.cginc中的定义也是需要

a._ShadowCoord

的,所以在v2f结构体中定义了SHADOW_COORDS(0)满足了TRANSFER_SHADOW和UNITY_LIGHT_ATTENUATION两段宏的需要。

原文地址:http://blog.51cto.com/shuxiayeshou/2065497

时间: 2024-11-08 15:51:51

ShaderLab学习小结(五)接收平行光阴影的相关文章

ShaderLab学习小结(六)接收点光源阴影

运行环境:Win10 x64Unity 5.5.4在"ShaderLab学习小结(五)接收平行光阴影"中,通过修改代码,让原来的shader可以接收主平行光的阴影.但是没有点光源的阴影. 在原有的pass通道下面再添加一个pass通道,用于接收点光源阴影.直接将原来的pass复制粘贴成新的pass通道,再进行一些修改即可. pass { tags{ "lightmode" = "forwardadd" } //1. blend one one /

ShaderLab学习小结(四)简单产生阴影

在之前"ShaderLab学习小结(三)漫反射+高光+点光源请添加链接描述"中的场景加入一个plane,如下图:被平行光和点光源照射的球体没有产生阴影代码方面很好解决,就是在原先的shader最末尾加个fallback即可 Shader "shadername"{ SubShader{} fallback "Diffuse" } 效果如下图:在两个光源照射下,球体产生了阴影,并投射到下面的plane上.注:plane用的是系统默认的standar

ShaderLab学习小结(十九)RenderToCubemap创建能反射周围环境的效果

继续用"ShaderLab学习小结(十八)cubemap"中所做的shader想要让它能对周围的环境进行反射思路就是要让它的cubemap里的内容是周围环境的映射shader不变,就要想办法进行映射,这就要用到c#代码了核心的方法是 public bool Camera.RenderToCubemap(RenderTexture cubemap) 顾名思义,这个是Camera的一个方法,渲染至cubemap返回ture则渲染成功看unity document中的说明即从这个camera

ShaderLab学习小结(十六)泡泡

一个球体,不论从哪去看,都是中心透明,向边上越来越不透,最后纯白,大概就像个泡泡一样效果如下图:先看shader代码: Shader "Custom/PaoPao" { //1. Properties{ _MainColor("Main",color)=(1,1,1,1) _Scale("Scale",range(1,8))=1 } SubShader { Tags{"queue"="Transparent"

ShaderLab学习小结(三)漫反射+高光+点光源

场景中有一个平行光,一个×××点光源,设高光颜色为绿,效果如下:Shader代码: Shader "Custom/DifSpecPoint" { Properties { _Spec ("Spec", Color) = (1,1,1,1) //高光颜色 _Shin ("Shin", range(1,32)) = 2 //高光强度系数 } SubShader { pass { tags{ "lightmode" = "

ShaderLab学习小结(一)最简化的顶点片断Shader

Shader "Custom/SimpleShader" //Shader最外层,""中定义Shader名 { SubShader{ //必要组成部分 pass{ //通道 CGPROGRAM //CG程序开头 #pragma vertex vert //定义顶点级程序 #pragma fragment frag //定义片断级程序 #include "unitycg.cginc" //引用unitycg文件,unity自定的结构体和函数等 st

ShaderLab学习小结(十八)cubemap

现在还没用到cubemap(除了天空盒子),只是初步的学了一下最简单的cubemap的shader找一幅图好吧,有点大,在unity里把它设置成cubemapOK,图有了,看shader,代码挺简单的,其实和普通的2D贴图shader挺像 Shader "Custom/Cubemap" { Properties { _Cube ("Cubemap", cube) = "" {} //1. } SubShader { pass { CGPROGRA

Maven学习小结(五 pom.xml详解[转])

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0http://maven.apache.org/maven-v4_0_0.xsd"> <!--父项目的坐标.如果项目

ShaderLab学习小结(七)用插值函数lerp渐变颜色

运行环境:Win10 x64Unity 5.5.4在场景中创建一个cube,使它的颜色产生简单的两种颜色过渡的渐变效果,如下图:先说一下CG语言中的lerp函数lerp(a, b, w); a与b为同类形,即都是float或者float2之类的,那lerp函数返回的结果也是与ab同类型的值.w是比重,在0到1之间当w为0时返回a,为1时返回b,在01之间时,以比重w将ab进行线性插值计算. 功能很简单,实现也很简单.Shader代码: Shader "Custom/TestRedYellow&q