//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//===========================================================================//

#include "cbase.h"
#include "team_train_watcher.h"
#include "team_control_point.h"
#include "trains.h"
#include "team_objectiveresource.h"
#include "teamplayroundbased_gamerules.h"
#include "team_control_point.h"
#include "team_control_point_master.h"
#include "engine/IEngineSound.h"
#include "soundenvelope.h"
#include "mp_shareddefs.h"
#include "props.h"
#include "physconstraint.h"

#ifdef TF_DLL
#include "tf_shareddefs.h"
#endif

// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
/*
#define TWM_FIRSTSTAGEOUTCOME01	"Announcer.PLR_FirstStageOutcome01"
#define TWM_FIRSTSTAGEOUTCOME02	"Announcer.PLR_FirstStageOutcome02"
#define TWM_RACEGENERAL01	"Announcer.PLR_RaceGeneral01"
#define TWM_RACEGENERAL02	"Announcer.PLR_RaceGeneral02"
#define TWM_RACEGENERAL03	"Announcer.PLR_RaceGeneral03"
#define TWM_RACEGENERAL04	"Announcer.PLR_RaceGeneral04"
#define TWM_RACEGENERAL05	"Announcer.PLR_RaceGeneral05"
#define TWM_RACEGENERAL08	"Announcer.PLR_RaceGeneral08"
#define TWM_RACEGENERAL06	"Announcer.PLR_RaceGeneral06"
#define TWM_RACEGENERAL07	"Announcer.PLR_RaceGeneral07"
#define TWM_RACEGENERAL09	"Announcer.PLR_RaceGeneral09"
#define TWM_RACEGENERAL12	"Announcer.PLR_RaceGeneral12"
#define TWM_RACEGENERAL13	"Announcer.PLR_RaceGeneral13"
#define TWM_RACEGENERAL14	"Announcer.PLR_RaceGeneral14"
#define TWM_RACEGENERAL15	"Announcer.PLR_RaceGeneral15"
#define TWM_RACEGENERAL10	"Announcer.PLR_RaceGeneral10"
#define TWM_RACEGENERAL11	"Announcer.PLR_RaceGeneral11"
#define TWM_SECONDSTAGEOUTCOME01	"Announcer.PLR_SecondStageOutcome01"
#define TWM_SECONDSTAGEOUTCOME04	"Announcer.PLR_SecondStageOutcome04"
#define TWM_SECONDSTAGEOUTCOME02	"Announcer.PLR_SecondStageOutcome02"
#define TWM_SECONDSTAGEOUTCOME03	"Announcer.PLR_SecondStageOutcome03"
#define TWM_FINALSTAGEOUTCOME01		"Announcer.PLR_FinalStageOutcome01"
#define TWM_FINALSTAGEOUTCOME02		"Announcer.PLR_FinalStageOutcome02"
#define TWM_FINALSTAGESTART01	"Announcer.PLR_FinalStageStart01"
#define TWM_FINALSTAGESTART04	"Announcer.PLR_FinalStageStart04"
#define TWM_FINALSTAGESTART08	"Announcer.PLR_FinalStageStart08"
#define TWM_FINALSTAGESTART09	"Announcer.PLR_FinalStageStart09"
#define TWM_FINALSTAGESTART07	"Announcer.PLR_FinalStageStart07"
#define TWM_FINALSTAGESTART02	"Announcer.PLR_FinalStageStart02"
#define TWM_FINALSTAGESTART03	"Announcer.PLR_FinalStageStart03"
#define TWM_FINALSTAGESTART05	"Announcer.PLR_FinalStageStart05"
#define TWM_FINALSTAGESTART06	"Announcer.PLR_FinalStageStart06"

EHANDLE g_hTeamTrainWatcherMaster = NULL;
*/
#define MAX_ALARM_TIME_NO_RECEDE 18 // max amount of time to play the alarm if the train isn't going to recede

BEGIN_DATADESC( CTeamTrainWatcher )

	// Inputs.
	DEFINE_INPUTFUNC( FIELD_VOID, "RoundActivate", InputRoundActivate ),
	DEFINE_INPUTFUNC( FIELD_INTEGER, "SetNumTrainCappers", InputSetNumTrainCappers ),
	DEFINE_INPUTFUNC( FIELD_VOID, "OnStartOvertime", InputOnStartOvertime ),
	DEFINE_INPUTFUNC( FIELD_VOID, "Enable", InputEnable ),
	DEFINE_INPUTFUNC( FIELD_VOID, "Disable", InputDisable ),
	DEFINE_INPUTFUNC( FIELD_FLOAT, "SetSpeedForwardModifier", InputSetSpeedForwardModifier ),
	DEFINE_INPUTFUNC( FIELD_INTEGER, "SetTrainRecedeTime", InputSetTrainRecedeTime ),
	DEFINE_INPUTFUNC( FIELD_BOOLEAN, "SetTrainCanRecede", InputSetTrainCanRecede ),

	// Outputs
	DEFINE_OUTPUT( m_OnTrainStartRecede, "OnTrainStartRecede" ),

	// key
	DEFINE_KEYFIELD( m_iszTrain, FIELD_STRING, "train" ),
	DEFINE_KEYFIELD( m_iszStartNode, FIELD_STRING, "start_node" ),
	DEFINE_KEYFIELD( m_iszGoalNode, FIELD_STRING, "goal_node" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[0], FIELD_STRING, "linked_pathtrack_1" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[0], FIELD_STRING, "linked_cp_1" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[1], FIELD_STRING, "linked_pathtrack_2" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[1], FIELD_STRING, "linked_cp_2" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[2], FIELD_STRING, "linked_pathtrack_3" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[2], FIELD_STRING, "linked_cp_3" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[3], FIELD_STRING, "linked_pathtrack_4" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[3], FIELD_STRING, "linked_cp_4" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[4], FIELD_STRING, "linked_pathtrack_5" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[4], FIELD_STRING, "linked_cp_5" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[5], FIELD_STRING, "linked_pathtrack_6" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[5], FIELD_STRING, "linked_cp_6" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[6], FIELD_STRING, "linked_pathtrack_7" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[6], FIELD_STRING, "linked_cp_7" ),

	DEFINE_KEYFIELD( m_iszLinkedPathTracks[7], FIELD_STRING, "linked_pathtrack_8" ),
	DEFINE_KEYFIELD( m_iszLinkedCPs[7], FIELD_STRING, "linked_cp_8" ),

	DEFINE_KEYFIELD( m_bTrainCanRecede, FIELD_BOOLEAN, "train_can_recede" ),

	DEFINE_KEYFIELD( m_bHandleTrainMovement, FIELD_BOOLEAN, "handle_train_movement" ),

	// can be up to 8 links

	// min speed for train hud speed levels
	DEFINE_KEYFIELD( m_flSpeedLevels[0], FIELD_FLOAT, "hud_min_speed_level_1" ),
	DEFINE_KEYFIELD( m_flSpeedLevels[1], FIELD_FLOAT, "hud_min_speed_level_2" ),
	DEFINE_KEYFIELD( m_flSpeedLevels[2], FIELD_FLOAT, "hud_min_speed_level_3" ),

	DEFINE_KEYFIELD( m_bDisabled,	FIELD_BOOLEAN,	"StartDisabled" ),

	DEFINE_KEYFIELD( m_iszSparkName, FIELD_STRING, "env_spark_name" ),

	DEFINE_KEYFIELD( m_flSpeedForwardModifier, FIELD_FLOAT, "speed_forward_modifier" ),

	DEFINE_KEYFIELD( m_nTrainRecedeTime, FIELD_INTEGER, "train_recede_time" ),

END_DATADESC()


IMPLEMENT_SERVERCLASS_ST(CTeamTrainWatcher, DT_TeamTrainWatcher)

	SendPropFloat( SENDINFO( m_flTotalProgress ), 11, 0, 0.0f, 1.0f ),
	SendPropInt( SENDINFO( m_iTrainSpeedLevel ), 4 ),
	SendPropTime( SENDINFO( m_flRecedeTime ) ),
	SendPropInt( SENDINFO( m_nNumCappers ) ),
#ifdef GLOWS_ENABLE
	SendPropEHandle( SENDINFO( m_hGlowEnt ) ),
#endif // GLOWS_ENABLE

END_SEND_TABLE()


LINK_ENTITY_TO_CLASS( team_train_watcher, CTeamTrainWatcher );

IMPLEMENT_AUTO_LIST( ITFTeamTrainWatcher );

/*
LINK_ENTITY_TO_CLASS( team_train_watcher_master, CTeamTrainWatcherMaster );
PRECACHE_REGISTER( team_train_watcher_master );

CTeamTrainWatcherMaster::CTeamTrainWatcherMaster()
{
	m_pBlueWatcher = NULL;
	m_pRedWatcher = NULL;

	m_flBlueProgress = 0.0f;
	m_flRedProgress = 0.0f;

	ListenForGameEvent( "teamplay_round_start" );
	ListenForGameEvent( "teamplay_round_win" );
}

CTeamTrainWatcherMaster::~CTeamTrainWatcherMaster()
{
	if ( g_hTeamTrainWatcherMaster.Get() == this )
	{
		g_hTeamTrainWatcherMaster = NULL;
	}
}

void CTeamTrainWatcherMaster::Precache( void )
{
	PrecacheScriptSound( TWM_FIRSTSTAGEOUTCOME01 );
	PrecacheScriptSound( TWM_FIRSTSTAGEOUTCOME02 );
	PrecacheScriptSound( TWM_RACEGENERAL01 );
	PrecacheScriptSound( TWM_RACEGENERAL02 );
	PrecacheScriptSound( TWM_RACEGENERAL03 );
	PrecacheScriptSound( TWM_RACEGENERAL04 );
	PrecacheScriptSound( TWM_RACEGENERAL05 );
	PrecacheScriptSound( TWM_RACEGENERAL08 );
	PrecacheScriptSound( TWM_RACEGENERAL06 );
	PrecacheScriptSound( TWM_RACEGENERAL07 );
	PrecacheScriptSound( TWM_RACEGENERAL09 );
	PrecacheScriptSound( TWM_RACEGENERAL12 );
	PrecacheScriptSound( TWM_RACEGENERAL13 );
	PrecacheScriptSound( TWM_RACEGENERAL14 );
	PrecacheScriptSound( TWM_RACEGENERAL15 );
	PrecacheScriptSound( TWM_RACEGENERAL10 );
	PrecacheScriptSound( TWM_RACEGENERAL11 );
	PrecacheScriptSound( TWM_SECONDSTAGEOUTCOME01 );
	PrecacheScriptSound( TWM_SECONDSTAGEOUTCOME04 );
	PrecacheScriptSound( TWM_SECONDSTAGEOUTCOME02 );
	PrecacheScriptSound( TWM_SECONDSTAGEOUTCOME03 );
	PrecacheScriptSound( TWM_FINALSTAGEOUTCOME01 );
	PrecacheScriptSound( TWM_FINALSTAGEOUTCOME02 );
	PrecacheScriptSound( TWM_FINALSTAGESTART01 );
	PrecacheScriptSound( TWM_FINALSTAGESTART04 );
	PrecacheScriptSound( TWM_FINALSTAGESTART08 );
	PrecacheScriptSound( TWM_FINALSTAGESTART09 );
	PrecacheScriptSound( TWM_FINALSTAGESTART07 );
	PrecacheScriptSound( TWM_FINALSTAGESTART02 );
	PrecacheScriptSound( TWM_FINALSTAGESTART03 );
	PrecacheScriptSound( TWM_FINALSTAGESTART05 );
	PrecacheScriptSound( TWM_FINALSTAGESTART06 );

	BaseClass::Precache();
}

bool CTeamTrainWatcherMaster::FindTrainWatchers( void )
{
	m_pBlueWatcher = NULL;
	m_pRedWatcher = NULL;

	// find the train_watchers for this round
	CTeamTrainWatcher *pTrainWatcher = (CTeamTrainWatcher *)gEntList.FindEntityByClassname( NULL, "team_train_watcher" );
	while ( pTrainWatcher )
	{
		if ( pTrainWatcher->IsDisabled() == false )
		{
			if ( pTrainWatcher->GetTeamNumber() == TF_TEAM_BLUE )
			{
				m_pBlueWatcher = pTrainWatcher;
			}
			else if ( pTrainWatcher->GetTeamNumber() == TF_TEAM_RED )
			{
				m_pRedWatcher = pTrainWatcher;
			}
		}

		pTrainWatcher = (CTeamTrainWatcher *)gEntList.FindEntityByClassname( pTrainWatcher, "team_train_watcher" );
	}

	return ( m_pBlueWatcher && m_pRedWatcher );
}

void CTeamTrainWatcherMaster::TWMThink( void )
{
	if ( TeamplayRoundBasedRules() && TeamplayRoundBasedRules()->State_Get() != GR_STATE_RND_RUNNING )
	{
		// the next time we 'think'
		SetContextThink( &CTeamTrainWatcherMaster::TWMThink, gpGlobals->curtime + 0.2, TWMASTER_THINK );
		return;
	}



	// the next time we 'think'
	SetContextThink( &CTeamTrainWatcherMaster::TWMThink, gpGlobals->curtime + 0.2, TWMASTER_THINK );
}

void CTeamTrainWatcherMaster::FireGameEvent( IGameEvent *event )
{
	const char *eventname = event->GetName();`

	if ( FStrEq( "teamplay_round_start", eventname ) )
	{
		if ( TeamplayRoundBasedRules() && TeamplayRoundBasedRules()->HasMultipleTrains() )
		{
			if ( FindTrainWatchers() )
			{
				// we found train watchers so start thinking
				SetContextThink( &CTeamTrainWatcherMaster::TWMThink, gpGlobals->curtime + 0.2, TWMASTER_THINK );
			}
		}
	}
	else if ( FStrEq( "teamplay_round_win", eventname ) )
	{
		if ( TeamplayRoundBasedRules() )
		{
			int iWinningTeam = event->GetInt( "team" );
			int iLosingTeam = ( iWinningTeam == TF_TEAM_RED ) ? TF_TEAM_BLUE : TF_TEAM_RED;
			bool bFullRound = event->GetBool( "full_round" );

			CTeamRecipientFilter filterWinner( iWinningTeam, true );
			CTeamRecipientFilter filterLoser( iLosingTeam, true );

			if ( bFullRound )
			{
				EmitSound( filterWinner, entindex(), TWM_FINALSTAGEOUTCOME01 );
				EmitSound( filterLoser, entindex(), TWM_FINALSTAGEOUTCOME02 );
			}
			else
			{
				EmitSound( filterWinner, entindex(), TWM_FIRSTSTAGEOUTCOME01 );
				EmitSound( filterLoser, entindex(), TWM_FIRSTSTAGEOUTCOME02 );
			}
		}
	}
}
*/
CTeamTrainWatcher::CTeamTrainWatcher()
{
	m_bDisabled = false;
	m_flRecedeTime = 0;
	m_bWaitingToRecede = false;
	m_bCapBlocked = false;

	m_flNextSpeakForwardConceptTime = 0;
	m_hAreaCap = NULL;

	m_bTrainCanRecede = true;
	m_bAlarmPlayed = false;
	m_pAlarm = NULL;
	m_flAlarmEndTime = -1;

	m_bHandleTrainMovement = false;
	m_flSpeedForwardModifier = 1.0f;
	m_iCurrentHillType = HILL_TYPE_NONE;
	m_flCurrentSpeed = 0.0f;
	m_bReceding = false;

	m_flTrainDistanceFromStart = 0.0f;

	m_nTrainRecedeTime = 0;

#ifdef GLOWS_ENABLE
	m_hGlowEnt.Set( NULL );
#endif // GLOWS_ENABLE

#ifdef TF_DLL
	ChangeTeam( TF_TEAM_BLUE );
#else
	ChangeTeam( TEAM_UNASSIGNED );
#endif
/*
	// create a CTeamTrainWatcherMaster entity
	if ( g_hTeamTrainWatcherMaster.Get() == NULL )
	{
		g_hTeamTrainWatcherMaster = CreateEntityByName( "team_train_watcher_master" );
	}
*/
	ListenForGameEvent( "path_track_passed" );
}

CTeamTrainWatcher::~CTeamTrainWatcher()
{
	m_Sparks.Purge();
}

void CTeamTrainWatcher::UpdateOnRemove( void )
{
	StopCaptureAlarm();

	BaseClass::UpdateOnRemove();
}

int CTeamTrainWatcher::UpdateTransmitState()
{
	if ( m_bDisabled )
	{
		return SetTransmitState( FL_EDICT_DONTSEND );
	}

	return SetTransmitState( FL_EDICT_ALWAYS );
}

void CTeamTrainWatcher::InputRoundActivate( inputdata_t &inputdata )
{
	StopCaptureAlarm();

	if ( !m_bDisabled )
	{
		WatcherActivate();
	}
}

void CTeamTrainWatcher::InputEnable( inputdata_t &inputdata )
{
	StopCaptureAlarm();

	m_bDisabled = false;

	WatcherActivate();

	UpdateTransmitState();
}

void CTeamTrainWatcher::InputDisable( inputdata_t &inputdata )
{
	StopCaptureAlarm();

	m_bDisabled = true;
	SetContextThink( NULL, 0, TW_THINK );

	m_bWaitingToRecede = false;

	m_Sparks.Purge();

#ifdef GLOWS_ENABLE
	m_hGlowEnt.Set( NULL );
#endif // GLOWS_ENABLE

	// if we're moving the train, let's shut it down
	if ( m_bHandleTrainMovement )
	{
		m_flCurrentSpeed = 0.0f;

		if ( m_hTrain )
		{
			m_hTrain->SetSpeedDirAccel( m_flCurrentSpeed );
		}

		// handle the sparks under the train
		HandleSparks( false );
	}

	UpdateTransmitState();
}

ConVar tf_escort_recede_time( "tf_escort_recede_time", "30", 0, "", true, 0, false, 0 );
ConVar tf_escort_recede_time_overtime( "tf_escort_recede_time_overtime", "5", 0, "", true, 0, false, 0 );

void CTeamTrainWatcher::FireGameEvent( IGameEvent *event )
{
	if ( IsDisabled() || !m_bHandleTrainMovement )
		return;

	const char *pszEventName = event->GetName();
	if ( FStrEq( pszEventName, "path_track_passed" ) )
	{
		int iIndex = event->GetInt( "index" );
		CPathTrack *pNode = dynamic_cast< CPathTrack* >( UTIL_EntityByIndex( iIndex ) );

		if ( pNode )
		{
			bool bHandleEvent = false;
			CPathTrack *pTempNode = m_hStartNode.Get();

			// is this a node in the track we're watching?
			while ( pTempNode )
			{
				if ( pTempNode == pNode )
				{
					bHandleEvent = true;
					break;
				}

				pTempNode = pTempNode->GetNext();
			}

			if ( bHandleEvent )
			{
				// If we're receding and we've hit a node but the next node (going backwards) is disabled 
				// the train is going to stop (like at the base of a downhill section) when we start forward 
				// again we won't pass this node again so don't change our hill state based on this node.
				if ( m_bReceding )
				{
					if ( pNode->GetPrevious() && pNode->GetPrevious()->IsDisabled() )
					{
						return;
					}
				}

				int iHillType = pNode->GetHillType();
				bool bUpdate = ( m_iCurrentHillType != iHillType );

				if ( !bUpdate )
				{
					// the hill settings are the same, but are we leaving an uphill or downhill segment?
					if ( m_iCurrentHillType != HILL_TYPE_NONE )
					{
						// let's peek at the next node
						CPathTrack *pNextNode = pNode->GetNext();
						if ( m_flCurrentSpeed < 0 )
						{
							// we're going backwards
							pNextNode = pNode->GetPrevious();
						}

						if ( pNextNode )
						{
							int iNextHillType = pNextNode->GetHillType();
							if ( m_iCurrentHillType != iNextHillType )
							{
								// we're leaving an uphill or downhill segment...so reset our state until we pass the next node
								bUpdate = true;
								iHillType = HILL_TYPE_NONE;
							}
						}
					}
				}

				if ( bUpdate )
				{
					m_iCurrentHillType = iHillType;
					HandleTrainMovement();
				}
			}
		}
	}
}

void CTeamTrainWatcher::HandleSparks( bool bSparks )
{
	if ( IsDisabled() || !m_bHandleTrainMovement )
		return;

	for ( int i = 0 ; i < m_Sparks.Count() ; i++ )
	{
		CEnvSpark* pSpark = m_Sparks[i].Get();
		if ( pSpark && ( pSpark->IsSparking() != bSparks ) )
		{
			if ( bSparks )
			{
				pSpark->StartSpark();
			}
			else
			{
				pSpark->StopSpark();
			}
		}
	}
}

void CTeamTrainWatcher::HandleTrainMovement( bool bStartReceding /* = false */ )
{
	if ( IsDisabled() || !m_bHandleTrainMovement )
		return;

	if ( m_hTrain )
	{
		float flSpeed = 0.0f;

		if ( bStartReceding )
		{
			flSpeed = -0.1f;
			m_bReceding = true;
		}
		else
		{
			// do we have cappers on the train?
			if ( m_nNumCappers > 0 )
			{
				m_bReceding = false;

				if ( m_iCurrentHillType == HILL_TYPE_DOWNHILL )
				{
					flSpeed = 1.0f;
				}
				else
				{
					switch( m_nNumCappers )
					{
					case 1:
						flSpeed = 0.55f;
						break;
					case 2:
						flSpeed = 0.77f;
						break;
					case 3:
					default:
						flSpeed = 1.0f;
						break;
					}
				}
			}
			else if ( m_nNumCappers == -1 )
			{
				// we'll get a -1 for a blocked cart (speed should be 0 for that unless we're on a hill)
				if ( m_iCurrentHillType == HILL_TYPE_DOWNHILL )
				{
					flSpeed = 1.0f;
				}
			}
			else
			{
				// there's nobody on the train, what should it be doing?
				if ( m_flCurrentSpeed > 0 )
				{
					if ( m_iCurrentHillType == HILL_TYPE_DOWNHILL )
					{
						flSpeed = 1.0f;
					}
				}
				else
				{
					// we're rolling backwards
					if ( m_iCurrentHillType == HILL_TYPE_UPHILL )
					{
						flSpeed = -1.0f;
					}
					else
					{
						if ( m_bReceding )
						{
							// resume our previous backup speed
							flSpeed = -0.1f;
						}
					}
				}
			}
		}

		// only need to update the train if our speed has changed
		if ( m_flCurrentSpeed != flSpeed )
		{
			if ( flSpeed >= 0.0f )
			{
				m_bReceding = false;
			}

			m_flCurrentSpeed = flSpeed;
			m_hTrain->SetSpeedDirAccel( m_flCurrentSpeed );

			// handle the sparks under the train
			bool bSparks = false;
			if ( m_flCurrentSpeed < 0 )
			{
				bSparks = true;
			}

			HandleSparks( bSparks );
		}
	}
}

void CTeamTrainWatcher::InputSetSpeedForwardModifier( inputdata_t &inputdata )
{
	InternalSetSpeedForwardModifier( inputdata.value.Float() );
}

void CTeamTrainWatcher::InternalSetSpeedForwardModifier( float flModifier )
{
	if ( IsDisabled() || !m_bHandleTrainMovement )
		return;

	// store the passed value
	float flSpeedForwardModifier = flModifier;
	flSpeedForwardModifier = fabs( flSpeedForwardModifier );

	m_flSpeedForwardModifier = clamp( flSpeedForwardModifier, 0.f, 1.f );

	if ( m_hTrain )
	{
		m_hTrain->SetSpeedForwardModifier( m_flSpeedForwardModifier );
	}
}

void CTeamTrainWatcher::InternalSetNumTrainCappers( int iNumCappers, CBaseEntity *pTrigger )
{
	if ( IsDisabled() )
		return;

	m_nNumCappers = iNumCappers;

	// inputdata.pCaller is hopefully an area capture
	// lets see if its blocked, and not start receding if it is
	CTriggerAreaCapture *pAreaCap = dynamic_cast<CTriggerAreaCapture *>( pTrigger );
	if ( pAreaCap )
	{
		m_bCapBlocked = pAreaCap->IsBlocked();
		m_hAreaCap = pAreaCap;
	}

	if ( iNumCappers <= 0 && !m_bCapBlocked && m_bTrainCanRecede )
	{
		if ( !m_bWaitingToRecede )
		{
			// start receding in [tf_escort_cart_recede_time] seconds
			m_bWaitingToRecede = true;

			if ( TeamplayRoundBasedRules() && TeamplayRoundBasedRules()->InOvertime() )
			{
				m_flRecedeTotalTime = tf_escort_recede_time_overtime.GetFloat();
			}
			else
			{
				m_flRecedeTotalTime = tf_escort_recede_time.GetFloat();
				if ( m_nTrainRecedeTime > 0 )
				{
					m_flRecedeTotalTime = m_nTrainRecedeTime;
				}
			}

			m_flRecedeStartTime = gpGlobals->curtime;
			m_flRecedeTime = m_flRecedeStartTime + m_flRecedeTotalTime;
		}		
	}
	else
	{
		// cancel receding
		m_bWaitingToRecede = false;
		m_flRecedeTime = 0;
	}

	HandleTrainMovement();
}

// only used for train watchers that control the train movement
void CTeamTrainWatcher::SetNumTrainCappers( int iNumCappers, CBaseEntity *pTrigger )
{
	if ( IsDisabled() || !m_bHandleTrainMovement )
		return;

	InternalSetNumTrainCappers( iNumCappers, pTrigger );
}

void CTeamTrainWatcher::InputSetNumTrainCappers( inputdata_t &inputdata )
{
	InternalSetNumTrainCappers( inputdata.value.Int(), inputdata.pCaller );
}

void CTeamTrainWatcher::InputSetTrainRecedeTime( inputdata_t &inputdata )
{
	int nSeconds = inputdata.value.Int();
	if ( nSeconds >= 0 )
	{
		m_nTrainRecedeTime = nSeconds;
	}
	else
	{
		m_nTrainRecedeTime = 0;
	}
}

void CTeamTrainWatcher::InputSetTrainCanRecede( inputdata_t &inputdata )
{
	m_bTrainCanRecede = inputdata.value.Bool();
}

void CTeamTrainWatcher::InputOnStartOvertime( inputdata_t &inputdata )
{
	// recalculate the recede time
	if ( m_bWaitingToRecede )
	{
		float flRecedeTimeRemaining = m_flRecedeTime - gpGlobals->curtime;
		float flOvertimeRecedeLen = tf_escort_recede_time_overtime.GetFloat();

		// drop to overtime recede time if it's more than that
		if ( flRecedeTimeRemaining > flOvertimeRecedeLen )
		{
			m_flRecedeTotalTime = flOvertimeRecedeLen;
			m_flRecedeStartTime = gpGlobals->curtime;
			m_flRecedeTime = m_flRecedeStartTime + m_flRecedeTotalTime;
		}
	}
}

#ifdef GLOWS_ENABLE
void CTeamTrainWatcher::FindGlowEntity( void )
{
	if ( m_hTrain && ( m_hTrain->GetEntityName() != NULL_STRING ) )
	{
		string_t iszTrainName = m_hTrain->GetEntityName();
		CBaseEntity *pGlowEnt = NULL;

		// first try to find a phys_constraint relationship with the train
		CPhysFixed *pPhysConstraint = dynamic_cast<CPhysFixed*>( gEntList.FindEntityByClassname( NULL, "phys_constraint" ) );
		while ( pPhysConstraint )
		{
			string_t iszName1 = pPhysConstraint->GetNameAttach1();
			string_t iszName2 = pPhysConstraint->GetNameAttach2();

			if ( iszTrainName == iszName1 )
			{
				pGlowEnt = gEntList.FindEntityByName( NULL, STRING( iszName2 ) );
				break;
			}
			else if ( iszTrainName == iszName2 )
			{
				pGlowEnt = gEntList.FindEntityByName( NULL, STRING( iszName1 ) );
				break;
			}
			
			pPhysConstraint = dynamic_cast<CPhysFixed*>( gEntList.FindEntityByClassname( pPhysConstraint, "phys_constraint" ) );
		}

		if ( !pGlowEnt )
		{
			// if we're here, we haven't found the glow entity yet...try all of the prop_dynamic entities
			CDynamicProp *pPropDynamic = dynamic_cast<CDynamicProp*>( gEntList.FindEntityByClassname( NULL, "prop_dynamic" ) );
			while ( pPropDynamic )
			{
				if ( pPropDynamic->GetParent() == m_hTrain )
				{
					pGlowEnt = pPropDynamic;
					break;
				}

				pPropDynamic = dynamic_cast<CDynamicProp*>( gEntList.FindEntityByClassname( pPropDynamic, "prop_dynamic" ) );
			}
		}

		// if we still haven't found a glow entity, just have the CFuncTrackTrain glow
		if ( !pGlowEnt )
		{
			pGlowEnt = m_hTrain.Get();
		}

		if ( pGlowEnt )
		{
			pGlowEnt->SetTransmitState( FL_EDICT_ALWAYS );
			m_hGlowEnt.Set( pGlowEnt );
		}
	}
}
#endif // GLOWS_ENABLE

// ==========================================================
// given a start node and a list of goal nodes
// calculate the distance between each
// ==========================================================
void CTeamTrainWatcher::WatcherActivate( void )
{		
	m_flRecedeTime = 0;
	m_bWaitingToRecede = false;
	m_bCapBlocked = false;
	m_flNextSpeakForwardConceptTime = 0;
	m_hAreaCap = NULL;
	m_flTrainDistanceFromStart = 0.0f;

	m_bAlarmPlayed = false;

	m_Sparks.Purge();

	StopCaptureAlarm();

	// init our train
	m_hTrain = dynamic_cast<CFuncTrackTrain*>( gEntList.FindEntityByName( NULL, m_iszTrain ) );
	if ( !m_hTrain )
	{
		Warning("%s failed to find train named '%s'\n", GetClassname(), STRING( m_iszTrain ) );
	}

	// find the trigger area that will give us movement updates and find the sparks (if we're going to handle the train movement)
	if ( m_bHandleTrainMovement )
	{
		if ( m_hTrain )
		{
			for ( int i=0; i<ITriggerAreaCaptureAutoList::AutoList().Count(); ++i )
			{
				CTriggerAreaCapture *pArea = static_cast< CTriggerAreaCapture * >( ITriggerAreaCaptureAutoList::AutoList()[i] );
				if ( pArea->GetParent() == m_hTrain.Get() )
				{
					// this is the capture area we care about, so let it know that we want updates on the capture numbers
					pArea->SetTrainWatcher( this );
					break;
				}
			}
		}

		// init the sprites (if any)
		CEnvSpark *pSpark = dynamic_cast<CEnvSpark*>( gEntList.FindEntityByName( NULL, m_iszSparkName ) );
		while ( pSpark )
		{
			m_Sparks.AddToTail( pSpark );
			pSpark = dynamic_cast<CEnvSpark*>( gEntList.FindEntityByName( pSpark, m_iszSparkName ) );
		}
	}

	// init our array of path_tracks linked to control points
	m_iNumCPLinks = 0;

	int i;
	for ( i = 0 ; i < MAX_CONTROL_POINTS ; i++ )
	{
		CPathTrack *pPathTrack = dynamic_cast<CPathTrack*>( gEntList.FindEntityByName( NULL, m_iszLinkedPathTracks[i] ) );
		CTeamControlPoint *pCP = dynamic_cast<CTeamControlPoint*>( gEntList.FindEntityByName( NULL, m_iszLinkedCPs[i] ) );
		if ( pPathTrack && pCP )
		{
			m_CPLinks[m_iNumCPLinks].hPathTrack = pPathTrack;
			m_CPLinks[m_iNumCPLinks].hCP = pCP;
			m_CPLinks[m_iNumCPLinks].flDistanceFromStart = 0;	// filled in when we parse the nodes
			m_CPLinks[m_iNumCPLinks].bAlertPlayed = false;
			m_iNumCPLinks++;
		}
	}

	// init our start and goal nodes
	m_hStartNode = dynamic_cast<CPathTrack*>( gEntList.FindEntityByName( NULL, m_iszStartNode ) );
	if ( !m_hStartNode )
	{
		Warning("%s failed to find path_track named '%s'\n", GetClassname(), STRING(m_iszStartNode) );
	}

	m_hGoalNode = dynamic_cast<CPathTrack*>( gEntList.FindEntityByName( NULL, m_iszGoalNode ) );
	if ( !m_hGoalNode )
	{
		Warning("%s failed to find path_track named '%s'\n", GetClassname(), STRING(m_iszGoalNode) );
	}

	m_flTotalPathDistance = 0.0f;

	CUtlVector< float > hillData;
	bool bOnHill = false;

	bool bDownHillData[TEAM_TRAIN_MAX_HILLS];
	Q_memset( bDownHillData, 0, sizeof( bDownHillData ) );
	int iHillCount = 0;

	if( m_hStartNode.Get() && m_hGoalNode.Get() )
	{
		CPathTrack *pNode = m_hStartNode;
		CPathTrack *pPrev = pNode;
		CPathTrack *pHillStart = NULL;
		pNode = pNode->GetNext();
		int iHillType = HILL_TYPE_NONE;

		// don't check the start node for links. If it's linked, it will have 0 distance anyway
		while ( pNode )
		{
			Vector dir = pNode->GetLocalOrigin() - pPrev->GetLocalOrigin();
			float length = dir.Length();

			m_flTotalPathDistance += length;

			// gather our hill data for the HUD
			if ( pNode->GetHillType() != iHillType )
			{
				if ( !bOnHill ) // we're at the start of a hill
				{
					hillData.AddToTail( m_flTotalPathDistance );
					bOnHill = true;
					pHillStart = pNode;

					if ( iHillCount < TEAM_TRAIN_MAX_HILLS )
					{
						bDownHillData[iHillCount] = pNode->IsDownHill() ? true : false;
						iHillCount++;
					}
				}
				else // we're at the end of a hill
				{
					float flDistance = m_flTotalPathDistance - length; // subtract length because the prev node was the end of the hill (not this one)

					if ( pHillStart && ( pHillStart == pPrev ) )
					{
						flDistance = m_flTotalPathDistance; // we had a single node marked as a hill, so we'll use the current distance as the next marker
					}

					hillData.AddToTail( flDistance ); 

					// is our current node the start of another hill?
					if ( pNode->GetHillType() != HILL_TYPE_NONE )
					{
						hillData.AddToTail( m_flTotalPathDistance );
						bOnHill = true;
						pHillStart = pNode;

						if ( iHillCount < TEAM_TRAIN_MAX_HILLS )
						{
							bDownHillData[iHillCount] = pNode->IsDownHill() ? true : false;
							iHillCount++;
						}
					}
					else
					{
						bOnHill = false;
						pHillStart = NULL;
					}
				}

				iHillType = pNode->GetHillType();
			}

			// if pNode is one of our cp nodes, store its distance from m_hStartNode
			for ( i = 0 ; i < m_iNumCPLinks ; i++ )
			{
				if ( m_CPLinks[i].hPathTrack == pNode )
				{
					m_CPLinks[i].flDistanceFromStart = m_flTotalPathDistance;
					break;
				}
			}

			if ( pNode == m_hGoalNode )
				break;

			pPrev = pNode;
			pNode = pNode->GetNext();
		}
	}

	// if we don't have an even number of entries in our hill data (beginning/end) add the final distance
	if ( ( hillData.Count() % 2 ) != 0 )
	{
		hillData.AddToTail( m_flTotalPathDistance );
	}

	if ( ObjectiveResource() )
	{
		ObjectiveResource()->ResetHillData( GetTeamNumber() );

		// convert our hill data into 0-1 percentages for networking
		if ( m_flTotalPathDistance > 0 && hillData.Count() > 0 )
		{
			i = 0;
	 		while ( i < hillData.Count() )
			{
				if ( i < TEAM_TRAIN_HILLS_ARRAY_SIZE - 1 ) // - 1 because we want to use 2 entries
				{
					// add/subtract to the hill start/end to fix rounding errors in the HUD when the train
					// stops at the bottom/top of a hill but the HUD thinks the train is still on the hill
					ObjectiveResource()->SetHillData( GetTeamNumber(), (hillData[i] / m_flTotalPathDistance) + 0.005f, (hillData[i+1] / m_flTotalPathDistance) - 0.005f, bDownHillData[i/2] );
				}
				i = i + 2;
			}
		}
	}
 
	// We have total distance and increments in our links array
	for ( i=0;i<m_iNumCPLinks;i++ )
	{
		int iCPIndex = m_CPLinks[i].hCP.Get()->GetPointIndex();
// This can be pulled once DoD includes team_objectiveresource.* and c_team_objectiveresource.*
#ifndef DOD_DLL 
		ObjectiveResource()->SetTrainPathDistance( iCPIndex, m_CPLinks[i].flDistanceFromStart / m_flTotalPathDistance );
#endif
	}

#ifdef GLOWS_ENABLE
	FindGlowEntity();
#endif // GLOWS_ENABLE

	InternalSetSpeedForwardModifier( m_flSpeedForwardModifier );

	SetContextThink( &CTeamTrainWatcher::WatcherThink, gpGlobals->curtime + 0.1, TW_THINK );
}

void CTeamTrainWatcher::StopCaptureAlarm( void )
{
	if ( m_pAlarm )
	{
		CSoundEnvelopeController::GetController().SoundDestroy( m_pAlarm );
		m_pAlarm = NULL;
		m_flAlarmEndTime = -1.0f;
	}

	SetContextThink( NULL, 0, TW_ALARM_THINK );
}

void CTeamTrainWatcher::StartCaptureAlarm( CTeamControlPoint *pPoint )
{
	StopCaptureAlarm();

	if ( pPoint )
	{
		CReliableBroadcastRecipientFilter filter;
		CSoundEnvelopeController &controller = CSoundEnvelopeController::GetController();
		m_pAlarm = controller.SoundCreate( filter, pPoint->entindex(), CHAN_STATIC, TEAM_TRAIN_ALARM, ATTN_NORM );
		controller.Play( m_pAlarm, 1.0, PITCH_NORM );

		m_flAlarmEndTime = gpGlobals->curtime + MAX_ALARM_TIME_NO_RECEDE;
	}
}

void CTeamTrainWatcher::PlayCaptureAlert( CTeamControlPoint *pPoint, bool bFinalPointInMap )
{
	if ( !pPoint )
		return;

	if ( TeamplayRoundBasedRules() )
	{
		TeamplayRoundBasedRules()->PlayTrainCaptureAlert( pPoint, bFinalPointInMap );
	}
}


ConVar tf_show_train_path( "tf_show_train_path", "0", FCVAR_CHEAT );

void CTeamTrainWatcher::WatcherThink( void )
{
	if ( m_bWaitingToRecede )
	{
		if ( m_flRecedeTime < gpGlobals->curtime )
		{
			m_bWaitingToRecede = false;

			// don't actually recede in overtime
			if ( TeamplayRoundBasedRules() && !TeamplayRoundBasedRules()->InOvertime() )
			{
				// fire recede output
				m_OnTrainStartRecede.FireOutput( this, this );
				HandleTrainMovement( true );
			}
		}
	}

	bool bDisableAlarm = (TeamplayRoundBasedRules() && TeamplayRoundBasedRules()->State_Get() != GR_STATE_RND_RUNNING);
	if ( bDisableAlarm )
	{
		StopCaptureAlarm();
	}

	// given its next node, we can walk the nodes and find the linear
	// distance to the next cp node, or to the goal node

	CFuncTrackTrain *pTrain = m_hTrain;
	if ( pTrain )
	{
		int iOldTrainSpeedLevel = m_iTrainSpeedLevel;

		// how fast is the train moving?
		float flSpeed = pTrain->GetDesiredSpeed();

		// divide speed into regions
		// anything negative is -1

		if ( flSpeed < 0 )
		{
			m_iTrainSpeedLevel = -1;

			// even though our desired speed might be negative,
			// our actual speed might be zero if we're at a dead end...
			// this will turn off the < image when the train is done moving backwards
			if ( pTrain->GetCurrentSpeed() == 0 )
			{
				m_iTrainSpeedLevel = 0;
			}
		}
		else if ( flSpeed > m_flSpeedLevels[2] )
		{
			m_iTrainSpeedLevel = 3;
		}
		else if ( flSpeed > m_flSpeedLevels[1] )
		{
			m_iTrainSpeedLevel = 2;
		}
		else if ( flSpeed > m_flSpeedLevels[0] )
		{
			m_iTrainSpeedLevel = 1;
		}
		else
		{
			m_iTrainSpeedLevel = 0;
		}

		if ( m_iTrainSpeedLevel != iOldTrainSpeedLevel )
		{
			// make sure the sparks are off if we're not moving backwards anymore
			if ( m_bHandleTrainMovement )
			{
				if ( m_iTrainSpeedLevel == 0 && iOldTrainSpeedLevel != 0 )
				{
					HandleSparks( false );
				}
			}

			// play any concepts that we might need to play		
			if ( TeamplayRoundBasedRules() )
			{
				if ( m_iTrainSpeedLevel == 0 && iOldTrainSpeedLevel != 0 )
				{
					TeamplayRoundBasedRules()->HaveAllPlayersSpeakConceptIfAllowed( MP_CONCEPT_CART_STOP );
					m_flNextSpeakForwardConceptTime = 0;
				}
				else if ( m_iTrainSpeedLevel < 0 && iOldTrainSpeedLevel == 0 )
				{
					TeamplayRoundBasedRules()->HaveAllPlayersSpeakConceptIfAllowed( MP_CONCEPT_CART_MOVING_BACKWARD );
					m_flNextSpeakForwardConceptTime = 0;
				}
			}
		}

		if ( m_iTrainSpeedLevel > 0 && m_flNextSpeakForwardConceptTime < gpGlobals->curtime )
		{
			if ( m_hAreaCap.Get() )
			{
				for ( int i = 1; i <= gpGlobals->maxClients; i++ )
				{
					CBaseMultiplayerPlayer *pPlayer = ToBaseMultiplayerPlayer( UTIL_PlayerByIndex( i ) );
					if ( pPlayer )
					{
						if ( m_hAreaCap->IsTouching( pPlayer ) )
						{
							pPlayer->SpeakConceptIfAllowed( MP_CONCEPT_CART_MOVING_FORWARD );
						}
					}
				}
			}

			m_flNextSpeakForwardConceptTime = gpGlobals->curtime + 3.0;
		}

		// what percent progress are we at?
		CPathTrack *pNode = ( pTrain->m_ppath ) ? pTrain->m_ppath->GetNext() : NULL;

		// if we're moving backwards, GetNext is going to be wrong
		if ( flSpeed < 0 )
		{
			pNode = pTrain->m_ppath;
		}

		if ( pNode )
		{
			float flDistanceToGoal = 0;

			// distance to next node
			Vector vecDir = pNode->GetLocalOrigin() - pTrain->GetLocalOrigin();
			flDistanceToGoal = vecDir.Length();

			// distance of next node to goal node
			if ( pNode && pNode != m_hGoalNode )
			{
				// walk this until we get to goal node, or a dead end
				CPathTrack *pPrev = pNode;
				pNode = pNode->GetNext();
				while ( pNode )
				{
					vecDir = pNode->GetLocalOrigin() - pPrev->GetLocalOrigin();
					flDistanceToGoal += vecDir.Length();

					if ( pNode == m_hGoalNode )
						break;

					pPrev = pNode;
					pNode = pNode->GetNext();
				}
			}

			if ( m_flTotalPathDistance <= 0 )
			{
				Assert( !"No path distance in team_train_watcher\n" );
				m_flTotalPathDistance = 1;
			}

			m_flTotalProgress = clamp( 1.0 - ( flDistanceToGoal / m_flTotalPathDistance ), 0.0, 1.0 );

			m_flTrainDistanceFromStart = m_flTotalPathDistance - flDistanceToGoal;

			// play alert sounds if necessary
			for ( int iCount = 0 ; iCount < m_iNumCPLinks ; iCount++ )
			{
				if ( m_flTrainDistanceFromStart < m_CPLinks[iCount].flDistanceFromStart - TEAM_TRAIN_ALERT_DISTANCE )
				{
					// back up twice the alert distance before resetting our flag to play the warning again
					if ( ( m_flTrainDistanceFromStart < m_CPLinks[iCount].flDistanceFromStart - ( TEAM_TRAIN_ALERT_DISTANCE * 2 ) ) || // has receded back twice the alert distance or...
						 ( !m_bTrainCanRecede ) ) // used to catch the case where the train doesn't normally recede but has rolled back down a hill away from the CP
					{
						// reset our alert flag
						m_CPLinks[iCount].bAlertPlayed = false;
					}
				}
				else
				{
					if ( m_flTrainDistanceFromStart < m_CPLinks[iCount].flDistanceFromStart && !m_CPLinks[iCount].bAlertPlayed )
					{
						m_CPLinks[iCount].bAlertPlayed = true;
						bool bFinalPointInMap = false;

						CTeamControlPoint *pCurrentPoint = m_CPLinks[iCount].hCP.Get();
						CTeamControlPointMaster *pMaster = g_hControlPointMasters.Count() ? g_hControlPointMasters[0] : NULL;
						if ( pMaster )
						{
							// if we're not playing mini-rounds 
							if ( !pMaster->PlayingMiniRounds() )  
							{
								for ( int i = FIRST_GAME_TEAM ; i < MAX_CONTROL_POINT_TEAMS ; i++ )
								{
									if ( ObjectiveResource() && ObjectiveResource()->TeamCanCapPoint( pCurrentPoint->GetPointIndex(), i ) )
									{
										if ( pMaster->WouldNewCPOwnerWinGame( pCurrentPoint, i ) )
										{
											bFinalPointInMap = true;
										}
									}
								}
							}
							else 
							{
								// or this is the last round
								if ( pMaster->NumPlayableControlPointRounds() == 1 )
								{
									CTeamControlPointRound *pRound = pMaster->GetCurrentRound();
									if ( pRound )
									{
										for ( int i = FIRST_GAME_TEAM ; i < MAX_CONTROL_POINT_TEAMS ; i++ )
										{
											if ( ObjectiveResource() && ObjectiveResource()->TeamCanCapPoint( pCurrentPoint->GetPointIndex(), i ) )
											{
												if ( pRound->WouldNewCPOwnerWinGame( pCurrentPoint, i ) )
												{
													bFinalPointInMap = true;
												}
											}
										}
									}
								}
							}
						}

						PlayCaptureAlert( pCurrentPoint, bFinalPointInMap );
					}
				}
			}

			// check to see if we need to start or stop the alarm
			if ( flDistanceToGoal <= TEAM_TRAIN_ALARM_DISTANCE )
			{
				if ( ObjectiveResource() )
				{
					ObjectiveResource()->SetTrackAlarm( GetTeamNumber(), true );
				}

				if ( !bDisableAlarm )
				{
					if ( !m_pAlarm )
					{
						if ( m_iNumCPLinks > 0 && !m_bAlarmPlayed )
						{
							// start the alarm at the final point
							StartCaptureAlarm( m_CPLinks[m_iNumCPLinks-1].hCP.Get() );
							m_bAlarmPlayed = true; // used to prevent the alarm from starting again on maps where the train doesn't recede (alarm loops for short time then only plays singles)
						}
					}
					else
					{
						if ( !m_bTrainCanRecede ) // if the train won't recede, we only want to play the alarm for a short time
						{
							if ( m_flAlarmEndTime > 0 && m_flAlarmEndTime < gpGlobals->curtime )
							{
								StopCaptureAlarm();
								SetContextThink( &CTeamTrainWatcher::WatcherAlarmThink, gpGlobals->curtime + TW_ALARM_THINK_INTERVAL, TW_ALARM_THINK );
							}
						}
					}
				}
			}
			else
			{
				if ( ObjectiveResource() )
				{
					ObjectiveResource()->SetTrackAlarm( GetTeamNumber(), false );
				}

				StopCaptureAlarm();
				m_bAlarmPlayed = false;
			}
		}

		if ( tf_show_train_path.GetBool() )
		{
			CPathTrack *nextNode = NULL;
			CPathTrack *node = m_hStartNode;

			CPathTrack::BeginIteration();
			while( node )
			{
				node->Visit();
				nextNode = node->GetNext();

				if ( !nextNode || nextNode->HasBeenVisited() )
					break;

				NDebugOverlay::Line( node->GetAbsOrigin(), nextNode->GetAbsOrigin(), 255, 255, 0, true, NDEBUG_PERSIST_TILL_NEXT_SERVER );

				node = nextNode;
			}
			CPathTrack::EndIteration();

			// show segment of path train is actually on
			node = pTrain->m_ppath;
			if ( node && node->GetNext() )
			{
				NDebugOverlay::HorzArrow( node->GetAbsOrigin(), node->GetNext()->GetAbsOrigin(), 5.0f, 255, 0, 0, 255, true, NDEBUG_PERSIST_TILL_NEXT_SERVER );
			}
		}
	}

	SetContextThink( &CTeamTrainWatcher::WatcherThink, gpGlobals->curtime + 0.1, TW_THINK );
}

void CTeamTrainWatcher::WatcherAlarmThink( void )
{
	CTeamControlPoint *pPoint = m_CPLinks[m_iNumCPLinks-1].hCP.Get();
	if ( pPoint )
	{
		pPoint->EmitSound( TEAM_TRAIN_ALARM_SINGLE );
	}

	SetContextThink( &CTeamTrainWatcher::WatcherAlarmThink, gpGlobals->curtime + TW_ALARM_THINK_INTERVAL, TW_ALARM_THINK );
}

CBaseEntity *CTeamTrainWatcher::GetTrainEntity( void )
{
	return m_hTrain.Get();
}

bool CTeamTrainWatcher::TimerMayExpire( void )
{
	if ( IsDisabled() )
	{
		return true;
	}

	// Still in overtime if we're waiting to recede
	if ( m_bWaitingToRecede )
		return false;

	// capture blocked so we're not receding, but game shouldn't end
	if ( m_bCapBlocked )
		return false;

	// not waiting, so we're capping, in which case the area capture
	// will not let us expire
	return true;
}


// Project the given position onto the track and return the point and how far along that projected position is
void CTeamTrainWatcher::ProjectPointOntoPath( const Vector &pos, Vector *posOnPathResult, float *distanceAlongPathResult ) const
{
	CPathTrack *nextNode = NULL;
	CPathTrack *node = m_hStartNode;

	Vector toPos;
	Vector alongPath;
	float distanceAlong = 0.0f;

	Vector closestPointOnPath = vec3_origin;
	float closestPerpendicularDistanceSq = FLT_MAX;
	float closestDistanceAlongPath = FLT_MAX;

	CPathTrack::BeginIteration();
	while( node )
	{
		node->Visit();
		nextNode = node->GetNext();

		if ( !nextNode || nextNode->HasBeenVisited() )
			break;

		alongPath = nextNode->GetAbsOrigin() - node->GetAbsOrigin();
		float segmentLength = alongPath.NormalizeInPlace();

		toPos = pos - node->GetAbsOrigin();
		float segmentOverlap = DotProduct( toPos, alongPath );

		if ( segmentOverlap >= 0.0f && segmentOverlap < segmentLength )
		{
			// projection is within segment bounds
			Vector onPath = node->GetAbsOrigin() + alongPath * segmentOverlap;

			float perpendicularDistanceSq = ( onPath - pos ).LengthSqr();
			if ( perpendicularDistanceSq < closestPerpendicularDistanceSq )
			{
				closestPointOnPath = onPath;
				closestPerpendicularDistanceSq = perpendicularDistanceSq;
				closestDistanceAlongPath = distanceAlong + segmentOverlap;
			}
		}

		distanceAlong += segmentLength;
		node = nextNode;
	}
	CPathTrack::EndIteration();

	if ( posOnPathResult )
	{
		*posOnPathResult = closestPointOnPath;
	}

	if ( distanceAlongPathResult )
	{
		*distanceAlongPathResult = closestDistanceAlongPath;
	}
}


// Return true if the given position is farther down the track than the train is
bool CTeamTrainWatcher::IsAheadOfTrain( const Vector &pos ) const
{
	float distanceAlongPath;
	ProjectPointOntoPath( pos, NULL, &distanceAlongPath );

	return ( distanceAlongPath > m_flTrainDistanceFromStart );
}


// return true if the train is almost at the next checkpoint
bool CTeamTrainWatcher::IsTrainNearCheckpoint( void ) const
{
	for( int i = 0; i < m_iNumCPLinks ; ++i )
	{
		if ( m_flTrainDistanceFromStart > m_CPLinks[i].flDistanceFromStart - TEAM_TRAIN_ALERT_DISTANCE &&
			m_flTrainDistanceFromStart < m_CPLinks[i].flDistanceFromStart )
		{
			return true;
		}
	}

	return false;
}


// return true if the train hasn't left its starting position yet
bool CTeamTrainWatcher::IsTrainAtStart( void ) const
{
	return ( m_flTrainDistanceFromStart < TEAM_TRAIN_ALARM_DISTANCE );
}


// return world space location of next checkpoint along the path
Vector CTeamTrainWatcher::GetNextCheckpointPosition( void ) const
{
	for( int i = 0; i < m_iNumCPLinks ; ++i )
	{
		if ( m_flTrainDistanceFromStart < m_CPLinks[i].flDistanceFromStart )
		{
			return m_CPLinks[i].hPathTrack->GetAbsOrigin();
		}
	}

	Assert( !"No checkpoint found in team train watcher\n" );
	return vec3_origin;
}

#if defined( STAGING_ONLY ) && defined( TF_DLL )
CON_COMMAND_F( tf_dumptrainstats, "Dump the stats for the current train watcher to the console", FCVAR_GAMEDLL )
{
	// Listenserver host or rcon access only!
	if ( !UTIL_IsCommandIssuedByServerAdmin() )
		return;

	CTeamTrainWatcher *pWatcher = NULL;
	while( ( pWatcher = dynamic_cast< CTeamTrainWatcher * >( gEntList.FindEntityByClassname( pWatcher, "team_train_watcher" ) ) ) != NULL )
	{
		pWatcher->DumpStats();
	}
}

void CTeamTrainWatcher::DumpStats( void )
{
	float flLastPosition = 0.0f;
	float flTotalDistance = 0.0f;
 	char szOutput[2048];
	char szTemp[256];

	V_strcpy_safe( szOutput, "\n\nTrain Watcher stats for team " );
	V_strcat_safe( szOutput, ( GetTeamNumber() == TF_TEAM_RED ) ? "Red\n" : "Blue\n" );

	for( int i = 0; i < m_iNumCPLinks ; ++i )
	{
		float flDistance = m_CPLinks[i].flDistanceFromStart - flLastPosition;
		if ( i == 0 )
		{
			V_sprintf_safe( szTemp, "\tControl Point: %d\tDistance from start: %0.2f\n", i + 1, flDistance );
		}
		else
		{
			V_sprintf_safe( szTemp, "\tControl Point: %d\tDistance from previous point: %0.2f\n", i + 1, flDistance );
		}
		V_strcat_safe( szOutput, szTemp );
		flTotalDistance += flDistance;
		flLastPosition = m_CPLinks[i].flDistanceFromStart;
	}

	V_sprintf_safe( szTemp, "\tTotal Distance: %0.2f\n\n", flTotalDistance ); 
	V_strcat_safe( szOutput, szTemp );
	Msg( "%s", szOutput );
}
#endif // STAGING_ONLY && TF_DLL