csgo-2018-source/game/server/env_cascade_light.cpp
2021-07-24 21:11:47 -07:00

367 lines
8.6 KiB
C++

//========= Copyright © 1996-2010, Valve Corporation, All rights reserved. ============//
//
// Purpose: global dynamic light with cascaded shadow mapping
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "lights.h"
#include "env_cascade_light.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
//#define CsmDbgMsg Msg
#define CsmDbgMsg(x)
ConVar cl_csm_auto_entity( "cl_csm_auto_entity", "1", 0, "" );
CCascadeLight *g_pCascadeLight;
LINK_ENTITY_TO_CLASS(env_cascade_light, CCascadeLight);
BEGIN_DATADESC( CCascadeLight )
DEFINE_KEYFIELD( m_bEnabled, FIELD_BOOLEAN, "enabled" ),
DEFINE_KEYFIELD( m_bStartDisabled, FIELD_BOOLEAN, "StartDisabled" ),
DEFINE_FIELD( m_LightColor, FIELD_COLOR32 ),
DEFINE_FIELD( m_LightColorScale, FIELD_INTEGER ),
// Inputs
DEFINE_INPUTFUNC( FIELD_COLOR32, "LightColor", InputSetLightColor ),
DEFINE_INPUTFUNC( FIELD_INTEGER, "LightColorScale", InputSetLightColorScale ),
DEFINE_INPUTFUNC( FIELD_STRING, "SetAngles", InputSetAngles ),
DEFINE_INPUTFUNC( FIELD_VOID, "Enable", InputEnable ),
DEFINE_INPUTFUNC( FIELD_VOID, "Disable", InputDisable ),
END_DATADESC()
IMPLEMENT_SERVERCLASS_ST_NOBASE(CCascadeLight, DT_CascadeLight)
SendPropVector(SENDINFO(m_shadowDirection), -1, SPROP_NOSCALE ),
SendPropVector(SENDINFO(m_envLightShadowDirection), -1, SPROP_NOSCALE ),
SendPropBool(SENDINFO(m_bEnabled) ),
SendPropBool(SENDINFO(m_bUseLightEnvAngles) ),
SendPropInt(SENDINFO(m_LightColor), 32, SPROP_UNSIGNED, SendProxy_Color32ToInt32 ),
SendPropInt(SENDINFO(m_LightColorScale), 32, 0, SendProxy_Int32ToInt32 ),
SendPropFloat(SENDINFO(m_flMaxShadowDist), 0, SPROP_NOSCALE ),
END_SEND_TABLE()
float CCascadeLight::m_flEnvLightShadowPitch;
QAngle CCascadeLight::m_EnvLightShadowAngles;
bool CCascadeLight::m_bEnvLightShadowValid;
color32 CCascadeLight::m_EnvLightColor;
int CCascadeLight::m_EnvLightColorScale;
CCascadeLight::CCascadeLight() :
CBaseEntity()
{
CsmDbgMsg( "CCascadeLight::CCascadeLight\n" );
m_bEnabled = true;
color32 tmp = { 255, 255, 255, 1 };
m_LightColor = tmp;
m_LightColorScale = 255;
QAngle angles;
angles.Init( 50, 43, 0 );
Vector vForward;
AngleVectors( angles, &vForward );
m_shadowDirection = vForward;
//m_shadowDirection.Init( 0.0f, 0.0f, -1.0f );
m_envLightShadowDirection = m_shadowDirection;
m_bUseLightEnvAngles = true;
m_flMaxShadowDist = 400.0f;
g_pCascadeLight = this;
}
CCascadeLight::~CCascadeLight()
{
g_pCascadeLight = NULL;
CsmDbgMsg( "CCascadeLight::~CCascadeLight\n" );
}
//------------------------------------------------------------------------------
// Purpose : Send even though we don't have a model
//------------------------------------------------------------------------------
int CCascadeLight::UpdateTransmitState()
{
// ALWAYS transmit to all clients.
return SetTransmitState( FL_EDICT_ALWAYS );
}
bool CCascadeLight::KeyValue( const char *szKeyName, const char *szValue )
{
if ( FStrEq( szKeyName, "color" ) )
{
/* unused?
float tmp[4];
UTIL_StringToFloatArray( tmp, 4, szValue );
m_LightColor.SetR( tmp[0] );
m_LightColor.SetG( tmp[1] );
m_LightColor.SetB( tmp[2] );
m_LightColor.SetA( tmp[3] );*/
}
else if ( FStrEq( szKeyName, "angles" ) )
{
QAngle angles;
UTIL_StringToVector( angles.Base(), szValue );
if (angles == vec3_angle)
{
angles.Init( 50, 43, 0 );
}
Vector vForward;
AngleVectors( angles, &vForward );
m_shadowDirection = vForward;
return true;
}
else if ( FStrEq( szKeyName, "uselightenvangles" ) )
{
m_bUseLightEnvAngles = ( atoi( szValue ) != 0 );
return true;
}
else if ( FStrEq( szKeyName, "maxshadowdistance" ) )
{
m_flMaxShadowDist = atof( szValue );
return true;
}
return BaseClass::KeyValue( szKeyName, szValue );
}
bool CCascadeLight::GetKeyValue( const char *szKeyName, char *szValue, int iMaxLen )
{
if ( FStrEq( szKeyName, "color" ) )
{
// path unused?
Q_snprintf( szValue, iMaxLen, "%d %d %d %d", m_LightColor.GetR(), m_LightColor.GetG(), m_LightColor.GetB(), m_LightColor.GetA() );
return true;
}
return BaseClass::GetKeyValue( szKeyName, szValue, iMaxLen );
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CCascadeLight::Spawn( void )
{
Precache();
SetSolid( SOLID_NONE );
if( m_bStartDisabled )
{
m_bEnabled = false;
}
else
{
m_bEnabled = true;
}
if ( m_bEnvLightShadowValid )
{
UpdateEnvLight();
}
//SetClassname( "cascadelight" );
BaseClass::Spawn();
}
void CCascadeLight::Release( void )
{
g_pCascadeLight = NULL;
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CCascadeLight::OnActivate()
{
}
//------------------------------------------------------------------------------
// Purpose :
//------------------------------------------------------------------------------
void CCascadeLight::OnDeactivate()
{
}
//------------------------------------------------------------------------------
// Input values
//------------------------------------------------------------------------------
void CCascadeLight::InputSetAngles( inputdata_t &inputdata )
{
const char *pAngles = inputdata.value.String();
QAngle angles;
UTIL_StringToVector( angles.Base(), pAngles );
Vector vTemp;
AngleVectors( angles, &vTemp );
m_shadowDirection = vTemp;
}
//------------------------------------------------------------------------------
// Purpose : Input handlers
//------------------------------------------------------------------------------
void CCascadeLight::InputEnable( inputdata_t &inputdata )
{
m_bEnabled = true;
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
void CCascadeLight::InputDisable( inputdata_t &inputdata )
{
m_bEnabled = false;
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
void CCascadeLight::InputSetLightColor( inputdata_t &inputdata )
{
m_LightColor = inputdata.value.Color32();
}
void CCascadeLight::InputSetLightColorScale( inputdata_t &inputdata )
{
m_LightColorScale = inputdata.value.Int();
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
void CCascadeLight::SetLightColor( int r, int g, int b, int a )
{
m_EnvLightColor.r = r;
m_EnvLightColor.g = g;
m_EnvLightColor.b = b;
m_EnvLightColor.a = 0; // use light scale as potentially > 255
m_EnvLightColorScale = a;
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
void CCascadeLight::SetEnabled( bool bEnable )
{
m_bEnabled = bEnable;
}
void CCascadeLight::UpdateEnvLight()
{
QAngle angles;
angles.x = -m_flEnvLightShadowPitch;
angles.y = m_EnvLightShadowAngles.y;
angles.z = 0;
Vector vForward;
AngleVectors( angles, &vForward );
m_envLightShadowDirection = vForward;
m_LightColor = m_EnvLightColor;
m_LightColorScale = m_EnvLightColorScale;
}
void CCascadeLight::SetEnvLightShadowPitch( float flPitch )
{
m_flEnvLightShadowPitch = flPitch;
m_bEnvLightShadowValid = true;
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
void CCascadeLight::SetEnvLightShadowAngles( const QAngle &angles )
{
m_EnvLightShadowAngles = angles;
m_bEnvLightShadowValid = true;
if ( g_pCascadeLight )
{
g_pCascadeLight->UpdateEnvLight();
}
}
class CCSMLightManager : public CAutoGameSystemPerFrame
{
public:
CCSMLightManager()
{
}
virtual ~CCSMLightManager()
{
}
virtual void LevelInitPreEntity()
{
CsmDbgMsg( "**** LevelInitPreEntity\n" );
}
virtual void LevelInitPostEntity()
{
CsmDbgMsg( "**** LevelInitPostEntity\n" );
if ( !cl_csm_auto_entity.GetBool() )
return;
if ( g_pCascadeLight )
return;
// Create the env_cascade_light automatically for cs:go - this is a hack that will hopefully go away as we add the entity to all of our maps.
CBaseEntity *entity = dynamic_cast< CBaseEntity * >( CreateEntityByName( "env_cascade_light" ) );
if (entity)
{
entity->Precache();
entity->KeyValue( "targetname", "cascadelight" );
DispatchSpawn(entity);
}
}
virtual void LevelShutdownPreEntity()
{
CsmDbgMsg( "**** LevelShutdownPreEntity\n" );
}
virtual void LevelShutdownPostEntity()
{
CsmDbgMsg( "**** LevelShutdownPostEntity\n" );
}
virtual void Shutdown()
{
CsmDbgMsg( "**** Shutdown\n" );
}
};
CCSMLightManager g_CSMLightManager;
void C_CSM_Server_Status( const CCommand& args )
{
Msg( "Entity exists: %u\n", g_pCascadeLight != NULL );
}
static ConCommand cl_csm_server_status("cl_csm_server_status", C_CSM_Server_Status, "Usage:\n cl_csm_server_status\n", 0);