301 lines
12 KiB
Plaintext
301 lines
12 KiB
Plaintext
|
Shader "Hidden/TerrainTools/BrushPreview"
|
||
|
{
|
||
|
SubShader
|
||
|
{
|
||
|
ZTest Always Cull Back ZWrite Off
|
||
|
Blend SrcAlpha OneMinusSrcAlpha
|
||
|
|
||
|
CGINCLUDE
|
||
|
// Upgrade NOTE: excluded shader from OpenGL ES 2.0 because it uses non-square matrices
|
||
|
#pragma exclude_renderers gles
|
||
|
|
||
|
#include "UnityCG.cginc"
|
||
|
#include "TerrainPreview.cginc"
|
||
|
#include "Packages/com.unity.terrain-tools/Shaders/TerrainTools.hlsl"
|
||
|
|
||
|
#define CLIP_PREVIEW clip( IsPcUvPartOfValidTerrainTileTexelSobel( (i.pcPixels + (.5).xx) / _PcPixelRect.zw, _BrushTex_TexelSize.xy * .5 ) - 1 );
|
||
|
|
||
|
sampler2D _FilterTex;
|
||
|
sampler2D _BrushTex;
|
||
|
float4 _BrushTex_TexelSize;
|
||
|
|
||
|
float _HoleStripeThreshold;
|
||
|
float _UseAltColor;
|
||
|
float _IsPaintHolesTool;
|
||
|
ENDCG
|
||
|
|
||
|
Pass // 0
|
||
|
{
|
||
|
Name "TerrainPreviewProcedural"
|
||
|
|
||
|
CGPROGRAM
|
||
|
#pragma vertex vert
|
||
|
#pragma fragment frag
|
||
|
|
||
|
#pragma shader_feature_local TERRAINTOOLS_FILTERS_ENABLED
|
||
|
|
||
|
struct v2f {
|
||
|
float4 clipPosition : SV_POSITION;
|
||
|
float2 uv : TEXCOORD0;
|
||
|
float3 positionWorld : TEXCOORD1;
|
||
|
float3 positionWorldOrig : TEXCOORD2;
|
||
|
float2 pcPixels : TEXCOORD3;
|
||
|
float2 brushUV : TEXCOORD4;
|
||
|
};
|
||
|
|
||
|
v2f vert(uint vid : SV_VertexID)
|
||
|
{
|
||
|
// build a quad mesh, with one vertex per paint context pixel (pcPixel)
|
||
|
float2 pcPixels = BuildProceduralQuadMeshVertex(vid);
|
||
|
|
||
|
// compute heightmap UV and sample heightmap
|
||
|
float2 heightmapUV = PaintContextPixelsToHeightmapUV(pcPixels);
|
||
|
float heightmapSample = UnpackHeightmap(tex2Dlod(_Heightmap, float4(heightmapUV, 0, 0)));
|
||
|
|
||
|
// compute brush UV
|
||
|
float2 brushUV = PaintContextPixelsToBrushUV(pcPixels);
|
||
|
|
||
|
// compute object position (in terrain space) and world position
|
||
|
float3 positionObject = PaintContextPixelsToObjectPosition(pcPixels, heightmapSample);
|
||
|
float3 positionWorld = TerrainObjectToWorldPosition(positionObject);
|
||
|
|
||
|
v2f o;
|
||
|
o.uv = heightmapUV;
|
||
|
o.pcPixels = pcPixels;
|
||
|
o.positionWorld = positionWorld;
|
||
|
o.positionWorldOrig = positionWorld;
|
||
|
o.clipPosition = UnityWorldToClipPos(positionWorld);
|
||
|
o.brushUV = brushUV;
|
||
|
return o;
|
||
|
}
|
||
|
|
||
|
float4 frag(v2f i) : SV_Target
|
||
|
{
|
||
|
CLIP_PREVIEW
|
||
|
|
||
|
float brushSample = UnpackHeightmap(tex2D(_BrushTex, i.brushUV));
|
||
|
|
||
|
// out of bounds multiplier
|
||
|
float oob = all(saturate(i.brushUV) == i.brushUV) ? 1.0f : 0.0f;
|
||
|
|
||
|
// brush outline stripe
|
||
|
float stripeWidth = 1.0f; // pixels
|
||
|
float stripeLocation = 0.0025f;
|
||
|
float brushStripe = Stripe(brushSample, stripeLocation, stripeWidth);
|
||
|
float4 color = float4(0.5f, 0.5f, 1.0f, 1.0f);
|
||
|
float opacity = .75f * brushSample;
|
||
|
|
||
|
#if TERRAINTOOLS_FILTERS_ENABLED
|
||
|
float filter = UnpackHeightmap(tex2D(_FilterTex, i.uv));
|
||
|
#endif
|
||
|
|
||
|
if (_IsPaintHolesTool) //override the color on the pixels that will be affected by the tool
|
||
|
{
|
||
|
float holeStripeLocation = saturate(abs(_HoleStripeThreshold));
|
||
|
float4 holeColor = _UseAltColor ?
|
||
|
float4(0.5f, 0.0f, 1.0f, 1.0f) : //alternate color = purplish
|
||
|
float4(0.0f, 0.0f, 1.0f, 1.0f); //regular color = blue
|
||
|
|
||
|
//clearly mark which pixels will be affected by the PaintHoles brush
|
||
|
float val = brushSample;
|
||
|
#if TERRAINTOOLS_FILTERS_ENABLED
|
||
|
val = val * filter;
|
||
|
#endif
|
||
|
opacity = abs(val) > (abs(holeStripeLocation)) && abs(val) > .0001f ? sign(holeStripeLocation) * sign(val) : 0.0f;
|
||
|
if (opacity > 0.0f)
|
||
|
{
|
||
|
//Blend in some cyan based on the brushSample (to subtly visualize the brush texture)
|
||
|
color.rgb = lerp(holeColor.rgb, float3(0.0f, 1.0f, 1.0f), 0.5f*brushSample);
|
||
|
|
||
|
brushStripe = Stripe(brushSample, holeStripeLocation, stripeWidth);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if TERRAINTOOLS_FILTERS_ENABLED
|
||
|
|
||
|
float4 filterColor = float4(1.0f, 0.6f, 0.05f, 1);
|
||
|
float fstripe = Stripe(filter, stripeLocation, stripeWidth);
|
||
|
float t = saturate((filter + fstripe) * brushSample // mult by brushSample to hide filter pixels outside brush boundaries.
|
||
|
- brushStripe // brush outline stripe should take precendence.
|
||
|
- stripeLocation); // dim color contribution based on threshold which
|
||
|
// should keep filter-tinted pixels within filter stripe boundaries.
|
||
|
color.rgb = lerp(color.rgb, filterColor.rgb, t);
|
||
|
brushStripe = max(fstripe * brushSample, brushStripe); // ensure stripe pixels are always 1 alpha
|
||
|
|
||
|
#endif
|
||
|
|
||
|
color.a = lerp(opacity, 1, saturate(brushStripe));
|
||
|
|
||
|
return color * oob;
|
||
|
}
|
||
|
ENDCG
|
||
|
}
|
||
|
|
||
|
Pass // 1
|
||
|
{
|
||
|
Name "TerrainPreviewProceduralDeltaNormals"
|
||
|
|
||
|
CGPROGRAM
|
||
|
#pragma vertex vert
|
||
|
#pragma fragment frag
|
||
|
|
||
|
sampler2D _HeightmapOrig;
|
||
|
|
||
|
struct v2f {
|
||
|
float4 clipPosition : SV_POSITION;
|
||
|
float3 positionWorld : TEXCOORD0;
|
||
|
float3 positionWorldOrig : TEXCOORD1;
|
||
|
float2 pcPixels : TEXCOORD2;
|
||
|
float2 brushUV : TEXCOORD3;
|
||
|
};
|
||
|
|
||
|
v2f vert(uint vid : SV_VertexID)
|
||
|
{
|
||
|
// build a quad mesh, with one vertex per paint context pixel (pcPixel)
|
||
|
float2 pcPixels = BuildProceduralQuadMeshVertex(vid);
|
||
|
|
||
|
// compute heightmap UV and sample heightmap
|
||
|
float2 heightmapUV = PaintContextPixelsToHeightmapUV(pcPixels);
|
||
|
float heightmapSample = UnpackHeightmap(tex2Dlod(_Heightmap, float4(heightmapUV, 0, 0)));
|
||
|
float heightmapSampleOrig = UnpackHeightmap(tex2Dlod(_HeightmapOrig, float4(heightmapUV, 0, 0)));
|
||
|
|
||
|
// compute brush UV
|
||
|
float2 brushUV = PaintContextPixelsToBrushUV(pcPixels);
|
||
|
|
||
|
// compute object position (in terrain space) and world position
|
||
|
float3 positionObject = PaintContextPixelsToObjectPosition(pcPixels, heightmapSample);
|
||
|
float3 positionWorld = TerrainObjectToWorldPosition(positionObject);
|
||
|
|
||
|
float3 positionObjectOrig = PaintContextPixelsToObjectPosition(pcPixels, heightmapSampleOrig);
|
||
|
float3 positionWorldOrig = TerrainObjectToWorldPosition(positionObjectOrig);
|
||
|
|
||
|
v2f o;
|
||
|
o.pcPixels = pcPixels;
|
||
|
o.positionWorld = positionWorld;
|
||
|
o.positionWorldOrig = positionWorldOrig;
|
||
|
o.clipPosition = UnityWorldToClipPos(positionWorld);
|
||
|
o.brushUV = brushUV;
|
||
|
return o;
|
||
|
}
|
||
|
|
||
|
float4 frag(v2f i) : SV_Target
|
||
|
{
|
||
|
float brushSample = UnpackHeightmap(tex2D(_BrushTex, i.brushUV));
|
||
|
|
||
|
CLIP_PREVIEW
|
||
|
|
||
|
// out of bounds multiplier
|
||
|
float oob = all(saturate(i.brushUV) == i.brushUV) ? 1.0f : 0.0f;
|
||
|
|
||
|
float deltaHeight = abs(i.positionWorld.y - i.positionWorldOrig.y);
|
||
|
|
||
|
// normal based coloring
|
||
|
float3 dx = ddx(i.positionWorld);
|
||
|
float3 dy = ddy(i.positionWorld);
|
||
|
float3 normal = normalize(cross(dy, dx));
|
||
|
|
||
|
float3 lightDir = UnityWorldSpaceLightDir(i.positionWorld.xyz);
|
||
|
|
||
|
float4 color;
|
||
|
color.rgb = saturate(normal.xzy * float3(-0.5f, -0.5f, 0.5f) + 0.5f);
|
||
|
color.rgb = lerp(color.rgb, float3(1.0f, 1.0f, 1.0f), 0.3f);
|
||
|
color.rgb = color.rgb * (0.1f + 0.9f * dot(lightDir, normal));
|
||
|
color.a = 0.9f * saturate(0.2f * deltaHeight);
|
||
|
|
||
|
return color;
|
||
|
}
|
||
|
ENDCG
|
||
|
}
|
||
|
|
||
|
Pass // 2
|
||
|
{
|
||
|
Name "TerrainPreviewProceduralDeltaStripes"
|
||
|
|
||
|
CGPROGRAM
|
||
|
#pragma vertex vert
|
||
|
#pragma fragment frag
|
||
|
|
||
|
sampler2D _HeightmapOrig;
|
||
|
|
||
|
struct v2f {
|
||
|
float4 clipPosition : SV_POSITION;
|
||
|
float3 positionWorld : TEXCOORD0;
|
||
|
float3 positionWorldOrig : TEXCOORD1;
|
||
|
float2 pcPixels : TEXCOORD2;
|
||
|
float2 brushUV : TEXCOORD3;
|
||
|
};
|
||
|
|
||
|
v2f vert(uint vid : SV_VertexID)
|
||
|
{
|
||
|
// build a quad mesh, with one vertex per paint context pixel (pcPixel)
|
||
|
float2 pcPixels = BuildProceduralQuadMeshVertex(vid);
|
||
|
|
||
|
// compute heightmap UV and sample heightmap
|
||
|
float2 heightmapUV = PaintContextPixelsToHeightmapUV(pcPixels);
|
||
|
float heightmapSample = UnpackHeightmap(tex2Dlod(_Heightmap, float4(heightmapUV, 0, 0)));
|
||
|
float heightmapSampleOrig = UnpackHeightmap(tex2Dlod(_HeightmapOrig, float4(heightmapUV, 0, 0)));
|
||
|
|
||
|
// compute brush UV
|
||
|
float2 brushUV = PaintContextPixelsToBrushUV(pcPixels);
|
||
|
|
||
|
// compute object position (in terrain space) and world position
|
||
|
float3 positionObject = PaintContextPixelsToObjectPosition(pcPixels, heightmapSample);
|
||
|
float3 positionWorld = TerrainObjectToWorldPosition(positionObject);
|
||
|
|
||
|
float3 positionObjectOrig = PaintContextPixelsToObjectPosition(pcPixels, heightmapSampleOrig);
|
||
|
float3 positionWorldOrig = TerrainObjectToWorldPosition(positionObjectOrig);
|
||
|
|
||
|
v2f o;
|
||
|
o.pcPixels = pcPixels;
|
||
|
o.positionWorld = positionWorld;
|
||
|
o.positionWorldOrig = positionWorldOrig;
|
||
|
o.clipPosition = UnityWorldToClipPos(positionWorld);
|
||
|
o.brushUV = brushUV;
|
||
|
return o;
|
||
|
}
|
||
|
|
||
|
float MultiStripes(in float x, in float freq1, in float freq2)
|
||
|
{
|
||
|
float2 derivatives = float2(ddx(x), ddy(x));
|
||
|
float derivLen = length(derivatives);
|
||
|
|
||
|
float tweak = 0.5;
|
||
|
float sharpen = tweak / max(derivLen, 0.00001f);
|
||
|
|
||
|
float triwave1 = abs(frac(x * freq1) - 0.5f) - 0.25f;
|
||
|
float triwave2 = abs(frac(x * freq2) - 0.5f) - 0.25f;
|
||
|
|
||
|
float width = 0.95f;
|
||
|
|
||
|
float result1 = saturate((triwave1 - width * 0.25f) * sharpen / freq1 + 0.75f);
|
||
|
float result2 = saturate((triwave2 - width * 0.25f) * sharpen / freq2 + 0.75f);
|
||
|
|
||
|
return max(result1, result2);
|
||
|
}
|
||
|
|
||
|
float4 frag(v2f i) : SV_Target
|
||
|
{
|
||
|
float brushSample = UnpackHeightmap(tex2D(_BrushTex, i.brushUV));
|
||
|
|
||
|
CLIP_PREVIEW
|
||
|
|
||
|
// out of bounds multiplier
|
||
|
float oob = all(saturate(i.brushUV) == i.brushUV) ? 1.0f : 0.0f;
|
||
|
|
||
|
// brush outline stripe
|
||
|
float stripeWidth = 1.0f; // pixels
|
||
|
float stripeLocation = 0.2f; // at 20% alpha
|
||
|
float heightStripes = MultiStripes(i.positionWorld.y + 0.25f, 0.0625f, 1.0f);
|
||
|
float xStripes = MultiStripes(i.positionWorld.x, 0.03125f, 0.5f);
|
||
|
float zStripes = MultiStripes(i.positionWorld.z, 0.03125f, 0.5f);
|
||
|
|
||
|
float deltaHeight = saturate(abs(i.positionWorld.y - i.positionWorldOrig.y));
|
||
|
float4 color = lerp(float4(0.5f, 0.5f, 1.0f, 1.0f), float4(0.5f, 1.0f, 0.5f, 1.0f), deltaHeight);
|
||
|
return color * lerp(deltaHeight * 0.5f, (brushSample > 0.0f ? 1.0f : 0.0f), 0.5f * saturate(heightStripes + xStripes + zStripes));
|
||
|
}
|
||
|
ENDCG
|
||
|
}
|
||
|
}
|
||
|
Fallback Off
|
||
|
}
|