source-engine/game/client/tf2/c_gasoline_blob.cpp

351 lines
8.4 KiB
C++
Raw Normal View History

2020-04-22 16:56:21 +00:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "c_gasoline_blob.h"
#include "gasoline_shared.h"
#include "engine/IEngineSound.h"
#include "clienteffectprecachesystem.h"
static CUtlLinkedList<C_GasolineBlob*, int> g_GasolineBlobs;
// If multiple blobs are within this distance to each other, then only one will
// play a sound.
#define BLOB_SOUND_RELATED_DISTANCE 600
#define PUDDLE_START_SIZE 35
#define PUDDLE_END_SIZE 65
#define PUDDLE_GROW_TIME 0.5
#define PUDDLE_FADE_TIME 1.0
CLIENTEFFECT_REGISTER_BEGIN( PrecacheGasolineBlob )
CLIENTEFFECT_MATERIAL( "decals/puddle" )
CLIENTEFFECT_REGISTER_END()
// ------------------------------------------------------------------------------------------------ //
// CGasolineEmitter.
// ------------------------------------------------------------------------------------------------ //
CSmartPtr<CGasolineEmitter> CGasolineEmitter::Create( C_GasolineBlob *pBlob )
{
CGasolineEmitter *pEmitter = new CGasolineEmitter;
pEmitter->m_pBlob = pBlob;
pEmitter->m_hFireMaterial = pEmitter->GetPMaterial( "particle/fire" );
pEmitter->m_hUnlitMaterial = pEmitter->GetPMaterial( "sprites/env_particles" );
pEmitter->m_Timer.Init( 40 );
return pEmitter;
}
void CGasolineEmitter::UpdateFire( float frametime )
{
float flLifetime = gpGlobals->curtime - m_pBlob->m_flCreateTime;
float litPercent = 1;
if ( m_pBlob->IsLit() )
{
litPercent = 1 - (flLifetime / m_pBlob->m_flMaxLifetime);
if ( litPercent <= 0 )
return;
}
else
{
return;
}
// Don't show a burn effect for a blob that hasn't hit anything yet.
// If you do, it tends to make the flamethrower effect look weird.
if ( !m_pBlob->IsStopped() )
return;
// Make a coordinate system in which to spawn the particles. It
Vector vUp, vRight;
vUp.Init();
vRight.Init();
if ( m_pBlob->IsStopped() )
{
QAngle angles;
VectorAngles( m_pBlob->GetSurfaceNormal(), angles );
AngleVectors( angles, NULL, &vRight, &vUp );
}
PMaterialHandle hMaterial = m_hFireMaterial;
float flParticleLifetime = 1;
float flRadius = 7;
unsigned char uchColor[4] = { 255, 128, 0, 128 };
float flMaxZVel = 29;
float curDelta = frametime;
while ( m_Timer.NextEvent( curDelta ) )
{
// Based on how close we are to expiring, show less particles.
if ( RandomFloat( 0, 1 ) > litPercent )
continue;
Vector vPos = m_pBlob->GetAbsOrigin();
if ( m_pBlob->IsStopped() )
{
float flAngle = RandomFloat( 0, M_PI * 2 );
float flDist = RandomFloat( 0, GASOLINE_BLOB_RADIUS );
vPos += vRight * (cos( flAngle ) * flDist);
vPos += vUp * ( sin( flAngle ) * flDist );
}
else
{
vPos += RandomVector( -GASOLINE_BLOB_RADIUS, GASOLINE_BLOB_RADIUS );
}
SimpleParticle *pParticle = AddSimpleParticle( hMaterial, vPos, flParticleLifetime, flRadius );
if ( pParticle )
{
pParticle->m_uchColor[0] = uchColor[0];
pParticle->m_uchColor[1] = uchColor[1];
pParticle->m_uchColor[2] = uchColor[2];
pParticle->m_uchEndAlpha = 0;
pParticle->m_uchStartAlpha = uchColor[3];
pParticle->m_vecVelocity.x = RandomFloat( -2, 2 );
pParticle->m_vecVelocity.y = RandomFloat( -2, 2 );
pParticle->m_vecVelocity.z = RandomFloat( 3, flMaxZVel );
}
}
}
// ------------------------------------------------------------------------------------------------ //
// C_GasolineBlob.
// ------------------------------------------------------------------------------------------------ //
IMPLEMENT_CLIENTCLASS_DT_NOBASE( C_GasolineBlob, DT_GasolineBlob, CGasolineBlob )
RecvPropInt( RECVINFO( m_BlobFlags ) ),
RecvPropVector( RECVINFO_NAME( m_vecNetworkOrigin, m_vecOrigin ) ),
RecvPropInt( RECVINFO_NAME(m_hNetworkMoveParent, moveparent), 0, RecvProxy_IntToMoveParent ),
RecvPropFloat( RECVINFO( m_flLitStartTime ) ),
RecvPropFloat( RECVINFO( m_flCreateTime ) ),
RecvPropFloat( RECVINFO( m_flMaxLifetime ) ),
RecvPropInt( RECVINFO( m_iTeamNum ) ),
RecvPropVector( RECVINFO( m_vSurfaceNormal ) )
END_RECV_TABLE()
C_GasolineBlob::C_GasolineBlob()
{
m_pEmitter = CGasolineEmitter::Create( this );
m_vSurfaceNormal.Init();
m_flLitStartTime = 0;
m_bSoundOn = false;
g_GasolineBlobs.AddToTail( this );
m_flPuddleSize = PUDDLE_START_SIZE;
m_flPuddleFade = 1;
}
C_GasolineBlob::~C_GasolineBlob()
{
g_GasolineBlobs.FindAndRemove( this );
StopSound();
// If a bunch of nearby blobs weren't playing a sound because we were, have them start their sound now.
FOR_EACH_LL( g_GasolineBlobs, i )
{
C_GasolineBlob *pBlob = g_GasolineBlobs[i];
if ( pBlob->IsSoundRelatedTo( this ) )
pBlob->CheckStartSound();
}
}
bool C_GasolineBlob::IsLit() const
{
return (m_BlobFlags & BLOBFLAG_LIT) != 0;
}
bool C_GasolineBlob::IsStopped() const
{
return (m_BlobFlags & BLOBFLAG_STOPPED) != 0;
}
const Vector& C_GasolineBlob::GetSurfaceNormal() const
{
return m_vSurfaceNormal;
}
float C_GasolineBlob::GetLitStartTime() const
{
return m_flLitStartTime;
}
void C_GasolineBlob::OnDataChanged( DataUpdateType_t type )
{
BaseClass::OnDataChanged( type );
if ( type == DATA_UPDATE_CREATED )
{
SetNextClientThink( CLIENT_THINK_ALWAYS );
}
CheckStartSound();
}
void C_GasolineBlob::ClientThink()
{
if ( m_pEmitter.IsValid() )
m_pEmitter->UpdateFire( gpGlobals->frametime );
// Grow the puddle a little.
if ( IsStopped() )
{
if ( IsLit() )
{
// Fade out after we get lit.
m_flPuddleFade -= gpGlobals->frametime / PUDDLE_FADE_TIME;
m_flPuddleFade = MAX( m_flPuddleFade, 0 );
}
else
{
// Grow the puddle until it's at its max size.
m_flPuddleSize += gpGlobals->frametime * ( PUDDLE_END_SIZE - PUDDLE_START_SIZE ) / PUDDLE_GROW_TIME;
m_flPuddleSize = MIN( m_flPuddleSize, PUDDLE_END_SIZE );
}
}
}
bool C_GasolineBlob::ShouldDraw()
{
return IsStopped() && (m_flPuddleFade > 0);
}
int C_GasolineBlob::DrawModel( int flags )
{
// Generate a basis.
QAngle angles;
VectorAngles( m_vSurfaceNormal, angles );
Vector vRight, vUp;
AngleVectors( angles, NULL, &vRight, &vUp );
float flAlpha = m_flPuddleFade * RemapVal( m_flPuddleSize, PUDDLE_START_SIZE, PUDDLE_END_SIZE, 0, 1 );
if ( flAlpha <= 0 )
return 0;
// Draw the puddle.
IMaterial *pMat = materials->FindMaterial( "decals/puddle", TEXTURE_GROUP_DECAL );
IMesh *pMesh = materials->GetDynamicMesh( true, NULL, NULL, pMat );
CMeshBuilder mb;
mb.Begin( pMesh, MATERIAL_QUADS, 1 );
Vector v;
v = GetAbsOrigin() + m_vSurfaceNormal + vRight*m_flPuddleSize - vUp*m_flPuddleSize;
mb.Position3f( v.x, v.y, v.z );
mb.Color4f( 1, 1, 1, flAlpha );
mb.Normal3f( VectorExpand( m_vSurfaceNormal ) );
mb.TexCoord2f( 0, 1, 0 );
mb.AdvanceVertex();
v = GetAbsOrigin() + m_vSurfaceNormal + vRight*m_flPuddleSize + vUp*m_flPuddleSize;
mb.Position3f( v.x, v.y, v.z );
mb.Color4f( 1, 1, 1, flAlpha );
mb.Normal3f( VectorExpand( m_vSurfaceNormal ) );
mb.TexCoord2f( 0, 1, 1 );
mb.AdvanceVertex();
v = GetAbsOrigin() + m_vSurfaceNormal - vRight*m_flPuddleSize + vUp*m_flPuddleSize;
mb.Position3f( v.x, v.y, v.z );
mb.Color4f( 1, 1, 1, flAlpha );
mb.Normal3f( VectorExpand( m_vSurfaceNormal ) );
mb.TexCoord2f( 0, 0, 1 );
mb.AdvanceVertex();
v = GetAbsOrigin() + m_vSurfaceNormal - vRight*m_flPuddleSize - vUp*m_flPuddleSize;
mb.Position3f( v.x, v.y, v.z );
mb.Color4f( 1, 1, 1, flAlpha );
mb.Normal3f( VectorExpand( m_vSurfaceNormal ) );
mb.TexCoord2f( 0, 0, 0 );
mb.AdvanceVertex();
mb.End( false, true );
return 0;
}
bool C_GasolineBlob::IsSoundRelatedTo( const C_GasolineBlob *pBlob ) const
{
return pBlob->GetAbsOrigin().DistTo( GetAbsOrigin() ) < BLOB_SOUND_RELATED_DISTANCE;
}
bool C_GasolineBlob::IsPlayingBurningSound() const
{
return m_bSoundOn;
}
void C_GasolineBlob::CheckStartSound()
{
if ( IsPlayingBurningSound() || (m_BlobFlags & BLOBFLAG_STOPPED) == 0 || !IsLit() )
return;
// First, make sure no nearby blob is playing the sound.
FOR_EACH_LL( g_GasolineBlobs, i )
{
C_GasolineBlob *pBlob = g_GasolineBlobs[i];
if ( pBlob != this && pBlob->IsSoundRelatedTo( this ) )
{
// If it's already playing a sound, then don't start our sound.
if ( pBlob->IsPlayingBurningSound() )
return;
}
}
StartSound();
}
void C_GasolineBlob::StartSound()
{
if ( !m_bSoundOn )
{
EmitSound( "GasolineBlob.FlameSound" );
m_bSoundOn = true;
}
}
void C_GasolineBlob::StopSound()
{
if ( m_bSoundOn )
{
BaseClass::StopSound( "GasolineBlob.FlameSound" );
m_bSoundOn = false;
}
}