Unity | Shader基础知识(第十七集:学习Stencil并做出透视效果)

news2025/1/11 0:52:46

目录

一、前言

二、了解unity预制的材质

三、什么是Stencil

四、UGUI如何使用Stencil(无代码)

1.Canvas中Image使用Stencil制作透视效果

2.学习Stencil

3.分析透视效果的需求

五、模型如何使用Stencil

1.shader准备

2.渲染顺序

3.Stencil代码语法

4.完整代码

六、作者的碎碎念


一、前言

这次我们一起学习一个新知识,叫Stencil,也叫测试模版。这个知识点可以做一些透视效果。(如图1所示)

图1 透视效果

用语言描述一下这个效果就是:你可以选择,要不要看到被遮盖住的东西。

今天教两种做法:

1.UGUI版,就是在canvas下面做,用原本有的shader就行,不需要自己重新写。

2.非UGUI版,就是不通过canvas去做,我们需要写代码,但是很通用,模型什么的都能用,比如,你想做,透过人物看到骨头这种效果,都是可以的。

我们本次的素材都用上面两个,所以看上去是图片,实际方法2也通用于模型。

二、了解unity预制的材质

实际上,我们在Unity中创建的任何一个可以被看见的游戏物体,都是有材质的。在你没有给他们添加时,他都用的默认材质。(如图2所示)

图2 image默认材质
图3 sphere默认材质

但是,默认的材质是不允许你进行手动调节的,如果你想对它进行手动调节,你需要自己新创建一个一模一样的。假如我现在要创建一个和image一样的材质(如图2所示),它用的shader是UI/Default。那我们也创建一个新材质,选择这个shader。(如图4所示)。

图4 Image默认材质

之后,你就可以调节这个shader里允许调节的信息,并直接使用这个材质了。 (如图5所示)

图5 Image默认材质

三、什么是Stencil

叫做模版,使用它叫做模版测试。

官方解释:模板缓冲区为每个像素在帧缓冲区中。在执行片段之前着色器对于给定的像素,GPU 可以将模板缓冲区中的当前值与给定的参考值进行比较。这称为模板测试。如果模板测试通过,GPU 将执行深度测试。如果模板测试失败,GPU 将跳过该像素的其余处理。这意味着您可以使用模板缓冲区作为掩码来告诉 GPU 要绘制哪些像素以及要丢弃哪些像素。

可以不看的,如果你看得懂,你也不会看到这了,对吧~直接听我讲吧。

所有的工具都是因为需要所以产生的,学到目前为止,我们发现,我们的shader大部分时候只能和自己互动,如果两个shader之间想要互动,几乎没有什么好办法。

因此,stencil就应运而生,stencil就像一个白纸,允许每一个使用它的shader在上面留下痕迹(当然你也可以选择不留下痕迹),然后再传给下一个渲染的shader,让它根据痕迹来选择是否渲染自己,是否再次留下痕迹。

这样,就可以实现不同shader之间互动的功能

小时候传纸条都玩过吧,一个意思。

四、UGUI如何使用Stencil(无代码)

懂了传纸条,也得懂怎么用,对?

a.你需要知道,这个传纸条的模版,是一直都在的,而且它的大小肯定是比你所有需要渲染的物体都要大,所以这个纸总是够用的,不需要你去写代码单独创建。

b.有一个规则:当你决定好打算在小纸条上留下痕迹时,所有你需要渲染的大小,都会留下这个痕迹。(这里为了方便学懂,我们都用2D的物体来解释)

明白以上两个知识后,我们在应用里去加深理解吧!


1.Canvas中Image使用Stencil制作透视效果

事先准备:

a.建一个Canvas,再建三个大小不一样的Image,一个做前景(前面的小村庄风景照),一个做后景(二次元美女),一个做遮罩(透过遮罩可以看见美女而不是风景),把你们喜欢的图片放上去。(如图6所示)

因为ui的渲染顺序是从上到下,所以我们先和图中放一个顺序,放大镜→前景→背景,原因待会解释。

图6 三个Image

b.建两个材质,全部都使用图4中Image的默认材质(如图7所示),分别挂到对应的物体上(如图8所示),有两个,别少挂一个,图里只截图了一个。

图7 新建材质
图8 把材质挂上去
2.学习Stencil

我们只学三个主要的。

a.我们先看一下最重要的三个Stencil数值(如图9所示)

图9 Stencil

Stencil ID                         提前准备一个数,用来比较,回头如果留痕迹就留这个

(你可以选择写0-255之间任何一个数)

Stencil Comparison         比较之前的shader留下的数,看是否要渲染自己Shader里的内容

Never                1                不渲染

Less                  2                Stencil ID小于以前的痕迹就渲染

Equal                3                Stencil ID等于以前的痕迹就渲染

LEqual              4                Stencil ID小于等于以前的痕迹就渲染

Greater             5                Stencil ID大于以前的痕迹就渲染

NotEqual          6                Stencil ID不等于以前的痕迹就渲染

GEqual             7                Stencil ID不等于以前的痕迹就渲染

Always              8                一直渲染

Stencil  Operation            是否打算在纸上留下Stencil ID 

备注:它只会在自己可以渲染的那一片区域的测试留下痕迹,不会全部覆盖,擦除别的同理。

Keep            0       不留痕迹,保留着以前的      

Zero             1       把以前的擦了,留下0      

Replace       2       把以前的擦了,留下Stencil ID的数

IncrSat         3       把以前的擦了,留下数字(以前的+Stencil ID),若大于255,就留下255

DecrSat        4      把以前的擦了,留下数字(以前的-Stencil ID),若小于0,就留下0

Invert            5       以前的数所有位取反(这个如果不懂,可以百度,不看也可以,用得少)

IncrWrap       6       以前的数+1,超过255,就变成0

DecrWrap     7       以前的数-1,小于0,就变成255

备注:数字就是前面英文的枚举。

3.分析透视效果的需求

总体步骤:让遮罩在测试模版上留下1,让美女图片设置参数1,当自己的参数和遮罩留下的参数相等时,美女图片渲染。

a.我们让遮罩先渲染,让它在测试模版上直接留下痕迹1(这里是几都可以)。

那么在UI层级上,我们需要把遮罩放在最上面,在参数选择上:

Stencil ID        1

Stencil Comparison        8(一直渲染

Stencil  Operation        2(把以前的擦了,留下1

图10 设置参数

b. 让美女图片设置参数1,当自己的参数和遮罩留下的参数相等时,美女图片渲染。

美女图片(背景)显示的时候是在风景上方的,所以层级是最下面,在参数选择上:

Stencil ID        1

Stencil Comparison      3(当数字相等时渲染,因为之前是1,现在也是1,所以渲染

Stencil  Operation        0(因为我们没有别的步骤了,留不留痕迹已经不重要了,填几都行

图11 参数设置

然后我们就完成了,移动遮罩的位置,就可以得到透视效果了。 

五、模型如何使用Stencil

1.shader准备

我们用之前透明物体shader,继续加工,就不重新写了,文章如下: 

Unity | Shader基础知识(第十五集:透明效果)_unity shader入门与实战-CSDN博客

如果不想回看了,那代码如下:

Shader "Custom/013-2"
{
Properties
    {
        _MainTex ("MainTex", 2D) = "white" {}
    }
    SubShader
    {
        Tags
        {
        "Queue" = "Transparent"
        }
        Cull Off

        CGPROGRAM
        #pragma surface surf Lambert alpha:fade

        sampler2D _MainTex;

        struct Input
        {
            float2 uv_MainTex;
        };

        void surf (Input IN, inout SurfaceOutput o)
        {
            fixed4 c =tex2D (_MainTex, IN.uv_MainTex);
            o.Albedo = c.rgb;
            o.Alpha = c.a;
        }
        ENDCG
    }
    FallBack "Diffuse"
}

代码要建立两个,一个给遮罩,一个给美女图片。 

2.渲染顺序

和UGUI版本相同,遮罩是需要提前渲染 ,所以把遮罩中的"Queue" = "Transparent"

改成"Queue" = "Transparent-1",这样遮罩就会提前渲染。(之前的知识)

美女图片的shader保持不变。

3.Stencil代码语法

和Tags相同,stencil也是一个单独的命令,也需要单独建一个括号,可以放的位置很多,这次我们先放到Tags下面。

        Tags
        {
        "Queue" = "Transparent-1"
        }
        Stencil
		{
		}

 接下来就是用代码填写Stencil的内容,也是和UGUI版本一样,我们只需要学三个单词。

Stencil ID :在代码中是Ref

Stencil Comparison:在代码中是Comp

Stencil  Operation:在代码中是​​​​​​​Pass

根据我们第四节中遮罩的要求:

Stencil ID        1

Stencil Comparison        8    Always    

Stencil  Operation           2    replace

代码里不能用数字枚举,得出代码如下:

        Tags
        {
        "Queue" = "Transparent-1"
        }
        Stencil
		{
			Ref 1
			Comp Always
			Pass replace
		}

我们的遮罩部分就完成了。


美女的shader同理可得:

Stencil ID        1        

Stencil Comparison      3       Equal     

Stencil  Operation        0        Keep

代码如下:

        Tags
        {
        "Queue" = "Transparent"
        }
        Stencil
		{
			Ref 1
			Comp Equal
			Pass Keep
		}

 我们模型部分就也完成了。

4.完整代码

遮罩代码:

Shader "Custom/013-1"
{
Properties
    {
        _MainTex ("MainTex", 2D) = "white" {}
    }
    SubShader
    {
        Tags
        {
        "Queue" = "Transparent-1"
        }
        Stencil
		{
			Ref 1
			Comp Always
			Pass replace
		}

        Cull Off

        CGPROGRAM
        #pragma surface surf Lambert alpha:fade

        sampler2D _MainTex;

        struct Input
        {
            float2 uv_MainTex;
        };

        void surf (Input IN, inout SurfaceOutput o)
        {
            fixed4 c =tex2D (_MainTex, IN.uv_MainTex);
            o.Albedo = c.rgb;
            o.Alpha = c.a;
        }
        ENDCG
    }
    FallBack "Diffuse"
}

 美女图片代码:

Shader "Custom/013-2"
{
Properties
    {
        _MainTex ("MainTex", 2D) = "white" {}
    }
    SubShader
    {
        Tags
        {
        "Queue" = "Transparent"
        }
        Stencil
		{
			Ref 1
			Comp Equal
			Pass Keep
		}

        Cull Off

        CGPROGRAM
        #pragma surface surf Lambert alpha:fade

        sampler2D _MainTex;

        struct Input
        {
            float2 uv_MainTex;
        };

        void surf (Input IN, inout SurfaceOutput o)
        {
            fixed4 c =tex2D (_MainTex, IN.uv_MainTex);
            o.Albedo = c.rgb;
            o.Alpha = c.a;
        }
        ENDCG
    }
    FallBack "Diffuse"
}

六、作者的碎碎念

 这篇文章我真的准备了好久,因为真的不知道如何才能讲的简单一点,如果觉得好的话,给我点个赞吧~

unity相关参考:Unity - Manual: ShaderLab command: Stencil

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1892127.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

最近,被“AI”狠狠刷屏了......

最近,被“AI”狠狠刷屏了。 作为时下最热门的话题,AI画图、AI配音、AI写文案、AI做视频......AI在最近两年可谓是火遍全球。ChatGPT、Midjourney和SORA等技术不断涌现,不仅深刻改变着我们的生活方式,也推动了AI技术的飞速发展。 …

计算机网络之局域网

目录 1.局域网的基本概念 2.LAN的特性 3.局域网特点 4.拓扑结构 5.传输媒体的选择 6.传输媒体 7.传输技术 8.传输技术距离问题 9.LAN的逻辑结构 10.局域网工作原理 上篇文章内容:OSI七层体系结构 1.局域网的基本概念 局域网 是将分散在有限地 理范围内&…

【初阶数据结构】深入解析循环队列:探索底层逻辑

🔥引言 本篇将介绍如何实现循环队列并实现过程需要注意的事项,虽然篇幅较小,但是其中逻辑还是值得引人思考的,循环队列可以采用数组或链表实现,这篇将采用数组实现循环队列 🌈个人主页:是店小二…

Day04-jenkins-docker

Day04-jenkins-docker 9. 案例06: 基于docker的案例实现静态代码9.1 整体流程9.2 步骤与环境1) 步骤2) 环境 9.3 详细步骤1)代码准备2)书写dockerfile3)准备私有仓库4)创建jenkins任务5)web节点上启动对应的docker容器…

小学校园“闲书”交易平台的设计与实现-计算机毕业设计源码04282

小学校园“闲书”交易平台的设计与实现 摘 要 小学校园“闲书”交易平台是为了解决小学生之间的书籍交流和阅读兴趣培养而设计的。该平台通过使用现代技术手段,如移动应用开发和互联网技术,构建了一个功能齐全的交易平台。平台支持用户注册与登录&#x…

Python 面试【高难】

欢迎莅临我的博客 💝💝💝,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

[OC]萝卜圈Python手动机器人脚本

这是给机器人设置的端口,对照用 代码 # #作者:溥哥’ ##机器人驱动主程序 #请在main中编写您自己的机器人驱动代码 import msvcrt def main():a"none"while True:key_input msvcrt.getch()akey_inputif abw:print(a)robot_drv.set_motors(1,40,2,40,3,…

Qt Group与华为合作开发OpenHarmony版本,打造无缝跨设备操作系统

在华为开发者大会2024上,跨平台软件开发和质量保证工具的领先供应商 Qt Group(Nasdaq, Helsinki: QTCOM)荣幸地宣布成为OpenHarmony生态系统合作伙伴。这是继近几年华为采用Qt开发框架和自动化测试工具Squish的商业许可后,Qt Grou…

【数学建模】 数据处理与拟合模型

文章目录 数据处理与拟合模型1. 数据与大数据1.1 什么是数?什么是数据?1.2 数据与大数据1.3 数据科学的研究对象 2. 数据的预处理2.1 为什么需要数据预处理2.2 使用pandas处理数据的基础2.3 pandas常用方法总结2.4 数据的规约1) 维度规约2) 数值规约3) 数…

小型语言模型SLM:趋势和用例

前言 近年来,GPT 和 BERT 等大型语言模型 (LLM) 不断发展,参数数量从数亿飙升至 GPT-4 等后继者的一万亿以上。然而,不断增长的参数规模引出了一个问题:对于企业应用来说,参数规模越大就一定越好吗? 答案…

测试图片上传功能,使用postman提供的url

是不是有时候想要测试图片上传功能,但是没有后台url进行测试,这时候就可以使用postman提供的url: https://postman-echo.com/post接下来,我将教你在postman中,用该url测试图片上传功能。 1.发送图片上传请求 第一步…

数据结构 -AVL树

文章目录 AVL树左旋和右旋插入的四种情况(一)新数字插到了左子树,导致左子树比右子树高2;左孩子的左子树比其右子树高1(二)新数字插到了左子树,导致左子树比右子树高2;左孩子的右子树…

Echarts中的热力图和漏斗图(在Vue中使用热力图和漏斗图)

热力图 (Heatmap) Echarts的热力图用于展示两个维度数据矩阵中的值分布情况。它通过在平面上划分成多个矩形区域,并用不同的颜色填充这些区域来表示数据的大小或强度。颜色渐变从浅到深通常映射着数值从小到大,从而直观展示数据的集中程度和分布模式。热…

MIPI概述

MIPI MIPI(Mobile Industry Processor Interface)是移动行业处理器接口的缩写,由移动行业处理器接口联盟(MIPI Alliance)(一个标准化组织,官网:Interface Specifications for Mobil…

【web APIs】快速上手Day04(Dom节点)

目录 Web APIs - 第4天日期对象实例化方法案例-页面显示时间时间的另外一个写法 时间戳三种方式获取时间戳案例-毕业倒计时效果 节点操作DOM节点查找节点父节点查找案例-关闭广告子节点查找兄弟关系查找 增加节点创建节点追加节点案例-学成在线案例渲染克隆节点 删除节点 M端事…

IDEA 一键部署Docker

以部署示例服务(sevnce-demo)为例。 配置服务器 地址、账号、密码根据实际情况填写 配置镜像仓库 地址、账号、密码根据实际情况填写 编写Dockerfile 在sevnce-demo根目录下右键,选择创建Dockerfile。 # 基础镜像 FROM sevnce-registry.c…

如何为老化的汽车铅酸电池充电

一项小研究表明,汽车铅酸电池不同于深循环或固定电池。汽车电池旨在限度地提高启动电流容量,并且对深度放电或浮充(也称为第 3 阶段充电循环)反应不佳。起动电池的极板结构使表面积化,并且电解液比重 (SG) 高于其他电池,以提供高启…

14.优化算法之BFS解决FloodFill算法1

0.FloodFill简介 dfs:深度优先遍历(红色) bfs:宽度优先遍历 1.图像渲染 算法原理 class Solution {int[] dx { 0, 0, 1, -1 };int[] dy { 1, -1, 0, 0 };public int[][] floodFill(int[][] image, int sr, int sc, int color)…

小试牛刀-Solana合约账户详解

目录 一.Solana 三.账户详解 3.1 程序账户 3.2 系统所有账户 3.3 程序派生账户(PDA) 3.4 Token账户 四、相关学习文档 五、在线编辑器 Welcome to Code Blocks blog 本篇文章主要介绍了 [Solana合约账户详解] ❤博主广交技术好友,喜欢文章的可以关注一下❤ …

Ai的快速发展会是下一个风口嘛?

人工智能的快速发展确实是当前和未来科技领域的一个重要趋势,可以说是一个“风口”。以下是几个支持这一观点的理由: 1. **技术进步和应用广泛性**:人工智能技术在各个领域的应用越来越广泛,涵盖了医疗、金融、制造、交通、媒体等…