//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: 
//
// $Header: $
// $NoKeywords: $
//===========================================================================//

#include "BaseVSShader.h"
#include "convar.h"

#include "worldvertextransition.inc"
#include "worldvertextransition_vs14.inc"
#include "worldvertextransition_seamless.inc"
#include "lightmappedgeneric_vs11.inc"
#include "writevertexalphatodestalpha_vs11.inc"
#include "worldvertextransition_dx8_helper.h"

// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"

DEFINE_FALLBACK_SHADER( WorldVertexTransition, WorldVertexTransition_DX8 )

ConVar mat_fullbright( "mat_fullbright","0", FCVAR_CHEAT );

BEGIN_VS_SHADER( WorldVertexTransition_DX8,
			  "Help for WorldVertexTransition_DX8" )

	BEGIN_SHADER_PARAMS
		SHADER_PARAM( BASETEXTURE2, SHADER_PARAM_TYPE_TEXTURE, "shadertest/BaseTexture2", "base texture2 help" )
		SHADER_PARAM( FRAME2, SHADER_PARAM_TYPE_INTEGER, "0", "frame number for $baseTexture" )
		SHADER_PARAM( BASETEXTURETRANSFORM2, SHADER_PARAM_TYPE_MATRIX, "center .5 .5 scale 1 1 rotate 0 translate 0 0", "$baseTexture texcoord transform" )
		SHADER_PARAM( SELFILLUMTINT, SHADER_PARAM_TYPE_COLOR, "[1 1 1]", "Self-illumination tint" )
		SHADER_PARAM( DETAIL, SHADER_PARAM_TYPE_TEXTURE, "shadertest/detail", "detail texture" )
		SHADER_PARAM( DETAILFRAME, SHADER_PARAM_TYPE_INTEGER, "0", "frame number for $detail" )
		SHADER_PARAM( DETAILSCALE, SHADER_PARAM_TYPE_FLOAT, "4", "scale of the detail texture" )
		SHADER_PARAM( ENVMAP, SHADER_PARAM_TYPE_TEXTURE, "shadertest/shadertest_env", "envmap" )
		SHADER_PARAM( ENVMAPFRAME, SHADER_PARAM_TYPE_INTEGER, "", "" )
		SHADER_PARAM( ENVMAPMASK, SHADER_PARAM_TYPE_TEXTURE, "shadertest/shadertest_envmask", "envmap mask" )
		SHADER_PARAM( ENVMAPMASKFRAME, SHADER_PARAM_TYPE_INTEGER, "", "" )
		SHADER_PARAM( ENVMAPMASKSCALE, SHADER_PARAM_TYPE_FLOAT, "1", "envmap mask scale" )
		SHADER_PARAM( ENVMAPTINT, SHADER_PARAM_TYPE_COLOR, "[1 1 1]", "envmap tint" )
		SHADER_PARAM( BUMPMAP, SHADER_PARAM_TYPE_TEXTURE, "models/shadertest/shader1_normal", "bump map for BASETEXTURE" )
		SHADER_PARAM( BUMPFRAME, SHADER_PARAM_TYPE_INTEGER, "0", "frame number for $bumpmap" )
		SHADER_PARAM( BUMPTRANSFORM, SHADER_PARAM_TYPE_MATRIX, "center .5 .5 scale 1 1 rotate 0 translate 0 0", "$bumpmap texcoord transform" )
		SHADER_PARAM( ENVMAPCONTRAST, SHADER_PARAM_TYPE_FLOAT, "0.0", "contrast 0 == normal 1 == color*color" )
		SHADER_PARAM( ENVMAPSATURATION, SHADER_PARAM_TYPE_FLOAT, "1.0", "saturation 0 == greyscale 1 == normal" )
		SHADER_PARAM( BUMPBASETEXTURE2WITHBUMPMAP, SHADER_PARAM_TYPE_BOOL, "0", "" )
		SHADER_PARAM( FRESNELREFLECTION, SHADER_PARAM_TYPE_FLOAT, "0.0", "1.0 == mirror, 0.0 == water" )
		SHADER_PARAM( SSBUMP, SHADER_PARAM_TYPE_INTEGER, "0", "whether or not to use alternate bumpmap format with height" )
		SHADER_PARAM( SEAMLESS_SCALE, SHADER_PARAM_TYPE_FLOAT, "0", "Scale factor for 'seamless' texture mapping. 0 means to use ordinary mapping" )
		SHADER_PARAM( BLENDMODULATETEXTURE, SHADER_PARAM_TYPE_TEXTURE, "", "texture to use r/g channels for blend range for" )
		SHADER_PARAM( BLENDMASKTRANSFORM, SHADER_PARAM_TYPE_MATRIX, "center .5 .5 scale 1 1 rotate 0 translate 0 0", "$blendmodulatetexture texcoord transform" )
	END_SHADER_PARAMS

	SHADER_FALLBACK
	{
		if ( IsPC() && g_pHardwareConfig->GetDXSupportLevel() < 80 )
			return "WorldVertexTransition_DX6";
		return 0;
	}

	void SetupVars( WorldVertexTransitionEditor_DX8_Vars_t& info )
	{
		info.m_nBaseTextureVar = BASETEXTURE;
		info.m_nBaseTextureFrameVar = FRAME;
		info.m_nBaseTextureTransformVar = BASETEXTURETRANSFORM;
		info.m_nBaseTexture2Var = BASETEXTURE2;
		info.m_nBaseTexture2FrameVar = FRAME2;
		info.m_nBaseTexture2TransformVar = BASETEXTURETRANSFORM2;
	}

	SHADER_INIT_PARAMS()
	{
		// Initializes FLASHLIGHTTEXTURE + MATERIAL_VAR2_LIGHTING_LIGHTMAP
		WorldVertexTransitionEditor_DX8_Vars_t info;
		SetupVars( info );
		InitParamsWorldVertexTransitionEditor_DX8( params, info );

		// FLASHLIGHTFIXME
		params[FLASHLIGHTTEXTURE]->SetStringValue( "effects/flashlight001" );

		if( IsUsingGraphics() && params[ENVMAP]->IsDefined() && !CanUseEditorMaterials() )
		{
			if( stricmp( params[ENVMAP]->GetStringValue(), "env_cubemap" ) == 0 )
			{
				Warning( "env_cubemap used on world geometry without rebuilding map. . ignoring: %s\n", pMaterialName );
				params[ENVMAP]->SetUndefined();
			}
		}
		
		if( !params[ENVMAPMASKSCALE]->IsDefined() )
		{
			params[ENVMAPMASKSCALE]->SetFloatValue( 1.0f );
		}

		if( !params[ENVMAPTINT]->IsDefined() )
		{
			params[ENVMAPTINT]->SetVecValue( 1.0f, 1.0f, 1.0f );
		}

		if( !params[SELFILLUMTINT]->IsDefined() )
		{
			params[SELFILLUMTINT]->SetVecValue( 1.0f, 1.0f, 1.0f );
		}

		if( !params[DETAILSCALE]->IsDefined() )
		{
			params[DETAILSCALE]->SetFloatValue( 4.0f );
		}

		if( !params[FRESNELREFLECTION]->IsDefined() )
		{
			params[FRESNELREFLECTION]->SetFloatValue( 1.0f );
		}

		if( !params[ENVMAPMASKFRAME]->IsDefined() )
		{
			params[ENVMAPMASKFRAME]->SetIntValue( 0 );
		}
		
		if( !params[ENVMAPFRAME]->IsDefined() )
		{
			params[ENVMAPFRAME]->SetIntValue( 0 );
		}

		if( !params[BUMPFRAME]->IsDefined() )
		{
			params[BUMPFRAME]->SetIntValue( 0 );
		}

		if( !params[ENVMAPCONTRAST]->IsDefined() )
		{
			params[ENVMAPCONTRAST]->SetFloatValue( 0.0f );
		}
		
		if( !params[ENVMAPSATURATION]->IsDefined() )
		{
			params[ENVMAPSATURATION]->SetFloatValue( 1.0f );
		}
		
		// No texture means no self-illum or env mask in base alpha
		if ( !params[BASETEXTURE]->IsDefined() )
		{
			CLEAR_FLAGS( MATERIAL_VAR_SELFILLUM );
			CLEAR_FLAGS( MATERIAL_VAR_BASEALPHAENVMAPMASK );
		}

		// If in decal mode, no debug override...
		if (IS_FLAG_SET(MATERIAL_VAR_DECAL))
		{
			SET_FLAGS( MATERIAL_VAR_NO_DEBUG_OVERRIDE );
		}

		SET_FLAGS2( MATERIAL_VAR2_LIGHTING_LIGHTMAP );
		if( g_pConfig->UseBumpmapping() && params[BUMPMAP]->IsDefined() )
		{
			SET_FLAGS2( MATERIAL_VAR2_LIGHTING_BUMPED_LIGHTMAP );
		}

		if( !params[BUMPBASETEXTURE2WITHBUMPMAP]->IsDefined() )
		{
			params[BUMPBASETEXTURE2WITHBUMPMAP]->SetIntValue( 0 );
		}

		if( !params[DETAILSCALE]->IsDefined() )
		{
			params[DETAILSCALE]->SetFloatValue( 4.0f );
		}

		if( !params[DETAILFRAME]->IsDefined() )
		{
			params[DETAILFRAME]->SetIntValue( 0 );
		}

		if( params[SEAMLESS_SCALE]->IsDefined() && params[SEAMLESS_SCALE]->GetFloatValue() != 0.0f )
		{
			// seamless scale is going to be used, so kill some other features. . might implement with these features later.
			params[DETAIL]->SetUndefined();
			params[BUMPMAP]->SetUndefined();
			params[ENVMAP]->SetUndefined();
		}

		if ( !params[SEAMLESS_SCALE]->IsDefined() )
		{
			// zero means don't do seamless mapping.
			params[SEAMLESS_SCALE]->SetFloatValue( 0.0f );
		}

		if( params[SSBUMP]->IsDefined() && params[SSBUMP]->GetIntValue() != 0 )
		{
			// turn of normal mapping since we have ssbump defined, which 
			// means that we didn't make a dx8 fallback for this material.
			params[BUMPMAP]->SetUndefined();
		}
	}
	SHADER_INIT
	{
		// Loads BASETEXTURE, BASETEXTURE2
		WorldVertexTransitionEditor_DX8_Vars_t info;
		SetupVars( info );
		InitWorldVertexTransitionEditor_DX8( this, params, info );

		// FLASHLIGHTFIXME
		if ( params[FLASHLIGHTTEXTURE]->IsDefined() )
		{
			LoadTexture( FLASHLIGHTTEXTURE );
		}

		if (params[DETAIL]->IsDefined())
		{
			LoadTexture( DETAIL );
		}
		
		if ( g_pHardwareConfig->SupportsPixelShaders_1_4() && params[BLENDMODULATETEXTURE]->IsDefined() )
		{
			LoadTexture( BLENDMODULATETEXTURE );
		}

		SET_FLAGS2( MATERIAL_VAR2_LIGHTING_LIGHTMAP );
		if( params[ENVMAP]->IsDefined() && !params[BUMPMAP]->IsDefined() )
		{
			Warning( "must have $bumpmap if you have $envmap for worldvertextransition\n" );
			params[ENVMAP]->SetUndefined();
			params[BUMPMAP]->SetUndefined();
		}
		if( g_pConfig->UseBumpmapping() && params[BUMPMAP]->IsDefined() )
		{
			SET_FLAGS2( MATERIAL_VAR2_NEEDS_TANGENT_SPACES );
			SET_FLAGS2( MATERIAL_VAR2_LIGHTING_BUMPED_LIGHTMAP );
			LoadBumpMap( BUMPMAP );
		}
		if( params[ENVMAP]->IsDefined() )
		{
			if( !IS_FLAG_SET( MATERIAL_VAR_ENVMAPSPHERE ) )
			{
				LoadCubeMap( ENVMAP );
			}
			else
			{
				Warning( "$envmapsphere not supported by worldvertextransition\n" );
				params[ENVMAP]->SetUndefined();
			}
		}
	}

	void WriteVertexAlphaToDestAlpha( IMaterialVar** params, IShaderDynamicAPI *pShaderAPI, IShaderShadow* pShaderShadow )
	{
		if( pShaderShadow )
		{
			pShaderShadow->EnableDepthWrites( false );
			pShaderShadow->EnableAlphaWrites( true );
			pShaderShadow->EnableColorWrites( false );

			writevertexalphatodestalpha_vs11_Static_Index vshIndex;
			pShaderShadow->SetVertexShader( "writevertexalphatodestalpha_vs11", vshIndex.GetIndex() );
		
			pShaderShadow->SetPixelShader( "writevertexalphatodestalpha_ps11" );
			pShaderShadow->VertexShaderVertexFormat( 
				VERTEX_POSITION | VERTEX_COLOR, 2, 0, 0 );
		}
		else
		{
			writevertexalphatodestalpha_vs11_Dynamic_Index vshIndex;
			pShaderAPI->SetVertexShaderIndex( vshIndex.GetIndex() );
		}
		Draw();
	}

	void DrawFlashlightPass( IMaterialVar** params, IShaderDynamicAPI *pShaderAPI, IShaderShadow* pShaderShadow, int passID )
	{
		bool bBump = ( passID == 0 ) && ShouldUseBumpmapping( params ) && params[BUMPMAP]->IsTexture();
		DrawFlashlight_dx80( params, pShaderAPI, pShaderShadow, bBump, BUMPMAP, BUMPFRAME, BUMPTRANSFORM, 
			FLASHLIGHTTEXTURE, FLASHLIGHTTEXTUREFRAME, true, true, passID, BASETEXTURE2, FRAME2 );
	}

	bool ShouldUseBumpmapping( IMaterialVar **params ) 
	{ 
		return g_pConfig->UseBumpmapping() && params[BUMPMAP]->IsDefined(); 
	}

	void DrawFlashlight( IMaterialVar** params, IShaderDynamicAPI *pShaderAPI, IShaderShadow* pShaderShadow )
	{
		WriteVertexAlphaToDestAlpha( params, pShaderAPI, pShaderShadow );
		DrawFlashlightPass( params, pShaderAPI, pShaderShadow, 0 );
		DrawFlashlightPass( params, pShaderAPI, pShaderShadow, 1 );
	}
	
	SHADER_DRAW
	{
		bool bLightingOnly = mat_fullbright.GetInt() == 2 && !IS_FLAG_SET( MATERIAL_VAR_NO_DEBUG_OVERRIDE );
		bool bSupports14 = g_pHardwareConfig->SupportsPixelShaders_1_4();

		// FLASHLIGHTFIXME: need to make these the same.
		bool hasFlashlight = UsingFlashlight( params );
		if( hasFlashlight )
		{
			DrawFlashlight( params, pShaderAPI, pShaderShadow );
		}
		else if ( params[SEAMLESS_SCALE]->GetFloatValue() != 0.0f )
		{
			// This is the seamless_scale version, which doesn't use $detail or $bumpmap
			SHADOW_STATE
			{
				// three copies of the base texture for seamless blending
				pShaderShadow->EnableTexture( SHADER_SAMPLER0, true );
				pShaderShadow->EnableTexture( SHADER_SAMPLER1, true );
				pShaderShadow->EnableTexture( SHADER_SAMPLER2, true );
				
				// lightmap
				pShaderShadow->EnableTexture( SHADER_SAMPLER3, true );

				int fmt = VERTEX_POSITION;
				pShaderShadow->VertexShaderVertexFormat( fmt, 2, 0, 0 );

				worldvertextransition_seamless_Static_Index vshIndex;
				pShaderShadow->SetVertexShader( "WorldVertexTransition_Seamless", vshIndex.GetIndex() );

				int pshIndex = 0;
				pShaderShadow->SetPixelShader( "WorldVertexTransition_Seamless", pshIndex );

				FogToFogColor();
			}
			DYNAMIC_STATE
			{
				// Texture 0..2
				if( bLightingOnly )
				{
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER0, TEXTURE_GREY );
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER1, TEXTURE_GREY );
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER2, TEXTURE_GREY );
				}
				else
				{
					BindTexture( SHADER_SAMPLER0, BASETEXTURE, FRAME );
					BindTexture( SHADER_SAMPLER1, BASETEXTURE, FRAME );
					BindTexture( SHADER_SAMPLER2, BASETEXTURE, FRAME );
				}

				// Texture 3 = lightmap
				pShaderAPI->BindStandardTexture( SHADER_SAMPLER3, TEXTURE_LIGHTMAP );

				EnablePixelShaderOverbright( 0, true, true );

				float fSeamlessScale = params[SEAMLESS_SCALE]->GetFloatValue();
				float map_scale[4]= { fSeamlessScale, fSeamlessScale, fSeamlessScale, fSeamlessScale };
				pShaderAPI->SetVertexShaderConstant( VERTEX_SHADER_SHADER_SPECIFIC_CONST_0, map_scale );

				worldvertextransition_seamless_Dynamic_Index vshIndex;
				vshIndex.SetDOWATERFOG( pShaderAPI->GetSceneFogMode() == MATERIAL_FOG_LINEAR_BELOW_FOG_Z );
				pShaderAPI->SetVertexShaderIndex( vshIndex.GetIndex() );
			}
			Draw();
			SHADOW_STATE
			{
				// inherit state from previous pass

				EnableAlphaBlending( SHADER_BLEND_SRC_ALPHA, SHADER_BLEND_ONE_MINUS_SRC_ALPHA );
			}
			DYNAMIC_STATE
			{
				if( !bLightingOnly )
				{
					BindTexture( SHADER_SAMPLER0, BASETEXTURE2, FRAME2 );
					BindTexture( SHADER_SAMPLER1, BASETEXTURE2, FRAME2 );
					BindTexture( SHADER_SAMPLER2, BASETEXTURE2, FRAME2 );
				}
			}
			Draw();
		}
		else if( !params[BUMPMAP]->IsTexture() || !g_pConfig->UseBumpmapping() )
		{
			bool bDetail = params[DETAIL]->IsTexture();
			bool bBlendModulate = params[BLENDMODULATETEXTURE]->IsTexture();
			SHADOW_STATE
			{
				// This is the dx8, non-worldcraft version, non-bumped version
				pShaderShadow->EnableTexture( SHADER_SAMPLER0, true );
				pShaderShadow->EnableTexture( SHADER_SAMPLER1, true );
				pShaderShadow->EnableTexture( SHADER_SAMPLER2, true );
				if( bDetail )
				{
					pShaderShadow->EnableTexture( SHADER_SAMPLER3, true );
				}
				if ( bSupports14 && bBlendModulate )
				{
					pShaderShadow->EnableTexture( SHADER_SAMPLER4, true );
				}

				int fmt = VERTEX_POSITION | VERTEX_COLOR;
				pShaderShadow->VertexShaderVertexFormat( fmt, 2, 0, 0 );

				if ( !bSupports14 )
				{
					worldvertextransition_Static_Index vshIndex;
					pShaderShadow->SetVertexShader( "WorldVertexTransition", vshIndex.GetIndex() );

					int pshIndex = bDetail ? 1 : 0;
					pShaderShadow->SetPixelShader( "WorldVertexTransition", pshIndex );
				}
				else
				{
					worldvertextransition_vs14_Static_Index vshIndex;
					pShaderShadow->SetVertexShader( "WorldVertexTransition_vs14", vshIndex.GetIndex() );

					int pshIndex = bDetail ? 1 : 0;
					pshIndex += bBlendModulate ? 2 : 0;
					pShaderShadow->SetPixelShader( "WorldVertexTransition_ps14", pshIndex );
				}
			
				FogToFogColor();
			}

			DYNAMIC_STATE
			{
				// Texture 1
				if( bLightingOnly )
				{
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER0, TEXTURE_GREY );
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER1, TEXTURE_GREY );
				}
				else
				{
					BindTexture( SHADER_SAMPLER0, BASETEXTURE, FRAME );
					BindTexture( SHADER_SAMPLER1, BASETEXTURE2, FRAME2 );
				}
				if( bDetail )
				{
					BindTexture( SHADER_SAMPLER3, DETAIL, DETAILFRAME );
				}
				if ( bSupports14 && bBlendModulate )
				{
					BindTexture( SHADER_SAMPLER4, BLENDMODULATETEXTURE );
				}

				// always set the transform for detail textures since I'm assuming that you'll
				// always have a detailscale.
				// go ahead and set this even if we don't have a detail texture so the vertex shader doesn't 
				// barf chunks with unitialized data.
				SetVertexShaderTextureScaledTransform( VERTEX_SHADER_SHADER_SPECIFIC_CONST_4, BASETEXTURETRANSFORM, DETAILSCALE );

				if ( bSupports14 )
				{
					SetVertexShaderTextureTransform( VERTEX_SHADER_SHADER_SPECIFIC_CONST_6, BLENDMASKTRANSFORM );
				}

				// Texture 3 = lightmap
				pShaderAPI->BindStandardTexture( SHADER_SAMPLER2, TEXTURE_LIGHTMAP );
				
				EnablePixelShaderOverbright( 0, true, true );
				
				SetVertexShaderTextureTransform( VERTEX_SHADER_SHADER_SPECIFIC_CONST_0, BASETEXTURETRANSFORM );
				SetVertexShaderTextureTransform( VERTEX_SHADER_SHADER_SPECIFIC_CONST_2, BASETEXTURETRANSFORM2 );
				if ( !bSupports14 )
				{
					worldvertextransition_Dynamic_Index vshIndex;
					vshIndex.SetDOWATERFOG( pShaderAPI->GetSceneFogMode() == MATERIAL_FOG_LINEAR_BELOW_FOG_Z );
					pShaderAPI->SetVertexShaderIndex( vshIndex.GetIndex() );
				}
				else
				{
					worldvertextransition_vs14_Dynamic_Index vshIndex;
					vshIndex.SetDOWATERFOG( pShaderAPI->GetSceneFogMode() == MATERIAL_FOG_LINEAR_BELOW_FOG_Z );
					pShaderAPI->SetVertexShaderIndex( vshIndex.GetIndex() );
				}
			}
			Draw();
		}
		else
		{
			if( params[BUMPBASETEXTURE2WITHBUMPMAP]->GetIntValue() )
			{
				DrawWorldBumpedUsingVertexShader( BASETEXTURE, BASETEXTURETRANSFORM,
					BUMPMAP, BUMPFRAME, BUMPTRANSFORM, ENVMAPMASK, ENVMAPMASKFRAME, ENVMAP, 
					ENVMAPFRAME, ENVMAPTINT, COLOR, ALPHA, ENVMAPCONTRAST, ENVMAPSATURATION, FRAME,
					FRESNELREFLECTION, true, BASETEXTURE2, BASETEXTURETRANSFORM2, FRAME2, false );
			}
			else
			{
				// draw the base texture with everything else you normally would for
				// bumped world materials
				DrawWorldBumpedUsingVertexShader(
					BASETEXTURE, BASETEXTURETRANSFORM,
					BUMPMAP, BUMPFRAME, BUMPTRANSFORM,
					ENVMAPMASK, ENVMAPMASKFRAME, ENVMAP, ENVMAPFRAME, ENVMAPTINT,
					COLOR, ALPHA, ENVMAPCONTRAST, ENVMAPSATURATION, FRAME,
					FRESNELREFLECTION,
					false, -1, -1, -1, false );

				// blend basetexture 2 on top of everything using lightmap alpha.
				SHADOW_STATE
				{
					pShaderShadow->EnableTexture( SHADER_SAMPLER0, true );
					pShaderShadow->EnableTexture( SHADER_SAMPLER1, true );
					pShaderShadow->VertexShaderVertexFormat( 
						VERTEX_POSITION | VERTEX_COLOR, 2, 0, 0 );
					pShaderShadow->EnableBlending( true );
					pShaderShadow->BlendFunc( SHADER_BLEND_SRC_ALPHA, SHADER_BLEND_ONE_MINUS_SRC_ALPHA );

					lightmappedgeneric_vs11_Static_Index vshIndex;
					vshIndex.SetDETAIL( false );
					vshIndex.SetENVMAP( false );
					vshIndex.SetENVMAPCAMERASPACE( false );
					vshIndex.SetENVMAPSPHERE( false );
					vshIndex.SetVERTEXCOLOR( true );
					pShaderShadow->SetVertexShader( "LightmappedGeneric_vs11", vshIndex.GetIndex() );

					pShaderShadow->SetPixelShader( "WorldVertexTransition_BlendBase2" );
					FogToFogColor();
				}
				DYNAMIC_STATE
				{
					if( bLightingOnly )
					{
						pShaderAPI->BindStandardTexture( SHADER_SAMPLER0, TEXTURE_GREY );
					}
					else
					{
						BindTexture( SHADER_SAMPLER0, BASETEXTURE2, FRAME2 );
					}
					pShaderAPI->BindStandardTexture( SHADER_SAMPLER1, TEXTURE_LIGHTMAP );
					
					float half[4] = { 0.5f, 0.5f, 0.5f, 0.5f };
					pShaderAPI->SetPixelShaderConstant( 4, half );
					EnablePixelShaderOverbright( 0, true, true );
					SetVertexShaderTextureTransform( VERTEX_SHADER_SHADER_SPECIFIC_CONST_0, BASETEXTURETRANSFORM2 );

					lightmappedgeneric_vs11_Dynamic_Index vshIndex;
					vshIndex.SetDOWATERFOG( pShaderAPI->GetSceneFogMode() == MATERIAL_FOG_LINEAR_BELOW_FOG_Z );
					pShaderAPI->SetVertexShaderIndex( vshIndex.GetIndex() );
				}
				Draw();
			}
		}
	}
END_SHADER