Chapter8 透明效果——Shader入门精要学习笔记

news2024/11/23 8:14:38

一、基本概念

  • 在Unity中通常使用两种方法来实现透明效果
    • 透明度测试(无法达到真正的半透明效果)
    • 透明度混合(关闭了深度写入)

透明度测试

  • 基本原理:设置一个阈值,只要片元的透明度小于阈值,就会被舍弃(不会进行任何处理,也不会对颜色缓冲造成影响);否则,就会按照普通的不透明物体来处理(进行深度测试、深度写入等)
  • 效果极端,要么透明,要么完全不透明

透明度混合

  • 基本原理:使用当前片元的透明度作为混合因子,与已经存储在颜色缓冲中的颜色值进行混合,得到新的颜色
  • 需要关闭深度写入,但没有关闭深度测试——深度缓冲是只读的
    • 为什么关闭深度写入:如果开启,半透明物体在不透明物体前面时,半透明物体会挡住不透明物体,实现不了半透明效果
  • 可以实现半透明效果

二、渲染顺序

  • 对于半透明物体和不透明物体,因为关闭了深度写入,所以应该在不透明物体渲染完后再渲染半透明物体
  • 对于两个半透明物体,应该先渲染远处的,再渲染近处的(不准确)

1.渲染引擎的方法

  • 先渲染所有不透明物体,并开启他们的深度测试和深度写入
  • 把半透明物体按他们的距离摄像机的远近进行排序,再按照从后往前的顺序渲染,开启深度测试,关闭深度写入
    • 会有更加复杂的重叠情况,排序困难

2.Unity的渲染顺序

  • 使用SubShader的 Queue 标签来决定我们的模型将归于哪个渲染队列,使用整数索引来表示每个渲染队列,号小的越早被渲染
    在这里插入图片描述
  • 如果想要通过透明度测试来实现效果,代码:
SubShader{
	Tags{ "Queue" = "AlphaTest"}
	Pass{...}
	}
  • 如果想使用透明度混合
SubShader{
	Tags{ "Queue" = "Transparent"}
	Pass{
		ZWrite Off //关闭深度写入
		...}
	}

三、透明度测试

  • 基本原理:设置一个阈值,只要片元的透明度小于阈值,就会被舍弃(不会进行任何处理,也不会对颜色缓冲造成影响);否则,就会按照普通的不透明物体来处理(进行深度测试、深度写入等)
  • 通常会在片元着色器中使用 clip函数 进行透明度测试
    • 如果给定参数的任何一个分量是负数,则会舍弃当前像素的输出颜色
// Upgrade NOTE: replaced '_Object2World' with 'unity_ObjectToWorld'
// Upgrade NOTE: replaced 'mul(UNITY_MATRIX_MVP,*)' with 'UnityObjectToClipPos(*)'

Shader "Custom/Chapter8-AlphaTest"
{
   Properties
   {
		_Color ("Main Tint", Color) = (1,1,1,1)
		_MainTex ("Main Tex", 2D) = "white"{}
		_Cutoff ("Alpha Cutoff", Range(0,1)) = 0.5//用于决定我们调用clip进行透明度测试时使用的判决条件
   }
   SubShader
   {
		Tags {"Queue" = "AlphaTest" "IgnoreProjector" = "True" "RenderType" = "TransparentCutoff"}
		Pass
		{
			Tags{"LightMode" = "ForwardBase"}
				CGPROGRAM

				#pragma vertex vert 
				#pragma fragment frag

				#include "Lighting.cginc"

				fixed4 _Color;
				sampler2D _MainTex;
				float4 _MainTex_ST;
				fixed _Cutoff;

				struct a2v
				{
					float4 vertex : POSITION;
					float3 normal : NORMAL;
					float4 texcoord : TEXCOORD0;
				};
				struct v2f
				{
					float4 pos : SV_POSITION;
					float3 worldNormal : TEXCOORD0;
					float3 worldPos : TEXCOORD1;
					float2 uv : TEXCOORD2;
				};

				v2f vert(a2v v)
				{
					v2f o;
					o.pos = UnityObjectToClipPos(v.vertex);
					o.worldNormal = UnityObjectToWorldNormal(v.normal);
					o.worldPos = mul(unity_ObjectToWorld, v.vertex).xyz;
					o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);

					return o;
				}
				fixed4 frag(v2f i) : SV_Target 
				{
					fixed3 worldNormal = normalize(i.worldNormal);
					fixed3 worldLightDir = normalize(UnityWorldSpaceLightDir(i.worldPos));

					fixed4 texColor = tex2D(_MainTex, i.uv);

					//透明度测试
					clip(texColor.a - _Cutoff);
					fixed3 albedo = texColor.rgb * _Color.rgb;
					fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz * albedo;
					fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(worldNormal,worldLightDir));

					return fixed4(ambient + diffuse, 1.0);
				}
				ENDCG
		}
   }
   FallBack "Transparent/Cutout/VertexLit"
}

  • _Cutoff (“Alpha Cutoff”, Range(0,1)) = 0.5 用于决定我们调用clip进行透明度测试时使用的判决条件
  • clip(texColor.a - _Cutoff); 做透明度测试,判断texColor.a - _Cutoff是否为负数,负数的纹理颜色为全透明
    在这里插入图片描述

四、透明度混合

  • 基本原理:使用当前片元的透明度作为混合因子,与已经存储在颜色缓冲中的颜色值进行混合,得到新的颜色
  • 需要使用Unity提供的混合命令 Blend
    在这里插入图片描述
  • 本节中使用的是Blend SrcFactor DstFactor来进行混合,将源颜色混合因子SrcFactor设为SrcAlpha,目标颜色混合因子设为OneMinusSrcAlpha
    • 即混合后的新颜色为
    • D s t C o l o r n e w DstColor_{new} DstColornew = SrcAlpha × SrcColor + (1-SrcAlpha) × D s t C o l o r o l d DstColor_{old} DstColorold
  • ZWrite Off 关闭深度写入
Shader "Custom/Chapter8-AlphaBlend"
{
	Properties
	{
		_Color ("Main Tint", Color) = (1,1,1,1)
		_MainTex ("Main Tex", 2D) = "white"{}
		_AlphaScale ("Alpha Scale", Range(0,1)) = 1
	}
	SubShader
	{
		Tags {"Queue" = "Transparent" "IgnoreProjector" = "True" "RenderType" = "Transparent"}
		Pass
		{
			Tags {"LightMode" = "ForwardBase"}

			ZWrite Off
			Blend SrcAlpha OneMinusSrcAlpha

			CGPROGRAM

			#pragma vertex vert 
			#pragma fragment frag 

			#include "Lighting.cginc"
			
			fixed4 _Color;
			sampler2D _MainTex;
			float4 _MainTex_ST;
			fixed _AlphaScale;

				struct a2v
				{
					float4 vertex : POSITION;
					float3 normal : NORMAL;
					float4 texcoord : TEXCOORD0;
				};
				struct v2f
				{
					float4 pos : SV_POSITION;
					float3 worldNormal : TEXCOORD0;
					float3 worldPos : TEXCOORD1;
					float2 uv : TEXCOORD2;
				};

				v2f vert(a2v v)
				{
					v2f o;
					o.pos = UnityObjectToClipPos(v.vertex);
					o.worldNormal = UnityObjectToWorldNormal(v.normal);
					o.worldPos = mul(unity_ObjectToWorld, v.vertex).xyz;
					o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);

					return o;
				}
				fixed4 frag(v2f i) : SV_Target 
				{
					fixed3 worldNormal = normalize(i.worldNormal);
					fixed3 worldLightDir = normalize(UnityWorldSpaceLightDir(i.worldPos));

					fixed4 texColor = tex2D(_MainTex, i.uv);

					//透明度混合
					fixed3 albedo = texColor.rgb * _Color.rgb;
					fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz * albedo;
					fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(worldNormal,worldLightDir));


					return fixed4(ambient + diffuse, texColor.a * _AlphaScale);
				}
			ENDCG
		}
	}
	FallBack "Transparent/VertexLit"
}

在这里插入图片描述

五、开启深度写入的半透明效果

  • 对于复杂网格的半透明处理,避免错误排序
  • 方法:使用 两个Pass 来渲染模型
    • 第一个开启深度写入,但不输出颜色,仅仅为了把该模型的深度值写入深度缓存中
    • 第二个Pass 进行正常的透明度混合,根据第一个Pass,可以进行像素级别的深度排序
    • 缺点是 多个Pass会造成性能的影响
Shader "Custom/Chapter8-AlphaBlend"
{
	Properties
	{
		_Color ("Main Tint", Color) = (1,1,1,1)
		_MainTex ("Main Tex", 2D) = "white"{}
		_AlphaScale ("Alpha Scale", Range(0,1)) = 1
	}
	SubShader
	{
		Tags {"Queue" = "Transparent" "IgnoreProjector" = "True" "RenderType" = "Transparent"}
		Pass
		{
			ZWrite On 
			ColorMask 0
		}
		Pass
		{
			Tags {"LightMode" = "ForwardBase"}

			ZWrite Off
			Blend SrcAlpha OneMinusSrcAlpha

			CGPROGRAM

			#pragma vertex vert 
			#pragma fragment frag 

			#include "Lighting.cginc"
			
			fixed4 _Color;
			sampler2D _MainTex;
			float4 _MainTex_ST;
			fixed _AlphaScale;

				struct a2v
				{
					float4 vertex : POSITION;
					float3 normal : NORMAL;
					float4 texcoord : TEXCOORD0;
				};
				struct v2f
				{
					float4 pos : SV_POSITION;
					float3 worldNormal : TEXCOORD0;
					float3 worldPos : TEXCOORD1;
					float2 uv : TEXCOORD2;
				};

				v2f vert(a2v v)
				{
					v2f o;
					o.pos = UnityObjectToClipPos(v.vertex);
					o.worldNormal = UnityObjectToWorldNormal(v.normal);
					o.worldPos = mul(unity_ObjectToWorld, v.vertex).xyz;
					o.uv = TRANSFORM_TEX(v.texcoord, _MainTex);

					return o;
				}
				fixed4 frag(v2f i) : SV_Target 
				{
					fixed3 worldNormal = normalize(i.worldNormal);
					fixed3 worldLightDir = normalize(UnityWorldSpaceLightDir(i.worldPos));

					fixed4 texColor = tex2D(_MainTex, i.uv);

					//透明度混合
					fixed3 albedo = texColor.rgb * _Color.rgb;
					fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz * albedo;
					fixed3 diffuse = _LightColor0.rgb * albedo * max(0, dot(worldNormal,worldLightDir));


					return fixed4(ambient + diffuse, texColor.a * _AlphaScale);
				}
			ENDCG
		}
	}
	FallBack "Transparent/VertexLit"
}
  • 与透明度混合代码几乎一致,只是增加了一个Pass
Pass
		{
			ZWrite On 
			ColorMask 0
		}
  • ZWrite On 打开深度写入,ColorMask 0 意味着Pass不写入任何颜色通道,不会输出任何颜色
    在这里插入图片描述

六、ShaderLab的混合命令

  • 混合与两个操作数有关:源颜色(SourceColor)和目标颜色(DestinationColor)
    • 源颜色指片元着色器产生的颜色 S
    • 目标颜色指从颜色缓冲中读取到的颜色值 D
    • 混合后的颜色 O
    • 包含了 RGBA四个颜色通道

1.混合等式和参数

  • 将S和D进行混合的等式——混合等式,需要两个,一个用于混合RGB,一个用于混合A
  • 设置混合状态时,即设置等式中的 “操作” 和 “因子”
    • 一共有两个等式(分别用于混合RGB和A)
    • 一个等式里有两个因子(一个与S相乘,一个与D相乘)

在这里插入图片描述

  • 第一个命令中只有两个因子——RGB和A通道用相同的两个因子
    在这里插入图片描述
    混合因子可以是哪些值呢?
    在这里插入图片描述

2.混合操作

  • 可以使用 BlendOp BlendOperation命令
    在这里插入图片描述
    在这里插入图片描述
  • 当使用Min和Max操作时,混合因子不起作用的

3.常见的混合类型(效果)

在这里插入图片描述

七、双面渲染的透明效果

  • 观察到其内部结构
  • 可以使用Cull指令来控制需要剔除哪个面的渲染图元
    • Back:背对相机的渲染图元不会被渲染
    • Front:朝向相机的渲染图元不会被渲染
    • Off:关闭剔除功能
Cull Back | Front | Off

1.透明度测试的双面渲染

  • 在Pass中使用 Cull Off 来关闭剔除功能即可
  • 只在 AlphaTest.shader 里Pass中加一句 Cull Off
    在这里插入图片描述

2.透明度混合的双面渲染

  • 因为关闭了深度写入,如果直接加 Cull Off 的话会造成前后面混乱
  • 所以,写了两个Pass (与AlphaBlend.shader一样)
    • 只在Pass中,第一个Pass只渲染背面 ,添加了Cull Front;第二个只渲染正面 添加了Cull Back
    • SubShader是会按照顺序执行Pass
      在这里插入图片描述

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

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

相关文章

pandas数据分析(2)

列 执行df.columns获取DataFrame列信息: 如果在构造DataFrame时没有提供列名,那么pandas会用 从0开始的数字为列编号。我们也可以为列命名,和为索引命名类似: 同样也可以重命名列名: 使用df.drop删除列: 删…

Apple - Text Layout Programming Guide

本文翻译整理自:Text Layout Programming Guide(更新日期:2014-02-11 https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/TextLayout/TextLayout.html#//apple_ref/doc/uid/10000158i 文章目录 一、文本布局编程指…

小米集团25届校招留学生面试经验汇总及入职测评笔试题型分析

一、小米校招24年春招智能驾驶产品管理面试经验分享 ​ - **自我介绍**:准备一个精炼的自我介绍,突出自己的优势和适合岗位的特点。 - **项目经验**:详细回顾你在实习或项目中的具体角色和贡献,准备用成果和数据支撑。 - **行业…

【成都活动邀请函】7月6 | PowerData 数字经济-“成都“开源行!

【成都活动邀请函】7月6 | PowerData 数字经济-"成都"开源行! 活动介绍活动信息线上直播扫码报名往期活动回顾专注数据开源,推动大数据发展 活动介绍 九天开出一成都,万户千门入画图。 自古以来,成都便是国家发展的重要…

为什么在重写equals方法后还要再重写hashcode方法(面试题)

接着上篇文章说到(上篇文章地址:http://t.csdnimg.cn/udpsThttp://t.csdnimg.cn/udpsT)我们在代码中发现重写了equals方法后还需要重写hashcode方法,为什么呢? 对于set这种数据类型,里面的值是不允许有重复…

【游戏引擎之路】登神长阶(五)

5月20日-6月4日:攻克2D物理引擎。 6月4日-6月13日:攻克《3D数学基础》。 6月13日-6月20日:攻克《3D图形教程》。 6月21日-6月22日:攻克《Raycasting游戏教程》。 6月23日-6月30日:攻克《Windows游戏编程大师技巧》。 …

【WPF】Windows系统桌面应用程序编程开发新手入门-打造自己的小工具

电脑Windows系统上的桌面程序通常是用Visual Studio 开发工具编写出来的,有两种开发方式供选择,一种是WindowForm,简称WinForm,另一种是Windows Presentation Foundation,简称WPF,这里将学习WPF项目。 文章…

安全不“放假”!暑期安全老师就用秒报小程序提示学生的安全

随着暑假的到来,孩子们如同出笼的小鸟,迫不及待地投入到广阔天地的怀抱。然而,作为老师,我深知这段时间虽然孩子们得到了放松和游玩的机会,但安全问题却不容忽视。如何让孩子们在享受假期的同时,又能确保他…

SpringBoot(一)创建一个简单的SpringBoot工程

Spring框架常用注解简单介绍 SpringMVC常用注解简单介绍 SpringBoot(一)创建一个简单的SpringBoot工程 SpringBoot(二)SpringBoot多环境配置 SpringBoot(三)SpringBoot整合MyBatis SpringBoot(四…

FairGuard游戏加固无缝兼容 Android 15 预览版

2024年6月25日,谷歌发布了 Android 15 Beta 3 ,作为Android 15 “平台稳定性”的里程碑版本,谷歌建议所有应用、游戏、SDK、库和游戏引擎开发者都将“平台稳定性”里程碑版本作为规划最终兼容性测试和公开发布的目标。 安卓开发者博客提供的版…

PostgreSQL安装教程及文件介绍

Ubuntu 安装和配置 PostgreSQL 以 Ubuntu Server 20.04,PostgreSQL 12 版本为例。 1. 安装 使用如下命令,安装指定版本的 PostgreSQL sudo apt install postgresql-12在 Ubuntu 20.04 中安装 PostgreSQL 登录您的 Ubuntu 系统并使用以下 apt 命令更新…

[HBM] HBM TSV (Through Silicon Via) 结构与工艺

依公知及经验整理,原创保护,禁止转载。 专栏 《深入理解DDR》 全文 3300 字。 1 概念 1.1 什么是HBM TSV 使用 TSV 堆叠多个DDR DRAM成为一块HBM, 成倍提高了存储器位宽, 一条位宽相当于高速公路的一条车道, 车道越多&#xff…

npm安装依赖报错——npm ERR gyp verb cli的解决方法

1. 问题描述 1.1 npm安装依赖报错——npm ERR! gyp verb cli 登录后复制 npm MARN deprecated axiosQ0.18.1: critical security vuLnerability fixed in v0.21.1. For more information, npm WARN deprecated svg001.3.2: This SVGO version is no Longer supported. upgrade …

汽车电子工程师入门系列——汽车操作系统架构学习研究-AUTOSAR

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…

6-14题连接 - 高频 SQL 50 题基础版

目录 1. 相关知识点2. 例子2.6. 使用唯一标识码替换员工ID2.7- 产品销售分析 I2.8 - 进店却未进行过交易的顾客2.9 - 上升的温度2.10 - 每台机器的进程平均运行时间2.11- 员工奖金2.12-学生们参加各科测试的次数2.13-至少有5名直接下属的经理2.14 - 确认率 1. 相关知识点 left …

5 数字滤波器的基本结构

目录 系统框图表示法 系统框图求系统函数 系统框图的其他结构形式 IIR数字滤波器的基本结构 直接型系统框图 级联型系统框图 并联型系统框图 信号流图 信号流图表示法 信号流图代数运算基本规则 系统框图表示法 系统框图求系统函数 系统框图的其他结构形式 IIR数字滤…

30分钟学习如何搭建扩散模型的运行环境【pytorch版】【B站视频教程】【解决环境搭建问题】

30分钟学习如何搭建扩散模型的运行环境【B站视频教程】【解决环境搭建问题】 动手学习扩散模型 点击以下链接即可进入学习: B站视频教程附赠:环境配置安装(配套讲解文档) 视频 讲解主要内容 一、环境设置 1.本地安装&#xf…

grpc学习golang版( 二、入门示例 )

系列文章目录 第一章 grpc基本概念与安装 第二章 grpc入门示例 第三章 proto文件数据类型 第四章 多服务示例 第五章 多proto文件示例 第六章 服务器流式传输 第七章 客户端流式传输 第八章 双向流示例 文章目录 一、环境二、编写protobuf文件三、编写server服务端四、编写Clie…

SpringBoot入门实战:SpringBoot整合SpringSecurity

1.背景介绍 SpringBoot是Spring官方推出的一款快速开发框架,它基于Spring框架,采用了约定大于配置的开发方式,简化了开发过程,提高了开发效率。SpringBoot整合SpringSecurity是SpringBoot与SpringSecurity的集成,可以实…

Ubuntu24.04 C++ 编译并使用OpenCV

Ubuntu24.04 换源 Ubuntu24.04 apt换国内源 清华源 阿里源 中科大源 163源-CSDN博客文章浏览阅读49次。作为 LTS 版本,它会有 5 年的常规维护和安全更新。如果订阅 Ubuntu Pro,还能享受额外7年的支持。也就是说, Ubuntu Pro将享有长达 12 年的系统更新保障!接下来,我们将…