【Unity Shader】Unity阴影

这篇具有很好参考价值的文章主要介绍了【Unity Shader】Unity阴影。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

背景

记录下在unity中如果想实现阴影,有哪些路子可以选择,目前看有两种
1.经典的shadowmap
2.planar projection

ShadowMap方式

如果开启renderer组件的cast shadows为on,开启平行光的light组件的shadow type,那么就会在物体shader中寻找LightMode=ShadowCaster的Pass进行渲染
场景有两个物体,平面和球体,使用unity内置的函数
平面的shader为

Shader "Unlit/plane_shader3"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
    }
    SubShader
    {
        Tags
        {
            "RenderType"="Opaque"
        }
        LOD 100

        Pass
        {
            Tags
            {
                "LightMode"="Forwardbase"
            }

            CGPROGRAM
            #pragma multi_compile_fwdbase;
            #pragma vertex vert
            #pragma fragment frag
            #include "UnityCG.cginc"
            #include "AutoLight.cginc"

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

            struct v2f
            {
                float2 uv : TEXCOORD0;
                float4 pos : SV_POSITION;
                SHADOW_COORDS(3) // float4 _ShadowCoord : TEXCOORD3;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;

            v2f vert(appdata v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                TRANSFER_SHADOW(o) //  a._ShadowCoord = ComputeScreenPos(a.pos);
                return o;
            }

            fixed4 frag(v2f i) : SV_Target
            {
                fixed shadow = SHADOW_ATTENUATION(i);
                fixed4 col = tex2D(_MainTex, i.uv);
                fixed4 col2 = fixed4(col.xyz * shadow, 1.0);
                return col2;
            }
            ENDCG
        }
        // 下面是需要写入深度图的物体添加这个pass
        Pass
        {
            Tags
            {
                "LightMode"="ShadowCaster"
            }

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

            struct v2f
            {
                V2F_SHADOW_CASTER;
            };

            v2f vert(appdata_base v)
            {
                v2f o;
                TRANSFER_SHADOW_CASTER_NORMALOFFSET(o)
                return o;
            }

            float4 frag(v2f i) : SV_Target
            {
                SHADOW_CASTER_FRAGMENT(i)
            }
            ENDCG
        }
    }
    //    FallBack "Specular"
}

球体的shader为

Shader "Unlit/sphere_shader"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
    }
    SubShader
    {
        Tags
        {
            "RenderType"="Opaque"
        }
        LOD 100

        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma multi_compile_fwdbase

            #include "UnityCG.cginc"

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

            struct v2f
            {
                float2 uv : TEXCOORD0;
                float4 vertex : SV_POSITION;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;

            v2f vert(appdata v)
            {
                v2f o;
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                return o;
            }

            fixed4 frag(v2f i) : SV_Target
            {
                // sample the texture
                fixed4 col = tex2D(_MainTex, i.uv);
                // apply fog
                return col;
            }
            ENDCG
        }

        // 下面是需要写入深度图的物体添加这个pass
        Pass
        {
            Name "ShadowCaster"
            Tags
            {
                "LightMode"="ShadowCaster"
            }

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

            struct v2f
            {
                V2F_SHADOW_CASTER;
            };

            v2f vert(appdata_base v)
            {
                v2f o;
                TRANSFER_SHADOW_CASTER_NORMALOFFSET(o)
                return o;
            }

            float4 frag(v2f i) : SV_Target
            {
                SHADOW_CASTER_FRAGMENT(i)
            }
            ENDCG
        }
    }
}

planar projection 平面投影方式

【Unity Shader】Unity阴影
球体和立方体挂载下面这个shader,就是多渲染一个阴影投影到y=0的平面上

// shader,放在需要显示阴影的对象上
Shader "Custom/PlanarShadow1" {
	Properties{
	_Instensity("Shininess", Range(2, 4)) = 2.0  //光照强度
 
	_Diffuse("Diffuse Color",Color) = (1,1,1,1)
    //纹理贴图
	_MainTex("Main Tex",2D) = "white"{}
	//控制纹理颜色
	_Color("Color",Color) = (1,1,1,1)
	}
 
		SubShader{
//光照计算
	    Pass{
			Tags{"LightMode" = "ForwardBase"}
			    CGPROGRAM
	            #include "Lighting.cginc"
	            #pragma vertex vert
	            #pragma fragment frag
 
		//fixed4 _Diffuse;
		fixed4 _Color;
		sampler2D _MainTex;
		float4 _MainTex_ST;
		fixed4 _Specular;
		half _Gloss;
 
		struct a2v {
			float4 vertex:POSITION;//告诉unity把模型空间下的顶点坐标填充给vertex
			float3 normal:NORMAL;
			//纹理坐标  然后还要贴图,就可以取到该坐标的颜色,然后替换漫反射的颜色
			float4 texcoord:TEXCOORD0;
		};
		struct v2f {
			float4 svPos:SV_POSITION;
			float3 worldNormal:TEXCOORD0;
			float4 worldVertex:TEXCOORD1;
			float2 uv:TEXCOORD2;
		};
 
		v2f vert(a2v v) {
			v2f f;
			f.svPos = UnityObjectToClipPos(v.vertex);
			f.worldNormal = UnityObjectToWorldNormal(v.normal);
			f.worldVertex = mul(v.vertex, unity_WorldToObject);
			f.uv = v.texcoord.xy * _MainTex_ST.xy + _MainTex_ST.zw;
			return f;
		}
		fixed4 frag(v2f f) :SV_Target{
			//法线的方向
			fixed3 normalDir = normalize(f.worldNormal);
		    //光线的方向
			fixed3 lightDir = normalize(WorldSpaceLightDir(f.worldVertex));
			//返回的颜色值代替漫反射的颜色
			fixed3 texColor = tex2D(_MainTex, f.uv.xy)*_Color.rgb;
 
			fixed3 diffuse = _LightColor0.rgb * texColor * max(dot(normalDir, lightDir), 0);
			//反射光的方向
			//fixed3 reflectDir = normalize(reflect(-lightDir, normalDir));
			//视野方向
			fixed3 viewDir = normalize(UnityWorldSpaceViewDir(f.worldVertex));
			//blinn-Phong光照模型   计算平分线
			fixed3 halfDir = normalize(lightDir + viewDir);
 
			//漫反射光照+上环境光+纹理颜色
			fixed3 tempColor = diffuse + UNITY_LIGHTMODEL_AMBIENT.rgb*texColor;
 
			return fixed4(tempColor, 1);
 
		}
 
		 ENDCG
	   }
 
 
 
	//计算阴影
	Pass
	{
		Tags{"LightMode" = "ForwardBase"}
		  Stencil          //加个模板
			{
				Ref 0
				Comp equal
				Pass incrWrap
				Fail keep
				ZFail keep
			}
			ZWrite off
 
	//	Blend DstColor SrcColor
		Blend Srcalpha OneminusSrcAlpha
		Offset -1, -1		//使阴影在平面之上  
		CGPROGRAM
		#pragma vertex vert
		#pragma fragment frag
		#include "UnityCG.cginc"
 
		float4x4 _World2Ground;  //阴影接收平面(世界空间到模型空间的转换矩阵)
		float4x4 _Ground2World;	 //阴影接收平面(模型空间到世界空间的转换矩阵)
		float _Instensity;
 
		struct v2f {
			float4 pos:SV_POSITION;
			float atten : TEXCOORD0;
		};
 
	   v2f vert(float4 vertex:POSITION)
	   {
		   float3 litDir;
		   litDir = WorldSpaceLightDir(vertex);//世界空间主光照相对于当前物体的方向
		   litDir = mul(_World2Ground,float4(litDir,0)).xyz;//光源方向转换到接受阴影的平面空间
		   litDir = normalize(litDir);// 归一
		   float4 vt;
		   vt = mul(unity_ObjectToWorld,vertex); //将当前物体转换到世界空间
		   vt = mul(_World2Ground,vt); // 将物体在世界空间的矩阵转换到地面空间
		   vt.xz = vt.xz - (vt.y / litDir.y)*litDir.xz;// 用三角形相似计算沿光源方向投射后的XZ
		   vt.y = 0;// 使阴影保持在接受平面上
		   vt = mul(_Ground2World, vt); // 阴影顶点矩阵返回到世界空间
		   vt = mul(unity_WorldToObject, vt); // 返回到物体的坐标
		   v2f o;
		   o.pos = UnityObjectToClipPos(vt);//输出到裁剪空间
		   o.atten = distance(vertex, vt) / _Instensity;// 根据物体顶点到阴影的距离计算衰减
		   return o;
	   }
 
	   float4 frag(v2f i) :COLOR
	   {
		   return float4(0.3, 0.3, 0.3, 0.5);//一个灰色的阴影出来了
			//return smoothstep(0,1,i.atten / 2);
		}
 
		ENDCG
	   }
	}
}

备注

unity中屏幕空间的投影方式和传统shadowmap方式有区别,这里记录下
【Unity Shader】Unity阴影

参考

1.shadowmap https://www.bilibili.com/read/cv13646967/
2.平面投影
https://zhuanlan.zhihu.com/p/42781261
https://blog.csdn.net/cgy56191948/article/details/84990842文章来源地址https://www.toymoban.com/news/detail-493541.html

到了这里,关于【Unity Shader】Unity阴影的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Unity Shader 学习笔记(4)URP渲染管线带阴影PBR-Shader模板 -- 新增可自定义阴影颜色

    材质面板截图 功能实现(URP渲染管线下): 1、进一步优化Shader结构和算法; 2、包含PBR材质; 3、投射和接收阴影,并升级 支持自定义阴影颜色 ; 4、支持点光源照射(但不支持点光源阴影)。 通用渲染截图 自定义阴影颜色截图 完整代码: 写在最后: 1、在我的上一篇文

    2024年02月12日
    浏览(29)
  • Unity Shader:背景缓动

       这算是一个很常见的小功能,比如我们在玩横版游戏的时候,背景动画会以一定的频率运动,其实现方式也有很多种。   比如,使用UGUI的image+animtion动画的方式,自己k桢实现。   还可以使用材质球本身的功能来实现,关键函数如下:    配合的材质球如下:

    2024年02月13日
    浏览(27)
  • 【Unity Shader】Unity中利用GrabPass实现玻璃效果

    《入门精要》中模拟玻璃是用了Unity里的一个特殊的Pass来实现的,这个Pass就是 GrabPass ,比起上一篇博客实现镜子的方法,这个方法我认为相对复杂,因此在实现之前需要对GrabPass及实现原理做一个更加详细的介绍。 场景物体拜访和贴图完全参考《入门精要》: 以及当前场景

    2024年02月09日
    浏览(33)
  • unity GI Shader 实现

    之前分享了一篇对unity全局光照的解析,里面提到了一些东西,需要在Shader内实现,在这一篇补上。 要实现对全局GI的shader实现,我们可以通过对unity内置的Lit进行解析查看。 烘焙的方式有很多种,选择合适的方式烘焙和使用合适类型的光源尤为重要。 首先,我们先实现一下

    2024年02月10日
    浏览(28)
  • Unity用Shader实现边缘光效果

    《自学记录》 1、先创建一个Cube,再创建两个材质球Cube、Unilt 2、再创建一个shader代码UniltShader【Project右键Create-Shader-NewSurfaceShader】把里面原来的代码删除,写入下面的代码 3、把shader UniltShader拖给材质球Unilt 4、把Cube的Mesh Renderer中Materials的Size改为2,然后把材质球Cube、Unilt分

    2024年02月08日
    浏览(34)
  • Unity 使用Shader实现序列帧动画

    序列帧动画图片使用shader逐帧播放 可以直接使用,拿走不谢。 可以挂到材质上放入Image组件的材质中使用。

    2024年02月13日
    浏览(31)
  • Unity Shader学习(九)物体边缘实现

    根据前面的学习,我们了解到除了可以对点的颜色进行处理,还可以对点本身进行操作,例如我们可以改变点的位置,这样就可以实现对模型渲染的操控。物体边缘效果是我们常用的一种效果,要实现物体边缘,原理也很简单。 首先我们要了解到,模型在渲染时,有正面和背

    2024年02月16日
    浏览(37)
  • 实现窗户特效的Unity Shader解析

            本文将详细介绍一种使用Unity Shader实现窗户特效的方法。通过分析代码,我们将解释每个关键部分的作用,以及如何将其组合在一起以实现逼真的窗户效果。希望本文能为Shader编程初学者和Unity开发者提供一些有用的指导。   引言:         在游戏和虚拟现实

    2024年02月12日
    浏览(34)
  • unity shader 实现通用描边shader -文字描边-字体描边

    在制作游戏时,可以遇到要对字体添加描边的需求,unity 的UGUI自带的OutLine组件,描边效果不好,宽度过大会出现穿帮,顶点数量也会增加,性能不好,如果对于有几百字,顶点数量会很多,而且无法扩展功能 可以看出Outline创建了4个方向的文字 Unity5.2以前的版本要求,每一个Canvas下至多

    2024年02月03日
    浏览(33)
  • Unity中Shader的遮罩的实现

    Unity中Shader的遮罩的实现 效果演示: 修改后的代码 效果:

    2024年02月06日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包