崩坏3mmd中的渲染技术研究
http://youxiputao.com/articles/11839 主要是参考该篇文章做一个微小的复盘。
漫反射与高光
文章中的漫反射与高光并不是类似于普通的
resultCol = Diffuse(...) + Specular(...)
而是更贴近实际的cg绘画流程中的,基色+阴影色+高光的模式。
其中阴影色以正片叠底的方式叠加到基色上,之后再加上高光
这也是为什么文章中的DiffuseRamp跟SpecularRamp的两张贴图方向相反。因为Diffuse是作为阴影上色,因此其Ramp贴图在照度低的左部颜色更亮。
同时这种上色也使得传统的forward rendering中的复数光源用blend one one进行渲染变得不可能了(我猜的),因此只有一个主要的directional light作为光源。不过这样也够用了。
这里涉及到的正片叠底的模式,其实就是原有色与叠加色直接相乘。叠加色为1(白)时,结果不变,叠加色为0时变黑。文章中不止考虑了叠加的颜色,还考虑了叠加的强度(alpha通道)。此时叠加的代码如下:
//_DiffuseLayer1是叠加色,diffuse是在diffuseRamp上采样的结果
resultCol *= lerp(_DiffuseLayer1.rgb, fixed3(1, 1, 1), 1 - (_DiffuseLayer1.a * diffuse));
在cg绘画流程中,阴影图层往往不止一个,而是有多个来提升表现力。在文章中提到,DiffuseRamp的三个通道是分开使用进行上色的。设计师通过修改对应三个通道的Tint色来进行颜色的调整。此时代码如下:
fixed3 diffuse = tex2D(_DiffuseRamp, float2(diffuseAtten,0));
fixed3 white = fixed3(1, 1, 1);
resultCol *= lerp(_DiffuseLayer1.rgb, white, 1 - (_DiffuseLayer1.a * diffuse.r));
resultCol *= lerp(_DiffuseLayer2.rgb, white, 1 - (_DiffuseLayer2.a * diffuse.g));
resultCol *= lerp(_DiffuseLayer3.rgb, white, 1 - (_DiffuseLayer3.a * diffuse.b));
specular的部分和这个类似,区别在于不是正片叠底,而是直接相加。
软硬风格切换没什么好说的,根据采样到的顶点色去修改diffuseRamp的采样位置即可。
文章中提到的降低面部阴影的操作,我猜测是用了pow函数,将亮度采样整体提升,同时也可以保证亮度为0、1时不受干扰。
diffuseAtten = saturate(pow(diffuseAtten, 1 - 0.95 * _ShadowMask)); //so shadow will be less when shadowmask is high.
边缘光
不同于普通的边缘光,加入了对环境的采样。代码如下
half3 Fresnel(half3 normal, half3 viewDir,half exponent) {
//fresnel(rim light)
half fresnel = 1 - DotClamped(normal, viewDir);
fresnel = saturate(pow(fresnel, exponent));
half3 reflectionDir = reflect(-viewDir, normal);
float4 envSample = UNITY_SAMPLE_TEXCUBE(unity_SpecCube0, reflectionDir);
float3 envLight = DecodeHDR(envSample, unity_SpecCube0_HDR);
return envLight * fresnel;
}
眼睛折射
最简单的眼球渲染的模型就是一个圆形。
但是考虑到人眼的虹膜部分并不是跟眼球一个表面,而是内凹的一个形状,因此正确的渲染是虹膜部分向内部凹陷一定距离。
另外,虹膜到眼球表面的这段距离会造成一定的折射效果。但是这个折射效果跟凹陷效果其实不太容易分辨的出来,可以当做一种效果处理。
具体的实现就是根据视线方向去计算uv偏移。我们可以输入一个深度贴图,表示某个位置距离眼球表面的距离。
偏移的计算参考自 https://computergraphics.stackexchange.com/questions/4133/eye-parallax-refraction
//choose one to calculate uv offset(or use both)
float2 ReflectOffset(float2 ouv,half3 V) {
fixed depth = tex2D(_EyeDepthMask, ouv);
float2 offset = depth * mul((float2x3)unity_WorldToObject, V);
offset.y = -offset.y;
return ouv + offset * _EyeCenter.z;
}
float2 PBReflection(float2 ouv, half3 V, half3 frontW, half3 refractedW) {
half depth = tex2D(_EyeDepthMask, ouv).r * _EyeCenter.w;
float cosAlpha = dot(frontW, -refractedW);
float dist = depth/cosAlpha;
float3 offsetW = dist * refractedW;
float2 offsetL = mul((float2x3)unity_WorldToObject, offsetW);
offsetL.y = -offsetL.y;
return ouv + offsetL;
}
眼睛焦散
跟文章中提到的类似,根据反射后的lightDir计算一个diffuseterm,加上去就行了。
//caustic
half3 backLightDir = normalize(reflect(-lightDir, eyeForward));
half causticAtten = pow(DotClamped(backLightDir, normal),3);
half3 caustic = _EyeCausticTint * causticAtten * _LightColor0.rgb * s.EyeCaustic; //s.EyeCaustic是Caustic贴图的采样
resultCol += caustic;
头发高光
同时顶部的文章中的图片已经给出了代码。需要注意的是根据uv方向的不同,tangent可能要换成bitangent。
加入JitterMap后做抖动:
float tangentShift = tex2D(_SpecularJitterMap, IN.uv_SpecularJitterMap);
o.tangent += tangentShift * IN.worldNormal * _SpecularJitterStrength;
o.tangent = normalize(o.tangent);
文章中提到了两层高光,一层低频一层高频。可以配合SpecularRamp的两个通道去做,这样最终的效果会更贴近绘画的风格。
参考: http://amd-dev.wpengine.netdna-cdn.com/wordpress/media/2012/10/Scheuermann_HairSketchSlides.pdf
镜面反射
参考 http://gad.qq.com/article/detail/18544。
其中文末提到的bug可以通过斜投影解决,斜投影矩阵的计算可以参考http://gad.qq.com/article/detail/18612。
Unity内置了计算斜投影的函数,可以不用操心行列向量左右手坐标系之类琐碎的事情,具体的实现可以参考http://wiki.unity3d.com/index.php/MirrorReflection4。
屏幕空间反射
个人觉得挺有意思的效果,从头学了一下,重新实现了一遍,在 http://www.cnblogs.com/yangrouchuan/p/7574405.html
有个问题就是无法反射Forward rendering的物体。视频中似乎是只在远离人物的地方使用SSR。舞台平面都是用的镜面反射。
剩下的日后慢慢更新,目前还太菜了。。。
崩坏3mmd中的渲染技术研究的更多相关文章
- Unity Shader入门精要学习笔记 - 第16章 Unity中的渲染优化技术
转自冯乐乐的 <Unity Shader 入门精要> 移动平台的特点 为了尽可能一处那些隐藏的表面,减少overdraw(即一个像素被绘制多次),PowerVR芯片(通常用于ios设备和某 ...
- 基于.net的分布式系统限流组件 C# DataGridView绑定List对象时,利用BindingList来实现增删查改 .net中ThreadPool与Task的认识总结 C# 排序技术研究与对比 基于.net的通用内存缓存模型组件 Scala学习笔记:重要语法特性
基于.net的分布式系统限流组件 在互联网应用中,流量洪峰是常有的事情.在应对流量洪峰时,通用的处理模式一般有排队.限流,这样可以非常直接有效的保护系统,防止系统被打爆.另外,通过限流技术手段,可 ...
- geotrellis使用(三十四)矢量瓦片技术研究——矢栅一体化
前言 本文所涉及技术与Geotrellis并无太大关系,仅是矢量瓦片前端渲染和加载技术,但是其实我这是在为Geotrellis的矢量瓦片做铺垫.很多人可能会说,Geotrellis为什么要搞矢量瓦片, ...
- 基于Unity 5的次世代卡通渲染技术 -- Unite 2017 米哈游总监贺甲分享实录
在5月12日Unite2017开发者大会上,米哈游技术总监兼美术指导贺甲进行了主题为次世代卡通渲染的演讲.一下为详细分享内容: 大家好,首先自我介绍一下,我叫贺甲,在米哈游担任技术总监和美术指导工作, ...
- 剖析Unreal Engine超真实人类的渲染技术Part 1 - 概述和皮肤渲染
一.概述 1.1 数字人类的概要 数字人类(Digital Human)是利用计算机模拟真实人类的一种综合性的渲染技术.也被称为虚拟人类.超真实人类.照片级人类. 它是一种技术和艺术相结合的综合性模拟 ...
- 剖析Unreal Engine超真实人类的渲染技术Part 2 - 眼球渲染
目录 三.眼球渲染 3.1 眼球的构造及理论 3.1.1 眼球的构造 3.1.2 眼球的渲染理论 3.2 眼球的渲染技术 3.2.1 角膜的半透和光泽反射 3.2.2 瞳孔的次表面散射 3.2.3 瞳 ...
- 剖析Unreal Engine超真实人类的渲染技术Part 3 - 毛发渲染及其它
目录 四.毛发渲染 4.1 毛发的构造及渲染技术 4.1.1 毛发的构造 4.1.2 Marschner毛发渲染模型 4.1.3 毛发的间接光照 4.2 毛发的底层实现 4.3 毛发的材质解析 4.3 ...
- 三维场景中使用BillBoard技术
三维场景中对于渲染效果不是很精致的物体可以使用BillBoard技术实现,使用该技术需要将物体实时朝向摄像机,即计算billboard的旋转矩阵M. 首先根据摄像机位置cameraPos和billBo ...
- Web 开发中应用 HTML5 技术的10个实例教程
HTML5 作为下一代网站开发技术,无论你是一个 Web 开发人员或者想探索新的平台的游戏开发者,都值得去研究.借助尖端功能,技术和 API,HTML5 允许你创建响应性.创新性.互动性以及令人惊叹的 ...
随机推荐
- web前端之路 - 开篇
一 web发展历程 了解事物的历史有助于我们渐进式的从发展的思路清楚了解事物的来龙去脉. 这里有一篇网文写得比较清晰和完整:https://www.tianmaying.com/tutorial/we ...
- Uncaught ReferenceError: wx is not defined
程序的分享功能调用了微信的接口,但是忽然发现就报这个错误, Uncaught ReferenceError: wx is not defined 同时下方还有这个错误 This content sho ...
- 敏捷冲刺DAY3
一. 每日会议 1. 照片 2. 昨日完成工作 3. 今日完成工作 登录界面的进一步完善 服务器搭建 建立数据库 下一步任务的规划,展望 4. 工作中遇到的困难 工作中的困难:在进行模糊查询时,由于中 ...
- BAT批处理(六)
字符串处理 批处理有着具有非常强大的字符串处理能力,其功能绝不低于C语言里面的字符串函数集.批处理中可实现的字符串处理功能有:截取字符串内容.替换字符串特定字段.合并字符串.扩充字符串等功能.下面对这 ...
- 数据存储到MySQL并返回新插入的id值
当对数据库进行插入数据后,有时会需要刚插入的数据的id值,以作他用,整理如下: conn = pymysql.connect(, user=DB_USER, passwd=DB_PASSWORD, d ...
- 【Python】Python 过滤列表
如你所知,Python 具有通过列表解析将列表映射到其它列表的强大能力.这种能力同过滤机制结合使用,使列表中的有些元素被映射的同时跳过另外一些元素.过滤列表语法: [mapping-expressio ...
- 【bzoj3142】[Hnoi2013]数列 数学
题目描述 求满足 $1\le a_i\le n\ ,\ 1\le a_{i+1}-a_i\le m$ 的序列 $a_1...a_k$ 的个数模 $p$ 的值. 输入 只有一行用空格隔开的四个数:N.K ...
- BZOJ4260 Codechef REBXOR(trie)
用trie求出前缀最大区间异或和.后缀最大区间异或和即可.注意空间是nlog的. #include<iostream> #include<cstdio> #include< ...
- 个人vim配置
YouCompletMe支持golang cd .vim/bundle/YouCompleteMe ./install.sh --clang-completer --go-completer clan ...
- BZOJ1057:[ZJOI2007]棋盘制作——题解
http://www.lydsy.com/JudgeOnline/problem.php?id=1057 https://www.luogu.org/problemnew/show/P1169 国际象 ...