Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现
发表于2018-10-29
在Shader中,波形可以作为一种模拟动态的手段,例如颜色的波动,形状的波动,可以基于此创作出各种效果。
下文中就给大家介绍正弦、三角、锯齿、直角这几种基本波形以及变种的Shader实现代码,并配以函数图像和简单动画效果图。在效果图中,Shader代码计算出y值,在顶点着色器中赋给顶点的y坐标,或在片段着色器中插值后乘以颜色。
波形公式中参数的说明:
base:基点。波形的起始点。振幅从此点开始计算。
amplitude:振幅。振幅越大,波形峰值越大或谷值越小。
phase:相位。决定波形的起始值。波形随着phase整体平移。
freq:频率。单位时间内波形重复的次数。
正弦家族 Sin Family:
完美正弦 Perfect Sin:
公式描述:y=sin(x*freq+phase)*amplitude+base
Shader代码:y=sin(fmod(v.uv.x,1.0f)*2.0f*3.1415926f+_Time.y);
data:image/s3,"s3://crabby-images/860c6/860c6b81af78dfda21dd77b66f62fab6d28e9808" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/b48e6/b48e6bd797101bb02137bd5af4771d5b6127919b" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
山形 Hill:
公式描述:y=abs(sin(x*freq+phase))*amplitude+base
Shader代码:y=abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+_Time.y));
data:image/s3,"s3://crabby-images/daf43/daf4353c0d82f98013c1d3a07c28cd1b89b862a4" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/bbf4b/bbf4b82e62eebb04f7f4755b41a1d3e32d61cfaa" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
倒转山行 Inverse Hill:
公式描述:y=(1-abs(sin(x*freq+phase)))*amplitude+base
Shader代码:y=1.0f-abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+_Time.y));
data:image/s3,"s3://crabby-images/1c562/1c5625ca7b5c7eafddb2533ebb039abf9985df32" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/7ac4c/7ac4c0f30e77cfa988876e19c58218ede39519a2" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
锯齿家族 Sawtooth Family:
锯齿 SawTooth:
公式描述:y=max(min(((x+phase) mod freq),1),0)*amplitude+base
Shader代码:y=saturate(fmod(v.uv.x+_Time.y,1.0f));
data:image/s3,"s3://crabby-images/9998f/9998f69dad4345f0dc843fa45712ece73fec81e6" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/8153d/8153d5886053c495ea9a40d80a1c353814ff2bbd" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
倒转锯齿 Inverse SawTooth:
公式描述:y=(max(min(((x+phase) mod freq),1),0))*amplitude+base
Shader代码:y=1-saturate(fmod(v.uv.x+_Time.y,1.0f));
data:image/s3,"s3://crabby-images/a7ab0/a7ab02879f337627c01597be58889f98a6a7cb7c" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/8153d/8153d5886053c495ea9a40d80a1c353814ff2bbd" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
指数锯齿 Exponential SawTooth:
公式描述:y=max(min(((x+phase) mod freq)^exp,1),0)*amplitude+base
Shader代码:y=saturate(pow(frac(v.uv.x+_Time.y),10.0f));
data:image/s3,"s3://crabby-images/ae10e/ae10e8a901793ae4def028d98acd0ae6e54813a3" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/88427/88427b1a4b0409467427f1d49ca8188f7afea064" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
倒转指数锯齿 Inverse Exponential SawTooth:
公式描述:y=(1-max(min(((x+phase) mod freq)^exp,1),0))*amplitude+base
Shader代码:y=1-saturate(pow(fmod(v.uv.x+_Time.y,1.0f),10.0f));
data:image/s3,"s3://crabby-images/48464/4846495ebea59e4ee91186926cf75e2171536e2e" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/92422/92422017c8a72dd50f1c0faeba1da222ce29a882" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
饱和指数锯齿 Saturate Exponential SawTooth:
公式描述:y=min(max(min(((x+phase) mod freq)^10,1),0)*100,1)*amplitude+base
类似于指数锯齿,但单位时间内波形达到峰值时间更长
Shader代码:y=saturate(saturate(pow(fmod(x+_Time.y,1.0f),10.0f))*100.0f);
data:image/s3,"s3://crabby-images/bcbe2/bcbe2addc2c46725f711c6a9acadef844b139aef" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/effcc/effcc2a35988da651b0535fa16f72e473941ae71" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
三角形家族
三角形 Triangle
公式描述:y=abs(((x*freq+phase) mod 1)*2-1)*amplitude+base
Shader代码:y=abs(fmod(x+_Time.y,1.0f)*2.0f-1.0f);
data:image/s3,"s3://crabby-images/30034/30034ea87dbceab01e0e4000efde0ce82e3c4162" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/3be5b/3be5bc8f11e6b7b29e3f05054a6b4c898b63b930" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
梯形 Trapezium
公式描述:y=min(abs(((x*freq+phase)mod1)*2-1)*2,1)*amplitude+base
Shader代码:y=saturate(abs(fmod(x+_Time.y,1.0f)*2.0f-1.0f)*2.0f);
data:image/s3,"s3://crabby-images/50f33/50f33334cc594e277d590294ad74868fd216229a" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/a0e51/a0e51f85b85afc0713a1f65541b81fe131b73767" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
倒转梯形/不连续三角形 Inverse Trapezium/Discrete Triangles
公式描述:y=(1-min(abs(((x*freq+phase)mod 1)*2-1)*2,1))*amplitude+base
Shader代码:y=1-saturate(abs(fmod(x+_Time.y,1.0f)*2.0f-1.0f)*2.0f);
data:image/s3,"s3://crabby-images/adb90/adb90177da76467c4304bd6f08c098cf0e682496" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/d2ce5/d2ce55e8d637cbe59d74156fa47abe0af93c95e1" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
直角家族
直角
公式描述:y=round(sin(x*freq+phase))*amplitude+base
Shader代码:y=round(sin(x+_Time.y));
data:image/s3,"s3://crabby-images/3fdd0/3fdd01f8662c24ce934bb96ecad8f2bcb92617fc" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
data:image/s3,"s3://crabby-images/bc51d/bc51d73fa5b252cc39be05d1dc24d7ac809922e9" alt="Unity Shader:Waveform波形-基本波形:正弦、三角、锯齿、直角以及其变种的实现"
效果图Shader
效果图中模型的顶点的uv,x值以0.1向同一方向递增。
Shader "Unlit/WaveformTest" { Properties { _MainTex ("Texture", 2D) = "white" {} } SubShader { Tags { "RenderType"="Opaque" } LOD 100 Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag // make fog work #pragma multi_compile_fog #include "UnityCG.cginc" struct appdata { float4 vertex : POSITION; float2 uv : TEXCOORD0; }; struct v2f { float2 uv : TEXCOORD0; UNITY_FOG_COORDS(1) float4 vertex : SV_POSITION; }; sampler2D _MainTex; float4 _MainTex_ST; v2f vert (appdata v) { v2f o; float phase=_Time.y; float y; float x=v.uv.x; //perfect sin // y=sin(fmod(v.uv.x,1.0f)*2.0f*3.1415926f+phase); //hill // y=abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+phase)); //inverse hill // y=1.0f-abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+phase)); //SawTooth // y=saturate(fmod(v.uv.x+phase,1.0f)); //Inverse SawTooth // y=1-saturate(fmod(v.uv.x+phase,1.0f)); //Exponential SawTooth // y=saturate(pow(frac(v.uv.x+phase),10.0f)); //Inverse Exponential SawTooth // y=1-saturate(pow(fmod(v.uv.x+phase,1.0f),10.0f)); //Chang // y=saturate(saturate(pow(fmod(x+phase,1.0f),10.0f))*100.0f); //Triangle // y=abs(fmod(x+phase,1.0f)*2.0f-1.0f); //Tapezoid // y=saturate(abs(fmod(x+phase,1.0f)*2.0f-1.0f)*2.0f); //Discontinuous Triangles/Inverse Trapezoid // y=1-saturate(abs(fmod(x+phase,1.0f)*2.0f-1.0f)*2.0f); //Square y=round(sin(x+phase)); v.vertex.y=y; o.vertex = UnityObjectToClipPos(v.vertex); o.uv = TRANSFORM_TEX(v.uv, _MainTex); UNITY_TRANSFER_FOG(o,o.vertex); return o; } fixed4 frag (v2f i) : SV_Target { // sample the texture fixed4 col = tex2D(_MainTex, i.uv); // apply fog UNITY_APPLY_FOG(i.fogCoord, col); return col; } ENDCG } } }
Shader "Unlit/WaveformTestFrag" { Properties { _MainTex ("Texture", 2D) = "white" {} _Color("Color",color)=(1,1,1,1) } SubShader { Tags { "RenderType"="Opaque" } LOD 100 Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag // make fog work #pragma multi_compile_fog #include "UnityCG.cginc" struct appdata { float4 vertex : POSITION; float2 uv : TEXCOORD0; }; struct v2f { float2 uv : TEXCOORD0; float y:FLOAT; UNITY_FOG_COORDS(1) float4 vertex : SV_POSITION; }; sampler2D _MainTex; float4 _MainTex_ST; fixed4 _Color; v2f vert (appdata v) { v2f o; float phase=_Time.y; float y; float x=v.uv.x; //perfect sin // y=sin(fmod(v.uv.x,1.0f)*2.0f*3.1415926f+phase); //hill // y=abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+phase)); //inverse hill // y=1.0f-abs(sin(fmod(v.uv.x,1.0f)*3.1415926f+phase)); //SawTooth // y=saturate(fmod(v.uv.x+phase,1.0f)); //Inverse SawTooth // y=1-saturate(fmod(v.uv.x+phase,1.0f)); //Exponential SawTooth // y=saturate(pow(frac(v.uv.x+phase),10.0f)); //Inverse Exponential SawTooth // y=1-saturate(pow(fmod(v.uv.x+phase,1.0f),10.0f)); //Chang // y=saturate(saturate(pow(fmod(x+phase,1.0f),10.0f))*100.0f); //Triangle // y=abs(fmod(x+phase,1.0f)*2.0f-1.0f); //Tapezoid // y=saturate(abs(fmod(x+phase,1.0f)*2.0f-1.0f)*2.0f); //Discontinuous Triangles/Inverse Trapezoid // y=1-saturate(abs(fmod(x+phase,1.0f)*2.0f-1.0f)*2.0f); //Square y=round(sin(x+phase)); // v.vertex.y=y; o.y=y; o.vertex = UnityObjectToClipPos(v.vertex); o.uv = TRANSFORM_TEX(v.uv, _MainTex); UNITY_TRANSFER_FOG(o,o.vertex); return o; } fixed4 frag (v2f i) : SV_Target { // sample the texture fixed4 col = tex2D(_MainTex, i.uv); // apply fog UNITY_APPLY_FOG(i.fogCoord, col); return _Color*i.y; } ENDCG } } }
来自:https://blog.csdn.net/liu_if_else/article/details/77712935