shader之——多光源漫反射以及阴影

Shader "Unlit/MulLight"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
    }
    SubShader
    {
        //一盏主灯
        Pass
        {
            //Always: 总是渲染;没有光照模式。
            //ForwardBase: 适用于前渲染、环境、主要方向灯、光/sh光和烘焙图。
            //ForwardAdd: 适用于前渲染, 叠加每一盏灯,每一盏灯就多一个pass。
            //Deferred: 延迟渲染,渲染g缓冲区 。
            //ShadowCaster:将物体深度渲染到阴影贴图或者深度纹理上 。
            //PrepassBase: 用于传统的延迟光照,渲染法线和高光效果。
            //PrepassFinal:用于传统的延迟光照,通过结合文理、灯光、和法线来渲染最终的结果。
            //Vertex:当对象不是光映射时,用于遗留顶点的渲染,所有顶点灯都被利用。
            //VertexLMRGBM: 当对象被光映射时,在遗留的顶点上使用渲染,在LightMap是RGBM编码的平台上(pc和控制台)。
            //VertexLM: 当对象被光映射时,在遗留的顶点上使用渲染,在LightMap是双idr编码的(移动平台)平台上。
            Tags { "RenderType"="Opaque" "LightMode" = "ForwardBase"}//////
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "UnityCG.cginc"
             #pragma target 3.0 

            //衰减与阴影的实现
            #include "AutoLight.cginc"//////
              //fwdadd:ForwardBase的阴影显示,在下面的ForwardAdd里得用fwdadd; 必须结合fallback,两者缺一不可
            #pragma multi_compile_fwdadd_fullshadows//////
            sampler2D _MainTex;
            float4 _MainTex_ST;
            //定义一个灯光,名字为固定格式,会自动取场景中灯光
            float4 _LightColor0;//////

            struct appdata
            {
                float4 vertex : POSITION;
                float2 uv : TEXCOORD0;
                float4 normal:NORMAL;//////
            };

            struct v2f
            {
                float2 uv : TEXCOORD0;
                float4 pos: SV_POSITION;
                float3 normal :TEXCOORD1;//////
                 //点光源需要的衰减
                LIGHTING_COORDS(3,4)//////#include "AutoLight.cginc"
            };
            v2f vert (appdata v)
            {
                v2f o;
                //这里一般用 o.pos,用o.vertex有时候会报错
                o.pos= UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                o.normal = v.normal;//////
                //点光源需要的衰减
                TRANSFER_VERTEX_TO_FRAGMENT(o)//////#include "AutoLight.cginc"
                return o;
            }
            fixed4 frag (v2f i) : SV_Target
            {
                //物体法向量转化为世界法向量
                float3 N = normalize(UnityObjectToWorldNormal(i.normal));//////
                //世界光向量:unity封装好的光向量,会自动调用场景里面的存在的灯光
                float3 L =normalize( _WorldSpaceLightPos0.xyz);//////
                                                 //点光源需要的衰减系数
                float atten = LIGHT_ATTENUATION(i);//////#include "AutoLight.cginc"

                fixed4 col = tex2D(_MainTex, i.uv);
                //最终颜色 = 主颜色 x( 灯光颜色 x 漫反射系数 x衰减系数 + 环境光)
                col.rgb = col.rgb * (_LightColor0.rgb* saturate(dot(N,L)) *atten  + UNITY_LIGHTMODEL_AMBIENT);//////

                return col;
            }
            ENDCG
            }

            //多盏灯叠加
        Pass//////
        {
            Tags { "RenderType"="Opaque" "LightMode" = "ForwardAdd"} //ForwardAdd :多灯混合//////
            Blend One One//////
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "UnityCG.cginc"
            #pragma target 3.0

            //衰减与阴影的实现
            #include "AutoLight.cginc"//////
              //fwdadd:ForwardAdd的阴影显示,在上面的ForwardBase里得用fwdbase; 必须结合fallback,两者缺一不可
            #pragma multi_compile_fwdadd_fullshadows//////

            sampler2D _MainTex;
            float4 _MainTex_ST;
            //定义一个灯光,名字为固定格式,会自动取场景中灯光
            float4 _LightColor0;//////

            struct appdata
            {
                float4 vertex : POSITION;
                float2 uv : TEXCOORD0;
                float4 normal:NORMAL;//////
            };

            struct v2f
            {
                float2 uv : TEXCOORD0;
                float4 pos: SV_POSITION;
                float3 normal :TEXCOORD1;//////
                float4 wPos :TEXCOORD2;//////
                //点光源需要的衰减
                LIGHTING_COORDS(3,4)//////#include "AutoLight.cginc"
            };
            v2f vert (appdata v)
            {
                v2f o;
                 //这里一般用 o.pos,用o.vertex有时候会报错
                o.pos= UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                o.wPos= mul(unity_ObjectToWorld, v.vertex);//////
                o.normal = v.normal;//////
                //点光源需要的衰减
                TRANSFER_VERTEX_TO_FRAGMENT(o)//////#include "AutoLight.cginc"
                return o;
            }
            fixed4 frag (v2f i) : SV_Target
            {
                //物体法向量转化为世界法向量
                float3 N = normalize(UnityObjectToWorldNormal(i.normal));//////

                //世界光向量:这里计算的是点光源,按照灯光的距离来算衰减,第一个pass不需要
                float3 L = normalize (lerp(_WorldSpaceLightPos0.xyz , _WorldSpaceLightPos0.xyz - i.wPos.xyz , _WorldSpaceLightPos0.w));//////

                //点光源需要的衰减
                float atten = LIGHT_ATTENUATION(i);//////#include "AutoLight.cginc"

                fixed4 col = tex2D(_MainTex, i.uv);

                //最终颜色 = 主颜色 x 灯光颜色 x 漫反射系数 x 衰减系数  第一个pass已经有了环境色 这里就不能加了
                col.rgb = col.rgb * _LightColor0.rgb * saturate(dot(N,L))*atten;//////

                return col;
            }
        ENDCG
        }
    }
            //需要产生阴影
            FallBack "Diffuse"
}

 
时间: 2024-10-10 08:13:12

shader之——多光源漫反射以及阴影的相关文章

shader之——单光源漫反射

Shader "Unlit/long" { Properties { _MainTex ("Texture", 2D) = "white" {} } SubShader { Pass { //Always: 总是渲染:没有光照模式. //ForwardBase: 适用于前渲染.环境.主要方向灯.光/sh光和烘焙图. //ForwardAdd: 适用于前渲染, 叠加每一盏灯,每一盏灯就多一个pass. //Deferred: 延迟渲染,渲染g缓冲区

解读Unity中的CG编写Shader系列八(多光源漫反射)

转自http://www.itnose.net/detail/6117338.html 前文中完成最简单的漫反射shader只是单个光源下的漫反射,而往往场景中不仅仅只有一个光源,那么多个光源的情况下我们的物体表面的漫反射强度如何叠加在一起呢?前文打的tag "LightMode"="ForwardBase"又是什么意思呢? Unity内置的DiffuseShader,也就是我们创建一个Material出来时默认的Shader也是多光源的,所以这篇文章完成的shad

解读Unity中的CG编写Shader系列8——多光源漫反射

前文中完成最简单的漫反射shader只是单个光源下的漫反射,而往往场景中不仅仅只有一个光源,那么多个光源的情况下我们的物体表面的漫反射强度如何叠加在一起呢?前文打的tag "LightMode"="ForwardBase"又是什么意思呢? Unity内置的DiffuseShader,也就是我们创建一个Material出来时默认的Shader也是多光源的,所以这篇文章完成的shader与默认的diffuse shader基本效果一致. 首先引入几个概念 渲染路径 Re

关于Unity动态物体无法向使用使用custom shader和lightmap的物体投射阴影

最近在做unity shader forge和marmoset的优化,TA那边遇到了一个阴影显示的问题,具体如下: 在Forward Rendering状态下,静态场景使用了是shader forge生成的blendlayer类的shader,使用lightmap烘培贴图后,动态角色走到静态物体附近时,方向光在角色上的投影,无法投射到使用shader forge材质的物体上,但其他材质和使用marmoset的材质可以正常接收.查询了一些网站解决方案后,最终确定是custom shader 写法的

Cg入门17:Fragment shader - 片段级光照(添加阴影)

投射阴影 方法一:添加一个单独的pass通道 pass { Tags{"LightMode" = "ShadowCaster"} } ShadowCaster:阴影投射器,可以投射阴影 方法二:添加物体默认阴影投射 FallBack "Diffuse" 然后给光线开启阴影: 效果如下:(效果图为添加平行光) 如果要让我们的shader支持点光源阴影投射,就添加 FallBack "Diffuse" 添加点光源后效果: 接收阴影

解读Unity中的CG编写Shader系列6——漫反射

如果前面几个系列文章的内容过于冗长缺乏趣味着实见谅,由于时间原因前面的混合部分还没有写完,等以后再补充,现在开始关于反射的内容了. 折射与反射 在物理世界中,光的反射与折射往往是同时存在的,光源由真空或者空气中射入一种材料,光在进入这种材料的同时就发生了折射,折射的程度与各个介质的折射率有关,使光的传播路线偏离原来的路线: 继而如果光在通过不同传播介质的表面时,会像乒乓球一样弹回来,我们人眼能够看到东西,都是因为东西会反射光源,如果一种物质无法反射光,或者没有光源,我们就看不到东西.同样对于不同

Unity Shader——Writing Surface Shaders(0)

从今天起,开始翻译Unity关于shader的官方文档.翻译水平比较一般,目的主要是通过翻译来提升对shader的见解,也让其他人更容易的了解shader.以下开始正文内容: 编写Surface Shaders 和光交互的shader写起来很复杂,有不同的光照类型.阴影选项.渲染路径(正向渲染和延迟渲染),有时shader需要考虑所有的复杂性. Unity中的Surface Shader是一个代码生成器,用它来写光照shader(lit shader)相比于使用低阶的顶点/像素shader(ve

【浅墨Unity3D Shader编程】之三 光之城堡篇:子着色器、通道与标签的写法 & 纹理混合

本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/41175585 作者:毛星云(浅墨)    微博:http://weibo.com/u/1723155442 邮箱: [email protected] 本文介绍了Unity中子着色器.通道和标签相关的详细概念与写法,以及纹理的设置方法,基本的纹理混合写法,写了5个Shader作为本文Shader讲解的实战内容,最后创建了一个梦幻的光之

【Unity Shaders】Mobile Shader Adjustment—— 什么是高效的Shader

本系列主要参考<Unity Shaders and Effects Cookbook>一书(感谢原书作者),同时会加上一点个人理解或拓展. 这里是本书所有的插图.这里是本书所需的代码和资源(当然你也可以从官网下载). ========================================== 分割线 ========================================== 写在前面 之前学习的各种Shader时,我们从没有考虑在所有平台下的可用性.Unity是一个强大的跨