|
|
|
|
//====== Copyright <EFBFBD> 1996-2007, Valve Corporation, All rights reserved. =======//
|
|
|
|
|
//
|
|
|
|
|
// Purpose:
|
|
|
|
|
//
|
|
|
|
|
// $NoKeywords: $
|
|
|
|
|
//
|
|
|
|
|
//=============================================================================//
|
|
|
|
|
// STATIC: "CONVERT_TO_SRGB" "0..1" [ps20b][= g_pHardwareConfig->NeedsShaderSRGBConversion()] [PC]
|
|
|
|
|
// STATIC: "CONVERT_TO_SRGB" "0..0" [= 0] [XBOX]
|
|
|
|
|
// STATIC: "DETAILTEXTURE" "0..1"
|
|
|
|
|
// STATIC: "BUMPMAP" "0..1"
|
|
|
|
|
// STATIC: "VERTEXCOLOR" "0..1"
|
|
|
|
|
// STATIC: "SELFILLUM" "0..1"
|
|
|
|
|
// STATIC: "DIFFUSEBUMPMAP" "0..1"
|
|
|
|
|
// STATIC: "DETAIL_ALPHA_MASK_BASE_TEXTURE" "0..1"
|
|
|
|
|
// STATIC: "FLASHLIGHT" "0..1"
|
|
|
|
|
// STATIC: "SEAMLESS" "0..1"
|
|
|
|
|
// STATIC: "FLASHLIGHTDEPTHFILTERMODE" "0..2" [ps20b] [PC]
|
|
|
|
|
// STATIC: "FLASHLIGHTDEPTHFILTERMODE" "0..0" [ps20b] [XBOX]
|
|
|
|
|
|
|
|
|
|
// DYNAMIC: "WRITEWATERFOGTODESTALPHA" "0..1"
|
|
|
|
|
// DYNAMIC: "PIXELFOGTYPE" "0..1"
|
|
|
|
|
// DYNAMIC: "WRITE_DEPTH_TO_DESTALPHA" "0..1" [ps20b] [PC]
|
|
|
|
|
// DYNAMIC: "WRITE_DEPTH_TO_DESTALPHA" "0..0" [ps20b] [XBOX]
|
|
|
|
|
// DYNAMIC: "FLASHLIGHTSHADOWS" "0..1" [ps20b]
|
|
|
|
|
|
|
|
|
|
// SKIP: $DETAILTEXTURE && ( $BUMPMAP && !$DETAIL_ALPHA_MASK_BASE_TEXTURE )
|
|
|
|
|
// SKIP: !$BUMPMAP && $DIFFUSEBUMPMAP
|
|
|
|
|
// SKIP: $VERTEXCOLOR && $BUMPMAP
|
|
|
|
|
// SKIP: FLASHLIGHT && $SELFILLUM
|
|
|
|
|
// SKIP: FLASHLIGHT && $DETAIL_ALPHA_MASK_BASE_TEXTURE
|
|
|
|
|
// SKIP: FLASHLIGHT && ($BUMPMAP || $DIFFUSEBUMPMAP)
|
|
|
|
|
|
|
|
|
|
// We don't care about flashlight depth unless the flashlight is on
|
|
|
|
|
// SKIP: ( $FLASHLIGHT == 0 ) && ( $FLASHLIGHTSHADOWS == 1 ) [ps20b]
|
|
|
|
|
|
|
|
|
|
#if defined( SHADER_MODEL_PS_2_0 )
|
|
|
|
|
# define WRITE_DEPTH_TO_DESTALPHA 0
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define HDRTYPE HDR_TYPE_NONE
|
|
|
|
|
#include "common_flashlight_fxc.h"
|
|
|
|
|
#include "common_ps_fxc.h"
|
|
|
|
|
|
|
|
|
|
const HALF4 g_SelfIllumTint : register( c7 );
|
|
|
|
|
static const HALF g_OverbrightFactor = 2.0f;
|
|
|
|
|
|
|
|
|
|
const HALF3 g_EyePos : register( c10 );
|
|
|
|
|
const HALF4 g_FogParams : register( c11 );
|
|
|
|
|
|
|
|
|
|
const HALF3 g_FlashlightPos : register( c15 );
|
|
|
|
|
// flashlightfixme: Move this math into the vertex shader.
|
|
|
|
|
const float4x4 g_FlashlightWorldToTexture : register( c16 );
|
|
|
|
|
|
|
|
|
|
const float4 g_FlashlightAttenuationFactors : register( c20 );
|
|
|
|
|
|
|
|
|
|
sampler BaseTextureSampler : register( s0 );
|
|
|
|
|
sampler LightmapSampler : register( s1 );
|
|
|
|
|
sampler FlashlightSampler : register( s2 );
|
|
|
|
|
sampler DetailSampler : register( s3 );
|
|
|
|
|
sampler BumpmapSampler : register( s4 );
|
|
|
|
|
sampler NormalizeSampler : register( s6 );
|
|
|
|
|
|
|
|
|
|
struct PS_INPUT
|
|
|
|
|
{
|
|
|
|
|
HALF2 baseTexCoord : TEXCOORD0;
|
|
|
|
|
HALF4 detailOrBumpTexCoord : TEXCOORD1;
|
|
|
|
|
HALF4 lightmapTexCoord1And2 : TEXCOORD2; // CENTROID: TEXCOORD2
|
|
|
|
|
HALF2 lightmapTexCoord3 : TEXCOORD3; // CENTROID: TEXCOORD3
|
|
|
|
|
HALF4 worldPos_projPosZ : TEXCOORD4;
|
|
|
|
|
HALF3x3 tangentSpaceTranspose : TEXCOORD5;
|
|
|
|
|
// tangentSpaceTranspose : TEXCOORD6;
|
|
|
|
|
// tangentSpaceTranspose : TEXCOORD7;
|
|
|
|
|
HALF4 vertexColor : COLOR;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float4 main( PS_INPUT i ) : COLOR
|
|
|
|
|
{
|
|
|
|
|
bool bDetailTexture = DETAILTEXTURE ? true : false;
|
|
|
|
|
bool bBumpmap = BUMPMAP ? true : false;
|
|
|
|
|
bool bDiffuseBumpmap = DIFFUSEBUMPMAP ? true : false;
|
|
|
|
|
bool bVertexColor = VERTEXCOLOR ? true : false;
|
|
|
|
|
bool bSelfIllum = SELFILLUM ? true : false;
|
|
|
|
|
bool bDetailAlphaMaskBaseTexture = DETAIL_ALPHA_MASK_BASE_TEXTURE ? true : false;
|
|
|
|
|
bool bFlashlight = FLASHLIGHT ? true : false;
|
|
|
|
|
|
|
|
|
|
HALF3 lightmapColor1 = HALF3( 1.0f, 1.0f, 1.0f );
|
|
|
|
|
HALF3 lightmapColor2 = HALF3( 1.0f, 1.0f, 1.0f );
|
|
|
|
|
HALF3 lightmapColor3 = HALF3( 1.0f, 1.0f, 1.0f );
|
|
|
|
|
if( bBumpmap && bDiffuseBumpmap )
|
|
|
|
|
{
|
|
|
|
|
HALF2 bumpCoord1;
|
|
|
|
|
HALF2 bumpCoord2;
|
|
|
|
|
HALF2 bumpCoord3;
|
|
|
|
|
ComputeBumpedLightmapCoordinates( i.lightmapTexCoord1And2, i.lightmapTexCoord3.xy,
|
|
|
|
|
bumpCoord1, bumpCoord2, bumpCoord3 );
|
|
|
|
|
|
|
|
|
|
HALF4 lightmapSample1 = tex2D( LightmapSampler, bumpCoord1 );
|
|
|
|
|
lightmapColor1 = lightmapSample1.rgb;
|
|
|
|
|
lightmapColor2 = tex2D( LightmapSampler, bumpCoord2 );
|
|
|
|
|
lightmapColor3 = tex2D( LightmapSampler, bumpCoord3 );
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if( !bFlashlight )
|
|
|
|
|
{
|
|
|
|
|
HALF2 bumpCoord1 = ComputeLightmapCoordinates( i.lightmapTexCoord1And2, i.lightmapTexCoord3.xy );
|
|
|
|
|
HALF4 lightmapSample1 = tex2D( LightmapSampler, bumpCoord1 );
|
|
|
|
|
lightmapColor1 = lightmapSample1.rgb;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF4 detailColor = HALF4( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
|
|
|
if( bDetailTexture )
|
|
|
|
|
{
|
|
|
|
|
detailColor = tex2D( DetailSampler, i.detailOrBumpTexCoord.xy );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF4 baseColor = HALF4( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
|
|
|
baseColor = tex2D( BaseTextureSampler, i.baseTexCoord );
|
|
|
|
|
if ( bDetailAlphaMaskBaseTexture )
|
|
|
|
|
{
|
|
|
|
|
// This is what WorldTwoTextureBlend_DX6 does.
|
|
|
|
|
baseColor.rgb = saturate( saturate( baseColor * 2 ) * detailColor.a + (1 - detailColor.a) );
|
|
|
|
|
baseColor.rgb *= detailColor;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
baseColor.rgb = lerp( baseColor, detailColor, detailColor.a );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF3 normal = HALF3( 0.0f, 0.0f, 1.0f );
|
|
|
|
|
if( bBumpmap )
|
|
|
|
|
{
|
|
|
|
|
HALF3 normalTexel;
|
|
|
|
|
normalTexel = tex2D( BumpmapSampler, i.detailOrBumpTexCoord.xy );
|
|
|
|
|
normal = 2.0 * normalTexel - 1.0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF3 albedo = HALF3( 1.0f, 1.0f, 1.0f );
|
|
|
|
|
HALF alpha = 1.0f;
|
|
|
|
|
albedo *= baseColor;
|
|
|
|
|
if( !bSelfIllum )
|
|
|
|
|
{
|
|
|
|
|
alpha *= baseColor.a;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// The vertex color contains the modulation color + vertex color combined
|
|
|
|
|
albedo *= i.vertexColor;
|
|
|
|
|
alpha *= i.vertexColor.a; // not sure about this one
|
|
|
|
|
|
|
|
|
|
HALF3 diffuseLighting;
|
|
|
|
|
if( bFlashlight )
|
|
|
|
|
{
|
|
|
|
|
float3 worldSpaceNormal;
|
|
|
|
|
// Make the unbumped version not so fucking stupid and not need tangentSpaceTranspose you knob.
|
|
|
|
|
worldSpaceNormal = mul( normal, i.tangentSpaceTranspose );
|
|
|
|
|
|
|
|
|
|
int nShadowSampleLevel = 0;
|
|
|
|
|
bool bDoShadows = false;
|
|
|
|
|
// On ps_2_b, we can do shadow mapping
|
|
|
|
|
#if ( FLASHLIGHTSHADOWS && (defined(SHADER_MODEL_PS_2_B) || defined(SHADER_MODEL_PS_3_0) ) )
|
|
|
|
|
nShadowSampleLevel = FLASHLIGHTDEPTHFILTERMODE;
|
|
|
|
|
bDoShadows = true;
|
|
|
|
|
#endif
|
|
|
|
|
float4 flashlightSpacePosition = mul( float4( i.worldPos_projPosZ.xyz, 1.0f ), g_FlashlightWorldToTexture );
|
|
|
|
|
|
|
|
|
|
diffuseLighting = DoFlashlight( g_FlashlightPos, i.worldPos_projPosZ.xyz, flashlightSpacePosition,
|
|
|
|
|
worldSpaceNormal, g_FlashlightAttenuationFactors.xyz,
|
|
|
|
|
g_FlashlightAttenuationFactors.w, FlashlightSampler, FlashlightSampler, NormalizeSampler,
|
|
|
|
|
nShadowSampleLevel, bDoShadows, false, float2(0, 0), false );
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if( bBumpmap && bDiffuseBumpmap )
|
|
|
|
|
{
|
|
|
|
|
float dot1 = saturate( dot( normal, bumpBasis[0] ) );
|
|
|
|
|
float dot2 = saturate( dot( normal, bumpBasis[1] ) );
|
|
|
|
|
float dot3 = saturate( dot( normal, bumpBasis[2] ) );
|
|
|
|
|
|
|
|
|
|
float sum = dot1 + dot2 + dot3;
|
|
|
|
|
diffuseLighting = dot1 * lightmapColor1 +
|
|
|
|
|
dot2 * lightmapColor2 +
|
|
|
|
|
dot3 * lightmapColor3;
|
|
|
|
|
diffuseLighting *= 1.0f / sum;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
diffuseLighting = lightmapColor1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Only scale here since the flashlight will already be scaled properly
|
|
|
|
|
diffuseLighting *= g_OverbrightFactor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF3 diffuseComponent = albedo * diffuseLighting;
|
|
|
|
|
|
|
|
|
|
if( bSelfIllum )
|
|
|
|
|
{
|
|
|
|
|
HALF3 selfIllumComponent = g_SelfIllumTint * albedo;
|
|
|
|
|
diffuseComponent = lerp( diffuseComponent, selfIllumComponent, baseColor.a );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HALF3 specularLighting = HALF3( 0.0f, 0.0f, 0.0f );
|
|
|
|
|
HALF3 result = diffuseComponent + specularLighting;
|
|
|
|
|
|
|
|
|
|
float fogFactor = CalcPixelFogFactor( PIXELFOGTYPE, g_FogParams, g_EyePos.z, i.worldPos_projPosZ.z, i.worldPos_projPosZ.w );
|
|
|
|
|
|
|
|
|
|
#if WRITEWATERFOGTODESTALPHA && (PIXELFOGTYPE == PIXEL_FOG_TYPE_HEIGHT)
|
|
|
|
|
alpha = fogFactor;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return FinalOutput( float4( result, alpha ), fogFactor, PIXELFOGTYPE, TONEMAP_SCALE_LINEAR, (WRITE_DEPTH_TO_DESTALPHA != 0), i.worldPos_projPosZ.w );
|
|
|
|
|
}
|
|
|
|
|
|