Unity shader学习之菲涅耳反射

本文介绍了菲涅尔反射的基本概念,并给出了Schlick菲涅尔近似等式及Empricial菲涅尔近似等式的详细解析。此外,还提供了一个使用Schlick菲涅尔近似等式的具体Shader实现案例。

菲涅尔反射(Fresnel reflection),指光线照射物体表面时,一部分发生反射,一部分进入物体内部发生折射或散射,被反射的光和折射光之间存在一定的比率。

2个公式:

1. Schlick 菲涅耳近似等式:

FSchlick(v, n) = F0 + (1 - F0)(1 - dot(v, n))5

其中F0是一个反射系数,用于控制菲涅尔反射的强度,v 是视角方向, n 是表面法线。

2. Empricial 菲涅耳近似等式:

FEmpricial(v, n) = max(0, min(1, bias + scale * (1- dot(v, n)power)))

其中,bias, scale 和 power 是控制项。

如使用第一个公式的shader如下:

Shader "Custom/Fresnel Reflection"
{
    Properties
    {
        _Cubemap("Cubemap", Cube) = "" {}
        _RefractRatio("Reflect Ratio", Range(0, 1)) = 0.667
        _F0("F0", Range(0, 1)) = 0.5
    }

    SubShader
    {
        Pass
        {
            Tags
            {
                "LightMode" = "ForwardBase"
            }
            Cull Off

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

            #include "UnityCG.cginc"
            #include "Lighting.cginc"
            #include "AutoLight.cginc"

            samplerCUBE _Cubemap;
            float _RefractRatio;
            float _F0;

            struct appdata
            {
                float4 vertex : POSITION;
                fixed4 color : COLOR;
                float3 normal : NORMAl;
            };

            struct v2f
            {
                float4 pos : SV_POSITION;
                fixed4 color : COLOR;
                float3 worldNormal : TEXCOORD0;
                float3 worldPos : TEXCOORD1;
                SHADOW_COORDS(2)
            };

            v2f vert(appdata v)
            {
                v2f o;
                o.pos = mul(UNITY_MATRIX_MVP, v.vertex);
                o.color = v.color;
                o.worldNormal = normalize(UnityObjectToWorldNormal(v.normal));
                o.worldPos = mul(unity_ObjectToWorld, v.vertex);
                TRANSFER_SHADOW(o);
                return o;
            }

            fixed4 frag(v2f i) : SV_TARGET
            {
                fixed3 albedo = i.color.rgb;

                fixed3 ambient = albedo * UNITY_LIGHTMODEL_AMBIENT.rgb;

                float3 worldView = UnityWorldSpaceViewDir(i.worldPos);
                float3 reflectDir = reflect(-worldView, i.worldNormal);
                float3 refractDir = refract(-normalize(worldView), normalize(i.worldNormal), _RefractRatio);
                fixed3 reflectCol = texCUBE(_Cubemap, reflectDir);
                fixed3 refractCol = texCUBE(_Cubemap, refractDir);
                float schlick = _F0 + (1 - _F0) * pow(1 - dot(worldView, i.worldNormal), 5);

                UNITY_LIGHT_ATTENUATION(atten, i, i.worldPos);

                fixed3 col = ambient + lerp(reflectCol, refractCol, schlick) * atten;

                return fixed4(col, 1);
            }

            ENDCG
        }
    }

    Fallback "VertexLit"
}

效果如下:

 

转载于:https://www.cnblogs.com/jietian331/p/7200566.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值