当前位置:网站首页>【TA-霜狼_may-《百人计划》】先行部分 手搓视差体积云
【TA-霜狼_may-《百人计划》】先行部分 手搓视差体积云
2022-08-03 02:07:00 【zczplus】
【TA-霜狼_may-《百人计划》】先行部分 手搓体积云
【TA-霜狼_may-《百人计划》】先行部分 手搓体积云
最终效果:
对代码部分进行了部分修改,并添加了详细注释:
Shader "Custom/100 learning/MyCloudShader"
{
Properties
{
_Color ("Color", Color) = (1,1,1,1)
_MainTex ("MainTex", 2D) = "white" {
}
_Alpha("Alpha", Range(0,1)) = 0.5
_HeightOffset("HeightOffset", Range(0,1)) = 0.15
_StepLayer("StepLayer", Range(2,64)) = 16
}
SubShader
{
Tags{
"IgnoreProjector" = "True"
"Queue" = "Transparent-50"
"RenderType" = "Transparent"
}
Pass{
Name "FORWARD"
Tags{
"LightMode" = "ForwardBase"
}
Blend SrcAlpha OneMinusSrcAlpha
Cull Off
CGPROGRAM
#pragma vertex vert
#pragma fragment frag
#define UNITY_PASS_FORWARDBASE
#include "UnityCG.cginc"
#include "AutoLight.cginc"
#include "Lighting.cginc"
#pragma multi_compile_fwdbase
#pragma target 3.0
sampler2D _MainTex;
float4 _MainTex_ST;
half _HeightOffset;
half _HeightAmount;
half4 _Color;
half _Alpha;
half _StepLayer;
struct v2f{
float4 pos : SV_POSITION;
float2 uv : TEXCOORD0;
float3 normalDir : TEXCOORD1;
float3 viewDir : TEXCOORD2;
float4 posWorld : TEXCOORD3;
float2 uv2 : TEXCOORD4;
float4 color : TEXCOORD5;
UNITY_FOG_COORDS(7)
};
v2f vert(appdata_full v){
v2f o;
o.pos = UnityObjectToClipPos(v.vertex);
// 去除了原有的Frac函数,frac会导致纹理跳变,将纹理的重复方式修改为mirror即可
o.uv = TRANSFORM_TEX(v.texcoord, _MainTex) + float2(_Time.y * 0.02, 0);
o.uv2 = v.texcoord;
o.posWorld = mul(unity_ObjectToWorld, v.vertex);
o.normalDir = UnityObjectToWorldNormal(v.normal);
TANGENT_SPACE_ROTATION;
o.viewDir = mul(rotation, ObjSpaceViewDir(v.vertex));
o.color = v.color;
UNITY_TRANSFER_FOG(o, o.pos);
return o;
}
float4 frag(v2f i) : COLOR{
// 根据视角计算偏移,模拟出假的3D效果,SPM优化而来的POM
float3 viewDir = normalize(i.viewDir);
viewDir.xy *= _HeightOffset;
// 添加偏移值,减少狗牙感
viewDir.z += 0.4;
// uv用来计算陡峭视差映射需要用到z坐标
float3 uv = float3(i.uv, 0);
// uv2静止不需要z坐标
float2 uv2 = i.uv2;
// 静止贴图用于提供a分量
float4 MainTex = tex2D(_MainTex, uv2);
// 使用viewDir.xy 除以viewDir.z 可以得到uv的所需偏移方向,并平分成x层
// 分两段理解,先除以z则使整个贴图的竖直方向范围为1
// 再除以_StepLayer对竖直方向进行分层
float3 minOffset = viewDir / (viewDir.z * _StepLayer);
// 两段noise叠加,产生新的noise
float finiNoise = tex2D(_MainTex, uv.xy).r * MainTex.r;
float3 prev_uv = uv;
// 根据SPM的描述,循环体中也应该写prev_uv = uv才对,每次都对prev_uv进行更新
// 否则prev_uv中存放的位置是初始位置,当距离较远时可能误差会过大
while(finiNoise > uv.z){
prev_uv = uv;
uv += minOffset;
finiNoise = tex2Dlod(_MainTex, float4(uv.xy, 0, 0)).r * MainTex.r;
}
// 选取每层选用的UV进行加权计算位置 从而得到最终的uv点
// 下面这三句非常精髓:
float d1 = finiNoise -uv.z;
float d2 = finiNoise - prev_uv.z;
float w = d1 / (d1 - d2 + 0.0000001);
// d1 - d2 = prev_uv.z - uv.z = minOffset.z= 1/_StepLayer
// 经过测试发现这种实现方式会有严重的锯齿,弃用
// w = d1 / (1/_StepLayer + 0.0000001);
uv = lerp(uv, prev_uv, w);
half4 resultColor = tex2D(_MainTex, uv.xy) * MainTex;
// 透明度相关的一些小track
half rangeClt = MainTex.a * resultColor.r + _Alpha * 0.75;
half Alpha = abs(smoothstep(rangeClt, _Alpha, 1.0));
Alpha = Alpha*Alpha*Alpha*Alpha*Alpha;
// return MainTex;
return half4(resultColor.rgb * _Color.rgb, Alpha);
}
ENDCG
}
}
FallBack Off
}
要点
- 要想有好的效果,使用噪声图最好整体偏亮,并且在亮部依然有梯度变化,否则云没有立体效果;
- 纹理的重复方式需要注意,建议使用mirror然后对vs部分的代码进行调整;
- SPM陡峭视差贴图改进部分,在循环过程中应当对prev_uv进行每一次的更新,防止误差过大;
- 静止的uv贴图可以不用z坐标;
- 切线空间的使用!
边栏推荐
猜你喜欢
LabVIEW程序框图保存为图像
服务器在线测速系统源码
无法启动服务 错误 193 0xc1
QCheckBox、margin、border、pandding、QHoxLayout、QSplitter、QSpacerItem
一个循环,两个循环问题的思考及复现
The cornerstone of high concurrency: multithreading, daemon threading, thread safety, thread synchronization, mutual exclusion lock, all in one article!...
qt opengl 使用不同的颜色绘制线框三角形
initramfs详解----设备文件系统
The LVS load balancing cluster and the deployment of the LVS - NAT experiment
能添加任意贴图超级复布局的初级智能文本提示器
随机推荐
粘包与拆包
leetcode:153. 寻找旋转排序数组中的最小值
Kubernetes:(八)调度约束和故障排查
常用工具链和虚拟环境-WSL
lombok 下的@Builder和@EqualsAndHashCode(callSuper = true)注解
部门之间,互不信任正常吗?(你是否遇到过)
如何准备考pmp?
numpy PIL tensor之间的相互转换
孩子坐不住就是不专注?猿辅导揭秘专注力的三大误区
常见钓鱼手法及防范
公司封装方式导出excel过程
leetcode:151. 颠倒字符串中的单词
Fiddler基本使用
一次偶然的钓鱼文件分析
DJI内推码(2022年8月2日更新)
SPI机制是什么?
思维+启发式合并
【社媒营销】Facebook速推帖子如何运作?值得吗?
选中按钮上色
Get the first/last day of the current week, month, quarter in MySQL