Unity URP shader ———魔系符文宝石是如何练成的

各位同学大家好  

       我已经很久没有没有写教程了,最近项目比较忙。各种加班各种带小孩儿,不过,老师一有机会也在给尽可能服务大家,今天来一个硬菜:移动端高效魔系符文如何制作,国庆起来,老师抽了点儿时间做了一个宝石模型和贴图:

       老规矩,先看效果:

           

Unity URP Shader——魔系符文宝石是如何练成的

实现原理:

         1:菲尼尔实现边缘光,

               菲尼尔共公式:Fresnel =(1-dot (v,n))^power  ;

                        v  : ViewDir            n: normalWorld 

        2:模拟反射

              因为移动端处理,所以我们做模拟,不做真实的,大致原理是:

               取一张贴图*2次菲尼尔运算。最后去控制强度和颜色,实现我们想要的效果。 

     

 half4 Reflect =reflect * saturate(XionmgMaoWuDao_Fresnel(normalWorld,viewDir,3)*XionmgMaoWuDao_Fresnel(normalWorld,viewDir,3)) *_ReflectIntensity*_ReflectColor;

       3:模拟宝石内部效果

            UV计算: 宝石是在不同的视角下观看宝石,会看见 视乎宝石里面的画面与镜子相同,都是无穷大。那么计算机科学也是同样的原理,也是从观察角度去看,所以UV的计算方式如下:

           视角方向的(XY方向)-原点的XY方向,然后再做模型的Tiling运算。最后再计算出模型法线在世界空间中的XY方向  。加在一起实现视觉错觉运算。

        

   half2 InsideUV = (viewDir.xy - float3 (0,0,0).xy)*_InsideUvTiling.xy+_InsideUvTiling.zw;
			          InsideUV+= normalWorld.xy * _InsideUvIntensity ;

 

最终代码如下:

 

//熊猫悟道
Shader "XiongMaoWuDao/XiongMaoLaoShi_Gem"
{
    Properties
    {
	    _normalPaner ("normal Panner",vector) =(0,0,0,0)
        [MainColor] _BaseColor("BaseColor", Color) = (1,1,1,1)
		_ReflectColor ("Reflect Color ",Color) = (1,1,1,1)
		[HDR]_RimColor ("Rim Color" ,Color)= (1,1,1,1)
		_EimissionMap ("Eimiss Map",2D) = "white" {}
		_ReflectIntensity ("Reflect Intensity",float) =1
		_InsideUvTiling ("Inside UV Tiling" , vector) =(1,1,1,1)
		_InsideUvIntensity ("Inside Uv Intensity" ,float) = 1
		_InsideMap ("Inside Map",2D)="white"{}    
		
    }
    SubShader
    {
        Tags { "RenderType"="Opaque" "RenderPipeline"="UniversalRenderPipeline" "Queue"="Geometry"}

        Pass
        {
            HLSLPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            
            #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"

            struct Attributes
            {
                float4 positionOS   : POSITION;
				half3 normal        :NORMAL ;
                half2 uv           : TEXCOORD0;
            };

            struct Varyings
            {
                half2 uv           : TEXCOORD0;
                float4 positionHCS  : SV_POSITION;
				half3  viewDir      :TEXCOORD1;
				half3  normalWorld  :TEXCOORD2 ;
            };

            TEXTURE2D(_EimissionMap);
            SAMPLER(sampler_EimissionMap);
			TEXTURE2D(_InsideMap);
            SAMPLER(sampler_InsideMap);

            CBUFFER_START(UnityPerMaterial)
            float4 _EimissionMap_ST;
            half4 _RimColor;
			half3 _normalPaner;
			half  _ReflectIntensity;
			half4 _InsideUvTiling;
			half  _InsideUvIntensity;
			half4 _InsideMap_ST;
			half4 _BaseColor;
			half4 _ReflectColor;
            CBUFFER_END

			//宝石菲尼尔反射
			half XionmgMaoWuDao_Fresnel( half3 normal ,half3 viewDir ,half power)
			{
			//菲尼尔公式: (1-dot (v,n))^power ;
			return pow( 1.0- saturate(dot(normal,viewDir).x) ,power);
			}

            Varyings vert(Attributes v)
            {
                Varyings o;
				half3 posWS = TransformObjectToWorld (v.positionOS.xyz);
                o.positionHCS = TransformObjectToHClip(v.positionOS.xyz);
				o.normalWorld = TransformObjectToWorldNormal (v.normal +_normalPaner);
				o.viewDir  =GetWorldSpaceNormalizeViewDir(posWS);
                o.uv = TRANSFORM_TEX(v.uv, _EimissionMap);
                return o;
            }

            half4 frag(Varyings i) : SV_Target
            {
			   //初始化数据
			   half3 normalWorld = normalize (i.normalWorld);
			   half3 viewDir = normalize (i.viewDir);
			   //DIFFUSE
			   half4 Emiss = SAMPLE_TEXTURE2D(_EimissionMap, sampler_EimissionMap, i.uv);
			   half4 DiffuseA = (XionmgMaoWuDao_Fresnel(normalWorld,viewDir,3)+Emiss.r)*_RimColor;
			   //reflect
			   half reflect = SAMPLE_TEXTURE2D(_EimissionMap, sampler_EimissionMap, normalWorld.xy).g;
			   half4 Reflect =reflect * saturate(XionmgMaoWuDao_Fresnel(normalWorld,viewDir,3)*XionmgMaoWuDao_Fresnel(normalWorld,viewDir,3)) *_ReflectIntensity*_ReflectColor;
			   //Inside
			   half2 InsideUV = (viewDir.xy - float3 (0,0,0).xy)*_InsideUvTiling.xy+_InsideUvTiling.zw;
			          InsideUV+= normalWorld.xy * _InsideUvIntensity ;
		       half4 InsideMap = SAMPLE_TEXTURE2D(_InsideMap, sampler_InsideMap, InsideUV);
			   half4 totalColor = (DiffuseA+InsideMap)*_BaseColor + Reflect ;
                return totalColor;
            }
            ENDHLSL
        }
    }
}

              

     

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值