source-engine/game/client/tf/c_obj_dispenser.cpp

393 lines
12 KiB
C++
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Client's CObjectSentrygun
//
// $NoKeywords: $
//=============================================================================//
#include "cbase.h"
#include "c_baseobject.h"
#include "c_tf_player.h"
#include "vgui/ILocalize.h"
#include "c_obj_dispenser.h"
// NVNT haptics system interface
#include "c_tf_haptics.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
using namespace vgui;
//-----------------------------------------------------------------------------
// Purpose: RecvProxy that converts the Team's player UtlVector to entindexes
//-----------------------------------------------------------------------------
void RecvProxy_HealingList( const CRecvProxyData *pData, void *pStruct, void *pOut )
{
C_ObjectDispenser *pDispenser = (C_ObjectDispenser*)pStruct;
CBaseHandle *pHandle = (CBaseHandle*)(&(pDispenser->m_hHealingTargets[pData->m_iElement]));
RecvProxy_IntToEHandle( pData, pStruct, pHandle );
// update the heal beams
pDispenser->m_bUpdateHealingTargets = true;
}
void RecvProxyArrayLength_HealingArray( void *pStruct, int objectID, int currentArrayLength )
{
C_ObjectDispenser *pDispenser = (C_ObjectDispenser*)pStruct;
if ( pDispenser->m_hHealingTargets.Size() != currentArrayLength )
pDispenser->m_hHealingTargets.SetSize( currentArrayLength );
// update the heal beams
pDispenser->m_bUpdateHealingTargets = true;
}
//-----------------------------------------------------------------------------
// Purpose: Dispenser object
//-----------------------------------------------------------------------------
IMPLEMENT_CLIENTCLASS_DT(C_ObjectDispenser, DT_ObjectDispenser, CObjectDispenser)
RecvPropInt( RECVINFO( m_iState ) ),
RecvPropInt( RECVINFO( m_iAmmoMetal ) ),
RecvPropInt( RECVINFO( m_iMiniBombCounter ) ),
RecvPropArray2(
RecvProxyArrayLength_HealingArray,
RecvPropInt( "healing_array_element", 0, SIZEOF_IGNORE, 0, RecvProxy_HealingList ),
MAX_PLAYERS,
0,
"healing_array"
)
END_RECV_TABLE()
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
C_ObjectDispenser::C_ObjectDispenser()
{
m_bUpdateHealingTargets = false;
m_bPlayingSound = false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
C_ObjectDispenser::~C_ObjectDispenser()
{
StopSound( "Building_Dispenser.Heal" );
// NVNT see if local player is in the list of targets
// temp. fix if dispener is destroyed will stop all healers.
if(m_bPlayingSound)
{
if(tfHaptics.healingDispenserCount>0) {
tfHaptics.healingDispenserCount --;
if(tfHaptics.healingDispenserCount==0 && !tfHaptics.wasBeingHealedMedic)
tfHaptics.isBeingHealed = false;
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : updateType -
//-----------------------------------------------------------------------------
void C_ObjectDispenser::OnDataChanged( DataUpdateType_t updateType )
{
BaseClass::OnDataChanged( updateType );
#ifdef STAGING_ONLY
if ( updateType == DATA_UPDATE_CREATED )
{
SetNextClientThink( CLIENT_THINK_ALWAYS );
}
#endif // STAGING_ONLY
if ( m_bUpdateHealingTargets )
{
UpdateEffects();
m_bUpdateHealingTargets = false;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void C_ObjectDispenser::ClientThink()
{
BaseClass::ClientThink();
#ifdef STAGING_ONLY
C_TFPlayer *pTFOwner = GetOwner();
if ( pTFOwner && pTFOwner->m_Shared.IsEnteringOrExitingFullyInvisible() )
{
UpdateEffects();
}
#endif // STAGING_ONLY
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void C_ObjectDispenser::SetInvisibilityLevel( float flValue )
{
if ( IsEnteringOrExitingFullyInvisible( flValue ) )
{
UpdateEffects();
}
BaseClass::SetInvisibilityLevel( flValue );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void C_ObjectDispenser::UpdateEffects( void )
{
C_TFPlayer *pOwner = GetOwner();
if ( GetInvisibilityLevel() == 1.f || ( pOwner && pOwner->m_Shared.IsFullyInvisible() ) )
{
StopEffects( true );
return;
}
StopEffects();
// Now add any new targets
for ( int i = 0; i < m_hHealingTargets.Count(); i++ )
{
C_BaseEntity *pTarget = m_hHealingTargets[i].Get();
// Loops through the healing targets, and make sure we have an effect for each of them
if ( pTarget )
{
// don't want to show this effect for stealthed spies
C_TFPlayer *pPlayer = dynamic_cast< C_TFPlayer * >( pTarget );
if ( pPlayer && ( pPlayer->m_Shared.IsStealthed() || pPlayer->m_Shared.InCond( TF_COND_STEALTHED_BLINK ) ) )
continue;
bool bHaveEffect = false;
for ( int targets = 0; targets < m_hHealingTargetEffects.Count(); targets++ )
{
if ( m_hHealingTargetEffects[targets].pTarget == pTarget )
{
bHaveEffect = true;
break;
}
}
if ( bHaveEffect )
continue;
// NVNT if the dispenser has started to heal the local player
// notify the haptics system
if(pTarget==C_BasePlayer::GetLocalPlayer())
{
tfHaptics.healingDispenserCount++;
if(!tfHaptics.wasBeingHealedMedic) {
tfHaptics.isBeingHealed = true;
}
}
const char *pszEffectName;
if ( GetTeamNumber() == TF_TEAM_RED )
{
pszEffectName = "dispenser_heal_red";
}
else
{
pszEffectName = "dispenser_heal_blue";
}
CNewParticleEffect *pEffect;
// if we don't have a model, attach at the origin, otherwise use attachment 'heal_origin'
if ( FBitSet( GetObjectFlags(), OF_DOESNT_HAVE_A_MODEL ) )
{
// offset the origin to player's chest
if ( FBitSet( GetObjectFlags(), OF_PLAYER_DESTRUCTION ) )
{
pEffect = ParticleProp()->Create( pszEffectName, PATTACH_ABSORIGIN_FOLLOW, NULL, Vector( 0, 0, 50 ) );
}
else
{
pEffect = ParticleProp()->Create( pszEffectName, PATTACH_ABSORIGIN_FOLLOW );
}
}
else
{
pEffect = ParticleProp()->Create( pszEffectName, PATTACH_POINT_FOLLOW, "heal_origin" );
}
ParticleProp()->AddControlPoint( pEffect, 1, pTarget, PATTACH_ABSORIGIN_FOLLOW, NULL, Vector(0,0,50) );
int iIndex = m_hHealingTargetEffects.AddToTail();
m_hHealingTargetEffects[iIndex].pTarget = pTarget;
m_hHealingTargetEffects[iIndex].pEffect = pEffect;
// Start the sound over again every time we start a new beam
StopSound( "Building_Dispenser.Heal" );
CLocalPlayerFilter filter;
EmitSound( filter, entindex(), "Building_Dispenser.Heal" );
m_bPlayingSound = true;
}
}
// Stop the sound if we're not healing anyone
if ( m_bPlayingSound && m_hHealingTargets.Count() == 0 )
{
m_bPlayingSound = false;
// stop the sound
StopSound( "Building_Dispenser.Heal" );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void C_ObjectDispenser::StopEffects( bool bRemoveAll /* = false */ )
{
// Find all the targets we've stopped healing
bool bStillHealing[MAX_DISPENSER_HEALING_TARGETS] = { 0 };
for ( int i = 0; i < m_hHealingTargetEffects.Count(); i++ )
{
bStillHealing[i] = false;
// Are we still healing this target?
if ( !bRemoveAll )
{
for ( int target = 0; target < m_hHealingTargets.Count(); target++ )
{
if ( m_hHealingTargets[target] && m_hHealingTargets[target] == m_hHealingTargetEffects[i].pTarget )
{
bStillHealing[i] = true;
break;
}
}
}
}
// Now remove all the dead effects
for ( int i = m_hHealingTargetEffects.Count()-1; i >= 0; i-- )
{
if ( !bStillHealing[i] )
{
// NVNT if the healing target of this dispenser is the local player.
// inform the haptics system interface we are no longer healing.
if(m_hHealingTargetEffects[i].pTarget==C_BasePlayer::GetLocalPlayer())
{
if(tfHaptics.healingDispenserCount>0) {
tfHaptics.healingDispenserCount --;
if(tfHaptics.healingDispenserCount==0 && !tfHaptics.wasBeingHealedMedic)
tfHaptics.isBeingHealed = false;
}
}
ParticleProp()->StopEmission( m_hHealingTargetEffects[i].pEffect );
m_hHealingTargetEffects.Remove(i);
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Damage level has changed, update our effects
//-----------------------------------------------------------------------------
void C_ObjectDispenser::UpdateDamageEffects( BuildingDamageLevel_t damageLevel )
{
if ( m_hDamageEffects )
{
m_hDamageEffects->StopEmission( false, false );
m_hDamageEffects = NULL;
}
const char *pszEffect = "";
switch( damageLevel )
{
case BUILDING_DAMAGE_LEVEL_LIGHT:
pszEffect = "dispenserdamage_1";
break;
case BUILDING_DAMAGE_LEVEL_MEDIUM:
pszEffect = "dispenserdamage_2";
break;
case BUILDING_DAMAGE_LEVEL_HEAVY:
pszEffect = "dispenserdamage_3";
break;
case BUILDING_DAMAGE_LEVEL_CRITICAL:
pszEffect = "dispenserdamage_4";
break;
default:
break;
}
if ( Q_strlen(pszEffect) > 0 )
{
m_hDamageEffects = ParticleProp()->Create( pszEffect, PATTACH_ABSORIGIN );
}
}
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
int C_ObjectDispenser::GetMaxMetal( void )
{
return DISPENSER_MAX_METAL_AMMO;
}
//-----------------------------------------------------------------------------
// Control screen
//-----------------------------------------------------------------------------
DECLARE_VGUI_SCREEN_FACTORY( CDispenserControlPanel, "screen_obj_dispenser_blue" );
DECLARE_VGUI_SCREEN_FACTORY( CDispenserControlPanel_Red, "screen_obj_dispenser_red" );
//-----------------------------------------------------------------------------
// Constructor:
//-----------------------------------------------------------------------------
CDispenserControlPanel::CDispenserControlPanel( vgui::Panel *parent, const char *panelName )
: BaseClass( parent, "CDispenserControlPanel" )
{
m_pAmmoProgress = new RotatingProgressBar( this, "MeterArrow" );
}
//-----------------------------------------------------------------------------
// Deactivates buttons we can't afford
//-----------------------------------------------------------------------------
void CDispenserControlPanel::OnTickActive( C_BaseObject *pObj, C_TFPlayer *pLocalPlayer )
{
BaseClass::OnTickActive( pObj, pLocalPlayer );
Assert( dynamic_cast< C_ObjectDispenser* >( pObj ) );
m_hDispenser = static_cast< C_ObjectDispenser* >( pObj );
float flProgress = m_hDispenser ? m_hDispenser->GetMetalAmmoCount() / (float)m_hDispenser->GetMaxMetal() : 0.f;
m_pAmmoProgress->SetProgress( flProgress );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CDispenserControlPanel::IsVisible( void )
{
if ( m_hDispenser )
{
#ifdef STAGING_ONLY
if ( m_hDispenser->IsMiniBuilding() )
return false;
#endif // STAGING_ONLY
if ( m_hDispenser->GetInvisibilityLevel() == 1.f )
return false;
}
return BaseClass::IsVisible();
}
IMPLEMENT_CLIENTCLASS_DT(C_ObjectCartDispenser, DT_ObjectCartDispenser, CObjectCartDispenser)
END_RECV_TABLE()