1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
| Shader "Roystan/Grass" { Properties { _TopColor("草尖颜色", Color) = (1,1,1,1) _BottomColor("草根颜色", Color) = (1,1,1,1) _TranslucentGain("Translucent Gain", Range(0,1)) = 0.5 _BladeWidth("草宽度", Float) = 0.05 _BladeWidthRandom("草随机叠加宽度", Float) = 0.5 _BladeHeight("草高度", Float) = 0.5 _BladeHeightRandom("草随机叠加高度", Float) = 0.3 _BendRotationRandom("草随机旋转", Range(0, 1)) = 0.2 _BladeForward("草向前弯曲幅度", Float) = 0.38 _BladeCurve("草弯曲曲率", Range(1, 4)) = 2 _TessellationEdgeLength("草皮离相机距离影响细分因子", Range(1, 64)) = 1 _WindDistortionMap("风扰动贴图", 2D) = "white" {} _WindFrequency("风吹频率", Vector) = (0.05, 0.05, 0, 0) _WindStrength("风强度", Float) = 1 _InteractivePos("交互物体位置", Vector) = (0,0,0,0) _InteractiveRadius("交互影响半径", Float) = 1 _InteractiveStrength("交互影响强度", Float) = 1 }
HLSLINCLUDE #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl" #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Lighting.hlsl"
CBUFFER_START(UnityPerMaterial) float4 _TopColor; float4 _BottomColor; float _TranslucentGain; float _BendRotationRandom; float _BladeWidth; float _BladeWidthRandom; float _BladeHeight; float _BladeHeightRandom; float _BladeForward; float _BladeCurve; float4 _WindDistortionMap_ST; float4 _WindFrequency; float _WindStrength; float _TessellationEdgeLength; float4 _InteractivePos; float _InteractiveRadius; float _InteractiveStrength; CBUFFER_END
TEXTURE2D(_WindDistortionMap);SAMPLER(sampler_WindDistortionMap);
// Simple noise function, sourced from http://answers.unity.com/answers/624136/view.html // Extended discussion on this function can be found at the following link: // https://forum.unity.com/threads/am-i-over-complicating-this-random-function.454887/#post-2949326 // Returns a number in the 0...1 range. //从一个三维输入生成一个随机数 float rand(float3 co) { return frac(sin(dot(co.xyz, float3(12.9898, 78.233, 53.539))) * 43758.5453); }
// Construct a rotation matrix that rotates around the provided axis, sourced from: // https://gist.github.com/keijiro/ee439d5e7388f3aafc5296005c8c3f33 //接收一个角度(弧度制)并返回一个围绕提供轴旋转的矩阵 //应该是使用了罗德里格斯公式 float3x3 AngleAxis3x3(float angle, float3 axis) { float c, s; sincos(angle, s, c);
float t = 1 - c; float x = axis.x; float y = axis.y; float z = axis.z;
return float3x3( t * x * x + c, t * x * y - s * z, t * x * z + s * y, t * x * y + s * z, t * y * y + c, t * y * z - s * x, t * x * z - s * y, t * y * z + s * x, t * z * z + c ); }
struct vertexInput { float4 vertex : POSITION; float3 normal : NORMAL; float4 tangent : TANGENT; };
struct vertexOutput { float4 vertex : SV_POSITION; float3 normal : NORMAL; float4 tangent : TANGENT; };
struct TessellationFactors { float edge[3] : SV_TessFactor; float inside : SV_InsideTessFactor; }; // 用于几何着色器输出 struct geometryOutput { float4 pos : SV_POSITION; float2 uv : TEXCOORD0; float3 normal : NORMAL; float4 shadowCoord : TEXCOORD1; };
//===============曲面细分相关=================== vertexOutput tessVert(vertexInput v) { vertexOutput o; // Note that the vertex is NOT transformed to clip // space here; this is done in the grass geometry shader. o.vertex = v.vertex; o.normal = v.normal; o.tangent = v.tangent; return o; }
// 根据三角形离相机位置计算细分程度 // 1. 通过P0,P1的获取他们的世界空间下的中点G。 // 2. 然后获得G点到相机的距离,也就是类深度值了。 // 3. 然后通过影响深度来修改曲分因子。 // 4. 为了保证屏幕高度/近远的裁剪面距离比,加入_ScreenParams.y。 // 这样其实过远或者过近还是有点问题,最好是引入LOD分层 float GetTessSegments(float3 vertex0OS, float3 vertex1OS) { float3 p0 = mul(unity_ObjectToWorld, float4(vertex0OS, 1)).xyz; float3 p1 = mul(unity_ObjectToWorld, float4(vertex1OS, 1)).xyz; float edgeLength = distance(p0, p1);
float3 edgeCenter = (p0 + p1) * 0.5; float viewDistance = distance(edgeCenter, _WorldSpaceCameraPos);
return edgeLength * _ScreenParams.y/ (_TessellationEdgeLength * viewDistance); }
TessellationFactors patchConstantFunction (InputPatch<vertexInput, 3> patch) { TessellationFactors f; float tessSegments = GetTessSegments(patch[0].vertex, patch[1].vertex); f.edge[0] = tessSegments; f.edge[1] = tessSegments; f.edge[2] = tessSegments; f.inside = tessSegments; return f; }
[domain("tri")] [outputcontrolpoints(3)] [outputtopology("triangle_cw")] [partitioning("integer")] [patchconstantfunc("patchConstantFunction")] vertexInput hull (InputPatch<vertexInput, 3> patch, uint id : SV_OutputControlPointID) { return patch[id]; }
[domain("tri")] vertexOutput domain(TessellationFactors factors, OutputPatch<vertexInput, 3> patch, float3 barycentricCoordinates : SV_DomainLocation) { vertexInput v;
#define MY_DOMAIN_PROGRAM_INTERPOLATE(fieldName) v.fieldName = \ patch[0].fieldName * barycentricCoordinates.x + \ patch[1].fieldName * barycentricCoordinates.y + \ patch[2].fieldName * barycentricCoordinates.z;
MY_DOMAIN_PROGRAM_INTERPOLATE(vertex) MY_DOMAIN_PROGRAM_INTERPOLATE(normal) MY_DOMAIN_PROGRAM_INTERPOLATE(tangent)
return tessVert(v); }
//===============几何相关=================== // 计算与角色交互相关的数据 float3 GetInteractiveData(float3 pos) { float3 interactWS = TransformWorldToObject(_InteractivePos); float3 dis = distance(interactWS, pos); float radius = 1-saturate(dis/_InteractiveRadius); float3 sphereDisp = float3(pos.x-interactWS.x, pos.z-interactWS.z, 0); return float3(sphereDisp.xy, radius * _InteractiveStrength); } // 生产草叶数据 geometryOutput VertexOutput(float3 pos, float2 uv, float3 normalOS) { geometryOutput o = (geometryOutput)0; o.uv = uv; o.pos = TransformObjectToHClip(pos); float3 worldPos = TransformObjectToWorld(pos); o.shadowCoord = TransformWorldToShadowCoord(worldPos); o.normal = TransformObjectToWorldNormal(normalOS);
// 这块不太好看,后续重构下 #if UNITY_PASS_SHADOWCASTER worldPos = ApplyShadowBias(worldPos, o.normal, _LightDirection); o.pos = TransformWorldToHClip(worldPos); #endif return o; } geometryOutput GenerateGrassVertex(float3 vertexPosition, float width, float height, float forward, float2 uv, float3x3 transformMatrix) { float3 tangentPoint = float3(width, forward, height);
// 法线,当草叶曲率量设置为1时,草叶在切线空间中都面向同一个方向:向后的y轴 // 当草叶的曲率大于1时,每个顶点的切线Z位置将被传递到GenerateGrassVertex函数中的前量所抵消 float3 tangentNormal = float3(0, -1, forward); float3 normalOS = mul(transformMatrix, tangentNormal); float3 localPosition = vertexPosition + mul(transformMatrix, tangentPoint); return VertexOutput(localPosition, uv, normalOS); }
// 草叶要分成几段 #define BLADE_SEGMENTS 5 // 几何着色器处理函数 // 限制每个图元最多生成顶点数(可小于) [maxvertexcount(BLADE_SEGMENTS * 2 + 1)] void geo(triangle vertexOutput IN[3], inout TriangleStream<geometryOutput> triStream) { float3 pos = IN[0].vertex; float3 vNormal = IN[0].normal; float4 vTangent = IN[0].tangent; float3 vBinormal = cross(vNormal, vTangent) * vTangent.w;
//构建TBN矩阵 float3x3 tangentToLocal = float3x3( vTangent.x, vBinormal.x, vNormal.x, vTangent.y, vBinormal.y, vNormal.y, vTangent.z, vBinormal.z, vNormal.z );
// 整体旋转矩阵(pos作为随机种子) float3x3 facingRotationMatrix = AngleAxis3x3(rand(pos) * PI, float3(0, 0, 1));
//创建一个新的矩阵来沿让它着它的x轴旋转,以达到模拟弯曲的操作,0.5PI是为了得到0~90° float3x3 bendRotationMatrix = AngleAxis3x3(rand(pos.zzx) * _BendRotationRandom * PI * 0.5, float3(-1, 0, 0));
// 用pos来构建uv坐标的好处是:如果多个mesh使用这个shader,会感觉像它们受到同一种风的作用 float2 uv = pos.xz * _WindDistortionMap_ST.xy + _WindDistortionMap_ST.zw + _WindFrequency * _Time.y; float2 windSample = (SAMPLE_TEXTURE2D_LOD(_WindDistortionMap, sampler_WindDistortionMap, uv, 0).xy * 2 - 1) * _WindStrength; float3 wind = normalize(float3(windSample.x, windSample.y, 0)); float3x3 windRotationMatrix = AngleAxis3x3(windSample * PI, wind);
// 与交互物体产生的偏移 float3 interactData = GetInteractiveData(pos); float3x3 interactRotationMatrix = AngleAxis3x3(interactData.z * PI, normalize(float3(interactData.x, interactData.y, 0)));
// 底部点不需要弯曲和风吹拂效果 float3x3 transformationMatrixFacing = mul(tangentToLocal, facingRotationMatrix); float3x3 transformationMatrix = mul(mul(mul(mul(tangentToLocal, facingRotationMatrix), bendRotationMatrix), windRotationMatrix), interactRotationMatrix); geometryOutput o; // 控制草的高度宽度 float height = (rand(pos.zyx) * 2 - 1) * _BladeHeightRandom + _BladeHeight; float width = (rand(pos.xzy) * 2 - 1) * _BladeWidthRandom + _BladeWidth;
// 为草叶拆分三角形,更加真实 // - // - - // - - //- - float forward = rand(pos.yyz) * _BladeForward; for (int i = 0; i < BLADE_SEGMENTS; i++) { float t = i / (float)BLADE_SEGMENTS; float segmentHeight = height * t; float segmentWidth = width * (1 - t); float segmentForward = pow(t, _BladeCurve) * forward; // 只有底部两个顶点需要固定 float3x3 transformMatrix = i == 0 ? transformationMatrixFacing : transformationMatrix; triStream.Append(GenerateGrassVertex(pos, segmentWidth, segmentHeight, segmentForward, float2(0, t), transformMatrix)); triStream.Append(GenerateGrassVertex(pos, -segmentWidth, segmentHeight, segmentForward, float2(1, t), transformMatrix)); } triStream.Append(GenerateGrassVertex(pos, 0, height, forward, float2(0.5, 1), transformationMatrix)); }
ENDHLSL
SubShader { Cull Off
Pass { Tags { "RenderType" = "Opaque" "RenderPipeline"="UniversalPipeline" }
HLSLPROGRAM #pragma vertex tessVert #pragma geometry geo #pragma hull hull #pragma domain domain #pragma fragment frag #pragma target 4.6
#pragma multi_compile _ _MAIN_LIGHT_SHADOWS #pragma multi_compile _ _MAIN_LIGHT_SHADOWS_CASCADE float4 frag (geometryOutput i, half facing : VFACE) : SV_Target { float3 normal = facing > 0 ? i.normal : -i.normal; Light light = GetMainLight(i.shadowCoord); float3 lightDir = TransformObjectToWorld(light.direction); float NdotL = saturate(saturate(dot(normal, lightDir)) + _TranslucentGain); half3 ambient = SampleSH(normal); float3 lightIntensity = NdotL * light.color * ambient * light.shadowAttenuation;
float4 finalColor = lerp(_BottomColor, _TopColor * float4(lightIntensity, 1), i.uv.y);
return finalColor; } ENDHLSL } Pass { Tags{ "LightMode"="ShadowCaster"} ColorMask 0 HLSLPROGRAM #pragma vertex tessVert #pragma geometry geo #pragma hull hull #pragma domain domain #pragma fragment frag #pragma target 4.6 float4 frag (geometryOutput i) : SV_Target { return 0; } ENDHLSL } } }
|