csgo-2018-source/materialsystem/stdshaders/lightmappedpaint_ps2x.fxc
2021-07-24 21:11:47 -07:00

312 lines
12 KiB
Plaintext

// STATIC: "BUMPMAP" "0..2"
// STATIC: "CUBEMAP" "0..2"
// STATIC: "SEAMLESS" "0..1"
// STATIC: "THICKPAINT" "0..1"
// DYNAMIC: "FASTPATHENVMAPCONTRAST" "0..1"
// DYNAMIC: "FASTPATH" "0..1"
// SKIP: ( $BUMPMAP == 0 )
// Turning off 32bit lightmaps on Portal 2 to save shader perf. --Thorsten
//#define USE_32BIT_LIGHTMAPS_ON_360 // be sure to keep this in sync with the same #define in materialsystem/cmatlightmaps.cpp
#include "common_fog_ps_supportsvertexfog_fxc.h"
#include "common_ps_fxc.h"
#define PIXELSHADER
#include "common_lightmappedgeneric_fxc.h"
#if SEAMLESS
#define USE_FAST_PATH 1
#else
#define USE_FAST_PATH FASTPATH
#endif
const float4 g_EnvmapTint : register( c0 );
#if USE_FAST_PATH == 1
# if FASTPATHENVMAPCONTRAST == 0
static const float3 g_EnvmapContrast = { 0.0f, 0.0f, 0.0f };
# else
static const float3 g_EnvmapContrast = { 1.0f, 1.0f, 1.0f };
# endif
static const float3 g_EnvmapSaturation = { 1.0f, 1.0f, 1.0f };
static const float g_FresnelReflection = 1.0f;
static const float g_OneMinusFresnelReflection = 0.0f;
static const float4 g_SelfIllumTint = { 1.0f, 1.0f, 1.0f, 1.0f };
#else
const float3 g_EnvmapContrast : register( c2 );
const float3 g_EnvmapSaturation : register( c3 );
const float4 g_FresnelReflectionReg : register( c4 );
#define g_FresnelReflection g_FresnelReflectionReg.a
#define g_OneMinusFresnelReflection g_FresnelReflectionReg.b
const float4 g_SelfIllumTint : register( c7 );
#endif
const float3 g_EyePos : register( c10 );
const float4 g_FogParams : register( c11 );
const float4 g_TintValuesTimesLightmapScale : register( c12 );
#define g_flAlpha2 g_TintValuesTimesLightmapScale.w
#if PARALLAX_MAPPING || (CUBEMAP == 2)
const float4 g_ParallaxMappingControl : register( c20 );
#endif
#if (CUBEMAP == 2)
#define g_DiffuseCubemapScale g_ParallaxMappingControl.y
#endif
const float3 g_TintValuesWithoutLightmapScale : register( c21 );
// These constants are used to rotate the world space water normals around the up axis to align the
// normal with the camera and then give us a 2D offset vector to use for reflection and refraction uv's
const float3 g_vWorldToViewRefract0 : register( c22 );
const float3 g_vWorldToViewRefract1 : register( c23 );
sampler LightmapSampler : register( s1 );
samplerCUBE EnvmapSampler : register( s2 );
sampler BubbleLayoutSampler : register( s4 );
sampler BubbleSampler : register( s5 );
sampler SplatNormalSampler : register( s7 );
sampler PaintSampler : register( s9 );
#if defined( _X360 )
[maxtempreg(36)]
#endif // _X360
float4_color_return_type main( PS_INPUT i ) : COLOR
{
float3 worldPos = i.worldPos_projPosZ.xyz;
float3x3 tangenttranspose = float3x3( i.tangentSpaceTranspose0_vertexBlendX.xyz, i.tangentSpaceTranspose1_bumpTexCoord2u.xyz, i.tangentSpaceTranspose2_bumpTexCoord2v.xyz );
float3x3 worldToTangentSpace = transpose( tangenttranspose ); //yay, math both forwards and backwards, for kicks!
float3 worldVertToEyeVector = g_EyePos - worldPos;
float3 lightmapColor1 = float3( 1.0f, 1.0f, 1.0f );
float3 lightmapColor2 = float3( 1.0f, 1.0f, 1.0f );
float3 lightmapColor3 = float3( 1.0f, 1.0f, 1.0f );
float2 bumpCoord1;
float2 bumpCoord2;
float2 bumpCoord3;
ComputeBumpedLightmapCoordinates( i.lightmapTexCoord1And2, i.lightmapTexCoord3_bumpTexCoord.xy,
bumpCoord1, bumpCoord2, bumpCoord3 );
float2 paintCoord = bumpCoord1.xy - ( bumpCoord2.xy - bumpCoord1.xy ); // remove offset from 1st of the bumped coords for vanilla coords
float4 paintColor = tex2D( PaintSampler, paintCoord );
#if ( THICKPAINT )
{
lightmapColor1 = LightMapSample( LightmapSampler, bumpCoord1 ).rgb;
lightmapColor2 = LightMapSample( LightmapSampler, bumpCoord2 ).rgb;
lightmapColor3 = LightMapSample( LightmapSampler, bumpCoord3 ).rgb;
}
#else
{
lightmapColor1 = LightMapSample( LightmapSampler, paintCoord ).rgb; // use vanilla lightmap coords, not bumped
}
#endif
float2 coords = paintCoord.xy * 40.0f;
coords.y *= 0.5f; // paint map is built scaled in y, so must compensate in other texture fetches
#if defined( SHADER_MODEL_PS_2_0 )
return paintColor;
#endif
float alpha = paintColor.a;
clip( alpha - 0.01f );
//paint splat guts
float3 fvNormalTs = float3( 0.0f, 0.0f, 1.0f );
float4 fvSplats = tex2D( SplatNormalSampler, coords );
float3 fvViewDirectionTs = normalize( mul( worldVertToEyeVector, worldToTangentSpace ) );
alpha = (paintColor.a + fvSplats.a ) * 0.5f; // vanilla combined alpha
float fPaintThickness = smoothstep( 0.34f, 0.63f, alpha );
float4 cMixedPaint = paintColor.rgba;
#if( THICKPAINT )
{
float4 fvSplatsPushed = tex2D( SplatNormalSampler, coords + fvViewDirectionTs * fvSplats.a * 0.01 );
float bumpedPaintArea = smoothstep( 0.6f, 0.3f, alpha ); //area in which to super-bump, ie the edges of the paint
bumpedPaintArea *= fPaintThickness;
fvNormalTs = ( fvSplatsPushed.xyz * 2.0f ) - 1.0f ;
fvNormalTs.xy *= lerp( 0.1f, 4.0f, bumpedPaintArea );
float2 fvPaintCoordsDistorted = paintCoord.xy + ( 0.003 - 0.006 * alpha );
cMixedPaint = tex2D( PaintSampler, fvPaintCoordsDistorted ); // use distorted UVs so that the blend between different paint colors isn't blocky
cMixedPaint.rgb = lerp( paintColor.rgb, cMixedPaint.rgb, smoothstep( 0.9f, 1.0f, cMixedPaint.a ) ); // avoid fetching too far by fading out distorted paint towards edges
}
#else
{
fvNormalTs = ( fvSplats.xyz * 2.0f ) - 1.0f;
}
#endif
// Always simulate srgb reads in shader code for the tex2Dsrgb calls since the paint textures look better when filtered in gamma space
#if ( _X360 )
{
#if defined( CSTRIKE15 )
// [mariod] - no PWL for cstrike15
cMixedPaint.rgb = GammaToLinear( cMixedPaint.rgb );
#else
cMixedPaint.rgb = X360GammaToLinear( cMixedPaint.rgb );
#endif
}
#else
{
cMixedPaint.rgb = SrgbGammaToLinear( cMixedPaint.rgb );
}
#endif
float3 vNormalWs = mul( fvNormalTs.xyz, tangenttranspose );
#if ( THICKPAINT )
float4 vN;
vN.x = dot( g_vWorldToViewRefract0.xyz, vNormalWs.xyz );
vN.y = dot( g_vWorldToViewRefract1.xyz, vNormalWs.xyz );
vN.wz = vN.xy;
float flD = (paintColor.a + fvSplats.a );
flD *= smoothstep( 1.0f, 2.0f, flD );
float4 vDependentTexCoords = vN.xyzw * flD;
float2 vRefractTexCoord = vDependentTexCoords.wz;
float2 fvLayoutCoords = coords.xy * 6.5f + fvViewDirectionTs.xy * 0.0014f;
float2 fvDeepLayoutCoords = fvLayoutCoords * 1.76f + fvViewDirectionTs.xy * 0.0007f;
fvLayoutCoords += fvViewDirectionTs.xy * 0.03; //single-step parallax so that the bubbles look higher than the surface they're on
float4 fvLayout = tex2D( BubbleLayoutSampler, fvLayoutCoords );
float4 fvDeepLayout = tex2D( BubbleLayoutSampler, fvDeepLayoutCoords );
fvLayout.xy -= 0.5f;
fvLayout.z *= smoothstep( 0.750f, 0.80f, alpha );
fvDeepLayout.xy -= 0.5f;
fvDeepLayout.z *= smoothstep( 0.40f, 0.75f, alpha );
// generate matrix for transforming UV coordinates to be screen-facing
// used by the paint to render bubbles that act like facing particles, but are distributed
// using a texture that describes the UV layout. Causing the UVs to face the camera
// ensures the bubbles always appear round
float3x3 matCameraFaceUVs;
matCameraFaceUVs[2] = fvViewDirectionTs; // new normal
matCameraFaceUVs[1] = float3( 0.0f, 1.0f, 0.0f ); // tangent
matCameraFaceUVs[0] = normalize( cross( matCameraFaceUVs[1].xyz, matCameraFaceUVs[2].xyz ) ); //binormal
matCameraFaceUVs[1] = normalize( cross( matCameraFaceUVs[2].xyz, matCameraFaceUVs[0].xyz ) ); //re-square tangent
// Should try to do it in vertex shader but lack of tesselation means interpolation is not so pretty
//float2x2 matCameraFaceUVs;
//matCameraFaceUVs[0].xy = i.detailOrBumpAndEnvmapMaskTexCoord.xy;
//matCameraFaceUVs[1].xy = i.detailOrBumpAndEnvmapMaskTexCoord.zw;
float2 fvBubbleCoords;
fvBubbleCoords.x = dot( fvLayout.xy, matCameraFaceUVs[0].xy );
fvBubbleCoords.y = dot( fvLayout.xy, matCameraFaceUVs[1].xy );
float2 fvDeepBubbleCoords;
fvDeepBubbleCoords.x = dot( fvDeepLayout.xy, matCameraFaceUVs[0].xy );
fvDeepBubbleCoords.y = dot( fvDeepLayout.xy, matCameraFaceUVs[1].xy );
float2 fvBubbleSurfaceCoords = fvLayout.xy * 0.5f + 0.5f;
fvBubbleCoords.xy = fvBubbleCoords.xy + 0.5f + vRefractTexCoord * 0.05;
fvDeepBubbleCoords.xy = fvDeepBubbleCoords.xy + 0.5f + vRefractTexCoord * 0.2;
float4 fvDeepBubbles = tex2D( BubbleSampler, fvDeepBubbleCoords.xy );
fvDeepBubbles = lerp( float4( 0.5f, 0.5f, 1.0f, 0.0f ), fvDeepBubbles, fvDeepLayout.z );
float4 fvBubbles = tex2D( BubbleSampler, fvBubbleCoords.xy );
fvBubbles = lerp( float4( 0.5f, 0.5f, 1.0f, 0.0f ), fvBubbles, fvLayout.z );
float3 fvSurfaceBubbles = tex2D( BubbleSampler, fvBubbleSurfaceCoords.xy ).xyz;
fvSurfaceBubbles.xyz = ( fvSurfaceBubbles.xyz * 2.0f ) - 1.0f;
fvSurfaceBubbles.xy *= -2.0f;
fvNormalTs.xyz += fvSurfaceBubbles.xyz * pow( fvLayout.z, 3.5f ) * 0.4f;
fvNormalTs.xyz = normalize( fvNormalTs.xyz );
float4 worldSpaceBubblesNormal = lerp( fvDeepBubbles * float4( 1.0f, 1.0f, 1.0f, 0.4f ), fvBubbles, fvBubbles.a );
worldSpaceBubblesNormal.xyz = mul( worldSpaceBubblesNormal.xyz * 2.0f - 1.0f, tangenttranspose );
worldSpaceBubblesNormal = normalize( worldSpaceBubblesNormal );
#endif
float3 diffuseLighting = float3( 0.0f, 0.0f, 0.0f );
#if ( THICKPAINT )
{
float3 dp;
dp.x = saturate( dot( fvNormalTs.xyz, bumpBasis[0] ) );
dp.y = saturate( dot( fvNormalTs.xyz, bumpBasis[1] ) );
dp.z = saturate( dot( fvNormalTs.xyz, bumpBasis[2] ) );
dp *= dp;
diffuseLighting = dp.x * lightmapColor1 +
dp.y * lightmapColor2 +
dp.z * lightmapColor3;
float sum = dot( dp, float3( 1.0f, 1.0f, 1.0f ) );
diffuseLighting *= g_TintValuesTimesLightmapScale.rgb / sum;
}
#else
{
diffuseLighting = lightmapColor1 * g_TintValuesTimesLightmapScale.rgb;
}
#endif
float3 diffuseComponent = diffuseLighting;
float fLum = dot( cMixedPaint.rgb, float3( 0.299f, 0.587f, 0.114f ) ) * 0.5f; //desaturated, darken
float3 cPaint = lerp( float3( fLum, fLum, fLum ), cMixedPaint.rgb, fPaintThickness ); //less thick areas should be less saturated
vNormalWs.xyz = mul( fvNormalTs, tangenttranspose );
float3 specularLighting = float3( 0.0f, 0.0f, 0.0f );
// Calc Fresnel factor
float3 eyeVect = normalize(worldVertToEyeVector);
float fresnel = dot( vNormalWs, eyeVect );
float3 reflectVect = CalcReflectionVectorUnnormalized( vNormalWs, worldVertToEyeVector );
#if ( THICKPAINT )
float3 reflectVectBubbles = CalcReflectionVectorUnnormalized( worldSpaceBubblesNormal.xyz, worldVertToEyeVector );
#endif
float fBubbleReflectLum = 0.0f;
#if ( CUBEMAP )
// Must use environment map scale to get proper behavior across platforms with different texture formats (divide by 16 because code was written on PC where env_map_scale is 16.0)
specularLighting += ( ENV_MAP_SCALE / 16.0f )* texCUBE( EnvmapSampler, reflectVect ).rgb * ( 1.5f - fresnel ) * 0.67f; //TODO: remove magic numbers
#if ( THICKPAINT )
float3 bubbleSpecularLighting = ( ENV_MAP_SCALE / 16.0f )* texCUBE( EnvmapSampler, reflectVectBubbles ).rgb * ( cPaint ) * worldSpaceBubblesNormal.a * fresnel * 2.0f;
specularLighting.rgb += bubbleSpecularLighting.rgb;
#endif
specularLighting *= ( 1.0f + diffuseComponent );
#endif
float3 result;
alpha = smoothstep( 0.40f, 0.45f, alpha );
#if ( THICKPAINT )
float alphaBubbles = fvBubbles.a * 0.3f + fvDeepBubbles.a * 0.2f;
alpha = alpha - alphaBubbles * 1.25f;
#endif
alpha *= 0.85f;
float3 fvTotalDiffuse = ( ( diffuseComponent * 0.82f ) + 0.04f ) * cPaint.rgb; // Make paint slightly emissive so we can see paint color on darkly lit surfaces
fvTotalDiffuse.rgb *= float3( 0.8f, 0.85f, 1.0f ); // This matches the code in paintblob_ps20b.fxc
result = fvTotalDiffuse.rgb + specularLighting.rgb;
float flVertexFogFactor = 0.0f;
float fogFactor = CalcPixelFogFactorSupportsVertexFog( PIXELFOGTYPE, g_FogParams, g_EyePos.xyz, worldPos, i.worldPos_projPosZ.w, flVertexFogFactor );
return FinalOutput( float4( result.rgb, alpha ), fogFactor, PIXELFOGTYPE, TONEMAP_SCALE_LINEAR, false, i.worldPos_projPosZ.w );
}