SpeedTree8Passes.hlsl 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. #ifndef UNIVERSAL_SPEEDTREE8_PASSES_INCLUDED
  2. #define UNIVERSAL_SPEEDTREE8_PASSES_INCLUDED
  3. #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Lighting.hlsl"
  4. struct SpeedTreeVertexInput
  5. {
  6. float4 vertex : POSITION;
  7. float3 normal : NORMAL;
  8. float4 tangent : TANGENT;
  9. float4 texcoord : TEXCOORD0;
  10. float4 texcoord1 : TEXCOORD1;
  11. float4 texcoord2 : TEXCOORD2;
  12. float4 texcoord3 : TEXCOORD3;
  13. float4 color : COLOR;
  14. UNITY_VERTEX_INPUT_INSTANCE_ID
  15. };
  16. struct SpeedTreeVertexOutput
  17. {
  18. half2 uv : TEXCOORD0;
  19. half4 color : TEXCOORD1;
  20. half4 fogFactorAndVertexLight : TEXCOORD2; // x: fogFactor, yzw: vertex light
  21. #ifdef EFFECT_BUMP
  22. half4 normalWS : TEXCOORD3; // xyz: normal, w: viewDir.x
  23. half4 tangentWS : TEXCOORD4; // xyz: tangent, w: viewDir.y
  24. half4 bitangentWS : TEXCOORD5; // xyz: bitangent, w: viewDir.z
  25. #else
  26. half3 normalWS : TEXCOORD3;
  27. half3 viewDirWS : TEXCOORD4;
  28. #endif
  29. #ifdef _MAIN_LIGHT_SHADOWS
  30. float4 shadowCoord : TEXCOORD6;
  31. #endif
  32. float3 positionWS : TEXCOORD7;
  33. float4 clipPos : SV_POSITION;
  34. UNITY_VERTEX_INPUT_INSTANCE_ID
  35. UNITY_VERTEX_OUTPUT_STEREO
  36. };
  37. struct SpeedTreeVertexDepthOutput
  38. {
  39. half2 uv : TEXCOORD0;
  40. half4 color : TEXCOORD1;
  41. float4 clipPos : SV_POSITION;
  42. UNITY_VERTEX_INPUT_INSTANCE_ID
  43. UNITY_VERTEX_OUTPUT_STEREO
  44. };
  45. struct SpeedTreeFragmentInput
  46. {
  47. SpeedTreeVertexOutput interpolated;
  48. #ifdef EFFECT_BACKSIDE_NORMALS
  49. half facing : VFACE;
  50. #endif
  51. };
  52. void InitializeData(inout SpeedTreeVertexInput input, float lodValue)
  53. {
  54. // smooth LOD
  55. #if defined(LOD_FADE_PERCENTAGE) && !defined(EFFECT_BILLBOARD)
  56. input.vertex.xyz = lerp(input.vertex.xyz, input.texcoord2.xyz, lodValue);
  57. #endif
  58. // wind
  59. #if defined(ENABLE_WIND) && !defined(_WINDQUALITY_NONE)
  60. if (_WindEnabled > 0)
  61. {
  62. float3 rotatedWindVector = mul(_ST_WindVector.xyz, (float3x3)unity_ObjectToWorld);
  63. float windLength = length(rotatedWindVector);
  64. if (windLength < 1e-5)
  65. {
  66. // sanity check that wind data is available
  67. return;
  68. }
  69. rotatedWindVector /= windLength;
  70. float3 treePos = float3(unity_ObjectToWorld[0].w, unity_ObjectToWorld[1].w, unity_ObjectToWorld[2].w);
  71. float3 windyPosition = input.vertex.xyz;
  72. #ifndef EFFECT_BILLBOARD
  73. // geometry type
  74. float geometryType = (int)(input.texcoord3.w + 0.25);
  75. bool leafTwo = false;
  76. if (geometryType > GEOM_TYPE_FACINGLEAF)
  77. {
  78. geometryType -= 2;
  79. leafTwo = true;
  80. }
  81. // leaves
  82. if (geometryType > GEOM_TYPE_FROND)
  83. {
  84. // remove anchor position
  85. float3 anchor = float3(input.texcoord1.zw, input.texcoord2.w);
  86. windyPosition -= anchor;
  87. if (geometryType == GEOM_TYPE_FACINGLEAF)
  88. {
  89. // face camera-facing leaf to camera
  90. float offsetLen = length(windyPosition);
  91. windyPosition = mul(windyPosition.xyz, (float3x3)UNITY_MATRIX_IT_MV); // inv(MV) * windyPosition
  92. windyPosition = normalize(windyPosition) * offsetLen; // make sure the offset vector is still scaled
  93. }
  94. // leaf wind
  95. #if defined(_WINDQUALITY_FAST) || defined(_WINDQUALITY_BETTER) || defined(_WINDQUALITY_BEST)
  96. #ifdef _WINDQUALITY_BEST
  97. bool bBestWind = true;
  98. #else
  99. bool bBestWind = false;
  100. #endif
  101. float leafWindTrigOffset = anchor.x + anchor.y;
  102. windyPosition = LeafWind(bBestWind, leafTwo, windyPosition, input.normal, input.texcoord3.x, float3(0,0,0), input.texcoord3.y, input.texcoord3.z, leafWindTrigOffset, rotatedWindVector);
  103. #endif
  104. // move back out to anchor
  105. windyPosition += anchor;
  106. }
  107. // frond wind
  108. bool bPalmWind = false;
  109. #ifdef _WINDQUALITY_PALM
  110. bPalmWind = true;
  111. if (geometryType == GEOM_TYPE_FROND)
  112. {
  113. windyPosition = RippleFrond(windyPosition, input.normal, input.texcoord.x, input.texcoord.y, input.texcoord3.x, input.texcoord3.y, input.texcoord3.z);
  114. }
  115. #endif
  116. // branch wind (applies to all 3D geometry)
  117. #if defined(_WINDQUALITY_BETTER) || defined(_WINDQUALITY_BEST) || defined(_WINDQUALITY_PALM)
  118. float3 rotatedBranchAnchor = normalize(mul(_ST_WindBranchAnchor.xyz, (float3x3)unity_ObjectToWorld)) * _ST_WindBranchAnchor.w;
  119. windyPosition = BranchWind(bPalmWind, windyPosition, treePos, float4(input.texcoord.zw, 0, 0), rotatedWindVector, rotatedBranchAnchor);
  120. #endif
  121. #endif // !EFFECT_BILLBOARD
  122. // global wind
  123. float globalWindTime = _ST_WindGlobal.x;
  124. #if defined(EFFECT_BILLBOARD) && defined(UNITY_INSTANCING_ENABLED)
  125. globalWindTime += UNITY_ACCESS_INSTANCED_PROP(STWind, _GlobalWindTime);
  126. #endif
  127. windyPosition = GlobalWind(windyPosition, treePos, true, rotatedWindVector, globalWindTime);
  128. input.vertex.xyz = windyPosition;
  129. }
  130. #endif
  131. #if defined(EFFECT_BILLBOARD)
  132. float3 treePos = float3(UNITY_MATRIX_M[0].w, UNITY_MATRIX_M[1].w, UNITY_MATRIX_M[2].w);
  133. // crossfade faces
  134. bool topDown = (input.texcoord.z > 0.5);
  135. float3 viewDir = UNITY_MATRIX_IT_MV[2].xyz;
  136. float3 cameraDir = normalize(mul((float3x3)UNITY_MATRIX_M, _WorldSpaceCameraPos - treePos));
  137. float viewDot = max(dot(viewDir, input.normal), dot(cameraDir, input.normal));
  138. viewDot *= viewDot;
  139. viewDot *= viewDot;
  140. viewDot += topDown ? 0.38 : 0.18; // different scales for horz and vert billboards to fix transition zone
  141. // if invisible, avoid overdraw
  142. if (viewDot < 0.3333)
  143. {
  144. input.vertex.xyz = float3(0, 0, 0);
  145. }
  146. input.color = float4(1, 1, 1, clamp(viewDot, 0, 1));
  147. // adjust lighting on billboards to prevent seams between the different faces
  148. if (topDown)
  149. {
  150. input.normal += cameraDir;
  151. }
  152. else
  153. {
  154. half3 binormal = cross(input.normal, input.tangent.xyz) * input.tangent.w;
  155. float3 right = cross(cameraDir, binormal);
  156. input.normal = cross(binormal, right);
  157. }
  158. input.normal = normalize(input.normal);
  159. #endif
  160. }
  161. SpeedTreeVertexOutput SpeedTree8Vert(SpeedTreeVertexInput input)
  162. {
  163. SpeedTreeVertexOutput output = (SpeedTreeVertexOutput)0;
  164. UNITY_SETUP_INSTANCE_ID(input);
  165. UNITY_TRANSFER_INSTANCE_ID(input, output);
  166. UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(output);
  167. // handle speedtree wind and lod
  168. InitializeData(input, unity_LODFade.x);
  169. output.uv = input.texcoord.xy;
  170. output.color = input.color;
  171. // color already contains (ao, ao, ao, blend)
  172. // put hue variation amount in there
  173. #ifdef EFFECT_HUE_VARIATION
  174. float3 treePos = float3(UNITY_MATRIX_M[0].w, UNITY_MATRIX_M[1].w, UNITY_MATRIX_M[2].w);
  175. float hueVariationAmount = frac(treePos.x + treePos.y + treePos.z);
  176. output.color.g = saturate(hueVariationAmount * _HueVariationColor.a);
  177. #endif
  178. VertexPositionInputs vertexInput = GetVertexPositionInputs(input.vertex.xyz);
  179. half3 normalWS = TransformObjectToWorldNormal(input.normal);
  180. half3 vertexLight = VertexLighting(vertexInput.positionWS, normalWS);
  181. half fogFactor = ComputeFogFactor(vertexInput.positionCS.z);
  182. output.fogFactorAndVertexLight = half4(fogFactor, vertexLight);
  183. half3 viewDirWS = GetCameraPositionWS() - vertexInput.positionWS;
  184. #ifdef EFFECT_BUMP
  185. real sign = input.tangent.w * GetOddNegativeScale();
  186. output.normalWS.xyz = normalWS;
  187. output.tangentWS.xyz = TransformObjectToWorldDir(input.tangent.xyz);
  188. output.bitangentWS.xyz = cross(output.normalWS.xyz, output.tangentWS.xyz) * sign;
  189. // View dir packed in w.
  190. output.normalWS.w = viewDirWS.x;
  191. output.tangentWS.w = viewDirWS.y;
  192. output.bitangentWS.w = viewDirWS.z;
  193. #else
  194. output.normalWS = normalWS;
  195. output.viewDirWS = viewDirWS;
  196. #endif
  197. #ifdef _MAIN_LIGHT_SHADOWS
  198. output.shadowCoord = GetShadowCoord(vertexInput);
  199. #endif
  200. output.positionWS = vertexInput.positionWS;
  201. output.clipPos = vertexInput.positionCS;
  202. return output;
  203. }
  204. SpeedTreeVertexDepthOutput SpeedTree8VertDepth(SpeedTreeVertexInput input)
  205. {
  206. SpeedTreeVertexDepthOutput output = (SpeedTreeVertexDepthOutput)0;
  207. UNITY_SETUP_INSTANCE_ID(input);
  208. UNITY_TRANSFER_INSTANCE_ID(input, output);
  209. UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(output);
  210. // handle speedtree wind and lod
  211. InitializeData(input, unity_LODFade.x);
  212. output.uv = input.texcoord.xy;
  213. output.color = input.color;
  214. VertexPositionInputs vertexInput = GetVertexPositionInputs(input.vertex.xyz);
  215. #ifdef SHADOW_CASTER
  216. half3 normalWS = TransformObjectToWorldNormal(input.normal);
  217. float4 positionCS = TransformWorldToHClip(ApplyShadowBias(vertexInput.positionWS, normalWS, _LightDirection));
  218. output.clipPos = positionCS;
  219. #else
  220. output.clipPos = vertexInput.positionCS;
  221. #endif
  222. return output;
  223. }
  224. void InitializeInputData(SpeedTreeFragmentInput input, half3 normalTS, out InputData inputData)
  225. {
  226. inputData.positionWS = input.interpolated.positionWS.xyz;
  227. #ifdef EFFECT_BUMP
  228. inputData.normalWS = TransformTangentToWorld(normalTS, half3x3(input.interpolated.tangentWS.xyz, input.interpolated.bitangentWS.xyz, input.interpolated.normalWS.xyz));
  229. inputData.normalWS = NormalizeNormalPerPixel(inputData.normalWS);
  230. inputData.viewDirectionWS = half3(input.interpolated.normalWS.w, input.interpolated.tangentWS.w, input.interpolated.bitangentWS.w);
  231. #else
  232. inputData.normalWS = NormalizeNormalPerPixel(input.interpolated.normalWS);
  233. inputData.viewDirectionWS = input.interpolated.viewDirWS;
  234. #endif
  235. #if SHADER_HINT_NICE_QUALITY
  236. inputData.viewDirectionWS = SafeNormalize(inputData.viewDirectionWS);
  237. #endif
  238. #ifdef _MAIN_LIGHT_SHADOWS
  239. inputData.shadowCoord = input.interpolated.shadowCoord;
  240. #else
  241. inputData.shadowCoord = float4(0, 0, 0, 0);
  242. #endif
  243. inputData.fogCoord = input.interpolated.fogFactorAndVertexLight.x;
  244. inputData.vertexLighting = input.interpolated.fogFactorAndVertexLight.yzw;
  245. inputData.bakedGI = half3(0, 0, 0); // No GI currently.
  246. }
  247. half4 SpeedTree8Frag(SpeedTreeFragmentInput input) : SV_Target
  248. {
  249. UNITY_SETUP_INSTANCE_ID(input.interpolated);
  250. #if !defined(SHADER_QUALITY_LOW)
  251. #ifdef LOD_FADE_CROSSFADE // enable dithering LOD transition if user select CrossFade transition in LOD group
  252. #ifdef EFFECT_BILLBOARD
  253. LODDitheringTransition(input.interpolated.clipPos.xy, unity_LODFade.x);
  254. #endif
  255. #endif
  256. #endif
  257. half2 uv = input.interpolated.uv;
  258. half4 diffuse = SampleAlbedoAlpha(uv, TEXTURE2D_ARGS(_MainTex, sampler_MainTex)) * _Color;
  259. half alpha = diffuse.a * input.interpolated.color.a;
  260. AlphaDiscard(alpha - 0.3333, 0.0);
  261. half3 albedo = diffuse.rgb;
  262. half3 emission = 0;
  263. half metallic = 0;
  264. half smoothness = 0;
  265. half occlusion = 0;
  266. half3 specular = 0;
  267. // hue variation
  268. #ifdef EFFECT_HUE_VARIATION
  269. half3 shiftedColor = lerp(albedo, _HueVariationColor.rgb, input.interpolated.color.g);
  270. // preserve vibrance
  271. half maxBase = max(albedo.r, max(albedo.g, albedo.b));
  272. half newMaxBase = max(shiftedColor.r, max(shiftedColor.g, shiftedColor.b));
  273. maxBase /= newMaxBase;
  274. maxBase = maxBase * 0.5f + 0.5f;
  275. shiftedColor.rgb *= maxBase;
  276. albedo = saturate(shiftedColor);
  277. #endif
  278. // normal
  279. #ifdef EFFECT_BUMP
  280. half3 normalTs = SampleNormal(uv, TEXTURE2D_ARGS(_BumpMap, sampler_BumpMap));
  281. #else
  282. half3 normalTs = half3(0, 0, 1);
  283. #endif
  284. // flip normal on backsides
  285. #ifdef EFFECT_BACKSIDE_NORMALS
  286. if (input.facing < 0.5)
  287. {
  288. normalTs.z = -normalTs.z;
  289. }
  290. #endif
  291. // adjust billboard normals to improve GI and matching
  292. #ifdef EFFECT_BILLBOARD
  293. normalTs.z *= 0.5;
  294. normalTs = normalize(normalTs);
  295. #endif
  296. // extra
  297. #ifdef EFFECT_EXTRA_TEX
  298. half4 extra = tex2D(_ExtraTex, uv);
  299. smoothness = extra.r;
  300. metallic = extra.g;
  301. occlusion = extra.b * input.interpolated.color.r;
  302. #else
  303. smoothness = _Glossiness;
  304. metallic = _Metallic;
  305. occlusion = input.interpolated.color.r;
  306. #endif
  307. // subsurface (hijack emissive)
  308. #ifdef EFFECT_SUBSURFACE
  309. emission = tex2D(_SubsurfaceTex, uv).rgb * _SubsurfaceColor.rgb;
  310. #endif
  311. InputData inputData;
  312. InitializeInputData(input, normalTs, inputData);
  313. half4 color = UniversalFragmentPBR(inputData, albedo, metallic, specular, smoothness, occlusion, emission, alpha);
  314. color.rgb = MixFog(color.rgb, inputData.fogCoord);
  315. return color;
  316. }
  317. half4 SpeedTree8FragDepth(SpeedTreeVertexDepthOutput input) : SV_Target
  318. {
  319. UNITY_SETUP_INSTANCE_ID(input);
  320. #if !defined(SHADER_QUALITY_LOW)
  321. #ifdef LOD_FADE_CROSSFADE // enable dithering LOD transition if user select CrossFade transition in LOD group
  322. #ifdef EFFECT_BILLBOARD
  323. LODDitheringTransition(input.clipPos.xy, unity_LODFade.x);
  324. #endif
  325. #endif
  326. #endif
  327. half2 uv = input.uv;
  328. half4 diffuse = SampleAlbedoAlpha(uv, TEXTURE2D_ARGS(_MainTex, sampler_MainTex)) * _Color;
  329. half alpha = diffuse.a * input.color.a;
  330. AlphaDiscard(alpha - 0.3333, 0.0);
  331. #if defined(SCENESELECTIONPASS)
  332. // We use depth prepass for scene selection in the editor, this code allow to output the outline correctly
  333. return half4(_ObjectId, _PassValue, 1.0, 1.0);
  334. #else
  335. return half4(0, 0, 0, 0);
  336. #endif
  337. }
  338. #endif