在 Substance Painter中实现Unity Standard Shader

news2025/1/10 16:59:47

由于有需要在Substance Painter中显示什么样的效果,在Unity就要显示什么样的效果的需求,最近研究了几天,总算在Substance Painter中实现Unity standard的材质的渲染效果。具体效果如下:
在Unity中:
在这里插入图片描述

Substance Painter中:
在这里插入图片描述
相识度能够达到百分之八九十吧。主要是Unity的项目使用的是Gamma颜色空间,还是有很大的出入,而且还不好修改。

这一篇不再讲基础的如何在Substance Painter中自定义Shader了,不了解的可以翻一翻我之前写的。

版本相关

Unity 2019.4.40 内置渲染管线 Gamma颜色空间
Substance Painter 9

痛点

  1. 阴影问题,在sp里面,阴影是取环境光里面最亮的点作为主光源位置,这个我的解决方案就是自己调,设置主光源朝向配置项,可以同步unity和sp里面的主光源朝向,那么自动生成的shadow也是一个方向的。
  2. 间接光源的镜面反射,其它文章里面也说这个问题了,大都没有很好的解决方案,我这里的解决方案是通过shader去实现动态Mipmap实现,作为调试最终效果时使用。

接下来梳理一下实现过程

模型朝向同步

两个软件使用的坐标系不同,unity是左手坐标系,而sp是右手坐标系,解决方案是,在unity里面在y轴旋转180度。
在这里插入图片描述

保证天空球的位置正确

在unity里面,我专门搭建了一个测试环境,环境反射,直接设置了一张环境反射Cubemap
在这里插入图片描述
在sp里面,需要打开显示设置,然后设置背景贴图,背景曝光设置0,背景旋转设置为270,这样设置完,两个场景去采样环境全景图时,采样的位置是一致的。
在这里插入图片描述
贴图直接用的sp里面的,位置在项目文件夹下面
在这里插入图片描述

相机同步

在unity里面使用默认的透视相机,Field of View 为60
在这里插入图片描述
在sp里面,视角设置60度,与unity相同,记得把后期特效关闭
在这里插入图片描述

主光源同步

主光源同步我的思路是,在SP里面设置主光源参数,主光源参数需要一个旋转参数,主光源颜色,以及强度,主光源默认是方向光。
在这里插入图片描述
在SP里面,我们需要先在shader里面定义参数
在这里插入图片描述
显示效果
在这里插入图片描述
接下来比较重要的就是,在SP里面将主光源的旋转,修改为朝向,实现函数这里要感谢文心一言,它总算干了一件正事
在这里插入图片描述
由此获得主光源朝向。

定义一些Unity常量

要同步shader,需要将一些常量设置
在这里插入图片描述
SP是线性空间的,所以Standard里面一些Gamma相关的代码自动摒弃。
贴图直接使用内置的方法去获取
在这里插入图片描述
还有一些和Standard对应的参数
在这里插入图片描述
间接光漫反射的球谐光照参数,是直接写死在shader里面的
在这里插入图片描述
还有像在unity里面常用的一些函数,比如saturate和lerp,都直接定义出来
在这里插入图片描述

从SP导出贴图设置

我的设置是导出四张贴图,以后需要再加,分别是Albedo贴图,Normal,Emissive,MRA
在这里插入图片描述
MRA贴图由三张贴图拼接而成,三个通道分别是Metallic,Roughness,AO

在unity里面,也重写了FragmentSetup函数,修改了里面一些逻辑
在这里插入图片描述
本来standard比较笨重,所以,我们可以将一些没必要的设置改掉,比如工作流我确定使用金属工作里,那就不需要判断了,直接使用金属工作流
在这里插入图片描述
在SP里面,直接使用内置库函数去获取贴图,思路直接按照standard的思路来即可。
在这里插入图片描述

实现直接光照

直接光照在standard里面有多种方式,为了保证效果,我这里直接使用了效果最好,渲染最昂贵的双向表面分布函数实现,代码渲染逻辑没动,还是standard的哪一套
在这里插入图片描述
在SP里面,直接复制过来即可,除了一些没用的代码删除掉了,缺少什么函数,在unity复制过来改改就能用,glsl和cg区别不大。
在这里插入图片描述
最后,输出时,我只使用了内置diffuseShadingOutput函数输出渲染。
在这里插入图片描述
同样参数下,两个模型的渲染效果
在这里插入图片描述
在这里插入图片描述

实现间接光漫反射

间接光漫反射都是直接使用SH球谐光照,我在SP里面是直接写死的参数,参数是在unityframedebug里面抄的
在这里插入图片描述
直接将它抄给SP
在这里插入图片描述
实现很简单,直接用standard的函数实现即可
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

实现间接光镜面反射

间接光镜面反射是这里面最难实现的,主要还是mipmap的问题,如果粗糙度为0,也就是最光滑的平面的时候,可以看到,获取到的镜面反射的效果是一致的。
Unity Roughness=0
在这里插入图片描述
Substance Painter Roughness=0
在这里插入图片描述
如果Roughness加大,mipmap层级上去以后,区别就很明显了。
Unity Roughness=0.5
在这里插入图片描述
Substance Painter Roughness=0.5
在这里插入图片描述
造成这样的结果的原因是两个引擎内部的对Cubemap的mipmap采样造成的,看一下Roughness=1的结果,更加明显。
Unity Roughness= 1
在这里插入图片描述
Substance Painter Roughness=1
在这里插入图片描述
在这里,我还是先列一下实现方式,两个引擎都是通过采样环境反射球来实现的环境光的镜面反射。unity里面是通过Unity_GlossyEnvironment函数实现,支持两个Cubemap采样结果混合。
在这里插入图片描述
在Substance Painter里面,则是直接使用的SP内置的函数库 lib-env.glsl,里面有个方法envSample,传入方向和mipmap即可实现
在这里插入图片描述
在这里插入图片描述
粗糙度转Mipmap和Unity的也有所不同,这个代码抄至 nagnae blog

#define UNITY_SPECCUBE_LOD_STEPS_CUSTOM 6
real PerceptualRoughnessToMipmapLevel(real perceptualRoughness)
{
    half mip = perceptualRoughness * (1.7 - 0.7 * perceptualRoughness);
	//mip = pow( mip, 0.53 );
	mip = pow( mip, 0.4 );
	//mip = pow( mip, 0.3 );
	mip *= 0.97;
	
	//float unity_environment_lod_count = float(unity_specularprobe_lod_count);

	#if 0
		 float mipmap_end = environment_max_lod - environment_mipmap_bias;
		 float mipmap_start = mipmap_end - UNITY_SPECCUBE_LOD_STEPS_CUSTOM;
	#elif 0	
		 float mipmap_start = max( 0, environment_max_lod - unity_environment_lod_count );
		 float mipmap_end = min( unity_environment_lod_count, mipmap_start + UNITY_SPECCUBE_LOD_STEPS_CUSTOM );
	#elif 0
		 float mipmap_start = 0;
		 float mipmap_scale = environment_max_lod / unity_environment_lod_count;
		 float mipmap_tail = ( unity_environment_lod_count - UNITY_SPECCUBE_LOD_STEPS_CUSTOM ) * mipmap_scale - 1;
		 float mipmap_end = min( environment_max_lod, environment_max_lod - mipmap_tail*1.2 );
	#elif 1
		 float mipmap_start = 0;
		 float mipmap_end = environment_max_lod - 1.5;
	#endif

	return mip * ( mipmap_end - mipmap_start ) + mipmap_start;
}

如果按照Substance Painter 默认的效果,结果是不理想的,环境反射越强烈,效果区别越明显,所以,我的解决方案,就是在shader里面进行一次mipmap,这是我在learnopengl学习的结果,地址:https://learnopengl.com/PBR/IBL/Specular-IBL,我实现的原理也是粗暴的进行多重采样
在这里插入图片描述
每个片元要进行1024次采样,这样性能很低,所以,我增加了一个配置项,可以进行一次采样,如果开启配置,实现多重采样,实现Unity内的那种顺滑的mipmap
在这里插入图片描述
在Roughness=0时,效果还是一致,这里不再展示。
Unity Roughness= 1
在这里插入图片描述
Substance Painter Roughness=1
在这里插入图片描述
可以看的出来,在Roughness为1时,颜色基本上相近了,只有高光范围还是有一些不同,那先这样,这也算一种解决方案。如果有更好的解决方案,欢迎小伙伴们告诉我。
Unity Roughness=0.5
在这里插入图片描述

Substance Painter Roughness=0.5
在这里插入图片描述
最终,实现了所有的直接光漫反射,直接光镜面反射 和间接光漫反射 间接光镜面反射,我们最后按照Unity的standard将颜色合并到一起,就实现了对应的效果。
在这里插入图片描述
Substance Painter Combie Roughness = 0.5 Metallic = 1
在这里插入图片描述

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

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

相关文章

as 加载aar(包含so文件)遇到问题的解决方法

""" r java.lang.UnsatisfiedLinkEnror :No implementation found for void com.example.myapplication2.Model.init(java.lang .Storing, java.lang .string) 原来是so文件里的Model类的所属的包名,跟我java里的不一样。linux里,可以用 …

AI数字人:最强声音驱动面部表情模型VideoReTalking

目录 1 VideoReTalking论文解读 1.1 介绍 1.2 相关工作 1.2.1 视频编辑中的音频配音 1.2.2 基于音频的单图像面部动画 1.3 框架 1.3.1 语义引导重演网络 1.3.2 口型同步网络 1.3.3 身份感知增强网络 1.3.4 后期处理 1.4 训练 1.4.1 每个模块的训练 1.4.2 评估 1.…

vue组件页面没有加载出来

现象:加载一个组件,生命周期走完,界面没有出来。 由于生命周期都走完,接口也都有请求,最初怀疑,样式原因导致dom元素被吞,然后修改样式无效。 定位方法:dom元素一一移除&#xff0…

Python爬虫技术系列-01请求响应获取-urllib库

Python爬虫技术系列-01请求响应获取-urllib库 1 urllib库1.1 urllib概述1.1.1 urllib简介1.1.2 urllib的robotparser模块1.1.3 request模块1.1.4 Error1.1.5 parse模块 1.2 urllib高级应用1.2.1Opener1.2.2 代理设置 1 urllib库 参考连接: https://zhuanlan.zhihu.…

继往开来,图鸟UI又推出一款高颜值、兼容多平台、丰富组件的图表组件模板

大家好,今天分享的主题是图表统计。图表统计是使用图表和图形来可视化和呈现数据的方法。它通过将数据转化为柱状图、折线图、饼图等形式来展示各种统计指标和趋势。 图表统计可以帮助我们更好地理解和分析数据,发现数据之间的关联和规律,并…

Qt5开发及实例V2.0-第十七章-Qt版MyWord字处理软件

Qt5开发及实例V2.0-第十七章-Qt版MyWord字处理软件 第17章-Qt版MyWord字处理软件17.1 运行界面17.1.1 菜单设计基本操作17.1.2.MyWord系统菜单 17.2 工具栏设计17.2.1 与菜单对应的工具条17.2.2 附加功能的工具条 这段代码的作用是加载系统标准字号集,只要在主窗体构…

景联文科技:2023人工智能数据标注行业现状分析?

随着人工智能产业的快速发展,AI数据标注行业也呈现出蓬勃生机。人工智能算法的不断开发和迭代,离不开结构化数据的支持,AI数据标注行业的发展也得到了人工智能核心产业的带动。预计2027年市场规模有望达到130-160亿元。 随着算法的不断演进和…

记一次CTF入门练习

前言 学习CTF有一段时间了,现在做个小小的总结,加深印象。 习题下载(只含第一部分) 链接:https://pan.baidu.com/s/1j-_TlAek4Q9IlEwbMWv2GA?pwd47pk 提取码:47pk 正文 1.1、MISC——找到证据 解压缩…

JavaWeb 学习笔记 7:Filter

JavaWeb 学习笔记 7:Filter 1.快速开始 使用过滤器的方式与 Servlet 类似,要实现一个Filter接口: WebFilter("/*") public class FirstFilter implements Filter {public void init(FilterConfig filterConfig) throws ServletE…

UOS Deepin Ubuntu Linux 开启 ssh 远程登录

UOS Deepin Ubuntu Linux 开启 ssh 远程登录 打开控制台 安装 openssh-server sudo apt -y install openssh-server修改 /etc/ssh/ssh_config 文件 sudo vim /etc/ssh/ssh_config找到 # Port 22 去掉 # 注释后 保存 重启 ssh 服务 sudo systemctl restart ssh设置 ssh 服务 开机…

pytorch学习------常见的优化算法

优化算法 优化算法就是一种调整模型参数更新的策略,在深度学习和机器学习中,我们常常通过修改参数使得损失函数最小化或最大化。 优化算法介绍 1、梯度下降算法(batch gradient descent BGD) 每次迭代都需要把所有样本都送入&…

VINS中的观测性问题

文章目录 一、背景二、BA problem的观测性问题1、不可观方向2、解决方案3、优化问题中信息矩阵物理意义 三、Keyframe-based Visual-Inertial SLAM的观测性问题1、不可观问题2、解决方案 四、MSCKF观测性分析1、观测性分析2、解决方案3、小结 一、背景 本文档分析以下VINS中的…

行业追踪,2023-09-22

自动复盘 2023-09-22 凡所有相,皆是虚妄。若见诸相非相,即见如来。 k 线图是最好的老师,每天持续发布板块的rps排名,追踪板块,板块来开仓,板块去清仓,丢弃自以为是的想法,板块去留让…

软考软件设计师-存储管理-文件管理-计算机网络(中

文章目录 一、存储管理页面置换算法 (最佳OPT)存储页面-先进先出置换算法(FIFO)最久未使用算法(最近最久未使用LRU) 二、文件管理初识文件管理文件目录-绝对路径文件管理-文件的结构文件管理-索引的分配 空闲存储空间的管理(位示图法)三、计算…

Spring | 事件监听器应用与最佳实践

引言 在复杂的软件开发环境中,组件之间的通信和信息交流显得尤为重要。Spring框架,作为Java世界中最受欢迎的开发框架之一,提供了一种强大的事件监听器模型,使得组件间的通信变得更加灵活和解耦。本文主要探讨Spring事件监听器的…

数据治理-组织触点

协调工作的一部分包括为数据治理工作制定组织接触点 采购和合同:首席数据官与供应商/合作伙伴的管理部门或者采购部门合作,制定和执行有关数据管理合同的标准文本。 预算和资金:如果首席数据官没有直接孔子所有与数据采购相关的预算&#xff…

sns.load_dataset(“iris“)报错原因探究+解决办法

问题描述 import seaborn as sns # 读取数据 iris sns.load_dataset("iris")在代码中使用了seaborn ,并加载iris数据,结果产生了报错信息如下所示 问题分析 原因很简单,我们使用了sns.load_dataset("iris")来加载数据…

在Python中创建相关系数矩阵的6种方法

相关系数矩阵(Correlation matrix)是数据分析的基本工具。它们让我们了解不同的变量是如何相互关联的。在Python中,有很多个方法可以计算相关系数矩阵,今天我们来对这些方法进行一个总结 Pandas Pandas的DataFrame对象可以使用c…

解决编译中遇到的问题:Please port gnulib freadahead.c to your platform

今天在编译旧版的gzip-1.7时遇到了一个错误: error: #error "Please port gnulib freadahead.c to your platform! Look at the definition of fflush, fread, ungetc on your system, then report this to bug-gnulib." 在网上搜了一下解决方法&#xf…

C++核心编程——P39~P44-运算符重载

运算符重载的概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。 1.加号运算符重载 作用:实现两个自定义数据类型相加的运算。 例如:两个整型相加编译器知道该怎么进行运算,如果是两个自定义…