2008-09-15 01:07:45 -05:00
//===== Copyright <20> 1996-2005, Valve Corporation, All rights reserved. ======//
//
// Purpose:
//
//===========================================================================//
# include "cbase.h"
# include "soundscape.h"
# include "datamap.h"
# include "soundscape_system.h"
# include "triggers.h"
# include "saverestore_utlvector.h"
// memdbgon must be the last include file in a .cpp file!!!
# include "tier0/memdbgon.h"
ConVar soundscape_debug ( " soundscape_debug " , " 0 " , FCVAR_CHEAT , " When on, draws lines to all env_soundscape entities. Green lines show the active soundscape, red lines show soundscapes that aren't in range, and white lines show soundscapes that are in range, but not the active soundscape. " ) ;
// ----------------------------------------------------------------------------- //
// CEnvSoundscapeProxy stuff.
// ----------------------------------------------------------------------------- //
LINK_ENTITY_TO_CLASS ( env_soundscape_proxy , CEnvSoundscapeProxy ) ;
BEGIN_DATADESC ( CEnvSoundscapeProxy )
DEFINE_KEYFIELD ( m_MainSoundscapeName , FIELD_STRING , " MainSoundscapeName " )
END_DATADESC ( )
CEnvSoundscapeProxy : : CEnvSoundscapeProxy ( )
{
m_MainSoundscapeName = NULL_STRING ;
}
void CEnvSoundscapeProxy : : Activate ( )
{
if ( m_MainSoundscapeName ! = NULL_STRING )
{
CBaseEntity * pEntity = gEntList . FindEntityByName ( NULL , m_MainSoundscapeName ) ;
if ( pEntity )
{
m_hProxySoundscape = dynamic_cast < CEnvSoundscape * > ( pEntity ) ;
}
}
if ( m_hProxySoundscape )
{
// Copy the relevant parameters from our main soundscape.
m_soundscapeIndex = m_hProxySoundscape - > m_soundscapeIndex ;
2008-09-15 02:50:57 -05:00
for ( size_t i = 0 ; i < ARRAYSIZE ( m_positionNames ) ; i + + )
2008-09-15 01:07:45 -05:00
m_positionNames [ i ] = m_hProxySoundscape - > m_positionNames [ i ] ;
}
else
{
Warning ( " env_soundscape_proxy can't find target soundscape: '%s' \n " , STRING ( m_MainSoundscapeName ) ) ;
}
BaseClass : : Activate ( ) ;
}
// ----------------------------------------------------------------------------- //
// CEnvSoundscape stuff.
// ----------------------------------------------------------------------------- //
LINK_ENTITY_TO_CLASS ( env_soundscape , CEnvSoundscape ) ;
BEGIN_DATADESC ( CEnvSoundscape )
DEFINE_KEYFIELD ( m_flRadius , FIELD_FLOAT , " radius " ) ,
// don't save, recomputed on load
//DEFINE_FIELD( m_soundscapeIndex, FIELD_INTEGER ),
DEFINE_FIELD ( m_soundscapeName , FIELD_STRING ) ,
DEFINE_FIELD ( m_hProxySoundscape , FIELD_EHANDLE ) ,
// Silence, Classcheck!
// DEFINE_ARRAY( m_positionNames, FIELD_STRING, 4 ),
DEFINE_KEYFIELD ( m_positionNames [ 0 ] , FIELD_STRING , " position0 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 1 ] , FIELD_STRING , " position1 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 2 ] , FIELD_STRING , " position2 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 3 ] , FIELD_STRING , " position3 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 4 ] , FIELD_STRING , " position4 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 5 ] , FIELD_STRING , " position5 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 6 ] , FIELD_STRING , " position6 " ) ,
DEFINE_KEYFIELD ( m_positionNames [ 7 ] , FIELD_STRING , " position7 " ) ,
DEFINE_UTLVECTOR ( m_hPlayersInPVS , FIELD_EHANDLE ) ,
DEFINE_FIELD ( m_flNextUpdatePlayersInPVS , FIELD_TIME ) ,
DEFINE_KEYFIELD ( m_bDisabled , FIELD_BOOLEAN , " StartDisabled " ) ,
DEFINE_INPUTFUNC ( FIELD_VOID , " Enable " , InputEnable ) ,
DEFINE_INPUTFUNC ( FIELD_VOID , " Disable " , InputDisable ) ,
DEFINE_INPUTFUNC ( FIELD_VOID , " ToggleEnabled " , InputToggleEnabled ) ,
DEFINE_OUTPUT ( m_OnPlay , " OnPlay " ) ,
END_DATADESC ( )
CEnvSoundscape : : CEnvSoundscape ( )
{
m_soundscapeName = NULL_STRING ;
m_flNextUpdatePlayersInPVS = 0 ;
m_soundscapeIndex = - 1 ;
m_bDisabled = false ;
g_SoundscapeSystem . AddSoundscapeEntity ( this ) ;
}
CEnvSoundscape : : ~ CEnvSoundscape ( )
{
g_SoundscapeSystem . RemoveSoundscapeEntity ( this ) ;
}
void CEnvSoundscape : : InputEnable ( inputdata_t & inputdata )
{
if ( ! IsEnabled ( ) )
{
Enable ( ) ;
}
}
void CEnvSoundscape : : InputDisable ( inputdata_t & inputdata )
{
if ( IsEnabled ( ) )
{
Disable ( ) ;
}
}
void CEnvSoundscape : : InputToggleEnabled ( inputdata_t & inputdata )
{
if ( IsEnabled ( ) )
{
Disable ( ) ;
}
else
{
Enable ( ) ;
}
}
//-----------------------------------------------------------------------------
// Purpose: Returns whether the laser is currently active.
//-----------------------------------------------------------------------------
bool CEnvSoundscape : : IsEnabled ( void ) const
{
return ! m_bDisabled ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CEnvSoundscape : : Disable ( void )
{
m_bDisabled = true ;
// Reset if we are the currently active soundscape
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CEnvSoundscape : : Enable ( void )
{
m_bDisabled = false ;
// Force the player to recheck soundscapes
}
bool CEnvSoundscape : : KeyValue ( const char * szKeyName , const char * szValue )
{
if ( FStrEq ( szKeyName , " soundscape " ) )
{
m_soundscapeName = AllocPooledString ( szValue ) ;
}
else
return BaseClass : : KeyValue ( szKeyName , szValue ) ;
return true ;
}
// returns true if the given sound entity is in range
// and can see the given player entity (pTarget)
bool CEnvSoundscape : : InRangeOfPlayer ( CBasePlayer * pTarget )
{
Vector vecSpot1 = EarPosition ( ) ;
Vector vecSpot2 = pTarget - > EarPosition ( ) ;
// calc range from sound entity to player
Vector vecRange = vecSpot2 - vecSpot1 ;
float range = vecRange . Length ( ) ;
if ( m_flRadius > range | | m_flRadius = = - 1 )
{
trace_t tr ;
UTIL_TraceLine ( vecSpot1 , vecSpot2 , MASK_SOLID_BRUSHONLY | MASK_WATER , pTarget , COLLISION_GROUP_NONE , & tr ) ;
if ( tr . fraction = = 1 & & ! tr . startsolid )
{
return true ;
}
}
return false ;
}
int CEnvSoundscape : : UpdateTransmitState ( )
{
// Always transmit all soundscapes to the player.
return SetTransmitState ( FL_EDICT_ALWAYS ) ;
}
void CEnvSoundscape : : WriteAudioParamsTo ( audioparams_t & audio )
{
audio . ent . Set ( this ) ;
audio . soundscapeIndex = m_soundscapeIndex ;
audio . localBits = 0 ;
2008-09-15 02:50:57 -05:00
for ( int i = 0 ; i < ( int ) ARRAYSIZE ( m_positionNames ) ; i + + )
2008-09-15 01:07:45 -05:00
{
if ( m_positionNames [ i ] ! = NULL_STRING )
{
// We are a valid entity for a sound position
CBaseEntity * pEntity = gEntList . FindEntityByName ( NULL , m_positionNames [ i ] , this , this ) ;
if ( pEntity )
{
audio . localBits | = 1 < < i ;
audio . localSound . Set ( i , pEntity - > GetAbsOrigin ( ) ) ;
}
}
}
m_OnPlay . FireOutput ( this , this ) ;
}
bool CEnvSoundscape : : UpdatePlayersInPVS ( )
{
// Only update players in PVS every 2 seconds.
if ( gpGlobals - > curtime < m_flNextUpdatePlayersInPVS )
return false ;
m_flNextUpdatePlayersInPVS = gpGlobals - > curtime + 2.0f + RandomFloat ( - 0.3 , 0.3 ) ;
// Find the players in our PVS.
unsigned char myPVS [ 16 * 1024 ] ;
int pvsLen = engine - > GetPVSForCluster ( engine - > GetClusterForOrigin ( GetAbsOrigin ( ) ) , sizeof ( myPVS ) , myPVS ) ;
m_hPlayersInPVS . Purge ( ) ;
for ( int i = 1 ; i < = gpGlobals - > maxClients ; i + + )
{
CBasePlayer * pPlayer = UTIL_PlayerByIndex ( i ) ;
if ( ! pPlayer )
continue ;
Vector vecWorldMins , vecWorldMaxs ;
pPlayer - > CollisionProp ( ) - > WorldSpaceAABB ( & vecWorldMins , & vecWorldMaxs ) ;
if ( engine - > CheckBoxInPVS ( vecWorldMins , vecWorldMaxs , myPVS , pvsLen ) )
{
m_hPlayersInPVS . AddToTail ( pPlayer ) ;
}
}
return true ;
}
//
// A client that is visible and in range of a sound entity will
// have its soundscape set by that sound entity. If two or more
// sound entities are contending for a client, then the nearest
// sound entity to the client will set the client's soundscape.
// A client's soundscape will remain set to its prior value until
// a new in-range, visible sound entity resets a new soundscape.
//
// CONSIDER: if player in water state, autoset and underwater soundscape?
void CEnvSoundscape : : Update ( )
{
bool bUpdated = UpdatePlayersInPVS ( ) ;
if ( ! IsEnabled ( ) )
return ;
// Only update soundscapes in multiplayer when the PVS gets updated
if ( g_pGameRules - > IsMultiplayer ( ) & & ! bUpdated & & ! soundscape_debug . GetBool ( ) )
return ;
bool bDebugThis = soundscape_debug . GetInt ( ) = = 1 ;
for ( int i = 0 ; i < m_hPlayersInPVS . Count ( ) ; i + + )
{
CBasePlayer * pPlayer = m_hPlayersInPVS [ i ] ;
if ( ! pPlayer )
continue ;
if ( ! InRangeOfPlayer ( pPlayer ) )
continue ;
// check to see if this is the sound entity that is
// currently affecting this player
audioparams_t & audio = pPlayer - > GetAudioParams ( ) ;
// if we got this far, we're looking at an entity that is contending
// for current player sound. the closest entity to player wins.
CEnvSoundscape * pCurrent = ( CEnvSoundscape * ) audio . ent . Get ( ) ;
if ( ! pCurrent | |
! pCurrent - > IsEnabled ( ) | |
! pCurrent - > InRangeOfPlayer ( pPlayer ) )
{
// The old one is obscured or out of range.. take over.
WriteAudioParamsTo ( audio ) ;
}
else if ( pCurrent & &
EarPosition ( ) . DistTo ( pPlayer - > EarPosition ( ) ) < pCurrent - > EarPosition ( ) . DistTo ( pPlayer - > EarPosition ( ) ) )
{
// new entity is closer to player, so it wins.
WriteAudioParamsTo ( audio ) ;
}
if ( ! bDebugThis )
{
bDebugThis = soundscape_debug . GetInt ( ) = = 2 ;
}
}
if ( bDebugThis )
{
// draw myself
NDebugOverlay : : Box ( GetAbsOrigin ( ) , Vector ( - 10 , - 10 , - 10 ) , Vector ( 10 , 10 , 10 ) , 255 , 0 , 255 , 64 , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
// Don't use GetLocalPlayer(), because that prevents multiplayer games using this for testing with a single client in the game
CBasePlayer * pPlayer = UTIL_PlayerByIndex ( 1 ) ;
if ( pPlayer )
{
audioparams_t & audio = pPlayer - > GetAudioParams ( ) ;
if ( audio . ent . Get ( ) ! = this )
{
if ( InRangeOfPlayer ( pPlayer ) )
{
NDebugOverlay : : Line ( GetAbsOrigin ( ) , pPlayer - > WorldSpaceCenter ( ) , 255 , 255 , 255 , true , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
else
{
NDebugOverlay : : Line ( GetAbsOrigin ( ) , pPlayer - > WorldSpaceCenter ( ) , 255 , 0 , 0 , true , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
}
else
{
if ( InRangeOfPlayer ( pPlayer ) )
{
NDebugOverlay : : Line ( GetAbsOrigin ( ) , pPlayer - > WorldSpaceCenter ( ) , 0 , 255 , 0 , true , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
else
{
NDebugOverlay : : Line ( GetAbsOrigin ( ) , pPlayer - > WorldSpaceCenter ( ) , 255 , 170 , 0 , true , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
// also draw lines to each sound position.
// we don't store the number of local sound positions, just a bitvector of which ones are on.
unsigned int soundbits = audio . localBits . Get ( ) ;
float periodic = 2.0f * sin ( ( fmod ( gpGlobals - > curtime , 2.0f ) - 1.0f ) * M_PI ) ; // = -4f .. 4f
for ( int ii = 0 ; ii < NUM_AUDIO_LOCAL_SOUNDS ; + + ii )
{
if ( soundbits & ( 1 < < ii ) )
{
const Vector & soundLoc = audio . localSound . Get ( ii ) ;
NDebugOverlay : : Line ( GetAbsOrigin ( ) , soundLoc , 0 , 32 , 255 , false , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
NDebugOverlay : : Cross3D ( soundLoc , 16.0f + periodic , 0 , 0 , 255 , false , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
}
}
}
NDebugOverlay : : EntityTextAtPosition ( GetAbsOrigin ( ) , 0 , STRING ( m_soundscapeName ) , NDEBUG_PERSIST_TILL_NEXT_SERVER ) ;
}
}
//
// env_soundscape - spawn a sound entity that will set player soundscape
// when player moves in range and sight.
//
//
void CEnvSoundscape : : Spawn ( )
{
Precache ( ) ;
// Because the soundscape has no model, need to make sure it doesn't get culled from the PVS for this reason and therefore
// never exist on the client, etc.
AddEFlags ( EFL_FORCE_CHECK_TRANSMIT ) ;
}
void CEnvSoundscape : : Precache ( )
{
if ( m_soundscapeName = = NULL_STRING )
{
DevMsg ( " Found soundscape entity with no soundscape name. \n " , STRING ( m_soundscapeName ) ) ;
return ;
}
m_soundscapeIndex = g_SoundscapeSystem . GetSoundscapeIndex ( STRING ( m_soundscapeName ) ) ;
if ( IsX360 ( ) )
{
g_SoundscapeSystem . PrecacheSounds ( m_soundscapeIndex ) ;
}
if ( ! g_SoundscapeSystem . IsValidIndex ( m_soundscapeIndex ) )
{
DevWarning ( " Can't find soundscape: %s \n " , STRING ( m_soundscapeName ) ) ;
}
}
void CEnvSoundscape : : DrawDebugGeometryOverlays ( void )
{
if ( m_debugOverlays & ( OVERLAY_BBOX_BIT | OVERLAY_PIVOT_BIT | OVERLAY_ABSBOX_BIT ) )
{
CBasePlayer * pPlayer = UTIL_GetLocalPlayer ( ) ;
if ( pPlayer )
{
audioparams_t & audio = pPlayer - > GetAudioParams ( ) ;
if ( audio . ent . Get ( ) = = this )
{
NDebugOverlay : : Line ( GetAbsOrigin ( ) , pPlayer - > WorldSpaceCenter ( ) , 255 , 0 , 255 , false , 0 ) ;
}
}
}
BaseClass : : DrawDebugGeometryOverlays ( ) ;
}
// ---------------------------------------------------------------------------------------------------- //
// CEnvSoundscapeTriggerable
// ---------------------------------------------------------------------------------------------------- //
LINK_ENTITY_TO_CLASS ( env_soundscape_triggerable , CEnvSoundscapeTriggerable ) ;
BEGIN_DATADESC ( CEnvSoundscapeTriggerable )
END_DATADESC ( )
CEnvSoundscapeTriggerable : : CEnvSoundscapeTriggerable ( )
{
}
void CEnvSoundscapeTriggerable : : DelegateStartTouch ( CBaseEntity * pEnt )
{
CBasePlayer * pPlayer = dynamic_cast < CBasePlayer * > ( pEnt ) ;
if ( ! pPlayer )
return ;
// Just in case.. we shouldn't already be in the player's list because it should have
// called DelegateEndTouch, but this seems to happen when they're noclipping.
pPlayer - > m_hTriggerSoundscapeList . FindAndRemove ( this ) ;
// Add us to the player's list of soundscapes and
pPlayer - > m_hTriggerSoundscapeList . AddToHead ( this ) ;
WriteAudioParamsTo ( pPlayer - > GetAudioParams ( ) ) ;
}
void CEnvSoundscapeTriggerable : : DelegateEndTouch ( CBaseEntity * pEnt )
{
CBasePlayer * pPlayer = dynamic_cast < CBasePlayer * > ( pEnt ) ;
if ( ! pPlayer )
return ;
// Remove us from the ent's list of soundscapes.
pPlayer - > m_hTriggerSoundscapeList . FindAndRemove ( this ) ;
while ( pPlayer - > m_hTriggerSoundscapeList . Count ( ) > 0 )
{
CEnvSoundscapeTriggerable * pSS = dynamic_cast < CEnvSoundscapeTriggerable * > ( pPlayer - > m_hTriggerSoundscapeList [ 0 ] . Get ( ) ) ;
if ( pSS )
{
// Make this one current.
pSS - > WriteAudioParamsTo ( pPlayer - > GetAudioParams ( ) ) ;
return ;
}
else
{
pPlayer - > m_hTriggerSoundscapeList . Remove ( 0 ) ;
}
}
// No soundscapes left.
pPlayer - > GetAudioParams ( ) . ent = NULL ;
}
void CEnvSoundscapeTriggerable : : Think ( )
{
// Overrides the base class's think and prevents it from running at all.
}
// ---------------------------------------------------------------------------------------------------- //
// CTriggerSoundscape
// ---------------------------------------------------------------------------------------------------- //
class CTriggerSoundscape : public CBaseTrigger
{
public :
DECLARE_CLASS ( CTriggerSoundscape , CBaseTrigger ) ;
DECLARE_DATADESC ( ) ;
CTriggerSoundscape ( ) ;
virtual void StartTouch ( CBaseEntity * pOther ) ;
virtual void EndTouch ( CBaseEntity * pOther ) ;
virtual void Spawn ( ) ;
virtual void Activate ( ) ;
void PlayerUpdateThink ( ) ;
private :
CHandle < CEnvSoundscapeTriggerable > m_hSoundscape ;
string_t m_SoundscapeName ;
CUtlVector < CBasePlayerHandle > m_spectators ; // spectators in our volume
} ;
LINK_ENTITY_TO_CLASS ( trigger_soundscape , CTriggerSoundscape ) ;
BEGIN_DATADESC ( CTriggerSoundscape )
DEFINE_THINKFUNC ( PlayerUpdateThink ) ,
DEFINE_KEYFIELD ( m_SoundscapeName , FIELD_STRING , " soundscape " ) ,
DEFINE_FIELD ( m_hSoundscape , FIELD_EHANDLE ) ,
DEFINE_UTLVECTOR ( m_spectators , FIELD_EHANDLE ) ,
END_DATADESC ( )
CTriggerSoundscape : : CTriggerSoundscape ( )
{
}
void CTriggerSoundscape : : StartTouch ( CBaseEntity * pOther )
{
if ( m_hSoundscape )
m_hSoundscape - > DelegateStartTouch ( pOther ) ;
BaseClass : : StartTouch ( pOther ) ;
}
void CTriggerSoundscape : : EndTouch ( CBaseEntity * pOther )
{
if ( m_hSoundscape )
m_hSoundscape - > DelegateEndTouch ( pOther ) ;
BaseClass : : EndTouch ( pOther ) ;
}
void CTriggerSoundscape : : Spawn ( )
{
BaseClass : : Spawn ( ) ;
InitTrigger ( ) ;
SetThink ( & CTriggerSoundscape : : PlayerUpdateThink ) ;
SetNextThink ( gpGlobals - > curtime + 0.2 ) ;
}
void CTriggerSoundscape : : Activate ( )
{
m_hSoundscape = dynamic_cast < CEnvSoundscapeTriggerable * > ( gEntList . FindEntityByName ( NULL , m_SoundscapeName ) ) ;
BaseClass : : Activate ( ) ;
}
// look for dead/spectating players in our volume, to call touch on
void CTriggerSoundscape : : PlayerUpdateThink ( )
{
int i ;
SetNextThink ( gpGlobals - > curtime + 0.2 ) ;
CUtlVector < CBasePlayerHandle > oldSpectators ;
oldSpectators = m_spectators ;
m_spectators . RemoveAll ( ) ;
for ( i = 1 ; i < = gpGlobals - > maxClients ; + + i )
{
CBasePlayer * player = UTIL_PlayerByIndex ( i ) ;
if ( ! player )
continue ;
if ( player - > IsAlive ( ) )
continue ;
// if the spectator is intersecting the trigger, track it, and start a touch if it is just starting to touch
if ( Intersects ( player ) )
{
if ( ! oldSpectators . HasElement ( player ) )
{
StartTouch ( player ) ;
}
m_spectators . AddToTail ( player ) ;
}
}
// check for spectators who are no longer intersecting
for ( i = 0 ; i < oldSpectators . Count ( ) ; + + i )
{
CBasePlayer * player = oldSpectators [ i ] ;
if ( ! player )
continue ;
if ( ! m_spectators . HasElement ( player ) )
{
EndTouch ( player ) ;
}
}
}