当前位置: 首页 > ds >正文

漫反射实现+逐像素漫反射+逐像素漫反射实现

标准光照的构成结构

        自发光:材质本身发出的光,模拟环境使用的光

        漫反射光:光照在粗糙材质后,光的反射方向随机,还有一些光发生了折射,造成材质

                        表面没有明显的光斑。

        高光反射光:光照到材质表面后,无(低)损失直接反射给观察者眼睛,材质表面能观察

                        到光斑。

        环境光:模拟场景光照(简单理解为太阳光)

        裴祥凤提出的光照理论:标准光照=自发光+漫发射光+高光反射光+环境光这个理论是模拟

        光照效果,并不是真实效果。

        以他的名字命名:Phong光照模型

逐顶点光照和逐像素光照

        顶点着色器:会在模型渲染点上运行,其他的点会线性插值

        片元着色器:会在模型的所有像素点上运行。

        逐顶点光照:会在顶点着色器上进行光照运算(高洛德着色)

        逐像素光照:会在片元着色器上运行光照运算(phong着色)

        逐像素比逐顶点效果好,逐顶点比逐像素性能好

漫反射光照(兰伯特定律)

        漫反射光照=光源的颜色*材质的漫反射颜色*MAX(0,标准化后物体表面法向量·标准化

        后光源方向向量)

        光源颜色:场景中光GameObject取得

        材质的漫反射颜色:材质球配置

        Max是数学函数,标准化也有函数

        表面法线向量:CPU加载模型后,传递到GPU中的

        光源方向向量:场景中光GameObject取得

        实现:所有数据拿到,拿公式计算

半兰伯特定律:将整体颜色降低一半,再加一半

        漫反射光照=(光源的颜色·材质的漫反射颜色)·((标准化后物体表面法线向量·标准化后光源

        方向向量)·0.5+0.5)

高光反射光照

        高光光照=光源的颜色*材质高光反射颜色*MAX(0,标准化后的观察方向向量·标准化后的反射方向)^光晕系数

逐顶点光照测试相关代码如下所示:

// Upgrade NOTE: replaced '_Object2World' with 'unity_ObjectToWorld'
// Upgrade NOTE: replaced 'mul(UNITY_MATRIX_MVP,*)' with 'UnityObjectToClipPos(*)'Shader "CreateTest/DiffuseVertex"
{Properties{_DiffuseColor("漫反射颜色",Color) = (1,1,1,1)}SubShader{Pass{//设定光照模式为前向模式(才能正常获取光的颜色和光的方向)//Tags{"LightMode" = "ForwardBase"}CGPROGRAM#pragma vertex vert#pragma fragment frag//加载Cg语言的脚本,用来处理光照参数//处理光照的Cg库文件(cginc扩展名)//目录在Unity的安装目录下Editor/Data/CGIncludes/Lighting.cginc#include "Lighting.cginc"//导入材质颜色fixed4 _DiffuseColor;//如果在Cg编程中,顶点或片元着色器接收多个数值的时候,一般会用结构体实现//从CPU接收到的数据struct c2v{float4 vertex:POSITION;//从CPU接收到的模型空间下的点的位置float3 normal:NORMAL; //从CPU接收到的当前点的模型空间下的法线向量};//因为在顶点着色器中,需要计算裁剪空间下点的位置和Phong着色计算出的兰伯特定律计算后的颜色struct v2f{float4 pos:SV_POSITION;//经过顶点着色器计算后的,当前点的裁剪空间下的位置fixed3 color : COLOR;//经过兰伯特定律计算后的当前点的颜色};//高洛德着色(逐顶点光照),光照计算应该编写在顶点着色器中v2f vert(c2v data){//顶点着色器传递给片元着色器的数据结构体声明v2f r;//必须做的:将点从模型空间下,转换到裁剪空间下r.pos=UnityObjectToClipPos(data.vertex);//兰伯特定律计算//漫反射光照=光源的颜色*材质的漫反射颜色*MAX(0,标准化后物体表面法线向量*标准化后光源方向向量)//光照是在世界中发生,需要将所有的数值,变换到世界坐标系下,再运算// _Object2World矩阵是Unity提供的,用于转换模型空间下到世界空间下的转换矩阵//因为法线传递过来的是3x1的列矩阵,_Object2World是4x4的齐次矩阵,如果想做矩阵乘法,需要将齐次矩阵,变成3x3矩阵fixed3 worldNormal = normalize(mul((float3x3)unity_ObjectToWorld, data.normal));//获得直射光的光方向fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz);//公式运算fixed3 diffuse=_LightColor0.rgb* _DiffuseColor.rgb* max(0,dot(worldNormal,worldLightDir));//根据高洛德光照模型,将环境光追加在最终计算后的颜色上r.color = UNITY_LIGHTMODEL_AMBIENT.xyz + diffuse;//r.color = _LightColor0.rgb;return r;}//千万记得带"数字",若未带4可能会出现材质球颜色为红色情况fixed4 frag(v2f data) :SV_Target{return fixed4(data.color,1.0);}ENDCG}}Fallback "Diffuse"
}

逐像素光照测试相关代码如下:

// Upgrade NOTE: replaced '_Object2World' with 'unity_ObjectToWorld'
// Upgrade NOTE: replaced 'mul(UNITY_MATRIX_MVP,*)' with 'UnityObjectToClipPos(*)'Shader "CreateTest/DiffusePixel"
{Properties{_DiffuseColor("漫反射颜色",Color) = (1,1,1,1)}SubShader{Pass{//设定光照模式为前向模式(才能正常获取光的颜色和光的方向)//Tags{"LightMode" = "ForwardBase"}CGPROGRAM#pragma vertex vert#pragma fragment frag//加载Cg语言的脚本,用来处理光照参数//处理光照的Cg库文件(cginc扩展名)//目录在Unity的安装目录下Editor/Data/CGIncludes/Lighting.cginc#include "Lighting.cginc"//导入材质颜色fixed4 _DiffuseColor;//如果在Cg编程中,顶点或片元着色器接收多个数值的时候,一般会用结构体实现//从CPU接收到的数据struct c2v{float4 vertex:POSITION;//从CPU接收到的模型空间下的点的位置float3 normal:NORMAL; //从CPU接收到的当前点的模型空间下的法线向量};//因为在顶点着色器中,需要计算裁剪空间下点的位置和Phong着色计算出的兰伯特定律计算后的颜色struct v2f{float4 pos:SV_POSITION;//经过顶点着色器计算后的,当前点的裁剪空间下的位置//fixed3 color : COLOR;//经过兰伯特定律计算后的当前点的颜色float3 worldNormal:NORMAL;//经过矩阵转换后世界空间下的,法线向量};//将渲染点,从模型空间下,转换到裁剪空间下//将渲染点对应的法线,从模型空间下,转换到世界空间下v2f vert(c2v data){//顶点着色器传递给片元着色器的数据结构体声明v2f r;//必须做的:将点从模型空间下,转换到裁剪空间下r.pos=UnityObjectToClipPos(data.vertex);//几何运算,在顶点着色器中完成,再将运算好的数值,传递给片元着色器// _Object2World矩阵是Unity提供的,用于转换模型空间下到世界空间下的转换矩阵//因为法线传递过来的是3x1的列矩阵,_Object2World是4x4的齐次矩阵,如果想做矩阵乘法,需要将齐次矩阵,变成3x3矩阵r.worldNormal = mul((float3x3)unity_ObjectToWorld, data.normal);//r.color = _LightColor0.rgb;return r;}//phong着色(逐像素光照),光照计算应该编写在片元着色器中fixed4 frag(v2f data) :SV_Target{//兰伯特定律计算//漫反射光照=光源的颜色*材质的漫反射颜色*MAX(0,标准化后物体表面法线向量*标准化后光源方向向量)//光照是在世界中发生,需要将所有的数值,变换到世界坐标系下,再运算//世界空间下的,表面法线向量标准化fixed3 worldNormal = normalize(data.worldNormal);//获得直射光的光方向,标准化向量fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz);//公式运算fixed3 diffuse = _LightColor0.rgb * _DiffuseColor.rgb * max(0, dot(worldNormal, worldLightDir));//根据Phong光照模型,将环境光追加在最终计算后的颜色上fixed3 color = UNITY_LIGHTMODEL_AMBIENT.xyz + diffuse;return fixed4(color,1.0);}ENDCG}}Fallback "Diffuse"
}

两者之间的区别如下图所示:

 该系列专栏为网课课程笔记,仅用于学习参考。       

http://www.xdnf.cn/news/2905.html

相关文章:

  • 机器学习分类模型性能评估:应对类别不平衡的策略与指标
  • 数据结构 RBT 插入操作的 Python 代码实现
  • EMB量产首航!炯熠电子引领「线控底盘革命」
  • SOLIDWORKS修改模型默认颜色教程
  • Unity AI-使用Ollama本地大语言模型运行框架运行本地Deepseek等模型实现聊天对话(一)
  • WebXR教学 06 项目4 跳跃小游戏
  • for(auto it: vec)和for(auto it: vec)的区别以及使用场景
  • Java—— Arrays工具类及Lambda表达式
  • 联合体union的特殊之处
  • 软件测试实验报告3 | 自动化测试工具的基本操作
  • 局域网传文件——基于flask实现
  • 9.Three.js中 ArrayCamera 多视角相机详解+示例代码
  • RISCV学习(5)GD32VF103 MCU架构了解
  • 修改Hosts文件没有生效的解决办法
  • LM393比较器的比较翻转电压不对
  • seaborn数据统计可视化-介绍
  • 需要掌握的前端安全概念以及实操
  • 【React Native】精通 react native
  • 第十四届蓝桥杯Scratch03月stema选拔赛——九九乘法表
  • 城市群出行需求的时空分形
  • 工厂设计模式
  • GRPO vs SFT:强化学习提升大模型多模态推理泛化能力的原因研究
  • 2.2.1goweb内置的 HTTP 处理程序
  • Daytona - 开源开发环境管理器
  • 【前端】CSS实现div双背景图叠加
  • QTableView复选框居中
  • Java多线程入门案例详解:继承Thread类实现线程
  • 将本地Springboot项目部署到Linux服务器
  • Web开发之三层架构
  • 从困局到破局的AI+数据分析