找回密码
 立即注册
首页 业界区 业界 【URP】Unity3D物体遮罩的多种方案实现

【URP】Unity3D物体遮罩的多种方案实现

讥慰捷 昨天 10:17
【从UnityURP开始探索游戏渲染】专栏-直达
物体自身遮罩控制方案

基于Unity URP实现3D物体方向性动态遮罩

Shader核心功能‌:


  • 通过_MaskDirection向量定义遮罩方向
  • 使用_MaskProgress参数(0-1范围)控制显示比例
  • 基于顶点世界坐标在遮罩方向上的投影值进行裁剪(clip)
  • 兼容URP渲染管线标准结构
控制器特点‌:


  • 支持运行时动态调整遮罩方向和进度
  • 内置演示动画(旋转方向和进度波动)
  • 提供外部调用接口用于程序控制
  • 自动规范化方向向量确保计算准确
  • DirectionalMask.shader
    1. Shader "Custom/DirectionalMask"
    2. {
    3.     Properties
    4.     {
    5.         _MainTex("Texture", 2D) = "white" {}
    6.         _MaskDirection("Mask Direction", Vector) = (1,0,0,0)
    7.         _MaskProgress("Mask Progress", Range(0, 1)) = 0.5
    8.     }
    9.     SubShader
    10.     {
    11.         Tags { "RenderType"="Opaque" "RenderPipeline"="UniversalPipeline" }
    12.         HLSLINCLUDE
    13.         #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"
    14.         struct Attributes
    15.         {
    16.             float4 positionOS : POSITION;
    17.             float2 uv : TEXCOORD0;
    18.         };
    19.         struct Varyings
    20.         {
    21.             float4 positionCS : SV_POSITION;
    22.             float2 uv : TEXCOORD0;
    23.             float3 positionWS : TEXCOORD1;
    24.         };
    25.         TEXTURE2D(_MainTex);
    26.         SAMPLER(sampler_MainTex);
    27.         float4 _MainTex_ST;
    28.         float3 _MaskDirection;
    29.         float _MaskProgress;
    30.         Varyings Vert(Attributes input)
    31.         {
    32.             Varyings output;
    33.             output.positionCS = TransformObjectToHClip(input.positionOS.xyz);
    34.             output.uv = TRANSFORM_TEX(input.uv, _MainTex);
    35.             output.positionWS = TransformObjectToWorld(input.positionOS.xyz);
    36.             return output;
    37.         }
    38.         half4 Frag(Varyings input) : SV_Target
    39.         {
    40.             half4 col = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, input.uv);
    41.             
    42.             // 计算物体顶点在遮罩方向上的投影比例
    43.             float dotProduct = dot(normalize(input.positionWS), normalize(_MaskDirection));
    44.             float normalizedDot = (dotProduct + 1) * 0.5; // 映射到0-1范围
    45.             
    46.             // 根据进度参数裁剪
    47.             clip(normalizedDot - _MaskProgress);
    48.             
    49.             return col;
    50.         }
    51.         ENDHLSL
    52.         Pass
    53.         {
    54.             HLSLPROGRAM
    55.             #pragma vertex Vert
    56.             #pragma fragment Frag
    57.             ENDHLSL
    58.         }
    59.     }
    60. }
    复制代码
  • DirectionalMaskController.cs
    1. using UnityEngine;
    2. public class DirectionalMaskController : MonoBehaviour
    3. {
    4.     [SerializeField] private Renderer _targetRenderer;
    5.     [SerializeField] private Vector3 _maskDirection = Vector3.right;
    6.     [Range(0, 1)] public float maskProgress = 0.5f;
    7.     [SerializeField] private float _rotationSpeed = 30f;
    8.     [SerializeField] private float _progressSpeed = 0.2f;
    9.     private Material _material;
    10.     private static readonly int MaskDirection = Shader.PropertyToID("_MaskDirection");
    11.     private static readonly int MaskProgress = Shader.PropertyToID("_MaskProgress");
    12.     void Start()
    13.     {
    14.         _material = _targetRenderer.material;
    15.         UpdateShaderParams();
    16.     }
    17.     void Update()
    18.     {
    19.         // 动态旋转遮罩方向
    20.         _maskDirection = Quaternion.Euler(0, _rotationSpeed * Time.deltaTime, 0) * _maskDirection;
    21.         
    22.         // 动态调整遮罩进度
    23.         maskProgress = Mathf.PingPong(Time.time * _progressSpeed, 1f);
    24.         
    25.         UpdateShaderParams();
    26.     }
    27.     private void UpdateShaderParams()
    28.     {
    29.         _material.SetVector(MaskDirection, _maskDirection.normalized);
    30.         _material.SetFloat(MaskProgress, maskProgress);
    31.     }
    32.     // 外部调用接口
    33.     public void SetMaskDirection(Vector3 newDirection)
    34.     {
    35.         _maskDirection = newDirection.normalized;
    36.         UpdateShaderParams();
    37.     }
    38.     public void SetMaskProgress(float progress)
    39.     {
    40.         maskProgress = Mathf.Clamp01(progress);
    41.         UpdateShaderParams();
    42.     }
    43. }
    复制代码
扩展


  • 可添加_MaskSharpness参数控制边缘过渡柔和度
  • 结合Stencil Buffer实现多物体遮罩叠加
  • 通过AnimationCurve控制进度变化曲线
该方案相比基础遮罩实现增加了方向向量参数和动态进度控制,通过顶点投影计算实现方向性裁剪效果。控制器脚本支持运行时动态调整所有参数,并保持与Shader的高效数据交互。
HLSL Shader实现‌(基于透明度遮罩):


  • unity_urp遮罩shader
    1. Shader "Custom/TransparentColorURP" {
    2.     Properties {
    3.         _Color("Color Tint", Color) = (1,1,1,1)
    4.         _MainTex("Main Tex", 2D) = "white" {}
    5.         _AlphaScale("Alpha Scale", Range(0, 1)) = 0.5
    6.     }
    7.     SubShader {
    8.         Pass {
    9.             ZWrite On
    10.             ColorMask 0
    11.         }
    12.         Pass {
    13.             ZWrite Off
    14.             Blend SrcAlpha OneMinusSrcAlpha
    15.             // HLSL代码...
    16.         }
    17.     }
    18. }
    复制代码
控制脚本‌:


  • unity_遮罩控制脚本
    1. public class MaskController : MonoBehaviour {
    2.     public Renderer targetRenderer;
    3.     public float maskProgress = 0.5f;
    4.     void Update() {
    5.         targetRenderer.material.SetFloat("_AlphaScale",
    6.             Mathf.PingPong(Time.time, 1f));
    7.     }
    8. }
    复制代码
模板测试实现方案

模板Shader核心代码‌:

RenderFeature配置要点‌:


  • 需创建URP Renderer Feature
  • 设置Layer Mask匹配目标层级
  • 在后处理前添加Pass
三、其他实现方式

多Pass深度测试法‌:
  1. Shader "Custom/StencilMask"
  2. {
  3.     SubShader
  4.     {
  5.         Tags { "RenderType"="Opaque" "Queue"="Geometry-100" }
  6.         
  7.         Pass
  8.         {
  9.             ColorMask 0
  10.             ZWrite Off
  11.             
  12.             Stencil {
  13.                 Ref 1
  14.                 Comp Always
  15.                 Pass Replace
  16.             }
  17.         }
  18.     }
  19. }
复制代码
UGUI Mask混合方案‌:


  • 使用RectTransform确定遮罩区域
  • 通过第二摄像机渲染3D物体
  • 结合UI Mask组件控制显示范围
RenderTexture遮罩‌:


  • 创建临时RenderTexture
  • 使用特定Shader渲染遮罩通道
  • 通过后期处理合成最终效果
四、方案对比

方案类型优点适用场景模板测试高性能,精确控制角色轮廓/固定形状遮罩多Pass无需额外纹理简单遮挡提示UGUI混合可结合UI系统HUD元素遮罩RenderTexture效果最灵活复杂动态遮罩模板测试适合需要精确控制像素级显示的场景,而多Pass方案更适合简单的遮挡提示效果
【从UnityURP开始探索游戏渲染】专栏-直达

(欢迎点赞留言探讨,更多人加入进来能更加完善这个探索的过程,
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

您需要登录后才可以回帖 登录 | 立即注册