Unity3D Shader系列之模板测试

一、 模板测试原理

模板测试位于GPU渲染流水线的逐片元操作阶段,片元着色器完成之后就会进入模板测试,模板测试通过后再进入深度测试。我们的GPU中有一个模板缓冲区(Stencil Buffer)(Stencil即是模板的意思),其大小为整个屏幕大小*8位,即屏幕上的每一个像素点都存储有一个模板值,该模板值是8位的,所以其值范围为0~255。举个例子,如果我们屏幕的大小是1920×1080的,那么模板缓冲区的大小就为1920×1080×8位。

模板测试的原理很简单:如果我们在Shader中开启了模板测试,要渲染某个像素的时候,就会读取读取Shader中设置的参考值Ref(我们按需求设置),同时模板缓冲区中该像素对应的模板值StencilValueInBuffer,如果参考值Ref与模板缓冲区中的值StencilValueInBuffer满足某个条件(此条件是我们在Shader中按我们的需求指定的),则测试通过,该像素可以渲染;如果不满足,则该像素不可以渲染。无论深度测试是否通过,Shader中都有权利向模板缓冲区写入模板值(前提是开启了模板测试)。

二、 模板测试示例

如果Shader中模板测试关闭,即不进行模板测试,那么该Shader是没有权利写入模板缓冲区的。

下面来看看一个具体的例子,方便我们来理解模板测试的作用。

场景中,我们想要渲染两个物体Plane和Cube,我想要实现的效果是只有通过Plane才能够看到Cube,但是Plane本身是不可见的,相当于把Plane当作透视镜样。

场景如下图。

我们想要的结果如下。

想要的结果再具体点就是我们只想展示Cube被Plane挡住的部分,如下面的蓝色框部分,Plane也不显示。

要实现这一效果,就需要用到模板测试了。

①首先,我们需要先渲染Plane,再渲染Cube。注意这一点很重要,如果先渲染的是Cube,再渲染的Plane使用模板测试是不能实现这种效果的。如何保证一定是先渲染Plane再渲染Cube呢,很简单,将Plane的渲染队列(Render Queue)设置得比Cube的渲染队列小即可。关于渲染队列的知识点,我们在《Unity3D Shader系列之透视效果XRay》中专门有一节说明,这里不再赘述。

②然后,Plane的Shader关闭深度写入(ZWrite off)同时不输出颜色(ColorMask 0),开启模板测试,并一直测试通过,同时向模板缓冲区中写入值1。

解释一下上面这句话中各个指令的作用:

– 关闭深度写入(ZWrite off)

来保证后渲染的Cube总能通过深度测试。如果Plane开启了深度写入,那么后渲染的Cube中被Plane挡住的部分将永远不能通过深度测试,那部分也就永远不会绘制在屏幕上了,而我们想要的是刚好只显示这部分。

– 不输出颜色(ColorMask 0)

就是渲染Plane时,Plane的任何颜色都不写入颜色缓冲区,就相当于Plane看不见样(虽然Plane的确渲染了,只是看不到)

– 开启模板测试,并一直测试通过,同时向模板缓冲区中写入值1

GPU绘制完Plane后,模板缓冲区的值如下图。即Plane的部分值为1,模板缓冲区的其他部分仍为0。

③最后,我们渲染Cube,Cube中开启深度测试,模板测试参考值设置为1,只有当模板缓冲区中的值为1时,才能通过模板测试,从而实现我们想要的效果。

我们正常渲染Cube的范围是下面的白色框,但是由于只有蓝色框部分模板缓冲区中的值才是1,所以只有蓝色框部分能通过模板测试,左边的红色部分由于模板缓冲区中的值为0,所以不能通过模板测试,也就不能够渲染出来。

2.3 Unity3D中使用模板测试

Unity的Shader中使用模板测试很简单,在Shader中添加上Stencil代码块即可。如果没有Stencil代码块,就认为是关闭模板测试的。

Stencil
{}

从2.1节讲到的模板测试原理我们可以知道,模板测试并不是完全可编程的,我们只能去简单配置几个指令。模板测试的配置项包括如下几个

– 参考值 Ref

– 比较函数 Comp

– 若模板测试通过写入方式 Pass

– 若模板测试不通过写入方式 Fail

– 若模板测试通过但深度测试不通过模板缓冲区写入方式 ZFail

– 读掩码ReadMask和写掩码WriteMask

可参考官方文档

2.3.1 参考值

语法如下,Ref后面即参考值,值范围为0~255。

Stencil
{
    Ref 0}

当然很多时候我们想通过编辑器面板更改参考值,不想在Shader中直接写死,我们可以如下操作。

– 在Shader的Properties定义一个Int值_Ref

[_IntRange]用于限定编辑器面板中只能调节为整数值(ps:[_IntRange]是MaterialPropertyDrawer中的一种,可参考官方文档

Properties
{[IntRange]_Ref("Ref",Range(0,255))=0}
  • 然后将Ref 0改为Ref [_Ref]即可

Stencil
{
    Ref [_Ref]}

2.3.2 比较函数

语法如下,Comp后面跟的就是比较函数,默认值为Always,即模板测试一直通过。

Stencil
{Comp Always
}

可设置为以下函数。

2.3.3 模板操作

模板操作包括Pass、Fail、 ZFail。

Pass用来指定模板测试通过时写入什么值到模板缓冲区中;

Fail用来指定模板测试不通过时写入什么值到模板缓冲区中;

Pass用来指定板测试通过但深度测试不通过时写入什么值到模板缓冲区中。

Stencil
{PassReplaceFailKeepZFail Keep
}

可设置的值如下。三者默认值均为Keep。

2.3.4 读写掩码

读写掩码默认值均为255。

在进行比较时,是使用参考值&(按位与)读掩码来作为实际比较的参考值。

在进行模板操作时,是使用想写入的值&(按位与)写掩码来作为实际写入的值。

由于默认值都是255,任何8bit的值与255按位与&都等于原值,所以我们没明确指定读写掩码时,在进行比较时,使用的就是我们的参考值;在进行写入时,使用的就是我们的原始写入值。

Stencil
{
    ReadMask 255
    WriteMask 255}

3 示例代码

Plane用到的Shader。

Shader "LaoWang/Unlit/Plane"{
    Properties
    {[IntRange]_Ref("Ref",Range(0,255))=0}
    SubShader
    {
        Tags {"RenderType"="Opaque"}
        LOD 100

        Pass
        {
            Stencil
            {
                Ref [_Ref]
                Comp Always
                Pass Replace
            }

            ZWrite Off
            ColorMask 0

            CGPROGRAM
            #pragma vertex vert#pragma fragment frag#include"UnityCG.cginc"structappdata{
                float4 vertex : POSITION;};structv2f{
                float4 vertex : SV_POSITION;};

            fixed4 _Color;

            v2f vert (appdata v){
                v2f o;
                o.vertex =UnityObjectToClipPos(v.vertex);return o;}

            fixed4 frag (v2f i): SV_Target
            {returnfixed4(0,0,0,0);}
            ENDCG
        }}}

Cube用到的Shader。

Shader "LaoWang/Unlit/Cube"{
    Properties
    {_Color("Color", color)=(1.0,1.0,1.0,1.0)[IntRange]_Ref("Ref",Range(0,255))=0}
    SubShader
    {
        Tags {"RenderType"="Opaque"}
        LOD 100

        Pass
        {
            Stencil
            {
                Ref [_Ref]
                Comp Equal
                Pass IncrSat
            }

            CGPROGRAM
            #pragma vertex vert#pragma fragment frag#include"UnityCG.cginc"structappdata{
                float4 vertex : POSITION;};structv2f{
                float4 vertex : SV_POSITION;};

            fixed4 _Color;

            v2f vert (appdata v){
                v2f o;
                o.vertex =UnityObjectToClipPos(v.vertex);return o;}

            fixed4 frag (v2f i): SV_Target
            {return _Color;}
            ENDCG
        }}}

工程下载

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
以下是一个简单的Unity3D着色器,可以制作冰冻效果。它使用了一个简单的反射纹理和一个噪声纹理来创建冰的外观。 ``` Shader "Custom/IceShader" { Properties { _MainTex ("Texture", 2D) = "white" {} _ReflectTex ("Reflect Texture", 2D) = "white" {} _NoiseTex ("Noise Texture", 2D) = "white" {} _BumpScale ("Bump Scale", Range(0.01, 0.1)) = 0.05 _Reflectivity ("Reflectivity", Range(0.0, 1.0)) = 0.5 _NoiseScale ("Noise Scale", Range(0.1, 10.0)) = 1.0 _IceColor ("Ice Color", Color) = (1,1,1,1) } SubShader { Tags { "Queue"="Transparent" "RenderType"="Opaque" } LOD 100 CGPROGRAM #pragma surface surf Standard sampler2D _MainTex; sampler2D _ReflectTex; sampler2D _NoiseTex; float _BumpScale; float _Reflectivity; float _NoiseScale; fixed4 _IceColor; struct Input { float2 uv_MainTex; float2 uv_ReflectTex; float2 uv_NoiseTex; float3 worldPos; float3 worldNormal; }; void surf (Input IN, inout SurfaceOutputStandard o) { // Get the base color from the main texture fixed4 baseColor = tex2D(_MainTex, IN.uv_MainTex); // Get the reflection from the reflect texture fixed4 reflectColor = tex2D(_ReflectTex, IN.uv_ReflectTex); // Combine the base color and reflection fixed4 finalColor = lerp(baseColor, reflectColor, _Reflectivity); // Apply the ice color finalColor *= _IceColor; // Get the bump from the noise texture float4 noise = tex2D(_NoiseTex, IN.uv_NoiseTex); // Convert the noise to a normal vector float3 normal = UnpackNormal(noise.rgb); // Apply the bump to the surface normal IN.worldNormal += normal * _BumpScale; // Set output parameters o.Albedo = finalColor.rgb; o.Metallic = 0.0; o.Smoothness = 1.0; o.Normal = normalize(IN.worldNormal); o.Emission = finalColor.rgb; o.Occlusion = 1.0; o.Alpha = finalColor.a; } ENDCG } FallBack "Diffuse" } ``` 要使用这个着色器,您需要将三个纹理分别指定给_MainTex,_ReflectTex和_NoiseTex属性。您还可以调整_BumpScale,_Reflectivity和_NoiseScale参数来改变效果。最后,您可以使用_IceColor来指定冰的颜色。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一零壹0

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值