214 lines
5.8 KiB
C++
214 lines
5.8 KiB
C++
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose: Game-specific impact effect hooks
|
||
|
//
|
||
|
//=============================================================================//
|
||
|
#include "cbase.h"
|
||
|
|
||
|
#include "c_te_effect_dispatch.h"
|
||
|
#include "tempent.h"
|
||
|
#include "c_te_legacytempents.h"
|
||
|
#include "tf_shareddefs.h"
|
||
|
#include "c_rope.h"
|
||
|
|
||
|
// NOTE: Always include this last!
|
||
|
#include "tier0/memdbgon.h"
|
||
|
|
||
|
|
||
|
// Used for cycling so that they're sequencially ordered on each strand
|
||
|
int g_nHolidayLightColor = 0;
|
||
|
|
||
|
// 4 light colors
|
||
|
Color g_rgbaHolidayRed( 255, 0, 0, 255 );
|
||
|
Color g_rgbaHolidayYellow( 2, 110, 197, 255 );
|
||
|
Color g_rgbaHolidayGreen( 117, 193, 8, 255 );
|
||
|
Color g_rgbaHolidayBlue( 255, 151, 29, 255 );
|
||
|
|
||
|
Color *(rgbaHolidayLightColors[]) = { &g_rgbaHolidayRed,
|
||
|
&g_rgbaHolidayYellow,
|
||
|
&g_rgbaHolidayGreen,
|
||
|
&g_rgbaHolidayBlue };
|
||
|
|
||
|
|
||
|
struct HolidayLightData_t
|
||
|
{
|
||
|
Vector vOrigin;
|
||
|
int nID;
|
||
|
int nSubID;
|
||
|
float fScale;
|
||
|
};
|
||
|
|
||
|
|
||
|
void CreateHolidayLight( const HolidayLightData_t &holidayLight );
|
||
|
|
||
|
|
||
|
class CHolidayLightManager : public CAutoGameSystemPerFrame
|
||
|
{
|
||
|
public:
|
||
|
CHolidayLightManager( char const *name );
|
||
|
|
||
|
// Methods of IGameSystem
|
||
|
virtual void Update( float frametime );
|
||
|
|
||
|
virtual void LevelInitPostEntity( void );
|
||
|
virtual void LevelShutdownPreEntity();
|
||
|
|
||
|
void AddHolidayLight( const CEffectData &data );
|
||
|
|
||
|
private:
|
||
|
|
||
|
CUtlVector< HolidayLightData_t > m_PendingLightData;
|
||
|
};
|
||
|
|
||
|
CHolidayLightManager g_CHolidayLightManager( "CHolidayLightManager" );
|
||
|
|
||
|
|
||
|
CHolidayLightManager::CHolidayLightManager( char const *name ) : CAutoGameSystemPerFrame( name )
|
||
|
{
|
||
|
}
|
||
|
|
||
|
// Methods of IGameSystem
|
||
|
void CHolidayLightManager::Update( float frametime )
|
||
|
{
|
||
|
for ( int i = 0; i < m_PendingLightData.Count(); ++i )
|
||
|
{
|
||
|
CreateHolidayLight( m_PendingLightData[ i ] );
|
||
|
}
|
||
|
|
||
|
m_PendingLightData.RemoveAll();
|
||
|
}
|
||
|
|
||
|
void CHolidayLightManager::LevelInitPostEntity( void )
|
||
|
{
|
||
|
m_PendingLightData.RemoveAll();
|
||
|
}
|
||
|
|
||
|
void CHolidayLightManager::LevelShutdownPreEntity()
|
||
|
{
|
||
|
m_PendingLightData.RemoveAll();
|
||
|
}
|
||
|
|
||
|
void CHolidayLightManager::AddHolidayLight( const CEffectData &data )
|
||
|
{
|
||
|
C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
|
||
|
if ( !pPlayer )
|
||
|
return;
|
||
|
|
||
|
// Too far away?
|
||
|
if ( pPlayer->GetAbsOrigin().DistTo( data.m_vOrigin ) > 2000.0f )
|
||
|
return;
|
||
|
|
||
|
// In the skybox?
|
||
|
sky3dparams_t *pSky = &(pPlayer->m_Local.m_skybox3d);
|
||
|
if ( pSky->origin->DistTo( data.m_vOrigin ) < 2000.0f )
|
||
|
return;
|
||
|
|
||
|
HolidayLightData_t newData;
|
||
|
|
||
|
newData.vOrigin = data.m_vOrigin;
|
||
|
|
||
|
// HACK: Use these ints to ID the light later
|
||
|
newData.nID = data.m_nMaterial;
|
||
|
newData.nSubID = data.m_nHitBox;
|
||
|
|
||
|
// Skybox lights pass in a smaller scale
|
||
|
newData.fScale = data.m_flScale;
|
||
|
|
||
|
if ( m_PendingLightData.Count() < CTempEnts::MAX_TEMP_ENTITIES / 2 )
|
||
|
{
|
||
|
m_PendingLightData.AddToTail( newData );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void TF_HolidayLightCallback( const CEffectData &data )
|
||
|
{
|
||
|
g_CHolidayLightManager.AddHolidayLight( data );
|
||
|
}
|
||
|
|
||
|
void CreateHolidayLight( const HolidayLightData_t &holidayLight )
|
||
|
{
|
||
|
int nHolidayLightStyle = RopeManager()->GetHolidayLightStyle();
|
||
|
|
||
|
const model_t *pModel = ( nHolidayLightStyle == 0 ? engine->LoadModel( "effects/christmas_bulb.vmt" ) : engine->LoadModel( "effects/mtp_fluff.vmt" ) );
|
||
|
if ( !pModel )
|
||
|
return;
|
||
|
|
||
|
Assert( pModel );
|
||
|
|
||
|
C_LocalTempEntity *pTemp = tempents->FindTempEntByID( holidayLight.nID, holidayLight.nSubID );
|
||
|
|
||
|
if ( !pTemp )
|
||
|
{
|
||
|
// Didn't find one with that ID, so make a new one!
|
||
|
// Randomize the angle
|
||
|
QAngle angOrientation = ( nHolidayLightStyle == 0 ? QAngle( 0.0f, 0.0f, RandomFloat( -180.0f, 180.0f ) ) : vec3_angle );
|
||
|
pTemp = tempents->SpawnTempModel( pModel, holidayLight.vOrigin, angOrientation, vec3_origin, 2.0f, FTENT_NEVERDIE );
|
||
|
if ( !pTemp )
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
pTemp->clientIndex = 0;
|
||
|
|
||
|
// HACK: Use these ints to ID the light later
|
||
|
pTemp->m_nSkin = holidayLight.nID;
|
||
|
pTemp->hitSound = holidayLight.nSubID;
|
||
|
|
||
|
// Skybox lights pass in a smaller scale
|
||
|
pTemp->m_flSpriteScale = holidayLight.fScale;
|
||
|
|
||
|
// Smuggle the color index here
|
||
|
pTemp->m_nHitboxSet = g_nHolidayLightColor;
|
||
|
|
||
|
// Set the color
|
||
|
pTemp->SetRenderColor( rgbaHolidayLightColors[ g_nHolidayLightColor ]->r(),
|
||
|
rgbaHolidayLightColors[ g_nHolidayLightColor ]->g(),
|
||
|
rgbaHolidayLightColors[ g_nHolidayLightColor ]->b(),
|
||
|
rgbaHolidayLightColors[ g_nHolidayLightColor ]->a() );
|
||
|
|
||
|
// Next color in the pattern
|
||
|
g_nHolidayLightColor = ( g_nHolidayLightColor + 1 ) % ARRAYSIZE( rgbaHolidayLightColors );
|
||
|
|
||
|
// Animate if needed
|
||
|
pTemp->m_flFrameMax = modelinfo->GetModelFrameCount( pModel ) - 1;
|
||
|
pTemp->flags = ( FTENT_SPRANIMATE | FTENT_SPRANIMATELOOP );
|
||
|
pTemp->m_flFrameRate = 10;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Update the position
|
||
|
pTemp->SetAbsOrigin( holidayLight.vOrigin );
|
||
|
|
||
|
// Every 10 light strands have a blink cycle
|
||
|
if ( pTemp->m_nSkin % 5 == 0 )
|
||
|
{
|
||
|
// Magic! Basically this makes the on/off cycle of each color different and offsets it by the segment index.
|
||
|
// That way it looks like a timed pattern but is also chaotic.
|
||
|
int nCycle = ( pTemp->hitSound + static_cast< int >( gpGlobals->curtime * 2.0f ) ) % ( pTemp->m_nHitboxSet + ARRAYSIZE( rgbaHolidayLightColors ) + 1 );
|
||
|
pTemp->SetRenderColorA( nCycle < ARRAYSIZE( rgbaHolidayLightColors ) ? 255 : 64 );
|
||
|
}
|
||
|
|
||
|
// Update the scale
|
||
|
pTemp->m_flSpriteScale = holidayLight.fScale;
|
||
|
|
||
|
// Extend it's life
|
||
|
pTemp->die = gpGlobals->curtime + 2.0f;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DECLARE_CLIENT_EFFECT( "TF_HolidayLight", TF_HolidayLightCallback );
|
||
|
|
||
|
void RopesHolidayLightColor( const CCommand &args )
|
||
|
{
|
||
|
if ( args.ArgC() < 5 )
|
||
|
return;
|
||
|
|
||
|
int nLight = atoi( args[ 1 ] );
|
||
|
|
||
|
if ( nLight < 0 || nLight >= ARRAYSIZE( rgbaHolidayLightColors ) )
|
||
|
return;
|
||
|
|
||
|
rgbaHolidayLightColors[ nLight ]->SetColor( atoi( args[ 2 ] ), atoi( args[ 3 ] ), atoi( args[ 4 ] ) );
|
||
|
}
|
||
|
ConCommand r_ropes_holiday_light_color( "r_ropes_holiday_light_color", RopesHolidayLightColor, "Set each light's color: [light0-3] [r0-255] [g0-255] [b0-255]", FCVAR_NONE );
|