mirror of
https://github.com/mapbase-source/source-sdk-2013.git
synced 2025-01-07 20:55:31 +03:00
348 lines
12 KiB
C++
348 lines
12 KiB
C++
//========= Copyright 1996-2005, Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Sunlight shadow control entity.
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
#include "cbase.h"
|
|
|
|
#include "c_baseplayer.h"
|
|
#include "tier0/vprof.h"
|
|
#ifdef MAPBASE
|
|
#include "materialsystem/itexture.h"
|
|
#endif
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
extern ConVar cl_sunlight_ortho_size;
|
|
extern ConVar cl_sunlight_depthbias;
|
|
|
|
ConVar cl_globallight_freeze( "cl_globallight_freeze", "0" );
|
|
#ifdef MAPBASE
|
|
// I imagine these values might've been designed for the ASW view.
|
|
// You can set these as KV anyway.
|
|
ConVar cl_globallight_xoffset( "cl_globallight_xoffset", "0" );
|
|
ConVar cl_globallight_yoffset( "cl_globallight_yoffset", "0" );
|
|
|
|
static ConVar cl_globallight_slopescaledepthbias_shadowmap( "cl_globallight_slopescaledepthbias_shadowmap", "16", FCVAR_CHEAT );
|
|
static ConVar cl_globallight_shadowfiltersize( "cl_globallight_shadowfiltersize", "0.1", FCVAR_CHEAT );
|
|
static ConVar cl_globallight_depthbias_shadowmap( "cl_globallight_depthbias_shadowmap", "0.00001", FCVAR_CHEAT );
|
|
static ConVar cl_globallight_depthres( "cl_globallight_depthres", "8192", FCVAR_CHEAT );
|
|
#else
|
|
ConVar cl_globallight_xoffset( "cl_globallight_xoffset", "-800" );
|
|
ConVar cl_globallight_yoffset( "cl_globallight_yoffset", "1600" );
|
|
#endif
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Purpose : Sunlights shadow control entity
|
|
//------------------------------------------------------------------------------
|
|
class C_GlobalLight : public C_BaseEntity
|
|
{
|
|
public:
|
|
DECLARE_CLASS( C_GlobalLight, C_BaseEntity );
|
|
|
|
DECLARE_CLIENTCLASS();
|
|
|
|
virtual ~C_GlobalLight();
|
|
|
|
void OnDataChanged( DataUpdateType_t updateType );
|
|
void Spawn();
|
|
bool ShouldDraw();
|
|
|
|
void ClientThink();
|
|
|
|
private:
|
|
Vector m_shadowDirection;
|
|
bool m_bEnabled;
|
|
char m_TextureName[ MAX_PATH ];
|
|
#ifdef MAPBASE
|
|
int m_nSpotlightTextureFrame;
|
|
#endif
|
|
CTextureReference m_SpotlightTexture;
|
|
color32 m_LightColor;
|
|
#ifdef MAPBASE
|
|
float m_flBrightnessScale;
|
|
float m_flCurrentBrightnessScale;
|
|
#endif
|
|
Vector m_CurrentLinearFloatLightColor;
|
|
float m_flCurrentLinearFloatLightAlpha;
|
|
float m_flColorTransitionTime;
|
|
float m_flSunDistance;
|
|
float m_flFOV;
|
|
float m_flNearZ;
|
|
float m_flNorthOffset;
|
|
#ifdef MAPBASE
|
|
float m_flEastOffset;
|
|
float m_flForwardOffset;
|
|
float m_flOrthoSize;
|
|
#endif
|
|
bool m_bEnableShadows;
|
|
bool m_bOldEnableShadows;
|
|
|
|
static ClientShadowHandle_t m_LocalFlashlightHandle;
|
|
};
|
|
|
|
|
|
ClientShadowHandle_t C_GlobalLight::m_LocalFlashlightHandle = CLIENTSHADOW_INVALID_HANDLE;
|
|
|
|
|
|
IMPLEMENT_CLIENTCLASS_DT(C_GlobalLight, DT_GlobalLight, CGlobalLight)
|
|
RecvPropVector(RECVINFO(m_shadowDirection)),
|
|
RecvPropBool(RECVINFO(m_bEnabled)),
|
|
RecvPropString(RECVINFO(m_TextureName)),
|
|
#ifdef MAPBASE
|
|
RecvPropInt(RECVINFO(m_nSpotlightTextureFrame)),
|
|
#endif
|
|
/*RecvPropInt(RECVINFO(m_LightColor), 0, RecvProxy_Int32ToColor32),*/
|
|
RecvPropInt(RECVINFO(m_LightColor), 0, RecvProxy_IntToColor32),
|
|
#ifdef MAPBASE
|
|
RecvPropFloat(RECVINFO(m_flBrightnessScale)),
|
|
#endif
|
|
RecvPropFloat(RECVINFO(m_flColorTransitionTime)),
|
|
RecvPropFloat(RECVINFO(m_flSunDistance)),
|
|
RecvPropFloat(RECVINFO(m_flFOV)),
|
|
RecvPropFloat(RECVINFO(m_flNearZ)),
|
|
RecvPropFloat(RECVINFO(m_flNorthOffset)),
|
|
#ifdef MAPBASE
|
|
RecvPropFloat(RECVINFO(m_flEastOffset)),
|
|
RecvPropFloat(RECVINFO(m_flForwardOffset)),
|
|
RecvPropFloat(RECVINFO(m_flOrthoSize)),
|
|
#endif
|
|
RecvPropBool(RECVINFO(m_bEnableShadows)),
|
|
END_RECV_TABLE()
|
|
|
|
|
|
C_GlobalLight::~C_GlobalLight()
|
|
{
|
|
if ( m_LocalFlashlightHandle != CLIENTSHADOW_INVALID_HANDLE )
|
|
{
|
|
g_pClientShadowMgr->DestroyFlashlight( m_LocalFlashlightHandle );
|
|
m_LocalFlashlightHandle = CLIENTSHADOW_INVALID_HANDLE;
|
|
}
|
|
}
|
|
|
|
void C_GlobalLight::OnDataChanged( DataUpdateType_t updateType )
|
|
{
|
|
if ( updateType == DATA_UPDATE_CREATED )
|
|
{
|
|
m_SpotlightTexture.Init( m_TextureName, TEXTURE_GROUP_OTHER, true );
|
|
}
|
|
#ifdef MAPBASE
|
|
else //if ( updateType == DATA_UPDATE_DATATABLE_CHANGED )
|
|
{
|
|
// It could've been changed via input
|
|
if( !FStrEq(m_SpotlightTexture->GetName(), m_TextureName) )
|
|
{
|
|
m_SpotlightTexture.Init( m_TextureName, TEXTURE_GROUP_OTHER, true );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
BaseClass::OnDataChanged( updateType );
|
|
}
|
|
|
|
void C_GlobalLight::Spawn()
|
|
{
|
|
BaseClass::Spawn();
|
|
|
|
m_bOldEnableShadows = m_bEnableShadows;
|
|
|
|
SetNextClientThink( CLIENT_THINK_ALWAYS );
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// We don't draw...
|
|
//------------------------------------------------------------------------------
|
|
bool C_GlobalLight::ShouldDraw()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
void C_GlobalLight::ClientThink()
|
|
{
|
|
VPROF("C_GlobalLight::ClientThink");
|
|
|
|
bool bSupressWorldLights = false;
|
|
|
|
if ( cl_globallight_freeze.GetBool() == true )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( m_bEnabled )
|
|
{
|
|
Vector vLinearFloatLightColor( m_LightColor.r, m_LightColor.g, m_LightColor.b );
|
|
float flLinearFloatLightAlpha = m_LightColor.a;
|
|
|
|
#ifdef MAPBASE
|
|
if ( m_CurrentLinearFloatLightColor != vLinearFloatLightColor || m_flCurrentLinearFloatLightAlpha != flLinearFloatLightAlpha || m_flCurrentBrightnessScale != m_flBrightnessScale )
|
|
{
|
|
if (m_flColorTransitionTime != 0.0f)
|
|
{
|
|
float flColorTransitionSpeed = gpGlobals->frametime * m_flColorTransitionTime * 255.0f;
|
|
|
|
m_CurrentLinearFloatLightColor.x = Approach( vLinearFloatLightColor.x, m_CurrentLinearFloatLightColor.x, flColorTransitionSpeed );
|
|
m_CurrentLinearFloatLightColor.y = Approach( vLinearFloatLightColor.y, m_CurrentLinearFloatLightColor.y, flColorTransitionSpeed );
|
|
m_CurrentLinearFloatLightColor.z = Approach( vLinearFloatLightColor.z, m_CurrentLinearFloatLightColor.z, flColorTransitionSpeed );
|
|
m_flCurrentLinearFloatLightAlpha = Approach( flLinearFloatLightAlpha, m_flCurrentLinearFloatLightAlpha, flColorTransitionSpeed );
|
|
m_flCurrentBrightnessScale = Approach( m_flBrightnessScale, m_flCurrentBrightnessScale, flColorTransitionSpeed );
|
|
}
|
|
else
|
|
{
|
|
// Just do it instantly
|
|
m_CurrentLinearFloatLightColor.x = vLinearFloatLightColor.x;
|
|
m_CurrentLinearFloatLightColor.y = vLinearFloatLightColor.y;
|
|
m_CurrentLinearFloatLightColor.z = vLinearFloatLightColor.z;
|
|
m_flCurrentLinearFloatLightAlpha = flLinearFloatLightAlpha;
|
|
m_flCurrentBrightnessScale = m_flBrightnessScale;
|
|
}
|
|
}
|
|
#else
|
|
if ( m_CurrentLinearFloatLightColor != vLinearFloatLightColor || m_flCurrentLinearFloatLightAlpha != flLinearFloatLightAlpha )
|
|
{
|
|
float flColorTransitionSpeed = gpGlobals->frametime * m_flColorTransitionTime * 255.0f;
|
|
|
|
m_CurrentLinearFloatLightColor.x = Approach( vLinearFloatLightColor.x, m_CurrentLinearFloatLightColor.x, flColorTransitionSpeed );
|
|
m_CurrentLinearFloatLightColor.y = Approach( vLinearFloatLightColor.y, m_CurrentLinearFloatLightColor.y, flColorTransitionSpeed );
|
|
m_CurrentLinearFloatLightColor.z = Approach( vLinearFloatLightColor.z, m_CurrentLinearFloatLightColor.z, flColorTransitionSpeed );
|
|
m_flCurrentLinearFloatLightAlpha = Approach( flLinearFloatLightAlpha, m_flCurrentLinearFloatLightAlpha, flColorTransitionSpeed );
|
|
}
|
|
#endif
|
|
|
|
FlashlightState_t state;
|
|
|
|
Vector vDirection = m_shadowDirection;
|
|
VectorNormalize( vDirection );
|
|
|
|
//Vector vViewUp = Vector( 0.0f, 1.0f, 0.0f );
|
|
Vector vSunDirection2D = vDirection;
|
|
vSunDirection2D.z = 0.0f;
|
|
|
|
/*HACK_GETLOCALPLAYER_GUARD( "C_GlobalLight::ClientThink" );*/
|
|
|
|
if ( !C_BasePlayer::GetLocalPlayer() )
|
|
return;
|
|
|
|
Vector vPos;
|
|
QAngle EyeAngles;
|
|
float flZNear, flZFar, flFov;
|
|
|
|
C_BasePlayer::GetLocalPlayer()->CalcView( vPos, EyeAngles, flZNear, flZFar, flFov );
|
|
// Vector vPos = C_BasePlayer::GetLocalPlayer()->GetAbsOrigin();
|
|
|
|
// vPos = Vector( 0.0f, 0.0f, 500.0f );
|
|
vPos = ( vPos + vSunDirection2D * m_flNorthOffset ) - vDirection * m_flSunDistance;
|
|
#ifdef MAPBASE
|
|
vPos += Vector( m_flEastOffset + cl_globallight_xoffset.GetFloat(), m_flForwardOffset + cl_globallight_yoffset.GetFloat(), 0.0f );
|
|
#else
|
|
vPos += Vector( cl_globallight_xoffset.GetFloat(), cl_globallight_yoffset.GetFloat(), 0.0f );
|
|
#endif
|
|
|
|
QAngle angAngles;
|
|
VectorAngles( vDirection, angAngles );
|
|
|
|
Vector vForward, vRight, vUp;
|
|
AngleVectors( angAngles, &vForward, &vRight, &vUp );
|
|
|
|
state.m_fHorizontalFOVDegrees = m_flFOV;
|
|
state.m_fVerticalFOVDegrees = m_flFOV;
|
|
|
|
state.m_vecLightOrigin = vPos;
|
|
BasisToQuaternion( vForward, vRight, vUp, state.m_quatOrientation );
|
|
|
|
state.m_fQuadraticAtten = 0.0f;
|
|
state.m_fLinearAtten = m_flSunDistance * 2.0f;
|
|
state.m_fConstantAtten = 0.0f;
|
|
state.m_FarZAtten = m_flSunDistance * 2.0f;
|
|
#ifdef MAPBASE
|
|
float flAlpha = m_flCurrentLinearFloatLightAlpha * (1.0f / 255.0f);
|
|
state.m_Color[0] = (m_CurrentLinearFloatLightColor.x * ( 1.0f / 255.0f ) * flAlpha) * m_flCurrentBrightnessScale;
|
|
state.m_Color[1] = (m_CurrentLinearFloatLightColor.y * ( 1.0f / 255.0f ) * flAlpha) * m_flCurrentBrightnessScale;
|
|
state.m_Color[2] = (m_CurrentLinearFloatLightColor.z * ( 1.0f / 255.0f ) * flAlpha) * m_flCurrentBrightnessScale;
|
|
#else
|
|
state.m_Color[0] = m_CurrentLinearFloatLightColor.x * ( 1.0f / 255.0f ) * m_flCurrentLinearFloatLightAlpha;
|
|
state.m_Color[1] = m_CurrentLinearFloatLightColor.y * ( 1.0f / 255.0f ) * m_flCurrentLinearFloatLightAlpha;
|
|
state.m_Color[2] = m_CurrentLinearFloatLightColor.z * ( 1.0f / 255.0f ) * m_flCurrentLinearFloatLightAlpha;
|
|
#endif
|
|
state.m_Color[3] = 0.0f; // fixme: need to make ambient work m_flAmbient;
|
|
state.m_NearZ = 4.0f;
|
|
state.m_FarZ = m_flSunDistance * 2.0f;
|
|
state.m_fBrightnessScale = 2.0f;
|
|
state.m_bGlobalLight = true;
|
|
|
|
#ifdef MAPBASE
|
|
float flOrthoSize = m_flOrthoSize;
|
|
#else
|
|
float flOrthoSize = 1000.0f;
|
|
#endif
|
|
|
|
if ( flOrthoSize > 0 )
|
|
{
|
|
state.m_bOrtho = true;
|
|
state.m_fOrthoLeft = -flOrthoSize;
|
|
state.m_fOrthoTop = -flOrthoSize;
|
|
state.m_fOrthoRight = flOrthoSize;
|
|
state.m_fOrthoBottom = flOrthoSize;
|
|
}
|
|
else
|
|
{
|
|
state.m_bOrtho = false;
|
|
}
|
|
|
|
#ifdef MAPBASE
|
|
//state.m_bDrawShadowFrustum = true; // Don't draw that huge debug thing
|
|
state.m_flShadowMapResolution = cl_globallight_depthres.GetFloat();
|
|
state.m_flShadowFilterSize = cl_globallight_shadowfiltersize.GetFloat();
|
|
state.m_flShadowSlopeScaleDepthBias = cl_globallight_slopescaledepthbias_shadowmap.GetFloat();
|
|
state.m_flShadowDepthBias = cl_globallight_depthbias_shadowmap.GetFloat();
|
|
state.m_bEnableShadows = m_bEnableShadows;
|
|
state.m_pSpotlightTexture = m_SpotlightTexture;
|
|
state.m_nSpotlightTextureFrame = m_nSpotlightTextureFrame;
|
|
#else
|
|
state.m_bDrawShadowFrustum = true;
|
|
/*state.m_flShadowSlopeScaleDepthBias = g_pMaterialSystemHardwareConfig->GetShadowSlopeScaleDepthBias();;
|
|
state.m_flShadowDepthBias = g_pMaterialSystemHardwareConfig->GetShadowDepthBias();*/
|
|
state.m_bEnableShadows = m_bEnableShadows;
|
|
state.m_pSpotlightTexture = m_SpotlightTexture;
|
|
state.m_nSpotlightTextureFrame = 0;
|
|
#endif
|
|
|
|
state.m_nShadowQuality = 1; // Allow entity to affect shadow quality
|
|
// state.m_bShadowHighRes = true;
|
|
|
|
if ( m_bOldEnableShadows != m_bEnableShadows )
|
|
{
|
|
// If they change the shadow enable/disable, we need to make a new handle
|
|
if ( m_LocalFlashlightHandle != CLIENTSHADOW_INVALID_HANDLE )
|
|
{
|
|
g_pClientShadowMgr->DestroyFlashlight( m_LocalFlashlightHandle );
|
|
m_LocalFlashlightHandle = CLIENTSHADOW_INVALID_HANDLE;
|
|
}
|
|
|
|
m_bOldEnableShadows = m_bEnableShadows;
|
|
}
|
|
|
|
if( m_LocalFlashlightHandle == CLIENTSHADOW_INVALID_HANDLE )
|
|
{
|
|
m_LocalFlashlightHandle = g_pClientShadowMgr->CreateFlashlight( state );
|
|
}
|
|
else
|
|
{
|
|
g_pClientShadowMgr->UpdateFlashlightState( m_LocalFlashlightHandle, state );
|
|
g_pClientShadowMgr->UpdateProjectedTexture( m_LocalFlashlightHandle, true );
|
|
}
|
|
|
|
bSupressWorldLights = m_bEnableShadows;
|
|
}
|
|
else if ( m_LocalFlashlightHandle != CLIENTSHADOW_INVALID_HANDLE )
|
|
{
|
|
g_pClientShadowMgr->DestroyFlashlight( m_LocalFlashlightHandle );
|
|
m_LocalFlashlightHandle = CLIENTSHADOW_INVALID_HANDLE;
|
|
}
|
|
|
|
g_pClientShadowMgr->SetShadowFromWorldLightsEnabled( !bSupressWorldLights );
|
|
|
|
BaseClass::ClientThink();
|
|
}
|