Unity中Shader不同灯光类型的支持与区分

文章目录

  • 前言
  • 一、在开始之前做一些准备
    • 1、在上一篇文章的场景基础上,增加一个Unity默认的球体作为对照组
    • 2、创建一个点光源,用来看点光源的影响 对 Unity默认的Shader效果 和 我们实现的Shader效果 之间的不同
  • 二、点光源的适配
    • 把上一篇文章中 ForwardBase 的 Pass 复制粘贴 到 与 该Pass平行的程序块,然后再对其做之后点光源的灯光适配(因为点光源 和 聚光灯效果,是在ForwordAdd中实现的)
    • 按上面步骤修改后,小球变的 受点光源的影响 又受 主平行光的影响 ![请添加图片描述](https://img-blog.csdnimg.cn/ffaccd6cfcea4830b694372b23d36a5a.gif)
  • 三、不同灯光类型的支持与区分
    • 1、我们加入一个聚光灯
    • 2、使用内置的宏定义生成Shader变体来区分是什么类型的光照
    • 3、剔除无用的变体,节省性能
  • 最终测试代码:


前言

Unity中Shader不同灯光类型的支持与区分


一、在开始之前做一些准备

1、在上一篇文章的场景基础上,增加一个Unity默认的球体作为对照组

创建前:
在这里插入图片描述
创建后:
在这里插入图片描述

2、创建一个点光源,用来看点光源的影响 对 Unity默认的Shader效果 和 我们实现的Shader效果 之间的不同

二、点光源的适配

把上一篇文章中 ForwardBase 的 Pass 复制粘贴 到 与 该Pass平行的程序块,然后再对其做之后点光源的灯光适配(因为点光源 和 聚光灯效果,是在ForwordAdd中实现的)

Shader "MyShader/P1_5_4"
{Properties{//光照系数_DiffuseIntensity("Diffuse Intensity",float) = 1}SubShader{Tags { "RenderType"="Opaque" }Pass{Tags{"LightMode"="ForwardBase"}CGPROGRAM#pragma vertex vert#pragma fragment frag#include "UnityCG.cginc"#include "Lighting.cginc"struct appdata{float4 vertex : POSITION;float2 uv : TEXCOORD0;//在应用程序阶段传入到顶点着色器中,时加入顶点法向量信息half3 normal:NORMAL;};struct v2f{float2 uv : TEXCOORD0;float4 vertex : SV_POSITION;//定义一个3维向量,用于接受世界坐标顶点法向量信息half3 worldNormal:TEXCOORD1;};half _DiffuseIntensity;v2f vert (appdata v){v2f o;o.vertex = UnityObjectToClipPos(v.vertex);//把顶点法线本地坐标转化为世界坐标o.worldNormal = UnityObjectToWorldNormal(v.normal);return o;}fixed4 frag (v2f i) : SV_Target{//Lambert光照模型的结果//Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L))//使用 Unity 封装的参数 获取环境光色float Ambient = unity_AmbientSky;//在属性面板定义一个 可调节的参数 用来作为光照系数,调节效果的强弱half Kd = _DiffuseIntensity;//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//使用Lambert公式计算出光照//fixed4 Diffuse = Ambient + (Kd * LightColor * dot(N,L));//因为 当 顶点法线 与 反射点指向光源的向量 垂直 或成钝角时,光照效果就该忽略不计//所以,这里使用 max(a,b)函数来限制 点积的结果范围fixed4 Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L));return Diffuse;}ENDCG}Pass{Tags{"LightMode"="ForwardAdd"}CGPROGRAM#pragma vertex vert#pragma fragment frag#include "UnityCG.cginc"#include "Lighting.cginc"struct appdata{float4 vertex : POSITION;float2 uv : TEXCOORD0;//在应用程序阶段传入到顶点着色器中,时加入顶点法向量信息half3 normal:NORMAL;};struct v2f{float2 uv : TEXCOORD0;float4 vertex : SV_POSITION;//定义一个3维向量,用于接受世界坐标顶点法向量信息half3 worldNormal:TEXCOORD1;};half _DiffuseIntensity;v2f vert (appdata v){v2f o;o.vertex = UnityObjectToClipPos(v.vertex);//把顶点法线本地坐标转化为世界坐标o.worldNormal = UnityObjectToWorldNormal(v.normal);return o;}fixed4 frag (v2f i) : SV_Target{//Lambert光照模型的结果//Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L))//使用 Unity 封装的参数 获取环境光色float Ambient = unity_AmbientSky;//在属性面板定义一个 可调节的参数 用来作为光照系数,调节效果的强弱half Kd = _DiffuseIntensity;//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//使用Lambert公式计算出光照//fixed4 Diffuse = Ambient + (Kd * LightColor * dot(N,L));//因为 当 顶点法线 与 反射点指向光源的向量 垂直 或成钝角时,光照效果就该忽略不计//所以,这里使用 max(a,b)函数来限制 点积的结果范围fixed4 Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L));return Diffuse;}ENDCG}}
}

把复制后的光照模式改为ForwardAdd

Tags{“LightMode”=“ForwardAdd”}

修改后,点光源对我们的小球已经有了初步的影响
请添加图片描述
但是,会发现不受主平行光的影响了,所以需要进行修改

并且,由于默认的混合模式为 Blend One Zero。
渲染时,由于主平行光先渲染,点光源后渲染,所以颜色缓冲区会被后渲染的点光源覆盖。
所以修改ForwordAdd 的 Pass 中 混合模式为 Blend One One

Blend One One

因为计算点光源时不需要考虑环境光,所以在Lambert光照模型中删除环境光的影响

在ForwardAdd的Pass中的片元着色器中,把最后的输出结果修改为如下:

fixed4 Diffuse = LightColor * max(0,dot(N,L));

把片元着色器简化为:

fixed4 frag (v2f i) : SV_Target{//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//因为计算点光源时不需要考虑环境光,所以在Lambert光照模型中删除环境光的影响fixed4 Diffuse = LightColor * max(0,dot(N,L));return Diffuse;}

按上面步骤修改后,小球变的 受点光源的影响 又受 主平行光的影响
请添加图片描述

三、不同灯光类型的支持与区分

1、我们加入一个聚光灯

在这里插入图片描述
加入聚光灯后,会发现小球只渲染了聚光灯的效果
原因是:
目前场景只支持一盏逐像素灯,在 聚光灯 和 点光源 之间,谁的强度大,谁就变成逐像素灯

请添加图片描述

2、使用内置的宏定义生成Shader变体来区分是什么类型的光照

#pragma multi_compile_fwdadd
定义在LightMode=ForwardAdd的Pass中,在此Pass中用来计算其它的逐像素光照.而此指令的作用是一次性生成Unity在ForwardAdd中需要的各种内置宏.
DIRECTIONAL DIRECTIONAL_COOKIE POINT POINT_COOKIE SPOT

  1. DIRECTIONAL :判断当前灯是否为平行灯.
  2. DIRECTIONAL_COOKIE :判断当前灯是否为Cookie平行灯.
  3. POINT :判断当前灯是否为点灯.
  4. POINT_COOKIE :判断当前灯是否为Cookie点灯.
  5. SPOT :判断当前灯是否为聚光灯.

在 ForwardAdd 的 Pass 中加入这条宏

#pragma multi_compile_fwdadd

然后,我看可以看见该Shader生成了6个变体
在这里插入图片描述

在这里插入图片描述

我们在片元着色器中,测试使用一下这些变体
1、当为点光源时,返回绿色

fixed4 frag (v2f i) : SV_Target{#if POINTreturn fixed4(0,1,0,1);#endif//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//因为计算点光源时不需要考虑环境光,所以在Lambert光照模型中删除环境光的影响fixed4 Diffuse = LightColor * max(0,dot(N,L));return Diffuse;}

请添加图片描述

2、当为点光源时,返回黑色

fixed4 frag (v2f i) : SV_Target{#if POINTreturn fixed4(0,1,0,1);#elif SPOTreturn 0;#endif//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//因为计算点光源时不需要考虑环境光,所以在Lambert光照模型中删除环境光的影响fixed4 Diffuse = LightColor * max(0,dot(N,L));return Diffuse;}

请添加图片描述

3、剔除无用的变体,节省性能

因为Shader变体的数量是一般是倍数增加,所以在设计时,就要尽量减少Shader的变体数量
Shader变体的数量,会直接影响 ShaderLab 的内存,打包到手机会影响到 Native 内存

法一:手动声明我们需要的变体
#pragma multi_compile POINT SPOT	

在这里插入图片描述
在这里插入图片描述

法二:剔除不需要的变体
#pragma skip_variants XXX01 XXX02...
剔除指定的变体,可同时剔除多个

#pragma skip_variants DIRECTIONAL POINT_COOKIE DIRECTIONAL_COOKIE

效果是一样的:
在这里插入图片描述

最终测试代码:

Shader "MyShader/P1_5_4"
{Properties{//光照系数_DiffuseIntensity("Diffuse Intensity",float) = 1}SubShader{Tags { "RenderType"="Opaque" }Pass{Tags{"LightMode"="ForwardBase"}CGPROGRAM#pragma vertex vert#pragma fragment frag#include "UnityCG.cginc"#include "Lighting.cginc"struct appdata{float4 vertex : POSITION;float2 uv : TEXCOORD0;//在应用程序阶段传入到顶点着色器中,时加入顶点法向量信息half3 normal:NORMAL;};struct v2f{float2 uv : TEXCOORD0;float4 vertex : SV_POSITION;//定义一个3维向量,用于接受世界坐标顶点法向量信息half3 worldNormal:TEXCOORD1;};half _DiffuseIntensity;v2f vert (appdata v){v2f o;o.vertex = UnityObjectToClipPos(v.vertex);//把顶点法线本地坐标转化为世界坐标o.worldNormal = UnityObjectToWorldNormal(v.normal);return o;}fixed4 frag (v2f i) : SV_Target{//Lambert光照模型的结果//Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L))//使用 Unity 封装的参数 获取环境光色float Ambient = unity_AmbientSky;//在属性面板定义一个 可调节的参数 用来作为光照系数,调节效果的强弱half Kd = _DiffuseIntensity;//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//使用Lambert公式计算出光照//fixed4 Diffuse = Ambient + (Kd * LightColor * dot(N,L));//因为 当 顶点法线 与 反射点指向光源的向量 垂直 或成钝角时,光照效果就该忽略不计//所以,这里使用 max(a,b)函数来限制 点积的结果范围fixed4 Diffuse = Ambient + Kd * LightColor * max(0,dot(N,L));return Diffuse;}ENDCG}Pass{Tags{"LightMode"="ForwardAdd"}Blend One OneCGPROGRAM#pragma vertex vert#pragma fragment frag//加入Unity自带的宏,用于区分不同的光照//只声明我们需要的变体//#pragma multi_compile POINT SPOT#pragma multi_compile_fwdadd//剔除我们不需要的变体#pragma skip_variants DIRECTIONAL POINT_COOKIE DIRECTIONAL_COOKIE#include "UnityCG.cginc"#include "Lighting.cginc"struct appdata{float4 vertex : POSITION;float2 uv : TEXCOORD0;//在应用程序阶段传入到顶点着色器中,时加入顶点法向量信息half3 normal:NORMAL;};struct v2f{float2 uv : TEXCOORD0;float4 vertex : SV_POSITION;//定义一个3维向量,用于接受世界坐标顶点法向量信息half3 worldNormal:TEXCOORD1;};half _DiffuseIntensity;v2f vert (appdata v){v2f o;o.vertex = UnityObjectToClipPos(v.vertex);//把顶点法线本地坐标转化为世界坐标o.worldNormal = UnityObjectToWorldNormal(v.normal);return o;}fixed4 frag (v2f i) : SV_Target{#if POINTreturn fixed4(0,1,0,1);#elif SPOTreturn 0;#endif//获取主平行光的颜色fixed4 LightColor = _LightColor0;//获取顶点法线坐标(让其归一化)fixed3 N = normalize(i.worldNormal);//获取反射点指向光源的向量(因为内置了获取的方法,所以不用向量减法来计算)fixed3 L = _WorldSpaceLightPos0;//因为计算点光源时不需要考虑环境光,所以在Lambert光照模型中删除环境光的影响fixed4 Diffuse = LightColor * max(0,dot(N,L));return Diffuse;}ENDCG}}
}

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

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

相关文章

R语言的计量经济学实践技术应用

计量经济学通常使用较小样本,但这种区别日渐模糊,机器学习在经济学领域、特别是经济学与其它学科的交叉领域表现日益突出,R语言是用于统计建模的主流计算机语言,在本次培训中,我们将从实际应用出发,重点从数…

【arm实验2】按键中断事件控制实验

设置按键中断,按键1按下,LED亮,再次按下,灭 按键2按下,蜂鸣器叫,再次按下,停 按键3按下,风扇转,再次按下,停 主函数: linuxlinux:~/study/08-c$…

【基础篇】四、本地部署Flink

文章目录 1、本地独立部署会话模式的Flink2、本地独立部署会话模式的Flink集群3、向Flink集群提交作业4、Standalone方式部署单作业模式5、Standalone方式部署应用模式的Flink Flink的常见三种部署方式: 独立部署(Standalone部署)基于K8S部署…

使用Tortoisegit界面拉起master主分支以副分支以及命令行操作

文章目录 1、Gui操作界面2、命令行操作3、合并分支到master分支上面 1、Gui操作界面 "小乌龟"通常指的是Git的图形用户界面(GUI)工具,其中比较常见的是TortoiseGit。下面是使用TortoiseGit来拉取(checkout)一…

数据结构(一)—— 数据结构简介

文章目录 一、基本概念和术语?1.1、数据1.2、数据元素1.3、数据项(属性、字段)1.4、数据对象1.5、数据结构 二、逻辑结构和物理结构(存储结构)2.1、逻辑结构1、定义2、分类(线性结构和非线性结构&#xff0…

Netty 入门 — 亘古不变的Hello World

这篇文章我们正式开始学习 Netty,在入门之前我们还是需要了解什么是 Netty。 什么是 Netty 为什么很多人都推崇 Java boy 去研究 Netty?Netty 这么高大上,它到底是何方神圣? 用官方的话说:Netty 是一款异步的、基于事…

vue绑定style和class 对象写法

适用于:要绑定多个样式,个数确定,名字也确定,但不确定用不用。 绑定 class 样式【对象写法】: .box{width: 100px;height: 100px; } .aqua{background-color: aqua; } .border{border: 20px solid red; } .radius{bor…

【动态库】Ubuntu 添加动态库的搜索路径

在运行程序时,经常遇到下面这种动态库加载失败的情况,这时往往是系统在动态库的搜索路径下没有找到对应的库文件导致的。 目录 一、使用 LD_LIBRARY_PATH 二、修改 /etc/ld.so.conf 一、使用 LD_LIBRARY_PATH 环境变量 LD_LIBRARY_PATH是动态库的搜索…

Jenkins集成newman

一、Docker环境准备 二、Jenkins环境准备 三、登录Jenkins 安装NodeJs插件 四、Jenkins全局工具配置Nodejs 五、创建Jenkins自由风格项目 构建步骤1:选择Execute NodeJS script构建步骤2:选择执行shell脚本 六、将postman相关的脚本、环境变量数据、全局…

Java反射调用jar包

反射作为托管语言的特性,很多框架都依赖反射来实现容器和面向接口编程,对架构程序很重要,首先试验Java的反射部分。 首先创建一个ZLZJar的工程,让他打包成jar包,代码如下 public class Test {//加两个整数public int…

ELK 日志分析系统介绍与部署

目录 一、ELK 简介: 1.开源工具介绍: 2.其它组件: 2.1 Filebeat: 2.2 Fluentd: 2.3 缓存/消息队列(redis、kafka、RabbitMQ等): 3. filebeat 结合 logstash 带来好处: 二、为什么要…

[0xGameCTF 2023] web题解

文章目录 [Week 1]signinbaby_phphello_httprepo_leakping [Week 2]ez_sqli方法一&#xff08;十六进制绕过&#xff09;方法二&#xff08;字符串拼接&#xff09; ez_upload [Week 1] signin 打开题目&#xff0c;查看下js代码 在main.js里找到flag baby_php <?php /…

云安全——云计算基础

0x00 前言 学习云安全&#xff0c;那么必然要对云计算相关的内容进行学习和了解&#xff0c;所以云安全会分为两个部分来进行&#xff0c;首先是云计算先关的内容。 0x01 云计算 广泛传播 云计算最早大范围传播是2006年&#xff0c;8月&#xff0c;在圣何塞【1】举办的SES&a…

【Vue面试题二十八】、vue要做权限管理该怎么做?如果控制到按钮级别的权限怎么做?

文章底部有个人公众号&#xff1a;热爱技术的小郑。主要分享开发知识、学习资料、毕业设计指导等。有兴趣的可以关注一下。为何分享&#xff1f; 踩过的坑没必要让别人在再踩&#xff0c;自己复盘也能加深记忆。利己利人、所谓双赢。 面试官&#xff1a;vue要做权限管理该怎么做…

【Java 进阶篇】JavaScript电灯开关案例:从原理到实现

JavaScript是一门强大的编程语言&#xff0c;它可以用来创建各种交互式网页应用。在这篇博客中&#xff0c;我们将通过一个简单的电灯开关案例来深入了解JavaScript的基础概念&#xff0c;包括HTML、CSS和JavaScript的结合使用。我们将从头开始构建这个案例&#xff0c;逐步引入…

PYTHON进阶-面向对象编程

&#x1f308;write in front&#x1f308; &#x1f9f8;大家好&#xff0c;我是Aileen&#x1f9f8;.希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流. &#x1f194;本文由Aileen_0v0&#x1f9f8; 原创 CSDN首发&#x1f412; 如…

windows计划任务的配置文件

界面操作 创建计划 依次设置 命令行操作 SCHTASKS 命令简介 SCHTASKS 命令是由微软公司开发并内置于 Windows 系统中的一个命令行工具。该命令可用于设置、修改、查询和删除计划任务&#xff0c;或启动计划任务中所定义的程序或脚本。 SCHTASKS 命令的基本语法 SCHTASKS 命…

多输入多输出 | MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测

MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测 目录 MATLAB实现CNN-BiLSTM-Attention卷积神经网络-双向长短期记忆网络结合SE注意力机制的多输入多输出预测预测效果基本介绍程序设计往期精彩参考资料 预测效果 基本介绍 C…

SpringBoot面试题1:什么是SpringBoot?为什么要用SpringBoot?

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:什么是SpringBoot? Spring Boot 是一个用于快速开发独立的、基于 Spring 框架的应用程序的开源框架。它简化了 Spring 应用的配置和部署过程,使…

基于秃鹰优化的BP神经网络(分类应用) - 附代码

基于秃鹰优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码 文章目录 基于秃鹰优化的BP神经网络&#xff08;分类应用&#xff09; - 附代码1.鸢尾花iris数据介绍2.数据集整理3.秃鹰优化BP神经网络3.1 BP神经网络参数设置3.2 秃鹰算法应用 4.测试结果&#xff1a;5.M…