Sprite Mask inner workings?

The new Sprite Mask feature is very nice, but it is a black box. How exactly does it work? I can only guess that it makes heavy use of stencil buffer internally, but this isn’t documented anywhere. Frame Debugger shows nothing related to the stencil when rendering masks or masked sprites, and sprite shaders have no mention of stencil as well.
How safe is it to use this feature if I’m already using stencil extensively? It seems to be causing some troubles (read the first 2 paragraphs):

Also, is there any good way to use Sprite Mask with MeshRenderer? I mean, I can write a shader that will use the stencil mask, but for that, I must at least know what values are being written into stencil.

Thanks.

Hi, let me help clarify this:

SpriteMask component will render the masking sprite twice. The first time it will increment the stencil buffer. The second time it will decrement (FrameDebug does show that information, check the Stencil Pass for IncrementSaturate/DecrementSaturate).
SpriteRenderer offers an easy way to configure the interaction with the StencilBuffer (internally we are setting stencil states for you):

  • VisibleInsideMask will set StencilRef value to 1 and a CompareFunc to LessEqual. That means will render pixels where 1 is less or equal to the value of the StencilBuffer.
  • VisibleOutsideMask will set StencilRef value to 1 and a CompareFunc to Greater. That means will render pixels where 1 is greater to the value of the StencilBuffer.
  • None (no interaction) will use the stencil states defined by the material.

You can create your own Shader in order to interact with SpriteMasks. All you need to do is to set the right stencil states:

Shader "Custom/SpriteMaskInteraction"
{
    Properties
    {
        [PerRendererData] _MainTex ("Sprite Texture", 2D) = "white" {}
        _Color ("Tint", Color) = (1,1,1,1)
        [MaterialToggle] PixelSnap ("Pixel snap", Float) = 0
        [HideInInspector] _RendererColor ("RendererColor", Color) = (1,1,1,1)
        [HideInInspector] _Flip ("Flip", Vector) = (1,1,1,1)
        [PerRendererData] _AlphaTex ("External Alpha", 2D) = "white" {}
        [PerRendererData] _EnableExternalAlpha ("Enable External Alpha", Float) = 0
    }

    SubShader
    {
        Tags
        {
            "Queue"="Transparent"
            "IgnoreProjector"="True"
            "RenderType"="Transparent"
            "PreviewType"="Plane"
            "CanUseSpriteAtlas"="True"
        }

        Cull Off
        Lighting Off
        ZWrite Off
        Blend One OneMinusSrcAlpha

        Pass
        {
            Stencil {
                Ref 1  //Customize this value
                Comp Equal //Customize the compare function
                Pass Keep
            }

        CGPROGRAM
            #pragma vertex SpriteVert
            #pragma fragment SpriteFrag
            #pragma target 2.0
            #pragma multi_compile_instancing
            #pragma multi_compile _ PIXELSNAP_ON
            #pragma multi_compile _ ETC1_EXTERNAL_ALPHA
            #include "UnitySprites.cginc"
        ENDCG
        }
    }
}
11 Likes

Thanks for the detailed answer!

However, here’s how the Frame Debugger looks like for a simple scene with a mask and an object:
3211781--245899--upload_2017-9-7_11-38-25.png
There is nothing regarding Stencil anywhere. Also, this block is missing from built-in Sprites shaders, so how does it work? Is Unity magically setting the stencil state under the hood for SpriteRenderer only?

            Stencil {
                Ref 1  //Customize this value
                Comp Equal //Customize the compare function
                Pass Keep
            }

Also, it seems like the masking system doesn’t work at all for opaque sprites. I often use an opaque sprite shader with SpriteRenderer that uses z-test and does z-write to reduce overdraw.
Sprites-Opaque.shader

Shader "Sprites/Opaque"
{
    Properties
    {
        [PerRendererData] _MainTex ("Sprite Texture", 2D) = "white" {}
        _Color ("Tint", Color) = (1,1,1,1)
        [MaterialToggle] PixelSnap ("Pixel snap", Float) = 0
    }

    SubShader
    {
        Tags
        {
            "Queue" = "Geometry"
            "IgnoreProjector"="True"
            "RenderType"="Opaque"
            "PreviewType"="Plane"
            "CanUseSpriteAtlas"="True"
        }

        Cull Off
        Lighting Off
        ZWrite On
        ZTest LEqual
        Fog { Mode Off }
        Blend Off

        Pass
        {

        CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma multi_compile DUMMY PIXELSNAP_ON
            #include "UnityCG.cginc"

            struct appdata_t
            {
                float4 vertex   : POSITION;
                float4 color    : COLOR;
                float2 texcoord : TEXCOORD0;
            };

            struct v2f
            {
                float4 vertex   : SV_POSITION;
                fixed4 color    : COLOR;
                half2 texcoord  : TEXCOORD0;
            };

            fixed4 _Color;

            v2f vert(appdata_t IN)
            {
                v2f OUT;
                OUT.vertex = UnityObjectToClipPos(IN.vertex);
                OUT.texcoord = IN.texcoord;
                OUT.color = IN.color * _Color;
                #ifdef PIXELSNAP_ON
                OUT.vertex = UnityPixelSnap (OUT.vertex);
                #endif

                return OUT;
            }

            sampler2D _MainTex;

            fixed4 frag(v2f IN) : SV_Target
            {
                fixed4 c = tex2D(_MainTex, IN.texcoord) * IN.color;
                return c;
            }
        ENDCG
        }
    }
}

However, when I use this shader, the masking stops working. It seems to be that way because Sprites-Mask shader uses hardcoded Transparent queue. How can I change the render queue of the mask? Is it possible to use a custom shader for the mask itself?

I’ve also tried it with a MeshRenderer and it works fine… if the shader uses a transparent render queue.

About the FrameDebugger, my bad, 2017.1 does not show the information. It will in next versions.
Yes, we are setting stencil states for SpriteRenderer and ParticleSystemRenderer only.
SpriteMask is like any other renderer, you can access its material, change it or change its render queue.
I will check if we can solve some of the issues. Thanks for your feedback!

1 Like

Has anything been done about this yet? There is something definitely going on with the SpriteRenderer and custom shaders. I came across this issue by trying to add some multiplicative blending to the built-in Sprites-Default shader.

I’m on 2017.3.0f3

I created a new shader asset and copy pasted the Sprites-Default source from the Unity downloads page to it. I then created a new material with that new shader assigned. I then assigned that material to the SpriteRenderer component responsible for rendering the sprite that I wish to be masked. And my sprite-to-be-masked simply disappears. I’m using the SpriteMask component on another game object that represents my masking sprite and everything works when I use the actual built-in default sprite shader, but when I try and make SpriteRenderer use my own custom shader (which I can only assume is the exact same source as the actual built-in shader), it stops working.

Here are my discoveries:

  1. As long as the Mask Interaction setting is set to None, my custom shader works and my sprite is visible. It’s just not masked.

  2. If I set the Mask Interaction setting to “Visible Inside Mask”, my sprite disappears when it should be partially visible. At this point, if I simply switch the shader that my material uses from my custom shader to the built-in shader, the sprite appears masked as expected. Switching the shader used by my material back to my custom shader, again causes the sprite disappear.

  3. If I set the Mask Interaction setting to “Visible Outside Mask” (with my material using my custom shader), my sprite appears in full without being affected by the mask.

  4. If I have the Mask Interaction setting set to anything other than None, then it doesn’t seem to matter what Stencil states I set in my custom shader, they don’t appear to affect anything…my sprite remains invisible. If the setting is None however, then the stencil value at each pixel appears to remain at 0, since I can get my sprite to appear using my custom shader as long as my Stencil shader logic is the following:

Stencil {
    Ref 0  //Customize this value
    Comp Equal //Customize the compare function
    Pass Keep
}

I tried fiddling with what the Ref value is and the Comp method but none of it makes sense. It gives me the impression that the stencil value across all pixels under the sprite is zero, since comparing for equality to Ref 0 causes my sprite to appear, but comparing for equality to any other Ref value below or above 0 causes my sprite not to appear. But what’s strangest is that doing Comp Less or LEqual to Ref 1 doesn’t cause my sprite to appear, but doing Comp GEqual to 0 does cause it to appear. If GEqual to 0 works, I would’ve assumed Less and LEqual to 0 to work. But this I’m guessing is due to my lack of understanding of the stencil buffer.

Either way, the interaction between SpriteRenderer, SpriteMask, and my custom sprite shader (which I’ll repeat was copied verbatim from the built-in shader source available from the Unity downloads page) is baffling me to say the least.

I just wanted to change the blend mode of my sprites while retaining the same functionality inherent to the built-in sprites! What could possibly go wrong?

Thanks for listening.

I’m wondering if the issue I’m having has anything to do with the bug filed here. It says it was fixed in 2017.1, but there still seems to be some issues using non-built-in shaders with SpriteRenderer. Specifically when using the Mask Interaction setting.

Ok, so I just started up Unity and was prompted that a new update was available, 2017.3.1f1. My issue appears to be fixed with this new version! I realized one of the things I didn’t try yesterday was to simply close and restart Unity and or reimport my assets. I can’t say for sure, but it’s possible that’s all I originally had to do. Either way, I’m happy that things are working as expected again.

Crisis averted. :slight_smile:

Hi @Sergi_Valls , i know this is an old thread but I am having issues with the inner workings too.
I need to use SpriteMask with RenderTexture but it only takes in a Sprite and not a Texture. And i can’t create Sprite with any Texture other than Texture2D.

So i tried to use a script in hope that i can overwrite the _AlphaTex

public class RenderTextureMask : MonoBehaviour
{
    public RenderTexture rtexture;

    private Renderer _renderer;
    private MaterialPropertyBlock _propBlock;

    // Start is called before the first frame update
    void Start()
    {
        _propBlock = new MaterialPropertyBlock();
        _renderer = GetComponent<SpriteMask>();
    }

    // Update is called once per frame
    void LateUpdate()
    {
        var hasProperty = _renderer.HasPropertyBlock();
        // Get the current value of the material properties in the renderer.
        _renderer.GetPropertyBlock(_propBlock);
        // Assign our new value.
        _propBlock.SetTexture("_AlphaTex", rtexture);
        // Apply the edited values to the renderer.
        _renderer.SetPropertyBlock(_propBlock);
    }
}

However, this code does do not affect the SpriteMask any bit and i wonder if i did not understand the workings enough… Or maybe this alpha texture is being combined with all the other SpriteMasks in the scene before sending it to the shader?
And what is the best approach to use render texture as mask?

You could try using “_MainTex” instead.
You can create a Sprite from your RenderTexture. Copy RT’s pixels into a Texture2D:
https://answers.unity.com/questions/9969/convert-a-rendertexture-to-a-texture2d.html

Thanks for the reply!

Copying the RT’s pixels is definitely not what i’m going for cos the RT takes up half the screen and is changing every frame and i can’t keep performing this expensive process… (especially on mobile)

Will try out your suggestion with “_MainTex” when i can.
For now i’ve already got my own shader to perform what i needed.
Just thought that it is quite limiting that i can’t use the built-in SpriteMask to work with render textures.

Hi @Sergi_Valls , if I am creating my own shader, how is it possible to create a stencil buffer that mask only the given range of sorting order just like the option of “custom range” in the built-in SpriteMask. Thanks in advance!

You will need two renderers, sorted at the beginning and end of your range. The first one will draw to the stencil and the last one will clear it or revert the operation. Renderers sorted in between will be able to interact with the prepared stencil.

2 Likes

@Sergi_Valls Do you know if it would be possible to use a SpriteShape to SET the stencil in the same way the SpriteMask would with a Sprite?

I’m working on a 2D top-down game, and I’d love to use SpriteShape to make puddles that are on the ground, and have the puddles set the mask so that I can draw reflections of characters on top of them.

I was hoping I could look at the SpriteMask and then maybe make a customized version of it for the SpriteShape, but I can’t see the code for the SpriteMask to try to figure out what to do.

Thanks for any help!

In case this is still an issue, have a look at Packages/Universal RP/Shaders/2D/Sprite-Mask.shader and the referenced include file. The interesting part is the clip function in the fragment function combined with a stencil write. The clip uses an alpha cut-off value so that the stencil buffer is only affected when a pixel is actually set

It would be really useful if SpriteMask could be modified to allow for a SpriteShape’s fill mesh to be used for its mask sprite, so you can make a custom mesh shape for any stencil instead of needing a dedicated sprite for every shape you want.

2 Likes

Sorry to ping you in a 5 year old thread, but is it even possible to make a Shader Graph shader that can read from the Stencil Buffer in order to make a custom mask interaction shader that would interact with SpriteMask?

Would my only option be to copypaste all of my shader graphs’ generated code into written shaders and then add the stencil function? There has to be a better way.

I looked into Render Features/Render Objects, but there doesn’t seem to be a way to specify Sorting Layer and Sorting Order, which I believe is required to work with SpriteMask’s range option.

1 Like