5859 lines
162 KiB
C++
5859 lines
162 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: The TF Game rules
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "cs_gamerules.h"
|
|
#include "cs_ammodef.h"
|
|
#include "weapon_csbase.h"
|
|
#include "cs_shareddefs.h"
|
|
#include "KeyValues.h"
|
|
#include "cs_achievement_constants.h"
|
|
#include "fmtstr.h"
|
|
|
|
#ifdef CLIENT_DLL
|
|
|
|
#include "networkstringtable_clientdll.h"
|
|
#include "utlvector.h"
|
|
|
|
#else
|
|
|
|
#include "bot.h"
|
|
#include "utldict.h"
|
|
#include "cs_player.h"
|
|
#include "cs_team.h"
|
|
#include "cs_gamerules.h"
|
|
#include "voice_gamemgr.h"
|
|
#include "igamesystem.h"
|
|
#include "weapon_c4.h"
|
|
#include "mapinfo.h"
|
|
#include "shake.h"
|
|
#include "mapentities.h"
|
|
#include "game.h"
|
|
#include "cs_simple_hostage.h"
|
|
#include "cs_gameinterface.h"
|
|
#include "player_resource.h"
|
|
#include "info_view_parameters.h"
|
|
#include "cs_bot_manager.h"
|
|
#include "cs_bot.h"
|
|
#include "eventqueue.h"
|
|
#include "fmtstr.h"
|
|
#include "teamplayroundbased_gamerules.h"
|
|
#include "gameweaponmanager.h"
|
|
|
|
#include "cs_gamestats.h"
|
|
#include "cs_urlretrieveprices.h"
|
|
#include "networkstringtable_gamedll.h"
|
|
#include "player_resource.h"
|
|
#include "cs_player_resource.h"
|
|
|
|
#if defined( REPLAY_ENABLED )
|
|
#include "replay/ireplaysystem.h"
|
|
#include "replay/iserverreplaycontext.h"
|
|
#include "replay/ireplaysessionrecorder.h"
|
|
#endif // REPLAY_ENABLED
|
|
|
|
#endif
|
|
|
|
|
|
#include "cs_blackmarket.h"
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
#ifndef CLIENT_DLL
|
|
|
|
|
|
#define CS_GAME_STATS_UPDATE 79200 //22 hours
|
|
#define CS_GAME_STATS_UPDATE_PERIOD 7200 // 2 hours
|
|
|
|
extern IUploadGameStats *gamestatsuploader;
|
|
|
|
#if defined( REPLAY_ENABLED )
|
|
extern IReplaySystem *g_pReplay;
|
|
#endif // REPLAY_ENABLED
|
|
|
|
#endif
|
|
|
|
|
|
/**
|
|
* Player hull & eye position for standing, ducking, etc. This version has a taller
|
|
* player height, but goldsrc-compatible collision bounds.
|
|
*/
|
|
static CViewVectors g_CSViewVectors(
|
|
Vector( 0, 0, 64 ), // eye position
|
|
|
|
Vector(-16, -16, 0 ), // hull min
|
|
Vector( 16, 16, 62 ), // hull max
|
|
|
|
Vector(-16, -16, 0 ), // duck hull min
|
|
Vector( 16, 16, 45 ), // duck hull max
|
|
Vector( 0, 0, 47 ), // duck view
|
|
|
|
Vector(-10, -10, -10 ), // observer hull min
|
|
Vector( 10, 10, 10 ), // observer hull max
|
|
|
|
Vector( 0, 0, 14 ) // dead view height
|
|
);
|
|
|
|
|
|
#ifndef CLIENT_DLL
|
|
LINK_ENTITY_TO_CLASS(info_player_terrorist, CPointEntity);
|
|
LINK_ENTITY_TO_CLASS(info_player_counterterrorist,CPointEntity);
|
|
LINK_ENTITY_TO_CLASS(info_player_logo,CPointEntity);
|
|
#endif
|
|
|
|
REGISTER_GAMERULES_CLASS( CCSGameRules );
|
|
|
|
|
|
BEGIN_NETWORK_TABLE_NOBASE( CCSGameRules, DT_CSGameRules )
|
|
#ifdef CLIENT_DLL
|
|
RecvPropBool( RECVINFO( m_bFreezePeriod ) ),
|
|
RecvPropInt( RECVINFO( m_iRoundTime ) ),
|
|
RecvPropFloat( RECVINFO( m_fRoundStartTime ) ),
|
|
RecvPropFloat( RECVINFO( m_flGameStartTime ) ),
|
|
RecvPropInt( RECVINFO( m_iHostagesRemaining ) ),
|
|
RecvPropBool( RECVINFO( m_bMapHasBombTarget ) ),
|
|
RecvPropBool( RECVINFO( m_bMapHasRescueZone ) ),
|
|
RecvPropBool( RECVINFO( m_bLogoMap ) ),
|
|
RecvPropBool( RECVINFO( m_bBlackMarket ) )
|
|
#else
|
|
SendPropBool( SENDINFO( m_bFreezePeriod ) ),
|
|
SendPropInt( SENDINFO( m_iRoundTime ), 16 ),
|
|
SendPropFloat( SENDINFO( m_fRoundStartTime ), 32, SPROP_NOSCALE ),
|
|
SendPropFloat( SENDINFO( m_flGameStartTime ), 32, SPROP_NOSCALE ),
|
|
SendPropInt( SENDINFO( m_iHostagesRemaining ), 4 ),
|
|
SendPropBool( SENDINFO( m_bMapHasBombTarget ) ),
|
|
SendPropBool( SENDINFO( m_bMapHasRescueZone ) ),
|
|
SendPropBool( SENDINFO( m_bLogoMap ) ),
|
|
SendPropBool( SENDINFO( m_bBlackMarket ) )
|
|
#endif
|
|
END_NETWORK_TABLE()
|
|
|
|
|
|
LINK_ENTITY_TO_CLASS( cs_gamerules, CCSGameRulesProxy );
|
|
IMPLEMENT_NETWORKCLASS_ALIASED( CSGameRulesProxy, DT_CSGameRulesProxy )
|
|
|
|
|
|
#ifdef CLIENT_DLL
|
|
void RecvProxy_CSGameRules( const RecvProp *pProp, void **pOut, void *pData, int objectID )
|
|
{
|
|
CCSGameRules *pRules = CSGameRules();
|
|
Assert( pRules );
|
|
*pOut = pRules;
|
|
}
|
|
|
|
BEGIN_RECV_TABLE( CCSGameRulesProxy, DT_CSGameRulesProxy )
|
|
RecvPropDataTable( "cs_gamerules_data", 0, 0, &REFERENCE_RECV_TABLE( DT_CSGameRules ), RecvProxy_CSGameRules )
|
|
END_RECV_TABLE()
|
|
#else
|
|
void* SendProxy_CSGameRules( const SendProp *pProp, const void *pStructBase, const void *pData, CSendProxyRecipients *pRecipients, int objectID )
|
|
{
|
|
CCSGameRules *pRules = CSGameRules();
|
|
Assert( pRules );
|
|
return pRules;
|
|
}
|
|
|
|
BEGIN_SEND_TABLE( CCSGameRulesProxy, DT_CSGameRulesProxy )
|
|
SendPropDataTable( "cs_gamerules_data", 0, &REFERENCE_SEND_TABLE( DT_CSGameRules ), SendProxy_CSGameRules )
|
|
END_SEND_TABLE()
|
|
#endif
|
|
|
|
|
|
|
|
ConVar ammo_50AE_max( "ammo_50AE_max", "35", FCVAR_REPLICATED );
|
|
ConVar ammo_762mm_max( "ammo_762mm_max", "90", FCVAR_REPLICATED );
|
|
ConVar ammo_556mm_max( "ammo_556mm_max", "90", FCVAR_REPLICATED );
|
|
ConVar ammo_556mm_box_max( "ammo_556mm_box_max", "200", FCVAR_REPLICATED );
|
|
ConVar ammo_338mag_max( "ammo_338mag_max", "30", FCVAR_REPLICATED );
|
|
ConVar ammo_9mm_max( "ammo_9mm_max", "120", FCVAR_REPLICATED );
|
|
ConVar ammo_buckshot_max( "ammo_buckshot_max", "32", FCVAR_REPLICATED );
|
|
ConVar ammo_45acp_max( "ammo_45acp_max", "100", FCVAR_REPLICATED );
|
|
ConVar ammo_357sig_max( "ammo_357sig_max", "52", FCVAR_REPLICATED );
|
|
ConVar ammo_57mm_max( "ammo_57mm_max", "100", FCVAR_REPLICATED );
|
|
ConVar ammo_hegrenade_max( "ammo_hegrenade_max", "1", FCVAR_REPLICATED );
|
|
ConVar ammo_flashbang_max( "ammo_flashbang_max", "2", FCVAR_REPLICATED );
|
|
ConVar ammo_smokegrenade_max( "ammo_smokegrenade_max", "1", FCVAR_REPLICATED );
|
|
|
|
//ConVar mp_dynamicpricing( "mp_dynamicpricing", "0", FCVAR_REPLICATED, "Enables or Disables the dynamic weapon prices" );
|
|
|
|
|
|
extern ConVar sv_stopspeed;
|
|
|
|
ConVar mp_buytime(
|
|
"mp_buytime",
|
|
"1.5",
|
|
FCVAR_REPLICATED,
|
|
"How many minutes after round start players can buy items for.",
|
|
true, 0.25,
|
|
false, 0 );
|
|
|
|
ConVar mp_playerid(
|
|
"mp_playerid",
|
|
"0",
|
|
FCVAR_REPLICATED,
|
|
"Controls what information player see in the status bar: 0 all names; 1 team names; 2 no names",
|
|
true, 0,
|
|
true, 2 );
|
|
|
|
ConVar mp_playerid_delay(
|
|
"mp_playerid_delay",
|
|
"0.5",
|
|
FCVAR_REPLICATED,
|
|
"Number of seconds to delay showing information in the status bar",
|
|
true, 0,
|
|
true, 1 );
|
|
|
|
ConVar mp_playerid_hold(
|
|
"mp_playerid_hold",
|
|
"0.25",
|
|
FCVAR_REPLICATED,
|
|
"Number of seconds to keep showing old information in the status bar",
|
|
true, 0,
|
|
true, 1 );
|
|
|
|
ConVar mp_round_restart_delay(
|
|
"mp_round_restart_delay",
|
|
"5.0",
|
|
FCVAR_REPLICATED,
|
|
"Number of seconds to delay before restarting a round after a win",
|
|
true, 0.0f,
|
|
true, 10.0f );
|
|
|
|
ConVar sv_allowminmodels(
|
|
"sv_allowminmodels",
|
|
"1",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"Allow or disallow the use of cl_minmodels on this server." );
|
|
|
|
#ifdef CLIENT_DLL
|
|
|
|
ConVar cl_autowepswitch(
|
|
"cl_autowepswitch",
|
|
"1",
|
|
FCVAR_ARCHIVE | FCVAR_USERINFO,
|
|
"Automatically switch to picked up weapons (if more powerful)" );
|
|
|
|
ConVar cl_autohelp(
|
|
"cl_autohelp",
|
|
"1",
|
|
FCVAR_ARCHIVE | FCVAR_USERINFO,
|
|
"Auto-help" );
|
|
|
|
#else
|
|
|
|
// longest the intermission can last, in seconds
|
|
#define MAX_INTERMISSION_TIME 120
|
|
|
|
// Falling damage stuff.
|
|
#define CS_PLAYER_FATAL_FALL_SPEED 1100 // approx 60 feet
|
|
#define CS_PLAYER_MAX_SAFE_FALL_SPEED 580 // approx 20 feet
|
|
#define CS_DAMAGE_FOR_FALL_SPEED ((float)100 / ( CS_PLAYER_FATAL_FALL_SPEED - CS_PLAYER_MAX_SAFE_FALL_SPEED )) // damage per unit per second.
|
|
|
|
// These entities are preserved each round restart. The rest are removed and recreated.
|
|
static const char *s_PreserveEnts[] =
|
|
{
|
|
"ai_network",
|
|
"ai_hint",
|
|
"cs_gamerules",
|
|
"cs_team_manager",
|
|
"cs_player_manager",
|
|
"env_soundscape",
|
|
"env_soundscape_proxy",
|
|
"env_soundscape_triggerable",
|
|
"env_sun",
|
|
"env_wind",
|
|
"env_fog_controller",
|
|
"func_brush",
|
|
"func_wall",
|
|
"func_buyzone",
|
|
"func_illusionary",
|
|
"func_hostage_rescue",
|
|
"func_bomb_target",
|
|
"infodecal",
|
|
"info_projecteddecal",
|
|
"info_node",
|
|
"info_target",
|
|
"info_node_hint",
|
|
"info_player_counterterrorist",
|
|
"info_player_terrorist",
|
|
"info_map_parameters",
|
|
"keyframe_rope",
|
|
"move_rope",
|
|
"info_ladder",
|
|
"player",
|
|
"point_viewcontrol",
|
|
"scene_manager",
|
|
"shadow_control",
|
|
"sky_camera",
|
|
"soundent",
|
|
"trigger_soundscape",
|
|
"viewmodel",
|
|
"predicted_viewmodel",
|
|
"worldspawn",
|
|
"point_devshot_camera",
|
|
"", // END Marker
|
|
};
|
|
|
|
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
// Voice helper
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
|
|
class CVoiceGameMgrHelper : public IVoiceGameMgrHelper
|
|
{
|
|
public:
|
|
virtual bool CanPlayerHearPlayer( CBasePlayer *pListener, CBasePlayer *pTalker, bool &bProximity )
|
|
{
|
|
// Dead players can only be heard by other dead team mates
|
|
if ( pTalker->IsAlive() == false )
|
|
{
|
|
if ( pListener->IsAlive() == false )
|
|
return ( pListener->InSameTeam( pTalker ) );
|
|
|
|
return false;
|
|
}
|
|
|
|
return ( pListener->InSameTeam( pTalker ) );
|
|
}
|
|
};
|
|
CVoiceGameMgrHelper g_VoiceGameMgrHelper;
|
|
IVoiceGameMgrHelper *g_pVoiceGameMgrHelper = &g_VoiceGameMgrHelper;
|
|
|
|
|
|
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
// Globals.
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
|
|
// NOTE: the indices here must match TEAM_TERRORIST, TEAM_CT, TEAM_SPECTATOR, etc.
|
|
const char *sTeamNames[] =
|
|
{
|
|
"Unassigned",
|
|
"Spectator",
|
|
"TERRORIST",
|
|
"CT"
|
|
};
|
|
|
|
extern ConVar mp_maxrounds;
|
|
|
|
ConVar mp_startmoney(
|
|
"mp_startmoney",
|
|
"800",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"amount of money each player gets when they reset",
|
|
true, 800,
|
|
true, 16000 );
|
|
|
|
ConVar mp_roundtime(
|
|
"mp_roundtime",
|
|
"2.5",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"How many minutes each round takes.",
|
|
true, 1, // min value
|
|
true, 9 // max value
|
|
);
|
|
|
|
ConVar mp_freezetime(
|
|
"mp_freezetime",
|
|
"6",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"how many seconds to keep players frozen when the round starts",
|
|
true, 0, // min value
|
|
true, 60 // max value
|
|
);
|
|
|
|
ConVar mp_c4timer(
|
|
"mp_c4timer",
|
|
"45",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"how long from when the C4 is armed until it blows",
|
|
true, 10, // min value
|
|
true, 90 // max value
|
|
);
|
|
|
|
ConVar mp_limitteams(
|
|
"mp_limitteams",
|
|
"2",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"Max # of players 1 team can have over another (0 disables check)",
|
|
true, 0, // min value
|
|
true, 30 // max value
|
|
);
|
|
|
|
ConVar mp_tkpunish(
|
|
"mp_tkpunish",
|
|
"0",
|
|
FCVAR_REPLICATED,
|
|
"Will a TK'er be punished in the next round? {0=no, 1=yes}" );
|
|
|
|
ConVar mp_autokick(
|
|
"mp_autokick",
|
|
"1",
|
|
FCVAR_REPLICATED,
|
|
"Kick idle/team-killing players" );
|
|
|
|
ConVar mp_spawnprotectiontime(
|
|
"mp_spawnprotectiontime",
|
|
"5",
|
|
FCVAR_REPLICATED,
|
|
"Kick players who team-kill within this many seconds of a round restart." );
|
|
|
|
ConVar mp_humanteam(
|
|
"mp_humanteam",
|
|
"any",
|
|
FCVAR_REPLICATED,
|
|
"Restricts human players to a single team {any, CT, T}" );
|
|
|
|
ConVar mp_ignore_round_win_conditions(
|
|
"mp_ignore_round_win_conditions",
|
|
"0",
|
|
FCVAR_REPLICATED,
|
|
"Ignore conditions which would end the current round");
|
|
|
|
ConCommand EndRound( "endround", &CCSGameRules::EndRound, "End the current round.", FCVAR_CHEAT );
|
|
|
|
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
// Global helper functions.
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
|
|
void InitBodyQue(void)
|
|
{
|
|
// FIXME: Make this work
|
|
}
|
|
|
|
|
|
Vector DropToGround(
|
|
CBaseEntity *pMainEnt,
|
|
const Vector &vPos,
|
|
const Vector &vMins,
|
|
const Vector &vMaxs )
|
|
{
|
|
trace_t trace;
|
|
UTIL_TraceHull( vPos, vPos + Vector( 0, 0, -500 ), vMins, vMaxs, MASK_SOLID, pMainEnt, COLLISION_GROUP_NONE, &trace );
|
|
return trace.endpos;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: This function can be used to find a valid placement location for an entity.
|
|
// Given an origin to start looking from and a minimum radius to place the entity at,
|
|
// it will sweep out a circle around vOrigin and try to find a valid spot (on the ground)
|
|
// where mins and maxs will fit.
|
|
// Input : *pMainEnt - Entity to place
|
|
// &vOrigin - Point to search around
|
|
// fRadius - Radius to search within
|
|
// nTries - Number of tries to attempt
|
|
// &mins - mins of the Entity
|
|
// &maxs - maxs of the Entity
|
|
// &outPos - Return point
|
|
// Output : Returns true and fills in outPos if it found a spot.
|
|
//-----------------------------------------------------------------------------
|
|
bool EntityPlacementTest( CBaseEntity *pMainEnt, const Vector &vOrigin, Vector &outPos, bool bDropToGround )
|
|
{
|
|
// This function moves the box out in each dimension in each step trying to find empty space like this:
|
|
//
|
|
// X
|
|
// X X
|
|
// Step 1: X Step 2: XXX Step 3: XXXXX
|
|
// X X
|
|
// X
|
|
//
|
|
|
|
Vector mins, maxs;
|
|
pMainEnt->CollisionProp()->WorldSpaceAABB( &mins, &maxs );
|
|
mins -= pMainEnt->GetAbsOrigin();
|
|
maxs -= pMainEnt->GetAbsOrigin();
|
|
|
|
// Put some padding on their bbox.
|
|
float flPadSize = 5;
|
|
Vector vTestMins = mins - Vector( flPadSize, flPadSize, flPadSize );
|
|
Vector vTestMaxs = maxs + Vector( flPadSize, flPadSize, flPadSize );
|
|
|
|
// First test the starting origin.
|
|
if ( UTIL_IsSpaceEmpty( pMainEnt, vOrigin + vTestMins, vOrigin + vTestMaxs ) )
|
|
{
|
|
if ( bDropToGround )
|
|
{
|
|
outPos = DropToGround( pMainEnt, vOrigin, vTestMins, vTestMaxs );
|
|
}
|
|
else
|
|
{
|
|
outPos = vOrigin;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Vector vDims = vTestMaxs - vTestMins;
|
|
|
|
// Keep branching out until we get too far.
|
|
int iCurIteration = 0;
|
|
int nMaxIterations = 15;
|
|
|
|
int offset = 0;
|
|
do
|
|
{
|
|
for ( int iDim=0; iDim < 3; iDim++ )
|
|
{
|
|
float flCurOffset = offset * vDims[iDim];
|
|
|
|
for ( int iSign=0; iSign < 2; iSign++ )
|
|
{
|
|
Vector vBase = vOrigin;
|
|
vBase[iDim] += (iSign*2-1) * flCurOffset;
|
|
|
|
if ( UTIL_IsSpaceEmpty( pMainEnt, vBase + vTestMins, vBase + vTestMaxs ) )
|
|
{
|
|
// Ensure that there is a clear line of sight from the spawnpoint entity to the actual spawn point.
|
|
// (Useful for keeping things from spawning behind walls near a spawn point)
|
|
trace_t tr;
|
|
UTIL_TraceLine( vOrigin, vBase, MASK_SOLID, pMainEnt, COLLISION_GROUP_NONE, &tr );
|
|
|
|
if ( tr.fraction != 1.0 )
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ( bDropToGround )
|
|
outPos = DropToGround( pMainEnt, vBase, vTestMins, vTestMaxs );
|
|
else
|
|
outPos = vBase;
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
++offset;
|
|
} while ( iCurIteration++ < nMaxIterations );
|
|
|
|
// Warning( "EntityPlacementTest for ent %d:%s failed!\n", pMainEnt->entindex(), pMainEnt->GetClassname() );
|
|
return false;
|
|
}
|
|
|
|
int UTIL_HumansInGame( bool ignoreSpectators )
|
|
{
|
|
int iCount = 0;
|
|
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *entity = CCSPlayer::Instance( i );
|
|
|
|
if ( entity && !FNullEnt( entity->edict() ) )
|
|
{
|
|
if ( FStrEq( entity->GetPlayerName(), "" ) )
|
|
continue;
|
|
|
|
if ( FBitSet( entity->GetFlags(), FL_FAKECLIENT ) )
|
|
continue;
|
|
|
|
if ( ignoreSpectators && entity->GetTeamNumber() != TEAM_TERRORIST && entity->GetTeamNumber() != TEAM_CT )
|
|
continue;
|
|
|
|
if ( ignoreSpectators && entity->State_Get() == STATE_PICKINGCLASS )
|
|
continue;
|
|
|
|
iCount++;
|
|
}
|
|
}
|
|
|
|
return iCount;
|
|
}
|
|
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
// CCSGameRules implementation.
|
|
// --------------------------------------------------------------------------------------------------- //
|
|
|
|
CCSGameRules::CCSGameRules()
|
|
{
|
|
m_iRoundTime = 0;
|
|
m_iRoundWinStatus = WINNER_NONE;
|
|
m_iFreezeTime = 0;
|
|
|
|
m_fRoundStartTime = 0;
|
|
m_bAllowWeaponSwitch = true;
|
|
m_bFreezePeriod = true;
|
|
m_iNumTerrorist = m_iNumCT = 0; // number of players per team
|
|
m_flRestartRoundTime = 0.1f; // restart first round as soon as possible
|
|
m_iNumSpawnableTerrorist = m_iNumSpawnableCT = 0;
|
|
m_bFirstConnected = false;
|
|
m_bCompleteReset = false;
|
|
m_iAccountTerrorist = m_iAccountCT = 0;
|
|
m_iNumCTWins = 0;
|
|
m_iNumTerroristWins = 0;
|
|
m_iNumConsecutiveCTLoses = 0;
|
|
m_iNumConsecutiveTerroristLoses = 0;
|
|
m_bTargetBombed = false;
|
|
m_bBombDefused = false;
|
|
m_iTotalRoundsPlayed = -1;
|
|
m_iUnBalancedRounds = 0;
|
|
m_flGameStartTime = 0;
|
|
m_iHostagesRemaining = 0;
|
|
m_bLevelInitialized = false;
|
|
m_bLogoMap = false;
|
|
m_tmNextPeriodicThink = 0;
|
|
|
|
m_bMapHasBombTarget = false;
|
|
m_bMapHasRescueZone = false;
|
|
|
|
m_iSpawnPointCount_Terrorist = 0;
|
|
m_iSpawnPointCount_CT = 0;
|
|
|
|
m_bTCantBuy = false;
|
|
m_bCTCantBuy = false;
|
|
m_bMapHasBuyZone = false;
|
|
|
|
m_iLoserBonus = 0;
|
|
|
|
m_iHostagesRescued = 0;
|
|
m_iHostagesTouched = 0;
|
|
m_flNextHostageAnnouncement = 0.0f;
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN
|
|
// [dwenger] Reset rescue-related achievement values
|
|
//=============================================================================
|
|
|
|
// [tj] reset flawless and lossless round related flags
|
|
m_bNoTerroristsKilled = true;
|
|
m_bNoCTsKilled = true;
|
|
m_bNoTerroristsDamaged = true;
|
|
m_bNoCTsDamaged = true;
|
|
m_pFirstKill = NULL;
|
|
m_firstKillTime = 0;
|
|
|
|
// [menglish] Reset fun fact values
|
|
m_pFirstBlood = NULL;
|
|
m_firstBloodTime = 0;
|
|
|
|
m_bCanDonateWeapons = true;
|
|
|
|
// [dwenger] Reset rescue-related achievement values
|
|
m_pLastRescuer = NULL;
|
|
m_iNumRescuers = 0;
|
|
|
|
m_hostageWasInjured = false;
|
|
m_hostageWasKilled = false;
|
|
|
|
m_pFunFactManager = new CCSFunFactMgr();
|
|
m_pFunFactManager->Init();
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
m_iHaveEscaped = 0;
|
|
m_bMapHasEscapeZone = false;
|
|
m_iNumEscapers = 0;
|
|
m_iNumEscapeRounds = 0;
|
|
|
|
m_iMapHasVIPSafetyZone = 0;
|
|
m_pVIP = NULL;
|
|
m_iConsecutiveVIP = 0;
|
|
|
|
m_bMapHasBombZone = false;
|
|
m_bBombDropped = false;
|
|
m_bBombPlanted = false;
|
|
m_pLastBombGuy = NULL;
|
|
|
|
m_bAllowWeaponSwitch = true;
|
|
|
|
m_flNextHostageAnnouncement = gpGlobals->curtime; // asap.
|
|
|
|
ReadMultiplayCvars();
|
|
|
|
m_pPrices = NULL;
|
|
m_bBlackMarket = false;
|
|
m_bDontUploadStats = false;
|
|
|
|
// Create the team managers
|
|
for ( int i = 0; i < ARRAYSIZE( sTeamNames ); i++ )
|
|
{
|
|
CTeam *pTeam = static_cast<CTeam*>(CreateEntityByName( "cs_team_manager" ));
|
|
pTeam->Init( sTeamNames[i], i );
|
|
|
|
g_Teams.AddToTail( pTeam );
|
|
}
|
|
|
|
if ( filesystem->FileExists( UTIL_VarArgs( "maps/cfg/%s.cfg", STRING(gpGlobals->mapname) ) ) )
|
|
{
|
|
// Execute a map specific cfg file - as in Day of Defeat
|
|
// Map names cannot contain quotes or control characters so this is safe but silly that we have to do it.
|
|
engine->ServerCommand( UTIL_VarArgs( "exec \"%s.cfg\" */maps\n", STRING(gpGlobals->mapname) ) );
|
|
engine->ServerExecute();
|
|
}
|
|
|
|
#ifndef CLIENT_DLL
|
|
// stats
|
|
|
|
if ( g_flGameStatsUpdateTime == 0.0f )
|
|
{
|
|
memset( g_iWeaponPurchases, 0, sizeof( g_iWeaponPurchases) );
|
|
memset( g_iTerroristVictories, 0, sizeof( g_iTerroristVictories) );
|
|
memset( g_iCounterTVictories, 0, sizeof( g_iTerroristVictories) );
|
|
g_flGameStatsUpdateTime = CS_GAME_STATS_UPDATE; //Next update is between 22 and 24 hours.
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void CCSGameRules::AddPricesToTable( weeklyprice_t prices )
|
|
{
|
|
int iIndex = m_StringTableBlackMarket->FindStringIndex( "blackmarket_prices" );
|
|
|
|
if ( iIndex == INVALID_STRING_INDEX )
|
|
{
|
|
m_StringTableBlackMarket->AddString( CBaseEntity::IsServer(), "blackmarket_prices", sizeof( weeklyprice_t), &prices );
|
|
}
|
|
else
|
|
{
|
|
m_StringTableBlackMarket->SetStringUserData( iIndex, sizeof( weeklyprice_t), &prices );
|
|
}
|
|
|
|
SetBlackMarketPrices( false );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
CCSGameRules::~CCSGameRules()
|
|
{
|
|
// Note, don't delete each team since they are in the gEntList and will
|
|
// automatically be deleted from there, instead.
|
|
g_Teams.Purge();
|
|
if( m_pFunFactManager )
|
|
{
|
|
delete m_pFunFactManager;
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::UpdateClientData( CBasePlayer *player )
|
|
{
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: TF2 Specific Client Commands
|
|
// Input :
|
|
// Output :
|
|
//-----------------------------------------------------------------------------
|
|
bool CCSGameRules::ClientCommand( CBaseEntity *pEdict, const CCommand &args )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pEdict );
|
|
|
|
if ( FStrEq( args[0], "changeteam" ) )
|
|
{
|
|
return true;
|
|
}
|
|
else if ( FStrEq( args[0], "nextmap" ) )
|
|
{
|
|
if ( pPlayer->m_iNextTimeCheck < gpGlobals->curtime )
|
|
{
|
|
char szNextMap[32];
|
|
|
|
if ( nextlevel.GetString() && *nextlevel.GetString() )
|
|
{
|
|
Q_strncpy( szNextMap, nextlevel.GetString(), sizeof( szNextMap ) );
|
|
}
|
|
else
|
|
{
|
|
GetNextLevelName( szNextMap, sizeof( szNextMap ) );
|
|
}
|
|
|
|
ClientPrint( pPlayer, HUD_PRINTTALK, "#game_nextmap", szNextMap);
|
|
|
|
pPlayer->m_iNextTimeCheck = gpGlobals->curtime + 1;
|
|
}
|
|
return true;
|
|
}
|
|
else if( pPlayer->ClientCommand( args ) )
|
|
{
|
|
return true;
|
|
}
|
|
else if( BaseClass::ClientCommand( pEdict, args ) )
|
|
{
|
|
return true;
|
|
}
|
|
else if ( TheBots->ServerCommand( args.GetCommandString() ) )
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return TheBots->ClientCommand( pPlayer, args );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Player has just spawned. Equip them.
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::ClientCommandKeyValues( edict_t *pEntity, KeyValues *pKeyValues )
|
|
{
|
|
CCSPlayer *pPlayer = dynamic_cast< CCSPlayer * >( CBaseEntity::Instance( pEntity ) );
|
|
if ( pPlayer )
|
|
{
|
|
char const *pszCommand = pKeyValues->GetName();
|
|
if ( pszCommand && pszCommand[0] )
|
|
{
|
|
if ( FStrEq( pszCommand, "ClanTagChanged" ) )
|
|
{
|
|
pPlayer->SetClanTag( pKeyValues->GetString( "tag", "" ) );
|
|
|
|
const char *teamName = "UNKNOWN";
|
|
if ( pPlayer->GetTeam() )
|
|
{
|
|
teamName = pPlayer->GetTeam()->GetName();
|
|
}
|
|
UTIL_LogPrintf("\"%s<%i><%s><%s>\" triggered \"clantag\" (value \"%s\")\n",
|
|
pPlayer->GetPlayerName(),
|
|
pPlayer->GetUserID(),
|
|
pPlayer->GetNetworkIDString(),
|
|
teamName,
|
|
pKeyValues->GetString( "tag", "unknown" ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
BaseClass::ClientCommandKeyValues( pEntity, pKeyValues );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Player has just spawned. Equip them.
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::PlayerSpawn( CBasePlayer *pBasePlayer )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pBasePlayer );
|
|
if ( !pPlayer )
|
|
Error( "PlayerSpawn" );
|
|
|
|
if ( pPlayer->State_Get() != STATE_ACTIVE )
|
|
return;
|
|
|
|
pPlayer->EquipSuit();
|
|
|
|
bool addDefault = true;
|
|
|
|
CBaseEntity *pWeaponEntity = NULL;
|
|
while ( ( pWeaponEntity = gEntList.FindEntityByClassname( pWeaponEntity, "game_player_equip" )) != NULL )
|
|
{
|
|
if ( addDefault )
|
|
{
|
|
// remove all our weapons and armor before touching the first game_player_equip
|
|
pPlayer->RemoveAllItems( true );
|
|
}
|
|
pWeaponEntity->Touch( pPlayer );
|
|
addDefault = false;
|
|
}
|
|
|
|
|
|
if ( addDefault || pPlayer->m_bIsVIP )
|
|
pPlayer->GiveDefaultItems();
|
|
}
|
|
|
|
void CCSGameRules::BroadcastSound( const char *sound, int team )
|
|
{
|
|
CBroadcastRecipientFilter filter;
|
|
filter.MakeReliable();
|
|
|
|
if( team != -1 )
|
|
{
|
|
filter.RemoveAllRecipients();
|
|
filter.AddRecipientsByTeam( GetGlobalTeam(team) );
|
|
}
|
|
|
|
UserMessageBegin ( filter, "SendAudio" );
|
|
WRITE_STRING( sound );
|
|
MessageEnd();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Player has just spawned. Equip them.
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// return a multiplier that should adjust the damage done by a blast at position vecSrc to something at the position
|
|
// vecEnd. This will take into account the density of an entity that blocks the line of sight from one position to
|
|
// the other.
|
|
//
|
|
// this algorithm was taken from the HL2 version of RadiusDamage.
|
|
float CCSGameRules::GetExplosionDamageAdjustment(Vector & vecSrc, Vector & vecEnd, CBaseEntity *pEntityToIgnore)
|
|
{
|
|
float retval = 0.0;
|
|
trace_t tr;
|
|
|
|
UTIL_TraceLine(vecSrc, vecEnd, MASK_SHOT, pEntityToIgnore, COLLISION_GROUP_NONE, &tr);
|
|
if (tr.fraction == 1.0)
|
|
{
|
|
retval = 1.0;
|
|
}
|
|
else if (!(tr.DidHitWorld()) && (tr.m_pEnt != NULL) && (tr.m_pEnt != pEntityToIgnore) && (tr.m_pEnt->GetOwnerEntity() != pEntityToIgnore))
|
|
{
|
|
// if we didn't hit world geometry perhaps there's still damage to be done here.
|
|
|
|
CBaseEntity *blockingEntity = tr.m_pEnt;
|
|
|
|
// check to see if this part of the player is visible if entities are ignored.
|
|
UTIL_TraceLine(vecSrc, vecEnd, CONTENTS_SOLID, NULL, COLLISION_GROUP_NONE, &tr);
|
|
|
|
if (tr.fraction == 1.0)
|
|
{
|
|
if ((blockingEntity != NULL) && (blockingEntity->VPhysicsGetObject() != NULL))
|
|
{
|
|
int nMaterialIndex = blockingEntity->VPhysicsGetObject()->GetMaterialIndex();
|
|
|
|
float flDensity;
|
|
float flThickness;
|
|
float flFriction;
|
|
float flElasticity;
|
|
|
|
physprops->GetPhysicsProperties( nMaterialIndex, &flDensity,
|
|
&flThickness, &flFriction, &flElasticity );
|
|
|
|
const float DENSITY_ABSORB_ALL_DAMAGE = 3000.0;
|
|
float scale = flDensity / DENSITY_ABSORB_ALL_DAMAGE;
|
|
if ((scale >= 0.0) && (scale < 1.0))
|
|
{
|
|
retval = 1.0 - scale;
|
|
}
|
|
else if (scale < 0.0)
|
|
{
|
|
// should never happen, but just in case.
|
|
retval = 1.0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
retval = 0.75; // we're blocked by something that isn't an entity with a physics module or world geometry, just cut damage in half for now.
|
|
}
|
|
}
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
// returns the percentage of the player that is visible from the given point in the world.
|
|
// return value is between 0 and 1.
|
|
float CCSGameRules::GetAmountOfEntityVisible(Vector & vecSrc, CBaseEntity *entity)
|
|
{
|
|
float retval = 0.0;
|
|
|
|
const float damagePercentageChest = 0.40;
|
|
const float damagePercentageHead = 0.20;
|
|
const float damagePercentageFeet = 0.20;
|
|
const float damagePercentageRightSide = 0.10;
|
|
const float damagePercentageLeftSide = 0.10;
|
|
|
|
if (!(entity->IsPlayer()))
|
|
{
|
|
// the entity is not a player, so the damage is all or nothing.
|
|
Vector vecTarget;
|
|
vecTarget = entity->BodyTarget(vecSrc, false);
|
|
|
|
return GetExplosionDamageAdjustment(vecSrc, vecTarget, entity);
|
|
}
|
|
|
|
CCSPlayer *player = (CCSPlayer *)entity;
|
|
|
|
// check what parts of the player we can see from this point and modify the return value accordingly.
|
|
float chestHeightFromFeet;
|
|
|
|
float armDistanceFromChest = HalfHumanWidth;
|
|
|
|
// calculate positions of various points on the target player's body
|
|
Vector vecFeet = player->GetAbsOrigin();
|
|
|
|
Vector vecChest = player->BodyTarget(vecSrc, false);
|
|
chestHeightFromFeet = vecChest.z - vecFeet.z; // compute the distance from the chest to the feet. (this accounts for ducking and the like)
|
|
|
|
Vector vecHead = player->GetAbsOrigin();
|
|
vecHead.z += HumanHeight;
|
|
|
|
Vector vecRightFacing;
|
|
AngleVectors(player->GetAbsAngles(), NULL, &vecRightFacing, NULL);
|
|
|
|
vecRightFacing.NormalizeInPlace();
|
|
vecRightFacing = vecRightFacing * armDistanceFromChest;
|
|
|
|
Vector vecLeftSide = player->GetAbsOrigin();
|
|
vecLeftSide.x -= vecRightFacing.x;
|
|
vecLeftSide.y -= vecRightFacing.y;
|
|
vecLeftSide.z += chestHeightFromFeet;
|
|
|
|
Vector vecRightSide = player->GetAbsOrigin();
|
|
vecRightSide.x += vecRightFacing.x;
|
|
vecRightSide.y += vecRightFacing.y;
|
|
vecRightSide.z += chestHeightFromFeet;
|
|
|
|
// check chest
|
|
float damageAdjustment = GetExplosionDamageAdjustment(vecSrc, vecChest, entity);
|
|
retval += (damagePercentageChest * damageAdjustment);
|
|
|
|
// check top of head
|
|
damageAdjustment = GetExplosionDamageAdjustment(vecSrc, vecHead, entity);
|
|
retval += (damagePercentageHead * damageAdjustment);
|
|
|
|
// check feet
|
|
damageAdjustment = GetExplosionDamageAdjustment(vecSrc, vecFeet, entity);
|
|
retval += (damagePercentageFeet * damageAdjustment);
|
|
|
|
// check left "edge"
|
|
damageAdjustment = GetExplosionDamageAdjustment(vecSrc, vecLeftSide, entity);
|
|
retval += (damagePercentageLeftSide * damageAdjustment);
|
|
|
|
// check right "edge"
|
|
damageAdjustment = GetExplosionDamageAdjustment(vecSrc, vecRightSide, entity);
|
|
retval += (damagePercentageRightSide * damageAdjustment);
|
|
|
|
return retval;
|
|
}
|
|
|
|
void CCSGameRules::RadiusDamage( const CTakeDamageInfo &info, const Vector &vecSrcIn, float flRadius, int iClassIgnore, CBaseEntity * pEntityIgnore )
|
|
{
|
|
RadiusDamage( info, vecSrcIn, flRadius, iClassIgnore, false );
|
|
}
|
|
|
|
// Add the ability to ignore the world trace
|
|
void CCSGameRules::RadiusDamage( const CTakeDamageInfo &info, const Vector &vecSrcIn, float flRadius, int iClassIgnore, bool bIgnoreWorld )
|
|
{
|
|
CBaseEntity *pEntity = NULL;
|
|
trace_t tr;
|
|
float falloff, damagePercentage;
|
|
Vector vecSpot;
|
|
Vector vecToTarget;
|
|
Vector vecEndPos;
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
//=============================================================================
|
|
|
|
// [tj] The number of enemy players this explosion killed
|
|
int numberOfEnemyPlayersKilledByThisExplosion = 0;
|
|
|
|
// [tj] who we award the achievement to if enough players are killed
|
|
CCSPlayer* pCSExplosionAttacker = ToCSPlayer(info.GetAttacker());
|
|
|
|
// [tj] used to determine which achievement to award for sufficient kills
|
|
CBaseEntity* pInflictor = info.GetInflictor();
|
|
bool isGrenade = pInflictor && V_strcmp(pInflictor->GetClassname(), "hegrenade_projectile") == 0;
|
|
bool isBomb = pInflictor && V_strcmp(pInflictor->GetClassname(), "planted_c4") == 0;
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
|
|
vecEndPos.Init();
|
|
|
|
Vector vecSrc = vecSrcIn;
|
|
|
|
damagePercentage = 1.0;
|
|
|
|
if ( flRadius )
|
|
falloff = info.GetDamage() / flRadius;
|
|
else
|
|
falloff = 1.0;
|
|
|
|
int bInWater = (UTIL_PointContents ( vecSrc ) & MASK_WATER) ? true : false;
|
|
|
|
vecSrc.z += 1;// in case grenade is lying on the ground
|
|
|
|
// iterate on all entities in the vicinity.
|
|
for ( CEntitySphereQuery sphere( vecSrc, flRadius ); ( pEntity = sphere.GetCurrentEntity() ) != NULL; sphere.NextEntity() )
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] We have to save whether or not the player is killed so we don't give credit
|
|
// for pre-dead players.
|
|
//=============================================================================
|
|
bool wasAliveBeforeExplosion = false;
|
|
CCSPlayer* pCSExplosionVictim = ToCSPlayer(pEntity);
|
|
if (pCSExplosionVictim)
|
|
{
|
|
wasAliveBeforeExplosion = pCSExplosionVictim->IsAlive();
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
if ( pEntity->m_takedamage != DAMAGE_NO )
|
|
{
|
|
// UNDONE: this should check a damage mask, not an ignore
|
|
if ( iClassIgnore != CLASS_NONE && pEntity->Classify() == iClassIgnore )
|
|
{// houndeyes don't hurt other houndeyes with their attack
|
|
continue;
|
|
}
|
|
|
|
// blasts don't travel into or out of water
|
|
if ( !bIgnoreWorld )
|
|
{
|
|
if (bInWater && pEntity->GetWaterLevel() == 0)
|
|
continue;
|
|
if (!bInWater && pEntity->GetWaterLevel() == 3)
|
|
continue;
|
|
}
|
|
|
|
// radius damage can only be blocked by the world
|
|
vecSpot = pEntity->BodyTarget( vecSrc );
|
|
|
|
bool bHit = false;
|
|
|
|
if( bIgnoreWorld )
|
|
{
|
|
vecEndPos = vecSpot;
|
|
bHit = true;
|
|
}
|
|
else
|
|
{
|
|
// get the percentage of the target entity that is visible from the
|
|
// explosion position.
|
|
damagePercentage = GetAmountOfEntityVisible(vecSrc, pEntity);
|
|
if (damagePercentage > 0.0)
|
|
{
|
|
vecEndPos = vecSpot;
|
|
|
|
bHit = true;
|
|
}
|
|
}
|
|
|
|
if ( bHit )
|
|
{
|
|
// the explosion can 'see' this entity, so hurt them!
|
|
//vecToTarget = ( vecSrc - vecEndPos );
|
|
vecToTarget = ( vecEndPos - vecSrc );
|
|
|
|
// use a Gaussian function to describe the damage falloff over distance, with flRadius equal to 3 * sigma
|
|
// this results in the following values:
|
|
//
|
|
// Range Fraction Damage
|
|
// 0.0 100%
|
|
// 0.1 96%
|
|
// 0.2 84%
|
|
// 0.3 67%
|
|
// 0.4 49%
|
|
// 0.5 32%
|
|
// 0.6 20%
|
|
// 0.7 11%
|
|
// 0.8 6%
|
|
// 0.9 3%
|
|
// 1.0 1%
|
|
|
|
float fDist = vecToTarget.Length();
|
|
float fSigma = flRadius / 3.0f; // flRadius specifies 3rd standard deviation (0.0111 damage at this range)
|
|
float fGaussianFalloff = exp(-fDist * fDist / (2.0f * fSigma * fSigma));
|
|
float flAdjustedDamage = info.GetDamage() * fGaussianFalloff * damagePercentage;
|
|
|
|
if ( flAdjustedDamage > 0 )
|
|
{
|
|
CTakeDamageInfo adjustedInfo = info;
|
|
adjustedInfo.SetDamage( flAdjustedDamage );
|
|
|
|
Vector dir = vecToTarget;
|
|
VectorNormalize( dir );
|
|
|
|
// If we don't have a damage force, manufacture one
|
|
if ( adjustedInfo.GetDamagePosition() == vec3_origin || adjustedInfo.GetDamageForce() == vec3_origin )
|
|
{
|
|
CalculateExplosiveDamageForce( &adjustedInfo, dir, vecSrc, 1.5 /* explosion scale! */ );
|
|
}
|
|
else
|
|
{
|
|
// Assume the force passed in is the maximum force. Decay it based on falloff.
|
|
float flForce = adjustedInfo.GetDamageForce().Length() * falloff;
|
|
adjustedInfo.SetDamageForce( dir * flForce );
|
|
adjustedInfo.SetDamagePosition( vecSrc );
|
|
}
|
|
|
|
Vector vecTarget;
|
|
vecTarget = pEntity->BodyTarget(vecSrc, false);
|
|
|
|
UTIL_TraceLine(vecSrc, vecTarget, MASK_SHOT, NULL, COLLISION_GROUP_NONE, &tr);
|
|
|
|
// blasts always hit chest
|
|
tr.hitgroup = HITGROUP_GENERIC;
|
|
|
|
if (tr.fraction != 1.0)
|
|
{
|
|
// this has to be done to make breakable glass work.
|
|
ClearMultiDamage( );
|
|
pEntity->DispatchTraceAttack( adjustedInfo, dir, &tr );
|
|
ApplyMultiDamage();
|
|
}
|
|
else
|
|
{
|
|
pEntity->TakeDamage( adjustedInfo );
|
|
}
|
|
|
|
// Now hit all triggers along the way that respond to damage...
|
|
pEntity->TraceAttackToTriggers( adjustedInfo, vecSrc, vecEndPos, dir );
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [sbodenbender] Increment grenade damage stat
|
|
//=============================================================================
|
|
if (pCSExplosionVictim && pCSExplosionAttacker && isGrenade)
|
|
{
|
|
CCS_GameStats.IncrementStat(pCSExplosionAttacker, CSSTAT_GRENADE_DAMAGE, static_cast<int>(adjustedInfo.GetDamage()));
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
}
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Count up victims of area of effect damage for achievement purposes
|
|
//=============================================================================
|
|
|
|
if (pCSExplosionVictim)
|
|
{
|
|
//If the bomb is exploding, set the attacker to the planter (we can't put this in the CTakeDamageInfo, since
|
|
//players aren't supposed to get credit for bomb kills)
|
|
if (isBomb)
|
|
{
|
|
CPlantedC4* bomb = static_cast<CPlantedC4*> (pInflictor);
|
|
if (bomb)
|
|
{
|
|
pCSExplosionAttacker = bomb->GetPlanter();
|
|
}
|
|
}
|
|
|
|
//Count check to make sure we killed an enemy player
|
|
if( pCSExplosionAttacker &&
|
|
!pCSExplosionVictim->IsAlive() &&
|
|
wasAliveBeforeExplosion &&
|
|
pCSExplosionVictim->GetTeamNumber() != pCSExplosionAttacker->GetTeamNumber())
|
|
{
|
|
numberOfEnemyPlayersKilledByThisExplosion++;
|
|
}
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] //Depending on which type of explosion it was, award the appropriate achievement.
|
|
//=============================================================================
|
|
|
|
if (pCSExplosionAttacker && isGrenade && numberOfEnemyPlayersKilledByThisExplosion >= AchievementConsts::GrenadeMultiKill_MinKills)
|
|
{
|
|
pCSExplosionAttacker->AwardAchievement(CSGrenadeMultikill);
|
|
pCSExplosionAttacker->CheckMaxGrenadeKills(numberOfEnemyPlayersKilledByThisExplosion);
|
|
|
|
}
|
|
if (pCSExplosionAttacker && isBomb && numberOfEnemyPlayersKilledByThisExplosion >= AchievementConsts::BombMultiKill_MinKills)
|
|
{
|
|
pCSExplosionAttacker->AwardAchievement(CSBombMultikill);
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *pVictim -
|
|
// *pKiller -
|
|
// *pInflictor -
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::DeathNotice( CBasePlayer *pVictim, const CTakeDamageInfo &info )
|
|
{
|
|
// Work out what killed the player, and send a message to all clients about it
|
|
const char *killer_weapon_name = "world"; // by default, the player is killed by the world
|
|
int killer_ID = 0;
|
|
|
|
// Find the killer & the scorer
|
|
CBaseEntity *pInflictor = info.GetInflictor();
|
|
CBaseEntity *pKiller = info.GetAttacker();
|
|
CBasePlayer *pScorer = GetDeathScorer( pKiller, pInflictor );
|
|
CCSPlayer *pCSVictim = (CCSPlayer*)(pVictim);
|
|
|
|
bool bHeadshot = false;
|
|
|
|
if ( pScorer ) // Is the killer a client?
|
|
{
|
|
killer_ID = pScorer->GetUserID();
|
|
|
|
if( info.GetDamageType() & DMG_HEADSHOT )
|
|
{
|
|
//to enable drawing the headshot icon as well as the weapon icon,
|
|
bHeadshot = true;
|
|
}
|
|
|
|
if ( pInflictor )
|
|
{
|
|
if ( pInflictor == pScorer )
|
|
{
|
|
// If the inflictor is the killer, then it must be their current weapon doing the damage
|
|
if ( pScorer->GetActiveWeapon() )
|
|
{
|
|
killer_weapon_name = pScorer->GetActiveWeapon()->GetClassname(); //GetDeathNoticeName();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
killer_weapon_name = STRING( pInflictor->m_iClassname ); // it's just that easy
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
killer_weapon_name = STRING( pInflictor->m_iClassname );
|
|
}
|
|
|
|
// strip the NPC_* or weapon_* from the inflictor's classname
|
|
if ( strncmp( killer_weapon_name, "weapon_", 7 ) == 0 )
|
|
{
|
|
killer_weapon_name += 7;
|
|
}
|
|
else if ( strncmp( killer_weapon_name, "NPC_", 8 ) == 0 )
|
|
{
|
|
killer_weapon_name += 8;
|
|
}
|
|
else if ( strncmp( killer_weapon_name, "func_", 5 ) == 0 )
|
|
{
|
|
killer_weapon_name += 5;
|
|
}
|
|
else if( strncmp( killer_weapon_name, "hegrenade", 9 ) == 0 ) //"hegrenade_projectile"
|
|
{
|
|
killer_weapon_name = "hegrenade";
|
|
}
|
|
else if( strncmp( killer_weapon_name, "flashbang", 9 ) == 0 ) //"flashbang_projectile"
|
|
{
|
|
killer_weapon_name = "flashbang";
|
|
}
|
|
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "player_death" );
|
|
|
|
if ( event )
|
|
{
|
|
event->SetInt("userid", pVictim->GetUserID() );
|
|
event->SetInt("attacker", killer_ID );
|
|
event->SetString("weapon", killer_weapon_name );
|
|
event->SetInt("headshot", bHeadshot ? 1 : 0 );
|
|
event->SetInt("priority", bHeadshot ? 8 : 7 ); // HLTV event priority, not transmitted
|
|
if ( pCSVictim->GetDeathFlags() & CS_DEATH_DOMINATION )
|
|
{
|
|
event->SetInt( "dominated", 1 );
|
|
}
|
|
else if ( pCSVictim->GetDeathFlags() & CS_DEATH_REVENGE )
|
|
{
|
|
event->SetInt( "revenge", 1 );
|
|
}
|
|
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
}
|
|
|
|
//=========================================================
|
|
//=========================================================
|
|
void CCSGameRules::PlayerKilled( CBasePlayer *pVictim, const CTakeDamageInfo &info )
|
|
{
|
|
CBaseEntity *pInflictor = info.GetInflictor();
|
|
CBaseEntity *pKiller = info.GetAttacker();
|
|
CBasePlayer *pScorer = GetDeathScorer( pKiller, pInflictor );
|
|
CCSPlayer *pCSVictim = (CCSPlayer *)pVictim;
|
|
CCSPlayer *pCSScorer = (CCSPlayer *)pScorer;
|
|
|
|
CCS_GameStats.PlayerKilled( pVictim, info );
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Flag the round as non-lossless for the appropriate team.
|
|
// [menglish] Set the death flags depending on a nemesis system
|
|
//=============================================================================
|
|
|
|
if (pVictim->GetTeamNumber() == TEAM_TERRORIST)
|
|
{
|
|
m_bNoTerroristsKilled = false;
|
|
m_bNoTerroristsDamaged = false;
|
|
}
|
|
if (pVictim->GetTeamNumber() == TEAM_CT)
|
|
{
|
|
m_bNoCTsKilled = false;
|
|
m_bNoCTsDamaged = false;
|
|
}
|
|
|
|
m_bCanDonateWeapons = false;
|
|
|
|
if ( m_pFirstKill == NULL && pCSScorer != pVictim )
|
|
{
|
|
m_pFirstKill = pCSScorer;
|
|
m_firstKillTime = gpGlobals->curtime - m_fRoundStartTime;
|
|
}
|
|
|
|
// determine if this kill affected a nemesis relationship
|
|
int iDeathFlags = 0;
|
|
if ( pScorer )
|
|
{
|
|
CCS_GameStats.CalculateOverkill( pCSScorer, pCSVictim);
|
|
CCS_GameStats.CalcDominationAndRevenge( pCSScorer, pCSVictim, &iDeathFlags );
|
|
}
|
|
pCSVictim->SetDeathFlags( iDeathFlags );
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
// If we're killed by the C4, we do a subset of BaseClass::PlayerKilled()
|
|
// Specifically, we shouldn't lose any points or show death notices, to match goldsrc
|
|
if ( Q_strcmp(pKiller->GetClassname(), "planted_c4" ) == 0 )
|
|
{
|
|
// dvsents2: uncomment when removing all FireTargets
|
|
// variant_t value;
|
|
// g_EventQueue.AddEvent( "game_playerdie", "Use", value, 0, pVictim, pVictim );
|
|
FireTargets( "game_playerdie", pVictim, pVictim, USE_TOGGLE, 0 );
|
|
}
|
|
else
|
|
{
|
|
BaseClass::PlayerKilled( pVictim, info );
|
|
}
|
|
|
|
// check for team-killing, and give monetary rewards/penalties
|
|
// Find the killer & the scorer
|
|
if ( !pScorer )
|
|
return;
|
|
|
|
if ( IPointsForKill( pScorer, pVictim ) < 0 )
|
|
{
|
|
// team-killer!
|
|
pCSScorer->AddAccount( -3300 );
|
|
++pCSScorer->m_iTeamKills;
|
|
pCSScorer->m_bJustKilledTeammate = true;
|
|
|
|
ClientPrint( pCSScorer, HUD_PRINTCENTER, "#Killed_Teammate" );
|
|
if ( mp_autokick.GetBool() )
|
|
{
|
|
char strTeamKills[64];
|
|
Q_snprintf( strTeamKills, sizeof( strTeamKills ), "%d", pCSScorer->m_iTeamKills );
|
|
ClientPrint( pCSScorer, HUD_PRINTCONSOLE, "#Game_teammate_kills", strTeamKills ); // this includes a " of 3" in it
|
|
|
|
if ( pCSScorer->m_iTeamKills >= 3 )
|
|
{
|
|
ClientPrint( pCSScorer, HUD_PRINTCONSOLE, "#Banned_For_Killing_Teammates" );
|
|
engine->ServerCommand( UTIL_VarArgs( "kickid %d\n", pCSScorer->GetUserID() ) );
|
|
}
|
|
else if ( mp_spawnprotectiontime.GetInt() > 0 && GetRoundElapsedTime() < mp_spawnprotectiontime.GetInt() )
|
|
{
|
|
ClientPrint( pCSScorer, HUD_PRINTCONSOLE, "#Banned_For_Killing_Teammates" );
|
|
engine->ServerCommand( UTIL_VarArgs( "kickid %d\n", pCSScorer->GetUserID() ) );
|
|
}
|
|
}
|
|
|
|
if ( !(pCSScorer->m_iDisplayHistoryBits & DHF_FRIEND_KILLED) )
|
|
{
|
|
pCSScorer->m_iDisplayHistoryBits |= DHF_FRIEND_KILLED;
|
|
pCSScorer->HintMessage( "#Hint_careful_around_teammates", false );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Added a check to make sure we don't get money for suicides.
|
|
//=============================================================================
|
|
if (pCSScorer != pCSVictim)
|
|
{
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
if ( pCSVictim->IsVIP() )
|
|
{
|
|
pCSScorer->HintMessage( "#Hint_reward_for_killing_vip", true );
|
|
pCSScorer->AddAccount( 2500 );
|
|
}
|
|
else
|
|
{
|
|
pCSScorer->AddAccount( 300 );
|
|
}
|
|
}
|
|
|
|
if ( !(pCSScorer->m_iDisplayHistoryBits & DHF_ENEMY_KILLED) )
|
|
{
|
|
pCSScorer->m_iDisplayHistoryBits |= DHF_ENEMY_KILLED;
|
|
pCSScorer->HintMessage( "#Hint_win_round_by_killing_enemy", false );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void CCSGameRules::InitDefaultAIRelationships()
|
|
{
|
|
// Allocate memory for default relationships
|
|
CBaseCombatCharacter::AllocateDefaultRelationships();
|
|
|
|
// --------------------------------------------------------------
|
|
// First initialize table so we can report missing relationships
|
|
// --------------------------------------------------------------
|
|
int i, j;
|
|
for (i=0;i<NUM_AI_CLASSES;i++)
|
|
{
|
|
for (j=0;j<NUM_AI_CLASSES;j++)
|
|
{
|
|
// By default all relationships are neutral of priority zero
|
|
CBaseCombatCharacter::SetDefaultRelationship( (Class_T)i, (Class_T)j, D_NU, 0 );
|
|
}
|
|
}
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Purpose : Return classify text for classify type
|
|
//------------------------------------------------------------------------------
|
|
const char *CCSGameRules::AIClassText(int classType)
|
|
{
|
|
switch (classType)
|
|
{
|
|
case CLASS_NONE: return "CLASS_NONE";
|
|
case CLASS_PLAYER: return "CLASS_PLAYER";
|
|
default: return "MISSING CLASS in ClassifyText()";
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: When gaining new technologies in TF, prevent auto switching if we
|
|
// receive a weapon during the switch
|
|
// Input : *pPlayer -
|
|
// *pWeapon -
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CCSGameRules::FShouldSwitchWeapon( CBasePlayer *pPlayer, CBaseCombatWeapon *pWeapon )
|
|
{
|
|
bool bIsBeingGivenItem = false;
|
|
CCSPlayer *pCSPlayer = ToCSPlayer( pPlayer );
|
|
if ( pCSPlayer && pCSPlayer->IsBeingGivenItem() )
|
|
bIsBeingGivenItem = true;
|
|
|
|
if ( pPlayer->GetActiveWeapon() && pPlayer->IsNetClient() && !bIsBeingGivenItem )
|
|
{
|
|
// Player has an active item, so let's check cl_autowepswitch.
|
|
const char *cl_autowepswitch = engine->GetClientConVarValue( engine->IndexOfEdict( pPlayer->edict() ), "cl_autowepswitch" );
|
|
if ( cl_autowepswitch && atoi( cl_autowepswitch ) <= 0 )
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if ( pPlayer->IsBot() && !bIsBeingGivenItem )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if ( !GetAllowWeaponSwitch() )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return BaseClass::FShouldSwitchWeapon( pPlayer, pWeapon );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : allow -
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::SetAllowWeaponSwitch( bool allow )
|
|
{
|
|
m_bAllowWeaponSwitch = allow;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CCSGameRules::GetAllowWeaponSwitch()
|
|
{
|
|
return m_bAllowWeaponSwitch;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *pPlayer -
|
|
// Output : const char
|
|
//-----------------------------------------------------------------------------
|
|
const char *CCSGameRules::SetDefaultPlayerTeam( CBasePlayer *pPlayer )
|
|
{
|
|
Assert( pPlayer );
|
|
return BaseClass::SetDefaultPlayerTeam( pPlayer );
|
|
}
|
|
|
|
|
|
void CCSGameRules::LevelInitPreEntity()
|
|
{
|
|
BaseClass::LevelInitPreEntity();
|
|
|
|
// TODO for CZ-style hostages: TheHostageChatter->Precache();
|
|
}
|
|
|
|
|
|
void CCSGameRules::LevelInitPostEntity()
|
|
{
|
|
BaseClass::LevelInitPostEntity();
|
|
|
|
m_bLevelInitialized = false; // re-count CT and T start spots now that they exist
|
|
|
|
// Figure out from the entities in the map what kind of map this is (bomb run, prison escape, etc).
|
|
CheckMapConditions();
|
|
}
|
|
|
|
INetworkStringTable *g_StringTableBlackMarket = NULL;
|
|
|
|
void CCSGameRules::CreateCustomNetworkStringTables( void )
|
|
{
|
|
m_StringTableBlackMarket = g_StringTableBlackMarket;
|
|
|
|
if ( 0 )//mp_dynamicpricing.GetBool() )
|
|
{
|
|
m_bBlackMarket = BlackMarket_DownloadPrices();
|
|
|
|
if ( m_bBlackMarket == false )
|
|
{
|
|
Msg( "ERROR: mp_dynamicpricing set to 1 but couldn't download the price list!\n" );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_bBlackMarket = false;
|
|
SetBlackMarketPrices( true );
|
|
}
|
|
}
|
|
|
|
float CCSGameRules::FlPlayerFallDamage( CBasePlayer *pPlayer )
|
|
{
|
|
float fFallVelocity = pPlayer->m_Local.m_flFallVelocity - CS_PLAYER_MAX_SAFE_FALL_SPEED;
|
|
float fallDamage = fFallVelocity * CS_DAMAGE_FOR_FALL_SPEED * 1.25;
|
|
|
|
if ( fallDamage > 0.0f )
|
|
{
|
|
// let the bots know
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "player_falldamage" );
|
|
if ( event )
|
|
{
|
|
event->SetInt( "userid", pPlayer->GetUserID() );
|
|
event->SetFloat( "damage", fallDamage );
|
|
event->SetInt( "priority", 4 ); // HLTV event priority, not transmitted
|
|
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
}
|
|
|
|
return fallDamage;
|
|
}
|
|
|
|
|
|
void CCSGameRules::ClientDisconnected( edict_t *pClient )
|
|
{
|
|
BaseClass::ClientDisconnected( pClient );
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Clear domination data when a player disconnects
|
|
//=============================================================================
|
|
|
|
CCSPlayer *pPlayer = ToCSPlayer( GetContainingEntity( pClient ) );
|
|
if ( pPlayer )
|
|
{
|
|
pPlayer->RemoveNemesisRelationships();
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
|
|
CheckWinConditions();
|
|
}
|
|
|
|
|
|
// Called when game rules are destroyed by CWorld
|
|
void CCSGameRules::LevelShutdown()
|
|
{
|
|
int iLevelIndex = GetCSLevelIndex( STRING( gpGlobals->mapname ) );
|
|
|
|
if ( iLevelIndex != -1 )
|
|
{
|
|
g_iTerroristVictories[iLevelIndex] += m_iNumTerroristWins;
|
|
g_iCounterTVictories[iLevelIndex] += m_iNumCTWins;
|
|
}
|
|
|
|
BaseClass::LevelShutdown();
|
|
}
|
|
|
|
|
|
//---------------------------------------------------------------------------------------------------
|
|
/**
|
|
* Check if the scenario has been won/lost.
|
|
* Return true if the scenario is over, false if the scenario is still in progress
|
|
*/
|
|
bool CCSGameRules::CheckWinConditions( void )
|
|
{
|
|
if ( mp_ignore_round_win_conditions.GetBool() )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// If a winner has already been determined.. then get the heck out of here
|
|
if (m_iRoundWinStatus != WINNER_NONE)
|
|
{
|
|
// still check if we lost players to where we need to do a full reset next round...
|
|
int NumDeadCT, NumDeadTerrorist, NumAliveTerrorist, NumAliveCT;
|
|
InitializePlayerCounts( NumAliveTerrorist, NumAliveCT, NumDeadTerrorist, NumDeadCT );
|
|
|
|
bool bNeededPlayers = false;
|
|
NeededPlayersCheck( bNeededPlayers );
|
|
|
|
return true;
|
|
}
|
|
|
|
// Initialize the player counts..
|
|
int NumDeadCT, NumDeadTerrorist, NumAliveTerrorist, NumAliveCT;
|
|
InitializePlayerCounts( NumAliveTerrorist, NumAliveCT, NumDeadTerrorist, NumDeadCT );
|
|
|
|
|
|
/***************************** OTHER PLAYER's CHECK *********************************************************/
|
|
bool bNeededPlayers = false;
|
|
if ( NeededPlayersCheck( bNeededPlayers ) )
|
|
return false;
|
|
|
|
/****************************** ASSASINATION/VIP SCENARIO CHECK *******************************************************/
|
|
if ( VIPRoundEndCheck( bNeededPlayers ) )
|
|
return true;
|
|
|
|
/****************************** PRISON ESCAPE CHECK *******************************************************/
|
|
if ( PrisonRoundEndCheck() )
|
|
return true;
|
|
|
|
|
|
/****************************** BOMB CHECK ********************************************************/
|
|
if ( BombRoundEndCheck( bNeededPlayers ) )
|
|
return true;
|
|
|
|
|
|
/***************************** TEAM EXTERMINATION CHECK!! *********************************************************/
|
|
// CounterTerrorists won by virture of elimination
|
|
if ( TeamExterminationCheck( NumAliveTerrorist, NumAliveCT, NumDeadTerrorist, NumDeadCT, bNeededPlayers ) )
|
|
return true;
|
|
|
|
|
|
/******************************** HOSTAGE RESCUE CHECK ******************************************************/
|
|
if ( HostageRescueRoundEndCheck( bNeededPlayers ) )
|
|
return true;
|
|
|
|
// scenario not won - still in progress
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::NeededPlayersCheck( bool &bNeededPlayers )
|
|
{
|
|
// We needed players to start scoring
|
|
// Do we have them now?
|
|
if( !m_iNumSpawnableTerrorist || !m_iNumSpawnableCT )
|
|
{
|
|
Msg( "Game will not start until both teams have players.\n" );
|
|
UTIL_ClientPrintAll( HUD_PRINTCONSOLE, "#Game_scoring" );
|
|
bNeededPlayers = true;
|
|
|
|
m_bFirstConnected = false;
|
|
}
|
|
|
|
if ( !m_bFirstConnected && m_iNumSpawnableTerrorist && m_iNumSpawnableCT )
|
|
{
|
|
// Start the round immediately when the first person joins
|
|
// UTIL_LogPrintf( "World triggered \"Game_Commencing\"\n" );
|
|
|
|
m_bFreezePeriod = false; //Make sure we are not on the FreezePeriod.
|
|
m_bCompleteReset = true;
|
|
|
|
TerminateRound( 3.0f, Game_Commencing );
|
|
m_bFirstConnected = true;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
void CCSGameRules::InitializePlayerCounts(
|
|
int &NumAliveTerrorist,
|
|
int &NumAliveCT,
|
|
int &NumDeadTerrorist,
|
|
int &NumDeadCT
|
|
)
|
|
{
|
|
NumAliveTerrorist = NumAliveCT = NumDeadCT = NumDeadTerrorist = 0;
|
|
m_iNumTerrorist = m_iNumCT = m_iNumSpawnableTerrorist = m_iNumSpawnableCT = 0;
|
|
m_iHaveEscaped = 0;
|
|
|
|
// Count how many dead players there are on each team.
|
|
for ( int iTeam=0; iTeam < GetNumberOfTeams(); iTeam++ )
|
|
{
|
|
CTeam *pTeam = GetGlobalTeam( iTeam );
|
|
|
|
for ( int iPlayer=0; iPlayer < pTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pTeam->GetPlayer( iPlayer ) );
|
|
Assert( pPlayer );
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
Assert( pPlayer->GetTeamNumber() == pTeam->GetTeamNumber() );
|
|
|
|
switch ( pTeam->GetTeamNumber() )
|
|
{
|
|
case TEAM_CT:
|
|
m_iNumCT++;
|
|
|
|
if ( pPlayer->State_Get() != STATE_PICKINGCLASS )
|
|
m_iNumSpawnableCT++;
|
|
|
|
if ( pPlayer->m_lifeState != LIFE_ALIVE )
|
|
NumDeadCT++;
|
|
else
|
|
NumAliveCT++;
|
|
|
|
break;
|
|
|
|
case TEAM_TERRORIST:
|
|
m_iNumTerrorist++;
|
|
|
|
if ( pPlayer->State_Get() != STATE_PICKINGCLASS )
|
|
m_iNumSpawnableTerrorist++;
|
|
|
|
if ( pPlayer->m_lifeState != LIFE_ALIVE )
|
|
NumDeadTerrorist++;
|
|
else
|
|
NumAliveTerrorist++;
|
|
|
|
// Check to see if this guy escaped.
|
|
if ( pPlayer->m_bEscaped == true )
|
|
m_iHaveEscaped++;
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CCSGameRules::HostageRescueRoundEndCheck( bool bNeededPlayers )
|
|
{
|
|
// Check to see if 50% of the hostages have been rescued.
|
|
CHostage* hostage = NULL;
|
|
|
|
int iNumHostages = g_Hostages.Count();
|
|
int iNumLeftToRescue = 0;
|
|
int i;
|
|
|
|
for ( i=0; i<iNumHostages; i++ )
|
|
{
|
|
hostage = g_Hostages[i];
|
|
|
|
if ( hostage->m_iHealth > 0 && !hostage->IsRescued() ) // We've found a live hostage. don't end the round
|
|
iNumLeftToRescue++;
|
|
}
|
|
|
|
m_iHostagesRemaining = iNumLeftToRescue;
|
|
|
|
if ( (iNumLeftToRescue == 0) && (iNumHostages > 0) )
|
|
{
|
|
if ( m_iHostagesRescued >= (iNumHostages * 0.5) )
|
|
{
|
|
m_iAccountCT += 2500;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins ++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
CCS_GameStats.Event_AllHostagesRescued();
|
|
// tell the bots all the hostages have been rescued
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "hostage_rescued_all" );
|
|
if ( event )
|
|
{
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), All_Hostages_Rescued );
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::PrisonRoundEndCheck()
|
|
{
|
|
//MIKETODO: get this working when working on prison escape
|
|
/*
|
|
if (m_bMapHasEscapeZone == true)
|
|
{
|
|
float flEscapeRatio;
|
|
|
|
flEscapeRatio = (float) m_iHaveEscaped / (float) m_iNumEscapers;
|
|
|
|
if (flEscapeRatio >= m_flRequiredEscapeRatio)
|
|
{
|
|
BroadcastSound( "Event.TERWin" );
|
|
m_iAccountTerrorist += 3150;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumTerroristWins ++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
EndRoundMessage( "#Terrorists_Escaped", Terrorists_Escaped );
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), WINNER_TER );
|
|
return;
|
|
}
|
|
else if ( NumAliveTerrorist == 0 && flEscapeRatio < m_flRequiredEscapeRatio)
|
|
{
|
|
BroadcastSound( "Event.CTWin" );
|
|
m_iAccountCT += (1 - flEscapeRatio) * 3500; // CTs are rewarded based on how many terrorists have escaped...
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
EndRoundMessage( "#CTs_PreventEscape", CTs_PreventEscape );
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), WINNER_CT );
|
|
return;
|
|
}
|
|
|
|
else if ( NumAliveTerrorist == 0 && NumDeadTerrorist != 0 && m_iNumSpawnableCT > 0 )
|
|
{
|
|
BroadcastSound( "Event.CTWin" );
|
|
m_iAccountCT += (1 - flEscapeRatio) * 3250; // CTs are rewarded based on how many terrorists have escaped...
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
EndRoundMessage( "#Escaping_Terrorists_Neutralized", Escaping_Terrorists_Neutralized );
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), WINNER_CT );
|
|
return;
|
|
}
|
|
// else return;
|
|
}
|
|
*/
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::VIPRoundEndCheck( bool bNeededPlayers )
|
|
{
|
|
if (m_iMapHasVIPSafetyZone != 1)
|
|
return false;
|
|
|
|
if (m_pVIP == NULL)
|
|
return false;
|
|
|
|
if (m_pVIP->m_bEscaped == true)
|
|
{
|
|
m_iAccountCT += 3500;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins ++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
//MIKETODO: get this working when working on VIP scenarios
|
|
/*
|
|
MessageBegin( MSG_SPEC, SVC_DIRECTOR );
|
|
WRITE_BYTE ( 9 ); // command length in bytes
|
|
WRITE_BYTE ( DRC_CMD_EVENT ); // VIP rescued
|
|
WRITE_SHORT( ENTINDEX(m_pVIP->edict()) ); // index number of primary entity
|
|
WRITE_SHORT( 0 ); // index number of secondary entity
|
|
WRITE_LONG( 15 | DRC_FLAG_FINAL); // eventflags (priority and flags)
|
|
MessageEnd();
|
|
*/
|
|
|
|
// tell the bots the VIP got out
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "vip_escaped" );
|
|
if ( event )
|
|
{
|
|
event->SetInt( "userid", m_pVIP->GetUserID() );
|
|
event->SetInt( "priority", 9 );
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [menglish] If the VIP has escaped award him an MVP
|
|
//=============================================================================
|
|
|
|
m_pVIP->IncrementNumMVPs( CSMVP_UNDEFINED );
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), VIP_Escaped );
|
|
return true;
|
|
}
|
|
else if ( m_pVIP->m_lifeState == LIFE_DEAD ) // The VIP is dead
|
|
{
|
|
m_iAccountTerrorist += 3250;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumTerroristWins ++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
// tell the bots the VIP was killed
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "vip_killed" );
|
|
if ( event )
|
|
{
|
|
event->SetInt( "userid", m_pVIP->GetUserID() );
|
|
event->SetInt( "priority", 9 );
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), VIP_Assassinated );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::BombRoundEndCheck( bool bNeededPlayers )
|
|
{
|
|
// Check to see if the bomb target was hit or the bomb defused.. if so, then let's end the round!
|
|
if ( ( m_bTargetBombed == true ) && ( m_bMapHasBombTarget == true ) )
|
|
{
|
|
m_iAccountTerrorist += 3500;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumTerroristWins ++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Target_Bombed );
|
|
return true;
|
|
}
|
|
else
|
|
if ( ( m_bBombDefused == true ) && ( m_bMapHasBombTarget == true ) )
|
|
{
|
|
m_iAccountCT += 3250;
|
|
|
|
m_iAccountTerrorist += 800; // give the T's a little bonus for planting the bomb even though it was defused.
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Bomb_Defused );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::TeamExterminationCheck(
|
|
int NumAliveTerrorist,
|
|
int NumAliveCT,
|
|
int NumDeadTerrorist,
|
|
int NumDeadCT,
|
|
bool bNeededPlayers
|
|
)
|
|
{
|
|
if ( ( m_iNumCT > 0 && m_iNumSpawnableCT > 0 ) && ( m_iNumTerrorist > 0 && m_iNumSpawnableTerrorist > 0 ) )
|
|
{
|
|
if ( NumAliveTerrorist == 0 && NumDeadTerrorist != 0 && m_iNumSpawnableCT > 0 )
|
|
{
|
|
bool nowin = false;
|
|
|
|
for ( int iGrenade=0; iGrenade < g_PlantedC4s.Count(); iGrenade++ )
|
|
{
|
|
CPlantedC4 *pC4 = g_PlantedC4s[iGrenade];
|
|
|
|
if ( pC4->IsBombActive() )
|
|
nowin = true;
|
|
}
|
|
|
|
if ( !nowin )
|
|
{
|
|
if ( m_bMapHasBombTarget )
|
|
m_iAccountCT += 3250;
|
|
else
|
|
m_iAccountCT += 3000;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumCTWins++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), CTs_Win );
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Terrorists WON
|
|
if ( NumAliveCT == 0 && NumDeadCT != 0 && m_iNumSpawnableTerrorist > 0 )
|
|
{
|
|
if ( m_bMapHasBombTarget )
|
|
m_iAccountTerrorist += 3250;
|
|
else
|
|
m_iAccountTerrorist += 3000;
|
|
|
|
if ( !bNeededPlayers )
|
|
{
|
|
m_iNumTerroristWins++;
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Terrorists_Win );
|
|
return true;
|
|
}
|
|
}
|
|
else if ( NumAliveCT == 0 && NumAliveTerrorist == 0 )
|
|
{
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Round_Draw );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
void CCSGameRules::PickNextVIP()
|
|
{
|
|
// MIKETODO: work on this when getting VIP maps running.
|
|
/*
|
|
if (IsVIPQueueEmpty() != true)
|
|
{
|
|
// Remove the current VIP from his VIP status and make him a regular CT.
|
|
if (m_pVIP != NULL)
|
|
ResetCurrentVIP();
|
|
|
|
for (int i = 0; i <= 4; i++)
|
|
{
|
|
if (VIPQueue[i] != NULL)
|
|
{
|
|
m_pVIP = VIPQueue[i];
|
|
m_pVIP->MakeVIP();
|
|
|
|
VIPQueue[i] = NULL; // remove this player from the VIP queue
|
|
StackVIPQueue(); // and re-organize the queue
|
|
m_iConsecutiveVIP = 0;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
else if (m_iConsecutiveVIP >= 3) // If it's been the same VIP for 3 rounds already.. then randomly pick a new one
|
|
{
|
|
m_iLastPick++;
|
|
|
|
if (m_iLastPick > m_iNumCT)
|
|
m_iLastPick = 1;
|
|
|
|
int iCount = 1;
|
|
|
|
CBaseEntity* pPlayer = NULL;
|
|
CBasePlayer* player = NULL;
|
|
CBasePlayer* pLastPlayer = NULL;
|
|
|
|
pPlayer = UTIL_FindEntityByClassname ( pPlayer, "player" );
|
|
while ( (pPlayer != NULL) && (!FNullEnt(pPlayer->edict())) )
|
|
{
|
|
if ( !(pPlayer->pev->flags & FL_DORMANT) )
|
|
{
|
|
player = GetClassPtr((CBasePlayer *)pPlayer->pev);
|
|
|
|
if ( (player->m_iTeam == CT) && (iCount == m_iLastPick) )
|
|
{
|
|
if ( (player == m_pVIP) && (pLastPlayer != NULL) )
|
|
player = pLastPlayer;
|
|
|
|
// Remove the current VIP from his VIP status and make him a regular CT.
|
|
if (m_pVIP != NULL)
|
|
ResetCurrentVIP();
|
|
|
|
player->MakeVIP();
|
|
m_iConsecutiveVIP = 0;
|
|
|
|
return;
|
|
}
|
|
else if ( player->m_iTeam == CT )
|
|
iCount++;
|
|
|
|
if ( player->m_iTeam != SPECTATOR )
|
|
pLastPlayer = player;
|
|
}
|
|
pPlayer = UTIL_FindEntityByClassname ( pPlayer, "player" );
|
|
}
|
|
}
|
|
else if (m_pVIP == NULL) // There is no VIP and there is no one waiting to be the VIP.. therefore just pick the first CT player we can find.
|
|
{
|
|
CBaseEntity* pPlayer = NULL;
|
|
CBasePlayer* player = NULL;
|
|
|
|
pPlayer = UTIL_FindEntityByClassname ( pPlayer, "player" );
|
|
while ( (pPlayer != NULL) && (!FNullEnt(pPlayer->edict())) )
|
|
{
|
|
if ( pPlayer->pev->flags != FL_DORMANT )
|
|
{
|
|
player = GetClassPtr((CBasePlayer *)pPlayer->pev);
|
|
|
|
if ( player->m_iTeam == CT )
|
|
{
|
|
player->MakeVIP();
|
|
m_iConsecutiveVIP = 0;
|
|
return;
|
|
}
|
|
}
|
|
pPlayer = UTIL_FindEntityByClassname ( pPlayer, "player" );
|
|
}
|
|
}
|
|
*/
|
|
}
|
|
|
|
|
|
void CCSGameRules::ReadMultiplayCvars()
|
|
{
|
|
m_iRoundTime = (int)(mp_roundtime.GetFloat() * 60);
|
|
m_iFreezeTime = mp_freezetime.GetInt();
|
|
}
|
|
|
|
|
|
void CCSGameRules::RestartRound()
|
|
{
|
|
#if defined( REPLAY_ENABLED )
|
|
if ( g_pReplay )
|
|
{
|
|
// Write replay and stop recording if appropriate
|
|
if ( g_pReplay->IsRecording() )
|
|
{
|
|
g_pReplay->SV_EndRecordingSession();
|
|
}
|
|
|
|
int nActivePlayerCount = m_iNumTerrorist + m_iNumCT;
|
|
if ( nActivePlayerCount && g_pReplay->SV_ShouldBeginRecording( false ) )
|
|
{
|
|
// Tell the replay manager that it should begin recording the new round as soon as possible
|
|
g_pReplay->SV_GetContext()->GetSessionRecorder()->StartRecording();
|
|
}
|
|
}
|
|
#endif
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Notify players that the round is about to be reset
|
|
//=============================================================================
|
|
for ( int clientIndex = 1; clientIndex <= gpGlobals->maxClients; clientIndex++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( clientIndex );
|
|
if(pPlayer)
|
|
{
|
|
pPlayer->OnPreResetRound();
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
if ( !IsFinite( gpGlobals->curtime ) )
|
|
{
|
|
Warning( "NaN curtime in RestartRound\n" );
|
|
gpGlobals->curtime = 0.0f;
|
|
}
|
|
|
|
int i;
|
|
|
|
m_iTotalRoundsPlayed++;
|
|
|
|
//ClearBodyQue();
|
|
|
|
// Hardlock the player accelaration to 5.0
|
|
//CVAR_SET_FLOAT( "sv_accelerate", 5.0 );
|
|
//CVAR_SET_FLOAT( "sv_friction", 4.0 );
|
|
//CVAR_SET_FLOAT( "sv_stopspeed", 75 );
|
|
|
|
sv_stopspeed.SetValue( 75.0f );
|
|
|
|
// Tabulate the number of players on each team.
|
|
int NumDeadCT, NumDeadTerrorist, NumAliveTerrorist, NumAliveCT;
|
|
InitializePlayerCounts( NumAliveTerrorist, NumAliveCT, NumDeadTerrorist, NumDeadCT );
|
|
|
|
m_bBombDropped = false;
|
|
m_bBombPlanted = false;
|
|
|
|
if ( GetHumanTeam() != TEAM_UNASSIGNED )
|
|
{
|
|
MoveHumansToHumanTeam();
|
|
}
|
|
|
|
/*************** AUTO-BALANCE CODE *************/
|
|
if ( mp_autoteambalance.GetInt() != 0 &&
|
|
(m_iUnBalancedRounds >= 1) )
|
|
{
|
|
if ( GetHumanTeam() == TEAM_UNASSIGNED )
|
|
{
|
|
BalanceTeams();
|
|
}
|
|
}
|
|
|
|
if ( ((m_iNumSpawnableCT - m_iNumSpawnableTerrorist) >= 2) ||
|
|
((m_iNumSpawnableTerrorist - m_iNumSpawnableCT) >= 2) )
|
|
{
|
|
m_iUnBalancedRounds++;
|
|
}
|
|
else
|
|
{
|
|
m_iUnBalancedRounds = 0;
|
|
}
|
|
|
|
// Warn the players of an impending auto-balance next round...
|
|
if ( mp_autoteambalance.GetInt() != 0 &&
|
|
(m_iUnBalancedRounds == 1) )
|
|
{
|
|
if ( GetHumanTeam() == TEAM_UNASSIGNED )
|
|
{
|
|
UTIL_ClientPrintAll( HUD_PRINTCENTER,"#Auto_Team_Balance_Next_Round");
|
|
}
|
|
}
|
|
|
|
/*************** AUTO-BALANCE CODE *************/
|
|
|
|
if ( m_bCompleteReset )
|
|
{
|
|
// bounds check
|
|
if ( mp_timelimit.GetInt() < 0 )
|
|
{
|
|
mp_timelimit.SetValue( 0 );
|
|
}
|
|
m_flGameStartTime = gpGlobals->curtime;
|
|
if ( !IsFinite( m_flGameStartTime.Get() ) )
|
|
{
|
|
Warning( "Trying to set a NaN game start time\n" );
|
|
m_flGameStartTime.GetForModify() = 0.0f;
|
|
}
|
|
|
|
// Reset total # of rounds played
|
|
m_iTotalRoundsPlayed = 0;
|
|
|
|
// Reset score info
|
|
m_iNumTerroristWins = 0;
|
|
m_iNumCTWins = 0;
|
|
m_iNumConsecutiveTerroristLoses = 0;
|
|
m_iNumConsecutiveCTLoses = 0;
|
|
|
|
|
|
// Reset team scores
|
|
UpdateTeamScores();
|
|
|
|
|
|
// Reset the player stats
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = CCSPlayer::Instance( i );
|
|
|
|
if ( pPlayer && !FNullEnt( pPlayer->edict() ) )
|
|
pPlayer->Reset();
|
|
}
|
|
}
|
|
|
|
m_bFreezePeriod = true;
|
|
|
|
ReadMultiplayCvars();
|
|
|
|
// Check to see if there's a mapping info paramater entity
|
|
if ( g_pMapInfo )
|
|
{
|
|
switch ( g_pMapInfo->m_iBuyingStatus )
|
|
{
|
|
case 0:
|
|
m_bCTCantBuy = false;
|
|
m_bTCantBuy = false;
|
|
Msg( "EVERYONE CAN BUY!\n" );
|
|
break;
|
|
|
|
case 1:
|
|
m_bCTCantBuy = false;
|
|
m_bTCantBuy = true;
|
|
Msg( "Only CT's can buy!!\n" );
|
|
break;
|
|
|
|
case 2:
|
|
m_bCTCantBuy = true;
|
|
m_bTCantBuy = false;
|
|
Msg( "Only T's can buy!!\n" );
|
|
break;
|
|
|
|
case 3:
|
|
m_bCTCantBuy = true;
|
|
m_bTCantBuy = true;
|
|
Msg( "No one can buy!!\n" );
|
|
break;
|
|
|
|
default:
|
|
m_bCTCantBuy = false;
|
|
m_bTCantBuy = false;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// by default everyone can buy
|
|
m_bCTCantBuy = false;
|
|
m_bTCantBuy = false;
|
|
}
|
|
|
|
|
|
// Check to see if this map has a bomb target in it
|
|
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_bomb_target" ) )
|
|
{
|
|
m_bMapHasBombTarget = true;
|
|
m_bMapHasBombZone = true;
|
|
}
|
|
else if ( gEntList.FindEntityByClassname( NULL, "info_bomb_target" ) )
|
|
{
|
|
m_bMapHasBombTarget = true;
|
|
m_bMapHasBombZone = false;
|
|
}
|
|
else
|
|
{
|
|
m_bMapHasBombTarget = false;
|
|
m_bMapHasBombZone = false;
|
|
}
|
|
|
|
// Check to see if this map has hostage rescue zones
|
|
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_hostage_rescue" ) )
|
|
m_bMapHasRescueZone = true;
|
|
else
|
|
m_bMapHasRescueZone = false;
|
|
|
|
|
|
// See if the map has func_buyzone entities
|
|
// Used by CBasePlayer::HandleSignals() to support maps without these entities
|
|
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_buyzone" ) )
|
|
m_bMapHasBuyZone = true;
|
|
else
|
|
m_bMapHasBuyZone = false;
|
|
|
|
|
|
// GOOSEMAN : See if this map has func_escapezone entities
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_escapezone" ) )
|
|
{
|
|
m_bMapHasEscapeZone = true;
|
|
m_iHaveEscaped = 0;
|
|
m_iNumEscapers = 0; // Will increase this later when we count how many Ts are starting
|
|
if (m_iNumEscapeRounds >= 3)
|
|
{
|
|
SwapAllPlayers();
|
|
m_iNumEscapeRounds = 0;
|
|
}
|
|
|
|
m_iNumEscapeRounds++; // Increment the number of rounds played... After 8 rounds, the players will do a whole sale switch..
|
|
}
|
|
else
|
|
m_bMapHasEscapeZone = false;
|
|
|
|
// Check to see if this map has VIP safety zones
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_vip_safetyzone" ) )
|
|
{
|
|
PickNextVIP();
|
|
m_iConsecutiveVIP++;
|
|
m_iMapHasVIPSafetyZone = 1;
|
|
}
|
|
else
|
|
m_iMapHasVIPSafetyZone = 2;
|
|
|
|
// Update accounts based on number of hostages remaining..
|
|
int iRescuedHostageBonus = 0;
|
|
|
|
for ( int iHostage=0; iHostage < g_Hostages.Count(); iHostage++ )
|
|
{
|
|
CHostage *pHostage = g_Hostages[iHostage];
|
|
|
|
if( pHostage->IsRescuable() ) //Alive and not rescued
|
|
{
|
|
iRescuedHostageBonus += 150;
|
|
}
|
|
|
|
if ( iRescuedHostageBonus >= 2000 )
|
|
break;
|
|
}
|
|
|
|
//*******Catch up code by SupraFiend. Scale up the loser bonus when teams fall into losing streaks
|
|
if (m_iRoundWinStatus == WINNER_TER) // terrorists won
|
|
{
|
|
//check to see if they just broke a losing streak
|
|
if(m_iNumConsecutiveTerroristLoses > 1)
|
|
m_iLoserBonus = 1500;//this is the default losing bonus
|
|
|
|
m_iNumConsecutiveTerroristLoses = 0;//starting fresh
|
|
m_iNumConsecutiveCTLoses++;//increment the number of wins the CTs have had
|
|
}
|
|
else if (m_iRoundWinStatus == WINNER_CT) // CT Won
|
|
{
|
|
//check to see if they just broke a losing streak
|
|
if(m_iNumConsecutiveCTLoses > 1)
|
|
m_iLoserBonus = 1500;//this is the default losing bonus
|
|
|
|
m_iNumConsecutiveCTLoses = 0;//starting fresh
|
|
m_iNumConsecutiveTerroristLoses++;//increment the number of wins the Terrorists have had
|
|
}
|
|
|
|
//check if the losing team is in a losing streak & that the loser bonus hasen't maxed out.
|
|
if((m_iNumConsecutiveTerroristLoses > 1) && (m_iLoserBonus < 3000))
|
|
m_iLoserBonus += 500;//help out the team in the losing streak
|
|
else
|
|
if((m_iNumConsecutiveCTLoses > 1) && (m_iLoserBonus < 3000))
|
|
m_iLoserBonus += 500;//help out the team in the losing streak
|
|
|
|
// assign the wining and losing bonuses
|
|
if (m_iRoundWinStatus == WINNER_TER) // terrorists won
|
|
{
|
|
m_iAccountTerrorist += iRescuedHostageBonus;
|
|
m_iAccountCT += m_iLoserBonus;
|
|
}
|
|
else if (m_iRoundWinStatus == WINNER_CT) // CT Won
|
|
{
|
|
m_iAccountCT += iRescuedHostageBonus;
|
|
if (m_bMapHasEscapeZone == false) // only give them the bonus if this isn't an escape map
|
|
m_iAccountTerrorist += m_iLoserBonus;
|
|
}
|
|
|
|
|
|
//Update CT account based on number of hostages rescued
|
|
m_iAccountCT += m_iHostagesRescued * 750;
|
|
|
|
|
|
// Update individual players accounts and respawn players
|
|
|
|
//**********new code by SupraFiend
|
|
//##########code changed by MartinO
|
|
//the round time stamp must be set before players are spawned
|
|
m_fRoundStartTime = gpGlobals->curtime + m_iFreezeTime;
|
|
|
|
if ( !IsFinite( m_fRoundStartTime.Get() ) )
|
|
{
|
|
Warning( "Trying to set a NaN round start time\n" );
|
|
m_fRoundStartTime.GetForModify() = 0.0f;
|
|
}
|
|
|
|
//Adrian - No cash for anyone at first rounds! ( well, only the default. )
|
|
if ( m_bCompleteReset )
|
|
{
|
|
m_iAccountTerrorist = m_iAccountCT = 0; //No extra cash!.
|
|
|
|
//We are starting fresh. So it's like no one has ever won or lost.
|
|
m_iNumTerroristWins = 0;
|
|
m_iNumCTWins = 0;
|
|
m_iNumConsecutiveTerroristLoses = 0;
|
|
m_iNumConsecutiveCTLoses = 0;
|
|
m_iLoserBonus = 1400;
|
|
}
|
|
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
pPlayer->m_iNumSpawns = 0;
|
|
pPlayer->m_bTeamChanged = false;
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT )
|
|
{
|
|
if (pPlayer->DoesPlayerGetRoundStartMoney())
|
|
{
|
|
pPlayer->AddAccount( m_iAccountCT );
|
|
}
|
|
}
|
|
else if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST )
|
|
{
|
|
m_iNumEscapers++; // Add another potential escaper to the mix!
|
|
if (pPlayer->DoesPlayerGetRoundStartMoney())
|
|
{
|
|
pPlayer->AddAccount( m_iAccountTerrorist );
|
|
}
|
|
}
|
|
|
|
// tricky, make players non solid while moving to their spawn points
|
|
if ( (pPlayer->GetTeamNumber() == TEAM_CT) || (pPlayer->GetTeamNumber() == TEAM_TERRORIST) )
|
|
{
|
|
pPlayer->AddSolidFlags( FSOLID_NOT_SOLID );
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Keep track of number of players per side and if they have the same uniform
|
|
//=============================================================================
|
|
|
|
int terroristUniform = -1;
|
|
bool allTerroristsWearingSameUniform = true;
|
|
int numberOfTerrorists = 0;
|
|
int ctUniform = -1;
|
|
bool allCtsWearingSameUniform = true;
|
|
int numberOfCts = 0;
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
// know respawn all players
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT && pPlayer->PlayerClass() >= FIRST_CT_CLASS && pPlayer->PlayerClass() <= LAST_CT_CLASS )
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Increment CT count and check CT uniforms.
|
|
//=============================================================================
|
|
|
|
numberOfCts++;
|
|
if (ctUniform == -1)
|
|
{
|
|
ctUniform = pPlayer->PlayerClass();
|
|
}
|
|
else if (pPlayer->PlayerClass() != ctUniform)
|
|
{
|
|
allCtsWearingSameUniform = false;
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
pPlayer->RoundRespawn();
|
|
}
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST && pPlayer->PlayerClass() >= FIRST_T_CLASS && pPlayer->PlayerClass() <= LAST_T_CLASS )
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] Increment terrorist count and check terrorist uniforms
|
|
//=============================================================================
|
|
|
|
numberOfTerrorists++;
|
|
if (terroristUniform == -1)
|
|
{
|
|
terroristUniform = pPlayer->PlayerClass();
|
|
}
|
|
else if (pPlayer->PlayerClass() != terroristUniform)
|
|
{
|
|
allTerroristsWearingSameUniform = false;
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
pPlayer->RoundRespawn();
|
|
}
|
|
else
|
|
{
|
|
pPlayer->ObserverRoundRespawn();
|
|
}
|
|
|
|
if ( pPlayer->m_iAccount > pPlayer->m_iShouldHaveCash )
|
|
{
|
|
m_bDontUploadStats = true;
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
//=============================================================================
|
|
|
|
// [tj] Award same uniform achievement for qualifying teams
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT && allCtsWearingSameUniform && numberOfCts >= AchievementConsts::SameUniform_MinPlayers)
|
|
{
|
|
pPlayer->AwardAchievement(CSSameUniform);
|
|
}
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST && allTerroristsWearingSameUniform && numberOfTerrorists >= AchievementConsts::SameUniform_MinPlayers)
|
|
{
|
|
pPlayer->AwardAchievement(CSSameUniform);
|
|
}
|
|
}
|
|
|
|
// [menglish] reset per-round achievement variables for each player
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
if( pPlayer )
|
|
{
|
|
pPlayer->ResetRoundBasedAchievementVariables();
|
|
}
|
|
}
|
|
|
|
// [pfreese] Reset all round or match stats, depending on type of restart
|
|
if ( m_bCompleteReset )
|
|
{
|
|
CCS_GameStats.ResetAllStats();
|
|
CCS_GameStats.ResetPlayerClassMatchStats();
|
|
}
|
|
else
|
|
{
|
|
CCS_GameStats.ResetRoundStats();
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
// Respawn entities (glass, doors, etc..)
|
|
CleanUpMap();
|
|
|
|
// now run a tkpunish check, after the map has been cleaned up
|
|
for ( i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT && pPlayer->PlayerClass() >= FIRST_CT_CLASS && pPlayer->PlayerClass() <= LAST_CT_CLASS )
|
|
{
|
|
pPlayer->CheckTKPunishment();
|
|
}
|
|
if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST && pPlayer->PlayerClass() >= FIRST_T_CLASS && pPlayer->PlayerClass() <= LAST_T_CLASS )
|
|
{
|
|
pPlayer->CheckTKPunishment();
|
|
}
|
|
}
|
|
|
|
// Give C4 to the terrorists
|
|
if (m_bMapHasBombTarget == true )
|
|
GiveC4();
|
|
|
|
// Reset game variables
|
|
m_flIntermissionEndTime = 0;
|
|
m_flRestartRoundTime = 0.0;
|
|
m_iAccountTerrorist = m_iAccountCT = 0;
|
|
m_iHostagesRescued = 0;
|
|
m_iHostagesTouched = 0;
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN
|
|
// [dwenger] Reset rescue-related achievement values
|
|
//=============================================================================
|
|
|
|
// [tj] reset flawless and lossless round related flags
|
|
m_bNoTerroristsKilled = true;
|
|
m_bNoCTsKilled = true;
|
|
m_bNoTerroristsDamaged = true;
|
|
m_bNoCTsDamaged = true;
|
|
m_pFirstKill = NULL;
|
|
m_pFirstBlood = NULL;
|
|
|
|
m_bCanDonateWeapons = true;
|
|
|
|
// [dwenger] Reset rescue-related achievement values
|
|
m_iHostagesRemaining = 0;
|
|
m_pLastRescuer = NULL;
|
|
|
|
m_hostageWasInjured = false;
|
|
m_hostageWasKilled = false;
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
m_iNumRescuers = 0;
|
|
m_iRoundWinStatus = WINNER_NONE;
|
|
m_bTargetBombed = m_bBombDefused = false;
|
|
m_bCompleteReset = false;
|
|
m_flNextHostageAnnouncement = gpGlobals->curtime;
|
|
|
|
m_iHostagesRemaining = g_Hostages.Count();
|
|
|
|
// fire global game event
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "round_start" );
|
|
if ( event )
|
|
{
|
|
event->SetInt("timelimit", m_iRoundTime );
|
|
event->SetInt("fraglimit", 0 );
|
|
event->SetInt( "priority", 6 ); // HLTV event priority, not transmitted
|
|
|
|
if ( m_bMapHasRescueZone )
|
|
{
|
|
event->SetString("objective","HOSTAGE RESCUE");
|
|
}
|
|
else if ( m_bMapHasEscapeZone )
|
|
{
|
|
event->SetString("objective","PRISON ESCAPE");
|
|
}
|
|
else if ( m_iMapHasVIPSafetyZone == 1 )
|
|
{
|
|
event->SetString("objective","VIP RESCUE");
|
|
}
|
|
else if ( m_bMapHasBombTarget || m_bMapHasBombZone )
|
|
{
|
|
event->SetString("objective","BOMB TARGET");
|
|
}
|
|
else
|
|
{
|
|
event->SetString("objective","DEATHMATCH");
|
|
}
|
|
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
UploadGameStats();
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [pfreese] I commented out this call to CreateWeaponManager, as the
|
|
// CGameWeaponManager object doesn't appear to be actually used by the CSS
|
|
// code, and in any case, the weapon manager does not support wildcards in
|
|
// entity names (as seemingly indicated) below. When the manager fails to
|
|
// create its factory, it removes itself in any case.
|
|
//=============================================================================
|
|
|
|
// CreateWeaponManager( "weapon_*", gpGlobals->maxClients * 2 );
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
}
|
|
|
|
void CCSGameRules::GiveC4()
|
|
{
|
|
enum {
|
|
ALL_TERRORISTS = 0,
|
|
HUMAN_TERRORISTS,
|
|
};
|
|
int iTerrorists[2][ABSOLUTE_PLAYER_LIMIT];
|
|
int numAliveTs[2] = { 0, 0 };
|
|
int lastBombGuyIndex[2] = { -1, -1 };
|
|
|
|
//Create an array of the indeces of bomb carrier candidates
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( UTIL_PlayerByIndex( i ) );
|
|
|
|
if( pPlayer && pPlayer->IsAlive() && pPlayer->GetTeamNumber() == TEAM_TERRORIST && numAliveTs[ALL_TERRORISTS] < ABSOLUTE_PLAYER_LIMIT )
|
|
{
|
|
if ( pPlayer == m_pLastBombGuy )
|
|
{
|
|
lastBombGuyIndex[ALL_TERRORISTS] = numAliveTs[ALL_TERRORISTS];
|
|
lastBombGuyIndex[HUMAN_TERRORISTS] = numAliveTs[HUMAN_TERRORISTS];
|
|
}
|
|
|
|
iTerrorists[ALL_TERRORISTS][numAliveTs[ALL_TERRORISTS]] = i;
|
|
numAliveTs[ALL_TERRORISTS]++;
|
|
if ( !pPlayer->IsBot() )
|
|
{
|
|
iTerrorists[HUMAN_TERRORISTS][numAliveTs[HUMAN_TERRORISTS]] = i;
|
|
numAliveTs[HUMAN_TERRORISTS]++;
|
|
}
|
|
}
|
|
}
|
|
|
|
int which = cv_bot_defer_to_human.GetBool();
|
|
if ( numAliveTs[HUMAN_TERRORISTS] == 0 )
|
|
{
|
|
which = ALL_TERRORISTS;
|
|
}
|
|
|
|
//pick one of the candidates randomly
|
|
if( numAliveTs[which] > 0 )
|
|
{
|
|
int index = random->RandomInt(0,numAliveTs[which]-1);
|
|
if ( lastBombGuyIndex[which] >= 0 )
|
|
{
|
|
// give the C4 sequentially
|
|
index = (lastBombGuyIndex[which] + 1) % numAliveTs[which];
|
|
}
|
|
CCSPlayer *pPlayer = ToCSPlayer( UTIL_PlayerByIndex( iTerrorists[which][index] ) );
|
|
|
|
Assert( pPlayer && pPlayer->GetTeamNumber() == TEAM_TERRORIST && pPlayer->IsAlive() );
|
|
|
|
pPlayer->GiveNamedItem( WEAPON_C4_CLASSNAME );
|
|
m_pLastBombGuy = pPlayer;
|
|
|
|
//pPlayer->SetBombIcon();
|
|
//pPlayer->pev->body = 1;
|
|
|
|
pPlayer->m_iDisplayHistoryBits |= DHF_BOMB_RETRIEVED;
|
|
pPlayer->HintMessage( "#Hint_you_have_the_bomb", false, true );
|
|
|
|
// Log this information
|
|
//UTIL_LogPrintf("\"%s<%i><%s><TERRORIST>\" triggered \"Spawned_With_The_Bomb\"\n",
|
|
// STRING( pPlayer->GetPlayerName() ),
|
|
// GETPLAYERUSERID( pPlayer->edict() ),
|
|
// GETPLAYERAUTHID( pPlayer->edict() ) );
|
|
}
|
|
|
|
m_bBombDropped = false;
|
|
}
|
|
|
|
void CCSGameRules::Think()
|
|
{
|
|
CGameRules::Think();
|
|
|
|
for ( int i = 0; i < GetNumberOfTeams(); i++ )
|
|
{
|
|
GetGlobalTeam( i )->Think();
|
|
}
|
|
|
|
///// Check game rules /////
|
|
if ( CheckGameOver() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
// have we hit the max rounds?
|
|
if ( CheckMaxRounds() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
// did somebaody hit the fraglimit ?
|
|
if ( CheckFragLimit() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( CheckWinLimit() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
|
|
// Check for the end of the round.
|
|
if ( IsFreezePeriod() )
|
|
{
|
|
CheckFreezePeriodExpired();
|
|
}
|
|
else
|
|
{
|
|
CheckRoundTimeExpired();
|
|
}
|
|
|
|
CheckLevelInitialized();
|
|
|
|
if ( m_flRestartRoundTime > 0.0f && m_flRestartRoundTime <= gpGlobals->curtime )
|
|
{
|
|
bool botSpeaking = false;
|
|
for ( int i=1; i <= gpGlobals->maxClients; ++i )
|
|
{
|
|
CBasePlayer *player = UTIL_PlayerByIndex( i );
|
|
if (player == NULL)
|
|
continue;
|
|
|
|
if (!player->IsBot())
|
|
continue;
|
|
|
|
CCSBot *bot = dynamic_cast< CCSBot * >(player);
|
|
if ( !bot )
|
|
continue;
|
|
|
|
if ( bot->IsUsingVoice() )
|
|
{
|
|
if ( gpGlobals->curtime > m_flRestartRoundTime + 10.0f )
|
|
{
|
|
Msg( "Ignoring speaking bot %s at round end\n", bot->GetPlayerName() );
|
|
}
|
|
else
|
|
{
|
|
botSpeaking = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( !botSpeaking )
|
|
{
|
|
RestartRound();
|
|
}
|
|
}
|
|
|
|
if ( gpGlobals->curtime > m_tmNextPeriodicThink )
|
|
{
|
|
CheckRestartRound();
|
|
m_tmNextPeriodicThink = gpGlobals->curtime + 1.0;
|
|
}
|
|
}
|
|
|
|
|
|
// The bots do their processing after physics simulation etc so their visibility checks don't recompute
|
|
// bone positions multiple times a frame.
|
|
void CCSGameRules::EndGameFrame( void )
|
|
{
|
|
TheBots->StartFrame();
|
|
|
|
BaseClass::EndGameFrame();
|
|
}
|
|
|
|
bool CCSGameRules::CheckGameOver()
|
|
{
|
|
if ( g_fGameOver ) // someone else quit the game already
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [Forrest] Calling ChangeLevel multiple times was causing IncrementMapCycleIndex
|
|
// to skip over maps in the list. Avoid this using a technique from CTeamplayRoundBasedRules::Think.
|
|
//=============================================================================
|
|
// check to see if we should change levels now
|
|
if ( m_flIntermissionEndTime && ( m_flIntermissionEndTime < gpGlobals->curtime ) )
|
|
{
|
|
ChangeLevel(); // intermission is over
|
|
|
|
// Don't run this code again
|
|
m_flIntermissionEndTime = 0.f;
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CCSGameRules::CheckFragLimit()
|
|
{
|
|
if ( fraglimit.GetInt() <= 0 )
|
|
return false;
|
|
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
|
|
|
|
if ( pPlayer && pPlayer->FragCount() >= fraglimit.GetInt() )
|
|
{
|
|
const char *teamName = "UNKNOWN";
|
|
if ( pPlayer->GetTeam() )
|
|
{
|
|
teamName = pPlayer->GetTeam()->GetName();
|
|
}
|
|
UTIL_LogPrintf("\"%s<%i><%s><%s>\" triggered \"Intermission_Kill_Limit\"\n",
|
|
pPlayer->GetPlayerName(),
|
|
pPlayer->GetUserID(),
|
|
pPlayer->GetNetworkIDString(),
|
|
teamName
|
|
);
|
|
GoToIntermission();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CCSGameRules::CheckMaxRounds()
|
|
{
|
|
if ( mp_maxrounds.GetInt() != 0 )
|
|
{
|
|
if ( m_iTotalRoundsPlayed >= mp_maxrounds.GetInt() )
|
|
{
|
|
UTIL_LogPrintf("World triggered \"Intermission_Round_Limit\"\n");
|
|
GoToIntermission();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::CheckWinLimit()
|
|
{
|
|
// has one team won the specified number of rounds?
|
|
if ( mp_winlimit.GetInt() != 0 )
|
|
{
|
|
if ( m_iNumCTWins >= mp_winlimit.GetInt() )
|
|
{
|
|
UTIL_LogPrintf("Team \"CT\" triggered \"Intermission_Win_Limit\"\n");
|
|
GoToIntermission();
|
|
return true;
|
|
}
|
|
if ( m_iNumTerroristWins >= mp_winlimit.GetInt() )
|
|
{
|
|
UTIL_LogPrintf("Team \"TERRORIST\" triggered \"Intermission_Win_Limit\"\n");
|
|
GoToIntermission();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
void CCSGameRules::CheckFreezePeriodExpired()
|
|
{
|
|
float startTime = m_fRoundStartTime;
|
|
if ( !IsFinite( startTime ) )
|
|
{
|
|
Warning( "Infinite round start time!\n" );
|
|
m_fRoundStartTime.GetForModify() = gpGlobals->curtime;
|
|
}
|
|
|
|
if ( IsFinite( startTime ) && gpGlobals->curtime < startTime )
|
|
{
|
|
return; // not time yet to start round
|
|
}
|
|
|
|
// Log this information
|
|
UTIL_LogPrintf("World triggered \"Round_Start\"\n");
|
|
|
|
char CT_sentence[40];
|
|
char T_sentence[40];
|
|
|
|
switch ( random->RandomInt( 0, 3 ) )
|
|
{
|
|
case 0:
|
|
Q_strncpy(CT_sentence,"radio.moveout", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence ,"radio.moveout", sizeof( T_sentence ) );
|
|
break;
|
|
|
|
case 1:
|
|
Q_strncpy(CT_sentence, "radio.letsgo", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence , "radio.letsgo", sizeof( T_sentence ) );
|
|
break;
|
|
|
|
case 2:
|
|
Q_strncpy(CT_sentence , "radio.locknload", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence , "radio.locknload", sizeof( T_sentence ) );
|
|
break;
|
|
|
|
default:
|
|
Q_strncpy(CT_sentence , "radio.go", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence , "radio.go", sizeof( T_sentence ) );
|
|
break;
|
|
}
|
|
|
|
// More specific radio commands for the new scenarios : Prison & Assasination
|
|
if (m_bMapHasEscapeZone == TRUE)
|
|
{
|
|
Q_strncpy(CT_sentence , "radio.elim", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence , "radio.getout", sizeof( T_sentence ) );
|
|
}
|
|
else if (m_iMapHasVIPSafetyZone == 1)
|
|
{
|
|
Q_strncpy(CT_sentence , "radio.vip", sizeof( CT_sentence ) );
|
|
Q_strncpy(T_sentence , "radio.locknload", sizeof( T_sentence ) );
|
|
}
|
|
|
|
// Freeze period expired: kill the flag
|
|
m_bFreezePeriod = false;
|
|
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "round_freeze_end" );
|
|
if ( event )
|
|
{
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
// Update the timers for all clients and play a sound
|
|
bool bCTPlayed = false;
|
|
bool bTPlayed = false;
|
|
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = CCSPlayer::Instance( i );
|
|
if ( pPlayer && !FNullEnt( pPlayer->edict() ) )
|
|
{
|
|
if ( pPlayer->State_Get() == STATE_ACTIVE )
|
|
{
|
|
if ( (pPlayer->GetTeamNumber() == TEAM_CT) && !bCTPlayed )
|
|
{
|
|
pPlayer->Radio( CT_sentence );
|
|
bCTPlayed = true;
|
|
}
|
|
else if ( (pPlayer->GetTeamNumber() == TEAM_TERRORIST) && !bTPlayed )
|
|
{
|
|
pPlayer->Radio( T_sentence );
|
|
bTPlayed = true;
|
|
}
|
|
|
|
}
|
|
|
|
//pPlayer->SyncRoundTimer();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void CCSGameRules::CheckRoundTimeExpired()
|
|
{
|
|
if ( mp_ignore_round_win_conditions.GetBool() )
|
|
return;
|
|
|
|
if ( GetRoundRemainingTime() > 0 || m_iRoundWinStatus != WINNER_NONE )
|
|
return; //We haven't completed other objectives, so go for this!.
|
|
|
|
if( !m_bFirstConnected )
|
|
return;
|
|
|
|
// New code to get rid of round draws!!
|
|
|
|
if ( m_bMapHasBombTarget )
|
|
{
|
|
//If the bomb is planted, don't let the round timer end the round.
|
|
//keep going until the bomb explodes or is defused
|
|
if( !m_bBombPlanted )
|
|
{
|
|
m_iAccountCT += 3250;
|
|
|
|
m_iNumCTWins++;
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Target_Saved );
|
|
UpdateTeamScores();
|
|
MarkLivingPlayersOnTeamAsNotReceivingMoneyNextRound(TEAM_TERRORIST);
|
|
}
|
|
}
|
|
else if ( m_bMapHasRescueZone )
|
|
{
|
|
m_iAccountTerrorist += 3250;
|
|
|
|
m_iNumTerroristWins++;
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Hostages_Not_Rescued );
|
|
UpdateTeamScores();
|
|
MarkLivingPlayersOnTeamAsNotReceivingMoneyNextRound(TEAM_CT);
|
|
}
|
|
else if ( m_bMapHasEscapeZone )
|
|
{
|
|
m_iNumCTWins++;
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), Terrorists_Not_Escaped );
|
|
UpdateTeamScores();
|
|
}
|
|
else if ( m_iMapHasVIPSafetyZone == 1 )
|
|
{
|
|
m_iAccountTerrorist += 3250;
|
|
m_iNumTerroristWins++;
|
|
|
|
TerminateRound( mp_round_restart_delay.GetFloat(), VIP_Not_Escaped );
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
#if defined( REPLAY_ENABLED )
|
|
if ( g_pReplay )
|
|
{
|
|
// Write replay and stop recording if appropriate
|
|
g_pReplay->SV_EndRecordingSession();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void CCSGameRules::GoToIntermission( void )
|
|
{
|
|
Msg( "Going to intermission...\n" );
|
|
|
|
IGameEvent *winEvent = gameeventmanager->CreateEvent( "cs_win_panel_match" );
|
|
|
|
if( winEvent )
|
|
{
|
|
for ( int teamIndex = TEAM_TERRORIST; teamIndex <= TEAM_CT; teamIndex++ )
|
|
{
|
|
CTeam *team = GetGlobalTeam( teamIndex );
|
|
if ( team )
|
|
{
|
|
float kills = CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_KILLS];
|
|
float deaths = CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_DEATHS];
|
|
// choose dialog variables to set depending on team
|
|
switch ( teamIndex )
|
|
{
|
|
case TEAM_TERRORIST:
|
|
winEvent->SetInt( "t_score", team->GetScore() );
|
|
if(deaths == 0)
|
|
{
|
|
winEvent->SetFloat( "t_kd", kills );
|
|
}
|
|
else
|
|
{
|
|
winEvent->SetFloat( "t_kd", kills / deaths );
|
|
}
|
|
winEvent->SetInt( "t_objectives_done", CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_OBJECTIVES_COMPLETED] );
|
|
winEvent->SetInt( "t_money_earned", CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_MONEY_EARNED] );
|
|
break;
|
|
case TEAM_CT:
|
|
winEvent->SetInt( "ct_score", team->GetScore() );
|
|
if(deaths == 0)
|
|
{
|
|
winEvent->SetFloat( "ct_kd", kills );
|
|
}
|
|
else
|
|
{
|
|
winEvent->SetFloat( "ct_kd", kills / deaths );
|
|
}
|
|
winEvent->SetInt( "ct_objectives_done", CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_OBJECTIVES_COMPLETED] );
|
|
winEvent->SetInt( "ct_money_earned", CCS_GameStats.GetTeamStats(teamIndex)[CSSTAT_MONEY_EARNED] );
|
|
break;
|
|
default:
|
|
Assert( false );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
gameeventmanager->FireEvent( winEvent );
|
|
}
|
|
|
|
BaseClass::GoToIntermission();
|
|
|
|
// set all players to FL_FROZEN
|
|
for ( int i = 1; i <= MAX_PLAYERS; i++ )
|
|
{
|
|
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
|
|
|
|
if ( pPlayer )
|
|
{
|
|
pPlayer->AddFlag( FL_FROZEN );
|
|
}
|
|
}
|
|
|
|
// freeze players while in intermission
|
|
m_bFreezePeriod = true;
|
|
}
|
|
|
|
int PlayerScoreInfoSort( const playerscore_t *p1, const playerscore_t *p2 )
|
|
{
|
|
// check frags
|
|
if ( p1->iScore > p2->iScore )
|
|
return -1;
|
|
if ( p2->iScore > p1->iScore )
|
|
return 1;
|
|
|
|
// check index
|
|
if ( p1->iPlayerIndex < p2->iPlayerIndex )
|
|
return -1;
|
|
|
|
return 1;
|
|
}
|
|
|
|
#if defined (_DEBUG)
|
|
void TestRoundWinpanel( void )
|
|
{
|
|
IGameEvent *event = gameeventmanager->CreateEvent( "round_end" );
|
|
event->SetInt( "winner", TEAM_TERRORIST );
|
|
|
|
if ( event )
|
|
{
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
|
|
IGameEvent *event2 = gameeventmanager->CreateEvent( "player_death" );
|
|
if ( event2 )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( UTIL_PlayerByIndex(1) );
|
|
|
|
// pCappingPlayers is a null terminated list of player indeces
|
|
event2->SetInt("userid", pPlayer->GetUserID() );
|
|
event2->SetInt("attacker", pPlayer->GetUserID() );
|
|
event2->SetString("weapon", "Bare Hands" );
|
|
event2->SetInt("headshot", 1 );
|
|
event2->SetInt( "revenge", 1 );
|
|
|
|
gameeventmanager->FireEvent( event2 );
|
|
}
|
|
|
|
IGameEvent *winEvent = gameeventmanager->CreateEvent( "cs_win_panel_round" );
|
|
|
|
if ( winEvent )
|
|
{
|
|
if ( 1 )
|
|
{
|
|
if ( 0 /*team == m_iTimerWinTeam */)
|
|
{
|
|
// timer expired, defenders win
|
|
// show total time that was defended
|
|
winEvent->SetBool( "show_timer_defend", true );
|
|
winEvent->SetInt( "timer_time", 0 /*m_pRoundTimer->GetTimerMaxLength() */);
|
|
}
|
|
else
|
|
{
|
|
// attackers win
|
|
// show time it took for them to win
|
|
winEvent->SetBool( "show_timer_attack", true );
|
|
|
|
int iTimeElapsed = 90; //m_pRoundTimer->GetTimerMaxLength() - (int)m_pRoundTimer->GetTimeRemaining();
|
|
winEvent->SetInt( "timer_time", iTimeElapsed );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
winEvent->SetBool( "show_timer_attack", false );
|
|
winEvent->SetBool( "show_timer_defend", false );
|
|
}
|
|
|
|
int iLastEvent = Terrorists_Win;
|
|
|
|
winEvent->SetInt( "final_event", iLastEvent );
|
|
|
|
// Set the fun fact data in the event
|
|
winEvent->SetString( "funfact_token", "#funfact_first_blood" );
|
|
winEvent->SetInt( "funfact_player", 1 );
|
|
winEvent->SetInt( "funfact_data1", 20 );
|
|
winEvent->SetInt( "funfact_data2", 31 );
|
|
winEvent->SetInt( "funfact_data3", 45 );
|
|
|
|
gameeventmanager->FireEvent( winEvent );
|
|
}
|
|
}
|
|
ConCommand test_round_winpanel( "test_round_winpanel", TestRoundWinpanel, "", FCVAR_DEVELOPMENTONLY | FCVAR_CHEAT );
|
|
|
|
void TestMatchWinpanel( void )
|
|
{
|
|
IGameEvent *event = gameeventmanager->CreateEvent( "round_end" );
|
|
event->SetInt( "winner", TEAM_TERRORIST );
|
|
|
|
if ( event )
|
|
{
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
IGameEvent *winEvent = gameeventmanager->CreateEvent( "cs_win_panel_match" );
|
|
|
|
if ( winEvent )
|
|
{
|
|
winEvent->SetInt( "t_score", 4 );
|
|
winEvent->SetInt( "ct_score", 1 );
|
|
|
|
winEvent->SetFloat( "t_kd", 1.8f );
|
|
winEvent->SetFloat( "ct_kd", 0.4f );
|
|
|
|
winEvent->SetInt( "t_objectives_done", 5 );
|
|
winEvent->SetInt( "ct_objectives_done", 2 );
|
|
|
|
winEvent->SetInt( "t_money_earned", 30000 );
|
|
winEvent->SetInt( "ct_money_earned", 19999 );
|
|
|
|
gameeventmanager->FireEvent( winEvent );
|
|
}
|
|
}
|
|
ConCommand test_match_winpanel( "test_match_winpanel", TestMatchWinpanel, "", FCVAR_DEVELOPMENTONLY | FCVAR_CHEAT );
|
|
|
|
void TestFreezePanel( void )
|
|
{
|
|
IGameEvent *event = gameeventmanager->CreateEvent( "freezecam_started" );
|
|
if ( event )
|
|
{
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
IGameEvent *winEvent = gameeventmanager->CreateEvent( "show_freezepanel" );
|
|
|
|
if ( winEvent )
|
|
{
|
|
winEvent->SetInt( "killer", 1 );
|
|
gameeventmanager->FireEvent( winEvent );
|
|
}
|
|
}
|
|
ConCommand test_freezepanel( "test_freezepanel", TestFreezePanel, "", FCVAR_DEVELOPMENTONLY | FCVAR_CHEAT );
|
|
#endif // _DEBUG
|
|
|
|
static void PrintToConsole( CBasePlayer *player, const char *text )
|
|
{
|
|
if ( player )
|
|
{
|
|
ClientPrint( player, HUD_PRINTCONSOLE, text );
|
|
}
|
|
else
|
|
{
|
|
Msg( "%s", text );
|
|
}
|
|
}
|
|
|
|
void CCSGameRules::DumpTimers( void ) const
|
|
{
|
|
extern ConVar bot_join_delay;
|
|
CBasePlayer *player = UTIL_GetCommandClient();
|
|
CFmtStr str;
|
|
|
|
PrintToConsole( player, str.sprintf( "Timers and related info at %f:\n", gpGlobals->curtime ) );
|
|
PrintToConsole( player, str.sprintf( "m_bCompleteReset: %d\n", m_bCompleteReset ) );
|
|
PrintToConsole( player, str.sprintf( "m_iTotalRoundsPlayed: %d\n", m_iTotalRoundsPlayed ) );
|
|
PrintToConsole( player, str.sprintf( "m_iRoundTime: %d\n", m_iRoundTime.Get() ) );
|
|
PrintToConsole( player, str.sprintf( "m_iRoundWinStatus: %d\n", m_iRoundWinStatus ) );
|
|
|
|
PrintToConsole( player, str.sprintf( "first connected: %d\n", m_bFirstConnected ) );
|
|
PrintToConsole( player, str.sprintf( "intermission end time: %f\n", m_flIntermissionEndTime ) );
|
|
PrintToConsole( player, str.sprintf( "freeze period: %d\n", m_bFreezePeriod.Get() ) );
|
|
PrintToConsole( player, str.sprintf( "round restart time: %f\n", m_flRestartRoundTime ) );
|
|
PrintToConsole( player, str.sprintf( "game start time: %f\n", m_flGameStartTime.Get() ) );
|
|
PrintToConsole( player, str.sprintf( "m_fRoundStartTime: %f\n", m_fRoundStartTime.Get() ) );
|
|
PrintToConsole( player, str.sprintf( "freeze time: %d\n", m_iFreezeTime ) );
|
|
PrintToConsole( player, str.sprintf( "next think: %f\n", m_tmNextPeriodicThink ) );
|
|
|
|
PrintToConsole( player, str.sprintf( "fraglimit: %d\n", fraglimit.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "mp_maxrounds: %d\n", mp_maxrounds.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "mp_winlimit: %d\n", mp_winlimit.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "bot_quota: %d\n", cv_bot_quota.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "bot_quota_mode: %s\n", cv_bot_quota_mode.GetString() ) );
|
|
PrintToConsole( player, str.sprintf( "bot_join_after_player: %d\n", cv_bot_join_after_player.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "bot_join_delay: %d\n", bot_join_delay.GetInt() ) );
|
|
PrintToConsole( player, str.sprintf( "nextlevel: %s\n", nextlevel.GetString() ) );
|
|
|
|
int humansInGame = UTIL_HumansInGame( true );
|
|
int botsInGame = UTIL_BotsInGame();
|
|
PrintToConsole( player, str.sprintf( "%d humans and %d bots in game\n", humansInGame, botsInGame ) );
|
|
|
|
PrintToConsole( player, str.sprintf( "num CTs (spawnable): %d (%d)\n", m_iNumCT, m_iNumSpawnableCT ) );
|
|
PrintToConsole( player, str.sprintf( "num Ts (spawnable): %d (%d)\n", m_iNumTerrorist, m_iNumSpawnableTerrorist ) );
|
|
|
|
if ( g_fGameOver )
|
|
{
|
|
PrintToConsole( player, str.sprintf( "Game is over!\n" ) );
|
|
}
|
|
PrintToConsole( player, str.sprintf( "\n" ) );
|
|
}
|
|
|
|
CON_COMMAND( mp_dump_timers, "Prints round timers to the console for debugging" )
|
|
{
|
|
if ( !UTIL_IsCommandIssuedByServerAdmin() )
|
|
return;
|
|
|
|
if ( CSGameRules() )
|
|
{
|
|
CSGameRules()->DumpTimers();
|
|
}
|
|
}
|
|
|
|
|
|
// living players on the given team need to be marked as not receiving any money
|
|
// next round.
|
|
void CCSGameRules::MarkLivingPlayersOnTeamAsNotReceivingMoneyNextRound(int team)
|
|
{
|
|
int playerNum;
|
|
for (playerNum = 1; playerNum <= gpGlobals->maxClients; ++playerNum)
|
|
{
|
|
CCSPlayer *player = (CCSPlayer *)UTIL_PlayerByIndex(playerNum);
|
|
if (player == NULL)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ((player->GetTeamNumber() == team) && (player->IsAlive()))
|
|
{
|
|
player->MarkAsNotReceivingMoneyNextRound();
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void CCSGameRules::CheckLevelInitialized( void )
|
|
{
|
|
if ( !m_bLevelInitialized )
|
|
{
|
|
// Count the number of spawn points for each team
|
|
// This determines the maximum number of players allowed on each
|
|
|
|
CBaseEntity* ent = NULL;
|
|
|
|
m_iSpawnPointCount_Terrorist = 0;
|
|
m_iSpawnPointCount_CT = 0;
|
|
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "info_player_terrorist" ) ) != NULL )
|
|
{
|
|
if ( IsSpawnPointValid( ent, NULL ) )
|
|
{
|
|
m_iSpawnPointCount_Terrorist++;
|
|
}
|
|
else
|
|
{
|
|
Warning("Invalid terrorist spawnpoint at (%.1f,%.1f,%.1f)\n",
|
|
ent->GetAbsOrigin()[0],ent->GetAbsOrigin()[2],ent->GetAbsOrigin()[2] );
|
|
}
|
|
}
|
|
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "info_player_counterterrorist" ) ) != NULL )
|
|
{
|
|
if ( IsSpawnPointValid( ent, NULL ) )
|
|
{
|
|
m_iSpawnPointCount_CT++;
|
|
}
|
|
else
|
|
{
|
|
Warning("Invalid counterterrorist spawnpoint at (%.1f,%.1f,%.1f)\n",
|
|
ent->GetAbsOrigin()[0],ent->GetAbsOrigin()[2],ent->GetAbsOrigin()[2] );
|
|
}
|
|
}
|
|
|
|
// Is this a logo map?
|
|
if ( gEntList.FindEntityByClassname( NULL, "info_player_logo" ) )
|
|
m_bLogoMap = true;
|
|
|
|
m_bLevelInitialized = true;
|
|
}
|
|
}
|
|
|
|
void CCSGameRules::ShowSpawnPoints( void )
|
|
{
|
|
CBaseEntity* ent = NULL;
|
|
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "info_player_terrorist" ) ) != NULL )
|
|
{
|
|
if ( IsSpawnPointValid( ent, NULL ) )
|
|
{
|
|
NDebugOverlay::Box( ent->GetAbsOrigin(), VEC_HULL_MIN, VEC_HULL_MAX, 0, 255, 0, 200, 600 );
|
|
}
|
|
else
|
|
{
|
|
NDebugOverlay::Box( ent->GetAbsOrigin(), VEC_HULL_MIN, VEC_HULL_MAX, 255, 0, 0, 200, 600);
|
|
}
|
|
}
|
|
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "info_player_counterterrorist" ) ) != NULL )
|
|
{
|
|
if ( IsSpawnPointValid( ent, NULL ) )
|
|
{
|
|
NDebugOverlay::Box( ent->GetAbsOrigin(), VEC_HULL_MIN, VEC_HULL_MAX, 0, 255, 0, 200, 600 );
|
|
}
|
|
else
|
|
{
|
|
NDebugOverlay::Box( ent->GetAbsOrigin(), VEC_HULL_MIN, VEC_HULL_MAX, 255, 0, 0, 200, 600 );
|
|
}
|
|
}
|
|
}
|
|
|
|
void CCSGameRules::CheckRestartRound( void )
|
|
{
|
|
// Restart the game if specified by the server
|
|
int iRestartDelay = mp_restartgame.GetInt();
|
|
|
|
if ( iRestartDelay > 0 )
|
|
{
|
|
if ( iRestartDelay > 60 )
|
|
iRestartDelay = 60;
|
|
|
|
// log the restart
|
|
UTIL_LogPrintf( "World triggered \"Restart_Round_(%i_%s)\"\n", iRestartDelay, iRestartDelay == 1 ? "second" : "seconds" );
|
|
|
|
UTIL_LogPrintf( "Team \"CT\" scored \"%i\" with \"%i\" players\n", m_iNumCTWins, m_iNumCT );
|
|
UTIL_LogPrintf( "Team \"TERRORIST\" scored \"%i\" with \"%i\" players\n", m_iNumTerroristWins, m_iNumTerrorist );
|
|
|
|
// let the players know
|
|
char strRestartDelay[64];
|
|
Q_snprintf( strRestartDelay, sizeof( strRestartDelay ), "%d", iRestartDelay );
|
|
UTIL_ClientPrintAll( HUD_PRINTCENTER, "#Game_will_restart_in", strRestartDelay, iRestartDelay == 1 ? "SECOND" : "SECONDS" );
|
|
UTIL_ClientPrintAll( HUD_PRINTCONSOLE, "#Game_will_restart_in", strRestartDelay, iRestartDelay == 1 ? "SECOND" : "SECONDS" );
|
|
|
|
m_flRestartRoundTime = gpGlobals->curtime + iRestartDelay;
|
|
m_bCompleteReset = true;
|
|
mp_restartgame.SetValue( 0 );
|
|
}
|
|
}
|
|
|
|
|
|
class SetHumanTeamFunctor
|
|
{
|
|
public:
|
|
SetHumanTeamFunctor( int targetTeam )
|
|
{
|
|
m_targetTeam = targetTeam;
|
|
m_sourceTeam = ( m_targetTeam == TEAM_CT ) ? TEAM_TERRORIST : TEAM_CT;
|
|
|
|
m_traitors.MakeReliable();
|
|
m_loyalists.MakeReliable();
|
|
m_loyalists.AddAllPlayers();
|
|
}
|
|
|
|
bool operator()( CBasePlayer *basePlayer )
|
|
{
|
|
CCSPlayer *player = ToCSPlayer( basePlayer );
|
|
if ( !player )
|
|
return true;
|
|
|
|
if ( player->IsBot() )
|
|
return true;
|
|
|
|
if ( player->GetTeamNumber() != m_sourceTeam )
|
|
return true;
|
|
|
|
if ( player->State_Get() == STATE_PICKINGCLASS )
|
|
return true;
|
|
|
|
if ( CSGameRules()->TeamFull( m_targetTeam ) )
|
|
return false;
|
|
|
|
if ( CSGameRules()->TeamStacked( m_targetTeam, m_sourceTeam ) )
|
|
return false;
|
|
|
|
player->SwitchTeam( m_targetTeam );
|
|
m_traitors.AddRecipient( player );
|
|
m_loyalists.RemoveRecipient( player );
|
|
|
|
return true;
|
|
}
|
|
|
|
void SendNotice( void )
|
|
{
|
|
if ( m_traitors.GetRecipientCount() > 0 )
|
|
{
|
|
UTIL_ClientPrintFilter( m_traitors, HUD_PRINTCENTER, "#Player_Balanced" );
|
|
UTIL_ClientPrintFilter( m_loyalists, HUD_PRINTCENTER, "#Teams_Balanced" );
|
|
}
|
|
}
|
|
|
|
private:
|
|
int m_targetTeam;
|
|
int m_sourceTeam;
|
|
|
|
CRecipientFilter m_traitors;
|
|
CRecipientFilter m_loyalists;
|
|
};
|
|
|
|
|
|
void CCSGameRules::MoveHumansToHumanTeam( void )
|
|
{
|
|
int targetTeam = GetHumanTeam();
|
|
if ( targetTeam != TEAM_TERRORIST && targetTeam != TEAM_CT )
|
|
return;
|
|
|
|
SetHumanTeamFunctor setTeam( targetTeam );
|
|
ForEachPlayer( setTeam );
|
|
|
|
setTeam.SendNotice();
|
|
}
|
|
|
|
|
|
void CCSGameRules::BalanceTeams( void )
|
|
{
|
|
int iTeamToSwap = TEAM_UNASSIGNED;
|
|
int iNumToSwap;
|
|
|
|
if (m_iMapHasVIPSafetyZone == 1) // The ratio for teams is different for Assasination maps
|
|
{
|
|
int iDesiredNumCT, iDesiredNumTerrorist;
|
|
|
|
if ( (m_iNumCT + m_iNumTerrorist)%2 != 0) // uneven number of players
|
|
iDesiredNumCT = (int)((m_iNumCT + m_iNumTerrorist) * 0.55) + 1;
|
|
else
|
|
iDesiredNumCT = (int)((m_iNumCT + m_iNumTerrorist)/2);
|
|
iDesiredNumTerrorist = (m_iNumCT + m_iNumTerrorist) - iDesiredNumCT;
|
|
|
|
if ( m_iNumCT < iDesiredNumCT )
|
|
{
|
|
iTeamToSwap = TEAM_TERRORIST;
|
|
iNumToSwap = iDesiredNumCT - m_iNumCT;
|
|
}
|
|
else if ( m_iNumTerrorist < iDesiredNumTerrorist )
|
|
{
|
|
iTeamToSwap = TEAM_CT;
|
|
iNumToSwap = iDesiredNumTerrorist - m_iNumTerrorist;
|
|
}
|
|
else
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
if (m_iNumCT > m_iNumTerrorist)
|
|
{
|
|
iTeamToSwap = TEAM_CT;
|
|
iNumToSwap = (m_iNumCT - m_iNumTerrorist)/2;
|
|
|
|
}
|
|
else if (m_iNumTerrorist > m_iNumCT)
|
|
{
|
|
iTeamToSwap = TEAM_TERRORIST;
|
|
iNumToSwap = (m_iNumTerrorist - m_iNumCT)/2;
|
|
}
|
|
else
|
|
{
|
|
return; // Teams are even.. Get out of here.
|
|
}
|
|
}
|
|
|
|
if (iNumToSwap > 3) // Don't swap more than 3 players at a time.. This is a naive method of avoiding infinite loops.
|
|
iNumToSwap = 3;
|
|
|
|
int iTragetTeam = TEAM_UNASSIGNED;
|
|
|
|
if ( iTeamToSwap == TEAM_CT )
|
|
{
|
|
iTragetTeam = TEAM_TERRORIST;
|
|
}
|
|
else if ( iTeamToSwap == TEAM_TERRORIST )
|
|
{
|
|
iTragetTeam = TEAM_CT;
|
|
}
|
|
else
|
|
{
|
|
// no valid team to swap
|
|
return;
|
|
}
|
|
|
|
CRecipientFilter traitors;
|
|
CRecipientFilter loyalists;
|
|
|
|
traitors.MakeReliable();
|
|
loyalists.MakeReliable();
|
|
loyalists.AddAllPlayers();
|
|
|
|
for (int i = 0; i < iNumToSwap; i++)
|
|
{
|
|
// last person to join the server
|
|
int iHighestUserID = -1;
|
|
CCSPlayer *pPlayerToSwap = NULL;
|
|
|
|
// check if target team is full, exit if so
|
|
if ( TeamFull(iTragetTeam) )
|
|
break;
|
|
|
|
// search for player with highest UserID = most recently joined to switch over
|
|
for ( int j = 1; j <= gpGlobals->maxClients; j++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer *)UTIL_PlayerByIndex( j );
|
|
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
CCSBot *bot = dynamic_cast< CCSBot * >(pPlayer);
|
|
if ( bot )
|
|
continue; // don't swap bots - the bot system will handle that
|
|
|
|
if ( pPlayer &&
|
|
( m_pVIP != pPlayer ) &&
|
|
( pPlayer->GetTeamNumber() == iTeamToSwap ) &&
|
|
( engine->GetPlayerUserId( pPlayer->edict() ) > iHighestUserID ) &&
|
|
( pPlayer->State_Get() != STATE_PICKINGCLASS ) )
|
|
{
|
|
iHighestUserID = engine->GetPlayerUserId( pPlayer->edict() );
|
|
pPlayerToSwap = pPlayer;
|
|
}
|
|
}
|
|
|
|
if ( pPlayerToSwap != NULL )
|
|
{
|
|
traitors.AddRecipient( pPlayerToSwap );
|
|
loyalists.RemoveRecipient( pPlayerToSwap );
|
|
pPlayerToSwap->SwitchTeam( iTragetTeam );
|
|
}
|
|
}
|
|
|
|
if ( traitors.GetRecipientCount() > 0 )
|
|
{
|
|
UTIL_ClientPrintFilter( traitors, HUD_PRINTCENTER, "#Player_Balanced" );
|
|
UTIL_ClientPrintFilter( loyalists, HUD_PRINTCENTER, "#Teams_Balanced" );
|
|
}
|
|
}
|
|
|
|
|
|
bool CCSGameRules::TeamFull( int team_id )
|
|
{
|
|
CheckLevelInitialized();
|
|
|
|
switch ( team_id )
|
|
{
|
|
case TEAM_TERRORIST:
|
|
return m_iNumTerrorist >= m_iSpawnPointCount_Terrorist;
|
|
|
|
case TEAM_CT:
|
|
return m_iNumCT >= m_iSpawnPointCount_CT;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int CCSGameRules::GetHumanTeam()
|
|
{
|
|
if ( FStrEq( "CT", mp_humanteam.GetString() ) )
|
|
{
|
|
return TEAM_CT;
|
|
}
|
|
else if ( FStrEq( "T", mp_humanteam.GetString() ) )
|
|
{
|
|
return TEAM_TERRORIST;
|
|
}
|
|
|
|
return TEAM_UNASSIGNED;
|
|
}
|
|
|
|
int CCSGameRules::SelectDefaultTeam( bool ignoreBots /*= false*/ )
|
|
{
|
|
if ( ignoreBots && ( FStrEq( cv_bot_join_team.GetString(), "T" ) || FStrEq( cv_bot_join_team.GetString(), "CT" ) ) )
|
|
{
|
|
ignoreBots = false; // don't ignore bots when they can't switch teams
|
|
}
|
|
|
|
if ( ignoreBots && !mp_autoteambalance.GetBool() )
|
|
{
|
|
ignoreBots = false; // don't ignore bots when they can't switch teams
|
|
}
|
|
|
|
int team = TEAM_UNASSIGNED;
|
|
int numTerrorists = m_iNumTerrorist;
|
|
int numCTs = m_iNumCT;
|
|
if ( ignoreBots )
|
|
{
|
|
numTerrorists = UTIL_HumansOnTeam( TEAM_TERRORIST );
|
|
numCTs = UTIL_HumansOnTeam( TEAM_CT );
|
|
}
|
|
|
|
// Choose the team that's lacking players
|
|
if ( numTerrorists < numCTs )
|
|
{
|
|
team = TEAM_TERRORIST;
|
|
}
|
|
else if ( numTerrorists > numCTs )
|
|
{
|
|
team = TEAM_CT;
|
|
}
|
|
// Choose the team that's losing
|
|
else if ( m_iNumTerroristWins < m_iNumCTWins )
|
|
{
|
|
team = TEAM_TERRORIST;
|
|
}
|
|
else if ( m_iNumCTWins < m_iNumTerroristWins )
|
|
{
|
|
team = TEAM_CT;
|
|
}
|
|
else
|
|
{
|
|
// Teams and scores are equal, pick a random team
|
|
if ( random->RandomInt( 0, 1 ) == 0 )
|
|
{
|
|
team = TEAM_CT;
|
|
}
|
|
else
|
|
{
|
|
team = TEAM_TERRORIST;
|
|
}
|
|
}
|
|
|
|
if ( TeamFull( team ) )
|
|
{
|
|
// Pick the opposite team
|
|
if ( team == TEAM_TERRORIST )
|
|
{
|
|
team = TEAM_CT;
|
|
}
|
|
else
|
|
{
|
|
team = TEAM_TERRORIST;
|
|
}
|
|
|
|
// No choices left
|
|
if ( TeamFull( team ) )
|
|
return TEAM_UNASSIGNED;
|
|
}
|
|
|
|
return team;
|
|
}
|
|
|
|
//checks to see if the desired team is stacked, returns true if it is
|
|
bool CCSGameRules::TeamStacked( int newTeam_id, int curTeam_id )
|
|
{
|
|
//players are allowed to change to their own team
|
|
if(newTeam_id == curTeam_id)
|
|
return false;
|
|
|
|
// if mp_limitteams is 0, don't check
|
|
if ( mp_limitteams.GetInt() == 0 )
|
|
return false;
|
|
|
|
switch ( newTeam_id )
|
|
{
|
|
case TEAM_TERRORIST:
|
|
if(curTeam_id != TEAM_UNASSIGNED && curTeam_id != TEAM_SPECTATOR)
|
|
{
|
|
if((m_iNumTerrorist + 1) > (m_iNumCT + mp_limitteams.GetInt() - 1))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if((m_iNumTerrorist + 1) > (m_iNumCT + mp_limitteams.GetInt()))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
break;
|
|
case TEAM_CT:
|
|
if(curTeam_id != TEAM_UNASSIGNED && curTeam_id != TEAM_SPECTATOR)
|
|
{
|
|
if((m_iNumCT + 1) > (m_iNumTerrorist + mp_limitteams.GetInt() - 1))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if((m_iNumCT + 1) > (m_iNumTerrorist + mp_limitteams.GetInt()))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
//=========================================================
|
|
//=========================================================
|
|
bool CCSGameRules::FPlayerCanRespawn( CBasePlayer *pBasePlayer )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pBasePlayer );
|
|
if ( !pPlayer )
|
|
Error( "FPlayerCanRespawn: pPlayer=0" );
|
|
|
|
// Player cannot respawn twice in a round
|
|
if ( pPlayer->m_iNumSpawns > 0 && m_bFirstConnected )
|
|
return false;
|
|
|
|
// If they're dead after the map has ended, and it's about to start the next round,
|
|
// wait for the round restart to respawn them.
|
|
if ( gpGlobals->curtime < m_flRestartRoundTime )
|
|
return false;
|
|
|
|
// Only valid team members can spawn
|
|
if ( pPlayer->GetTeamNumber() != TEAM_CT && pPlayer->GetTeamNumber() != TEAM_TERRORIST )
|
|
return false;
|
|
|
|
// Only players with a valid class can spawn
|
|
if ( pPlayer->GetClass() == CS_CLASS_NONE )
|
|
return false;
|
|
|
|
// Player cannot respawn until next round if more than 20 seconds in
|
|
|
|
// Tabulate the number of players on each team.
|
|
m_iNumCT = GetGlobalTeam( TEAM_CT )->GetNumPlayers();
|
|
m_iNumTerrorist = GetGlobalTeam( TEAM_TERRORIST )->GetNumPlayers();
|
|
|
|
if ( m_iNumTerrorist > 0 && m_iNumCT > 0 )
|
|
{
|
|
if ( gpGlobals->curtime > (m_fRoundStartTime + 20) )
|
|
{
|
|
//If this player just connected and fadetoblack is on, then maybe
|
|
//the server admin doesn't want him peeking around.
|
|
color32_s clr = {0,0,0,255};
|
|
if ( mp_fadetoblack.GetBool() )
|
|
{
|
|
UTIL_ScreenFade( pPlayer, clr, 3, 3, FFADE_OUT | FFADE_STAYOUT );
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Player cannot respawn while in the Choose Appearance menu
|
|
//if ( pPlayer->m_iMenu == Menu_ChooseAppearance )
|
|
// return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void CCSGameRules::TerminateRound(float tmDelay, int iReason )
|
|
{
|
|
variant_t emptyVariant;
|
|
int iWinnerTeam = WINNER_NONE;
|
|
const char *text = "UNKNOWN";
|
|
|
|
// UTIL_ClientPrintAll( HUD_PRINTCENTER, sentence );
|
|
|
|
switch ( iReason )
|
|
{
|
|
// Terror wins:
|
|
case Target_Bombed:
|
|
text = "#Target_Bombed";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
|
|
case VIP_Assassinated:
|
|
text = "#VIP_Assassinated";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
|
|
case Terrorists_Escaped:
|
|
text = "#Terrorists_Escaped";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
|
|
case Terrorists_Win:
|
|
text = "#Terrorists_Win";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
|
|
case Hostages_Not_Rescued:
|
|
text = "#Hostages_Not_Rescued";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
|
|
case VIP_Not_Escaped:
|
|
text = "#VIP_Not_Escaped";
|
|
iWinnerTeam = WINNER_TER;
|
|
break;
|
|
// CT wins:
|
|
case VIP_Escaped:
|
|
text = "#VIP_Escaped";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case CTs_PreventEscape:
|
|
text = "#CTs_PreventEscape";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case Escaping_Terrorists_Neutralized:
|
|
text = "#Escaping_Terrorists_Neutralized";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case Bomb_Defused:
|
|
text = "#Bomb_Defused";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case CTs_Win:
|
|
text = "#CTs_Win";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case All_Hostages_Rescued:
|
|
text = "#All_Hostages_Rescued";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case Target_Saved:
|
|
text = "#Target_Saved";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
|
|
case Terrorists_Not_Escaped:
|
|
text = "#Terrorists_Not_Escaped";
|
|
iWinnerTeam = WINNER_CT;
|
|
break;
|
|
// no winners:
|
|
case Game_Commencing:
|
|
text = "#Game_Commencing";
|
|
iWinnerTeam = WINNER_DRAW;
|
|
break;
|
|
|
|
case Round_Draw:
|
|
text = "#Round_Draw";
|
|
iWinnerTeam = WINNER_DRAW;
|
|
break;
|
|
|
|
default:
|
|
DevMsg("TerminateRound: unknown round end ID %i\n", iReason );
|
|
break;
|
|
}
|
|
|
|
m_iRoundWinStatus = iWinnerTeam;
|
|
m_flRestartRoundTime = gpGlobals->curtime + tmDelay;
|
|
|
|
if ( iWinnerTeam == WINNER_CT )
|
|
{
|
|
for( int i=0;i<g_Hostages.Count();i++ )
|
|
g_Hostages[i]->AcceptInput( "CTsWin", NULL, NULL, emptyVariant, 0 );
|
|
}
|
|
|
|
else if ( iWinnerTeam == WINNER_TER )
|
|
{
|
|
for( int i=0;i<g_Hostages.Count();i++ )
|
|
g_Hostages[i]->AcceptInput( "TerroristsWin", NULL, NULL, emptyVariant, 0 );
|
|
}
|
|
else
|
|
{
|
|
Assert( iWinnerTeam == WINNER_NONE || iWinnerTeam == WINNER_DRAW );
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
//=============================================================================
|
|
|
|
// [tj] Check for any non-player-specific achievements.
|
|
ProcessEndOfRoundAchievements(iWinnerTeam, iReason);
|
|
|
|
if( iReason != Game_Commencing )
|
|
{
|
|
// [pfreese] Setup and send win panel event (primarily funfact data)
|
|
|
|
FunFact funfact;
|
|
funfact.szLocalizationToken = "";
|
|
funfact.iPlayer = 0;
|
|
funfact.iData1 = 0;
|
|
funfact.iData2 = 0;
|
|
funfact.iData3 = 0;
|
|
|
|
m_pFunFactManager->GetRoundEndFunFact( iWinnerTeam, iReason, funfact);
|
|
|
|
//Send all the info needed for the win panel
|
|
IGameEvent *winEvent = gameeventmanager->CreateEvent( "cs_win_panel_round" );
|
|
|
|
if ( winEvent )
|
|
{
|
|
// determine what categories to send
|
|
if ( GetRoundRemainingTime() <= 0 )
|
|
{
|
|
// timer expired, defenders win
|
|
// show total time that was defended
|
|
winEvent->SetBool( "show_timer_defend", true );
|
|
winEvent->SetInt( "timer_time", m_iRoundTime );
|
|
}
|
|
else
|
|
{
|
|
// attackers win
|
|
// show time it took for them to win
|
|
winEvent->SetBool( "show_timer_attack", true );
|
|
|
|
int iTimeElapsed = m_iRoundTime - GetRoundRemainingTime();
|
|
winEvent->SetInt( "timer_time", iTimeElapsed );
|
|
}
|
|
|
|
winEvent->SetInt( "final_event", iReason );
|
|
|
|
// Set the fun fact data in the event
|
|
winEvent->SetString( "funfact_token", funfact.szLocalizationToken);
|
|
winEvent->SetInt( "funfact_player", funfact.iPlayer );
|
|
winEvent->SetInt( "funfact_data1", funfact.iData1 );
|
|
winEvent->SetInt( "funfact_data2", funfact.iData2 );
|
|
winEvent->SetInt( "funfact_data3", funfact.iData3 );
|
|
gameeventmanager->FireEvent( winEvent );
|
|
}
|
|
}
|
|
|
|
// [tj] Inform players that the round is over
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
if(pPlayer)
|
|
{
|
|
pPlayer->OnRoundEnd(iWinnerTeam, iReason);
|
|
}
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
IGameEvent * event = gameeventmanager->CreateEvent( "round_end" );
|
|
if ( event )
|
|
{
|
|
event->SetInt( "winner", iWinnerTeam );
|
|
event->SetInt( "reason", iReason );
|
|
event->SetString( "message", text );
|
|
event->SetInt( "priority", 6 ); // HLTV event priority, not transmitted
|
|
gameeventmanager->FireEvent( event );
|
|
}
|
|
|
|
if ( GetMapRemainingTime() == 0.0f )
|
|
{
|
|
UTIL_LogPrintf("World triggered \"Intermission_Time_Limit\"\n");
|
|
GoToIntermission();
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
//=============================================================================
|
|
|
|
// Helper to determine if all players on a team are playing for the same clan
|
|
static bool IsClanTeam( CTeam *pTeam )
|
|
{
|
|
uint32 iTeamClan = 0;
|
|
for ( int iPlayer = 0; iPlayer < pTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CBasePlayer *pPlayer = pTeam->GetPlayer( iPlayer );
|
|
if ( !pPlayer )
|
|
return false;
|
|
|
|
const char *pClanID = engine->GetClientConVarValue( pPlayer->entindex(), "cl_clanid" );
|
|
uint32 iPlayerClan = atoi( pClanID );
|
|
if ( iPlayer == 0 )
|
|
{
|
|
// Initialize the team clan
|
|
iTeamClan = iPlayerClan;
|
|
}
|
|
else
|
|
{
|
|
if ( iPlayerClan != iTeamClan || iPlayerClan == 0 )
|
|
return false;
|
|
}
|
|
}
|
|
return iTeamClan != 0;
|
|
}
|
|
|
|
// [tj] This is where we check non-player-specific that occur at the end of the round
|
|
void CCSGameRules::ProcessEndOfRoundAchievements(int iWinnerTeam, int iReason)
|
|
{
|
|
if (iWinnerTeam == WINNER_CT || iWinnerTeam == WINNER_TER)
|
|
{
|
|
int losingTeamId = (iWinnerTeam == TEAM_CT) ? TEAM_TERRORIST : TEAM_CT;
|
|
CTeam* losingTeam = GetGlobalTeam(losingTeamId);
|
|
|
|
|
|
//Check for players we should ignore when checking team size.
|
|
int ignoreCount = 0;
|
|
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer* pPlayer = (CCSPlayer*)UTIL_PlayerByIndex( i );
|
|
if (pPlayer)
|
|
{
|
|
int teamNum = pPlayer->GetTeamNumber();
|
|
if ( teamNum == losingTeamId )
|
|
{
|
|
if (pPlayer->WasNotKilledNaturally())
|
|
{
|
|
ignoreCount++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// [tj] Check extermination with no losses achievement
|
|
if ( ( ( iReason == CTs_Win && m_bNoCTsKilled ) || ( iReason == Terrorists_Win && m_bNoTerroristsKilled ) )
|
|
&& losingTeam && losingTeam->GetNumPlayers() - ignoreCount >= AchievementConsts::DefaultMinOpponentsForAchievement)
|
|
{
|
|
CTeam *pTeam = GetGlobalTeam( iWinnerTeam );
|
|
|
|
for ( int iPlayer=0; iPlayer < pTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pTeam->GetPlayer( iPlayer ) );
|
|
Assert( pPlayer );
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
pPlayer->AwardAchievement(CSLosslessExtermination);
|
|
}
|
|
}
|
|
|
|
// [tj] Check flawless victory achievement - currently requiring extermination
|
|
if (((iReason == CTs_Win && m_bNoCTsDamaged) || (iReason == Terrorists_Win && m_bNoTerroristsDamaged))
|
|
&& losingTeam && losingTeam->GetNumPlayers() - ignoreCount >= AchievementConsts::DefaultMinOpponentsForAchievement)
|
|
{
|
|
CTeam *pTeam = GetGlobalTeam( iWinnerTeam );
|
|
|
|
for ( int iPlayer=0; iPlayer < pTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pTeam->GetPlayer( iPlayer ) );
|
|
Assert( pPlayer );
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
pPlayer->AwardAchievement(CSFlawlessVictory);
|
|
}
|
|
}
|
|
|
|
// [tj] Check bloodless victory achievement
|
|
if (((iWinnerTeam == TEAM_TERRORIST && m_bNoCTsKilled) || (iWinnerTeam == Terrorists_Win && m_bNoTerroristsKilled))
|
|
&& losingTeam && losingTeam->GetNumPlayers() >= AchievementConsts::DefaultMinOpponentsForAchievement)
|
|
{
|
|
CTeam *pTeam = GetGlobalTeam( iWinnerTeam );
|
|
|
|
for ( int iPlayer=0; iPlayer < pTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pTeam->GetPlayer( iPlayer ) );
|
|
Assert( pPlayer );
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
pPlayer->AwardAchievement(CSBloodlessVictory);
|
|
}
|
|
}
|
|
|
|
// Check the clan match achievement
|
|
CTeam *pWinningTeam = GetGlobalTeam( iWinnerTeam );
|
|
if ( pWinningTeam && pWinningTeam->GetNumPlayers() >= AchievementConsts::DefaultMinOpponentsForAchievement &&
|
|
losingTeam && losingTeam->GetNumPlayers() - ignoreCount >= AchievementConsts::DefaultMinOpponentsForAchievement &&
|
|
IsClanTeam( pWinningTeam ) && IsClanTeam( losingTeam ) )
|
|
{
|
|
for ( int iPlayer=0; iPlayer < pWinningTeam->GetNumPlayers(); iPlayer++ )
|
|
{
|
|
CCSPlayer *pPlayer = ToCSPlayer( pWinningTeam->GetPlayer( iPlayer ) );
|
|
if ( !pPlayer )
|
|
continue;
|
|
|
|
pPlayer->AwardAchievement( CSWinClanMatch );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//[tj] Counts the number of players in each category in the struct (dead, alive, etc...)
|
|
void CCSGameRules::GetPlayerCounts(TeamPlayerCounts teamCounts[TEAM_MAXCOUNT])
|
|
{
|
|
memset(teamCounts, 0, sizeof(TeamPlayerCounts) * TEAM_MAXCOUNT);
|
|
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer* pPlayer = (CCSPlayer*)UTIL_PlayerByIndex( i );
|
|
if (pPlayer)
|
|
{
|
|
int iTeam = pPlayer->GetTeamNumber();
|
|
|
|
if (iTeam >= 0 && iTeam < TEAM_MAXCOUNT)
|
|
{
|
|
++teamCounts[iTeam].totalPlayers;
|
|
if (pPlayer->IsAlive())
|
|
{
|
|
++teamCounts[iTeam].totalAlivePlayers;
|
|
}
|
|
else
|
|
{
|
|
++teamCounts[iTeam].totalDeadPlayers;
|
|
|
|
//If the player has joined a team bit isn't in the game yet
|
|
if (pPlayer->State_Get() == STATE_PICKINGCLASS)
|
|
{
|
|
++teamCounts[iTeam].unenteredPlayers;
|
|
}
|
|
else if (pPlayer->WasNotKilledNaturally())
|
|
{
|
|
++teamCounts[iTeam].suicidedPlayers;
|
|
}
|
|
else
|
|
{
|
|
++teamCounts[iTeam].killedPlayers;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
void CCSGameRules::UpdateTeamScores()
|
|
{
|
|
CTeam *pTerrorists = GetGlobalTeam( TEAM_TERRORIST );
|
|
CTeam *pCTs = GetGlobalTeam( TEAM_CT );
|
|
|
|
Assert( pTerrorists && pCTs );
|
|
|
|
if( pTerrorists )
|
|
pTerrorists->SetScore( m_iNumTerroristWins );
|
|
|
|
if( pCTs )
|
|
pCTs->SetScore( m_iNumCTWins );
|
|
}
|
|
|
|
|
|
void CCSGameRules::CheckMapConditions()
|
|
{
|
|
// Check to see if this map has a bomb target in it
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_bomb_target" ) )
|
|
{
|
|
m_bMapHasBombTarget = true;
|
|
m_bMapHasBombZone = true;
|
|
}
|
|
else if ( gEntList.FindEntityByClassname( NULL, "info_bomb_target" ) )
|
|
{
|
|
m_bMapHasBombTarget = true;
|
|
m_bMapHasBombZone = false;
|
|
}
|
|
else
|
|
{
|
|
m_bMapHasBombTarget = false;
|
|
m_bMapHasBombZone = false;
|
|
}
|
|
|
|
// See if the map has func_buyzone entities
|
|
// Used by CBasePlayer::HandleSignals() to support maps without these entities
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_buyzone" ) )
|
|
{
|
|
m_bMapHasBuyZone = true;
|
|
}
|
|
else
|
|
{
|
|
m_bMapHasBuyZone = false;
|
|
}
|
|
|
|
// Check to see if this map has hostage rescue zones
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_hostage_rescue" ) )
|
|
{
|
|
m_bMapHasRescueZone = true;
|
|
}
|
|
else
|
|
{
|
|
m_bMapHasRescueZone = false;
|
|
}
|
|
|
|
// GOOSEMAN : See if this map has func_escapezone entities
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_escapezone" ) )
|
|
{
|
|
m_bMapHasEscapeZone = true;
|
|
}
|
|
else
|
|
{
|
|
m_bMapHasEscapeZone = false;
|
|
}
|
|
|
|
// Check to see if this map has VIP safety zones
|
|
if ( gEntList.FindEntityByClassname( NULL, "func_vip_safetyzone" ) )
|
|
{
|
|
m_iMapHasVIPSafetyZone = 1;
|
|
}
|
|
else
|
|
{
|
|
m_iMapHasVIPSafetyZone = 2;
|
|
}
|
|
}
|
|
|
|
|
|
void CCSGameRules::SwapAllPlayers()
|
|
{
|
|
// MOTODO we have to make sure that enought spaning points exits
|
|
Assert ( 0 );
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
/* CCSPlayer *pPlayer = CCSPlayer::Instance( i );
|
|
if ( pPlayer && !FNullEnt( pPlayer->edict() ) )
|
|
pPlayer->SwitchTeam(); */
|
|
}
|
|
|
|
// Swap Team victories
|
|
int iTemp;
|
|
|
|
iTemp = m_iNumCTWins;
|
|
m_iNumCTWins = m_iNumTerroristWins;
|
|
m_iNumTerroristWins = iTemp;
|
|
|
|
// Update the clients team score
|
|
UpdateTeamScores();
|
|
}
|
|
|
|
|
|
bool CS_FindInList( const char **pStrings, const char *pToFind )
|
|
{
|
|
return FindInList( pStrings, pToFind );
|
|
}
|
|
|
|
void CCSGameRules::CleanUpMap()
|
|
{
|
|
if (IsLogoMap())
|
|
return;
|
|
|
|
// Recreate all the map entities from the map data (preserving their indices),
|
|
// then remove everything else except the players.
|
|
|
|
// Get rid of all entities except players.
|
|
CBaseEntity *pCur = gEntList.FirstEnt();
|
|
while ( pCur )
|
|
{
|
|
CWeaponCSBase *pWeapon = dynamic_cast< CWeaponCSBase* >( pCur );
|
|
// Weapons with owners don't want to be removed..
|
|
if ( pWeapon )
|
|
{
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [dwenger] Handle round restart processing for the weapon.
|
|
//=============================================================================
|
|
|
|
pWeapon->OnRoundRestart();
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
if ( pWeapon->ShouldRemoveOnRoundRestart() )
|
|
{
|
|
UTIL_Remove( pCur );
|
|
}
|
|
}
|
|
// remove entities that has to be restored on roundrestart (breakables etc)
|
|
else if ( !CS_FindInList( s_PreserveEnts, pCur->GetClassname() ) )
|
|
{
|
|
UTIL_Remove( pCur );
|
|
}
|
|
|
|
pCur = gEntList.NextEnt( pCur );
|
|
}
|
|
|
|
// Really remove the entities so we can have access to their slots below.
|
|
gEntList.CleanupDeleteList();
|
|
|
|
// Cancel all queued events, in case a func_bomb_target fired some delayed outputs that
|
|
// could kill respawning CTs
|
|
g_EventQueue.Clear();
|
|
|
|
// Now reload the map entities.
|
|
class CCSMapEntityFilter : public IMapEntityFilter
|
|
{
|
|
public:
|
|
virtual bool ShouldCreateEntity( const char *pClassname )
|
|
{
|
|
// Don't recreate the preserved entities.
|
|
if ( !CS_FindInList( s_PreserveEnts, pClassname ) )
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
// Increment our iterator since it's not going to call CreateNextEntity for this ent.
|
|
if ( m_iIterator != g_MapEntityRefs.InvalidIndex() )
|
|
m_iIterator = g_MapEntityRefs.Next( m_iIterator );
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
virtual CBaseEntity* CreateNextEntity( const char *pClassname )
|
|
{
|
|
if ( m_iIterator == g_MapEntityRefs.InvalidIndex() )
|
|
{
|
|
// This shouldn't be possible. When we loaded the map, it should have used
|
|
// CCSMapLoadEntityFilter, which should have built the g_MapEntityRefs list
|
|
// with the same list of entities we're referring to here.
|
|
Assert( false );
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
CMapEntityRef &ref = g_MapEntityRefs[m_iIterator];
|
|
m_iIterator = g_MapEntityRefs.Next( m_iIterator ); // Seek to the next entity.
|
|
|
|
if ( ref.m_iEdict == -1 || engine->PEntityOfEntIndex( ref.m_iEdict ) )
|
|
{
|
|
// Doh! The entity was delete and its slot was reused.
|
|
// Just use any old edict slot. This case sucks because we lose the baseline.
|
|
return CreateEntityByName( pClassname );
|
|
}
|
|
else
|
|
{
|
|
// Cool, the slot where this entity was is free again (most likely, the entity was
|
|
// freed above). Now create an entity with this specific index.
|
|
return CreateEntityByName( pClassname, ref.m_iEdict );
|
|
}
|
|
}
|
|
}
|
|
|
|
public:
|
|
int m_iIterator; // Iterator into g_MapEntityRefs.
|
|
};
|
|
CCSMapEntityFilter filter;
|
|
filter.m_iIterator = g_MapEntityRefs.Head();
|
|
|
|
// DO NOT CALL SPAWN ON info_node ENTITIES!
|
|
|
|
MapEntity_ParseAllEntities( engine->GetMapEntitiesString(), &filter, true );
|
|
}
|
|
|
|
|
|
bool CCSGameRules::IsThereABomber()
|
|
{
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = CCSPlayer::Instance( i );
|
|
|
|
if ( pPlayer && !FNullEnt( pPlayer->edict() ) )
|
|
{
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT )
|
|
continue;
|
|
|
|
if ( pPlayer->HasC4() )
|
|
return true; //There you are.
|
|
}
|
|
}
|
|
|
|
//Didn't find a bomber.
|
|
return false;
|
|
}
|
|
|
|
|
|
void CCSGameRules::EndRound()
|
|
{
|
|
// fake a round end
|
|
CSGameRules()->TerminateRound( 0.0f, Round_Draw );
|
|
}
|
|
|
|
CBaseEntity *CCSGameRules::GetPlayerSpawnSpot( CBasePlayer *pPlayer )
|
|
{
|
|
// gat valid spwan point
|
|
CBaseEntity *pSpawnSpot = pPlayer->EntSelectSpawnPoint();
|
|
|
|
// drop down to ground
|
|
Vector GroundPos = DropToGround( pPlayer, pSpawnSpot->GetAbsOrigin(), VEC_HULL_MIN, VEC_HULL_MAX );
|
|
|
|
// Move the player to the place it said.
|
|
pPlayer->Teleport( &pSpawnSpot->GetAbsOrigin(), &pSpawnSpot->GetLocalAngles(), &vec3_origin );
|
|
pPlayer->m_Local.m_vecPunchAngle = vec3_angle;
|
|
|
|
return pSpawnSpot;
|
|
}
|
|
|
|
// checks if the spot is clear of players
|
|
bool CCSGameRules::IsSpawnPointValid( CBaseEntity *pSpot, CBasePlayer *pPlayer )
|
|
{
|
|
if ( !pSpot->IsTriggered( pPlayer ) )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Vector mins = GetViewVectors()->m_vHullMin;
|
|
Vector maxs = GetViewVectors()->m_vHullMax;
|
|
|
|
Vector vTestMins = pSpot->GetAbsOrigin() + mins;
|
|
Vector vTestMaxs = pSpot->GetAbsOrigin() + maxs;
|
|
|
|
// First test the starting origin.
|
|
return UTIL_IsSpaceEmpty( pPlayer, vTestMins, vTestMaxs );
|
|
}
|
|
|
|
|
|
bool CCSGameRules::IsThereABomb()
|
|
{
|
|
bool bBombFound = false;
|
|
|
|
/* are there any bombs, either laying around, or in someone's inventory? */
|
|
if( gEntList.FindEntityByClassname( NULL, WEAPON_C4_CLASSNAME ) != 0 )
|
|
{
|
|
bBombFound = true;
|
|
}
|
|
/* what about planted bombs!? */
|
|
else if( gEntList.FindEntityByClassname( NULL, PLANTED_C4_CLASSNAME ) != 0 )
|
|
{
|
|
bBombFound = true;
|
|
}
|
|
|
|
return bBombFound;
|
|
}
|
|
|
|
void CCSGameRules::HostageTouched()
|
|
{
|
|
if( gpGlobals->curtime > m_flNextHostageAnnouncement && m_iRoundWinStatus == WINNER_NONE )
|
|
{
|
|
//BroadcastSound( "Event.HostageTouched" );
|
|
m_flNextHostageAnnouncement = gpGlobals->curtime + 60.0;
|
|
}
|
|
}
|
|
|
|
void CCSGameRules::CreateStandardEntities()
|
|
{
|
|
// Create the player resource
|
|
g_pPlayerResource = (CPlayerResource*)CBaseEntity::Create( "cs_player_manager", vec3_origin, vec3_angle );
|
|
|
|
// Create the entity that will send our data to the client.
|
|
#ifdef DBGFLAG_ASSERT
|
|
CBaseEntity *pEnt =
|
|
#endif
|
|
CBaseEntity::Create( "cs_gamerules", vec3_origin, vec3_angle );
|
|
Assert( pEnt );
|
|
}
|
|
|
|
#define MY_USHRT_MAX 0xffff
|
|
#define MY_UCHAR_MAX 0xff
|
|
|
|
bool DataHasChanged( void )
|
|
{
|
|
for ( int i = 0; i < CS_NUM_LEVELS; i++ )
|
|
{
|
|
if ( g_iTerroristVictories[i] != 0 || g_iCounterTVictories[i] != 0 )
|
|
return true;
|
|
}
|
|
|
|
for ( int i = 0; i < WEAPON_MAX; i++ )
|
|
{
|
|
if ( g_iWeaponPurchases[i] != 0 )
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void CCSGameRules::UploadGameStats( void )
|
|
{
|
|
g_flGameStatsUpdateTime -= gpGlobals->curtime;
|
|
|
|
if ( g_flGameStatsUpdateTime > 0 )
|
|
return;
|
|
|
|
if ( IsBlackMarket() == false )
|
|
return;
|
|
|
|
if ( m_bDontUploadStats == true )
|
|
return;
|
|
|
|
if ( DataHasChanged() == true )
|
|
{
|
|
cs_gamestats_t stats;
|
|
memset( &stats, 0, sizeof(stats) );
|
|
|
|
// Header
|
|
stats.header.iVersion = CS_STATS_BLOB_VERSION;
|
|
Q_strncpy( stats.header.szGameName, "cstrike", sizeof(stats.header.szGameName) );
|
|
Q_strncpy( stats.header.szMapName, STRING( gpGlobals->mapname ), sizeof( stats.header.szMapName ) );
|
|
|
|
ConVar *hostip = cvar->FindVar( "hostip" );
|
|
if ( hostip )
|
|
{
|
|
int ip = hostip->GetInt();
|
|
stats.header.ipAddr[0] = ip >> 24;
|
|
stats.header.ipAddr[1] = ( ip >> 16 ) & MY_UCHAR_MAX;
|
|
stats.header.ipAddr[2] = ( ip >> 8 ) & MY_UCHAR_MAX;
|
|
stats.header.ipAddr[3] = ( ip ) & MY_UCHAR_MAX;
|
|
}
|
|
|
|
ConVar *hostport = cvar->FindVar( "hostip" );
|
|
if ( hostport )
|
|
{
|
|
stats.header.port = hostport->GetInt();
|
|
}
|
|
|
|
stats.header.serverid = 0;
|
|
|
|
stats.iMinutesPlayed = clamp( (short)( gpGlobals->curtime / 60 ), 0, MY_USHRT_MAX );
|
|
|
|
memcpy( stats.iTerroristVictories, g_iTerroristVictories, sizeof( g_iTerroristVictories) );
|
|
memcpy( stats.iCounterTVictories, g_iCounterTVictories, sizeof( g_iCounterTVictories) );
|
|
memcpy( stats.iBlackMarketPurchases, g_iWeaponPurchases, sizeof( g_iWeaponPurchases) );
|
|
|
|
stats.iAutoBuyPurchases = g_iAutoBuyPurchases;
|
|
stats.iReBuyPurchases = g_iReBuyPurchases;
|
|
|
|
stats.iAutoBuyM4A1Purchases = g_iAutoBuyM4A1Purchases;
|
|
stats.iAutoBuyAK47Purchases = g_iAutoBuyAK47Purchases;
|
|
stats.iAutoBuyFamasPurchases = g_iAutoBuyFamasPurchases;
|
|
stats.iAutoBuyGalilPurchases = g_iAutoBuyGalilPurchases;
|
|
stats.iAutoBuyVestHelmPurchases = g_iAutoBuyVestHelmPurchases;
|
|
stats.iAutoBuyVestPurchases = g_iAutoBuyVestPurchases;
|
|
|
|
const void *pvBlobData = ( const void * )( &stats );
|
|
unsigned int uBlobSize = sizeof( stats );
|
|
|
|
if ( gamestatsuploader )
|
|
{
|
|
gamestatsuploader->UploadGameStats(
|
|
STRING( gpGlobals->mapname ),
|
|
CS_STATS_BLOB_VERSION,
|
|
uBlobSize,
|
|
pvBlobData );
|
|
}
|
|
|
|
|
|
memset( g_iWeaponPurchases, 0, sizeof( g_iWeaponPurchases) );
|
|
memset( g_iTerroristVictories, 0, sizeof( g_iTerroristVictories) );
|
|
memset( g_iCounterTVictories, 0, sizeof( g_iTerroristVictories) );
|
|
|
|
g_iAutoBuyPurchases = 0;
|
|
g_iReBuyPurchases = 0;
|
|
|
|
g_iAutoBuyM4A1Purchases = 0;
|
|
g_iAutoBuyAK47Purchases = 0;
|
|
g_iAutoBuyFamasPurchases = 0;
|
|
g_iAutoBuyGalilPurchases = 0;
|
|
g_iAutoBuyVestHelmPurchases = 0;
|
|
g_iAutoBuyVestPurchases = 0;
|
|
}
|
|
|
|
g_flGameStatsUpdateTime = CS_GAME_STATS_UPDATE; //Next update is between 22 and 24 hours.
|
|
}
|
|
#endif // CLIENT_DLL
|
|
|
|
CBaseCombatWeapon *CCSGameRules::GetNextBestWeapon( CBaseCombatCharacter *pPlayer, CBaseCombatWeapon *pCurrentWeapon )
|
|
{
|
|
CBaseCombatWeapon *bestWeapon = NULL;
|
|
|
|
// search all the weapons looking for the closest next
|
|
for ( int i = 0; i < MAX_WEAPONS; i++ )
|
|
{
|
|
CBaseCombatWeapon *weapon = pPlayer->GetWeapon(i);
|
|
if ( !weapon )
|
|
continue;
|
|
|
|
if ( !weapon->CanBeSelected() || weapon == pCurrentWeapon )
|
|
continue;
|
|
|
|
#ifndef CLIENT_DLL
|
|
CCSPlayer *csPlayer = ToCSPlayer(pPlayer);
|
|
CWeaponCSBase *csWeapon = static_cast< CWeaponCSBase * >(weapon);
|
|
if ( csPlayer && csPlayer->IsBot() && !TheCSBots()->IsWeaponUseable( csWeapon ) )
|
|
continue;
|
|
#endif // CLIENT_DLL
|
|
|
|
if ( bestWeapon )
|
|
{
|
|
if ( weapon->GetSlot() < bestWeapon->GetSlot() )
|
|
{
|
|
bestWeapon = weapon;
|
|
}
|
|
else if ( weapon->GetSlot() == bestWeapon->GetSlot() && weapon->GetPosition() < bestWeapon->GetPosition() )
|
|
{
|
|
bestWeapon = weapon;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bestWeapon = weapon;
|
|
}
|
|
}
|
|
|
|
return bestWeapon;
|
|
}
|
|
|
|
float CCSGameRules::GetMapRemainingTime()
|
|
{
|
|
#ifdef GAME_DLL
|
|
if ( nextlevel.GetString() && *nextlevel.GetString() )
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
// if timelimit is disabled, return -1
|
|
if ( mp_timelimit.GetInt() <= 0 )
|
|
return -1;
|
|
|
|
// timelimit is in minutes
|
|
float flTimeLeft = ( m_flGameStartTime + mp_timelimit.GetInt() * 60 ) - gpGlobals->curtime;
|
|
|
|
// never return a negative value
|
|
if ( flTimeLeft < 0 )
|
|
flTimeLeft = 0;
|
|
|
|
return flTimeLeft;
|
|
}
|
|
|
|
float CCSGameRules::GetMapElapsedTime( void )
|
|
{
|
|
return gpGlobals->curtime;
|
|
}
|
|
|
|
float CCSGameRules::GetRoundRemainingTime()
|
|
{
|
|
return (float) (m_fRoundStartTime + m_iRoundTime) - gpGlobals->curtime;
|
|
}
|
|
|
|
float CCSGameRules::GetRoundStartTime()
|
|
{
|
|
return m_fRoundStartTime;
|
|
}
|
|
|
|
|
|
float CCSGameRules::GetRoundElapsedTime()
|
|
{
|
|
return gpGlobals->curtime - m_fRoundStartTime;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::ShouldCollide( int collisionGroup0, int collisionGroup1 )
|
|
{
|
|
if ( collisionGroup0 > collisionGroup1 )
|
|
{
|
|
// swap so that lowest is always first
|
|
::V_swap(collisionGroup0,collisionGroup1);
|
|
}
|
|
|
|
//Don't stand on COLLISION_GROUP_WEAPONs
|
|
if( collisionGroup0 == COLLISION_GROUP_PLAYER_MOVEMENT &&
|
|
collisionGroup1 == COLLISION_GROUP_WEAPON )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// TODO: make a CS-SPECIFIC COLLISION GROUP FOR PHYSICS PROPS THAT USE THIS COLLISION BEHAVIOR.
|
|
|
|
|
|
if ( (collisionGroup0 == COLLISION_GROUP_PLAYER || collisionGroup0 == COLLISION_GROUP_PLAYER_MOVEMENT) &&
|
|
collisionGroup1 == COLLISION_GROUP_PUSHAWAY )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if ( collisionGroup0 == COLLISION_GROUP_DEBRIS && collisionGroup1 == COLLISION_GROUP_PUSHAWAY )
|
|
{
|
|
// let debris and multiplayer objects collide
|
|
return true;
|
|
}
|
|
|
|
return BaseClass::ShouldCollide( collisionGroup0, collisionGroup1 );
|
|
}
|
|
|
|
|
|
bool CCSGameRules::IsFreezePeriod()
|
|
{
|
|
return m_bFreezePeriod;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::IsVIPMap() const
|
|
{
|
|
//MIKETODO: VIP mode
|
|
return false;
|
|
}
|
|
|
|
|
|
bool CCSGameRules::IsBombDefuseMap() const
|
|
{
|
|
return m_bMapHasBombTarget;
|
|
}
|
|
|
|
bool CCSGameRules::IsHostageRescueMap() const
|
|
{
|
|
return m_bMapHasRescueZone;
|
|
}
|
|
|
|
bool CCSGameRules::IsLogoMap() const
|
|
{
|
|
return m_bLogoMap;
|
|
}
|
|
|
|
float CCSGameRules::GetBuyTimeLength() const
|
|
{
|
|
return ( mp_buytime.GetFloat() * 60 );
|
|
}
|
|
|
|
bool CCSGameRules::IsBuyTimeElapsed()
|
|
{
|
|
return ( GetRoundElapsedTime() > GetBuyTimeLength() );
|
|
}
|
|
|
|
int CCSGameRules::DefaultFOV()
|
|
{
|
|
return 90;
|
|
}
|
|
|
|
const CViewVectors* CCSGameRules::GetViewVectors() const
|
|
{
|
|
return &g_CSViewVectors;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Init CS ammo definitions
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// shared ammo definition
|
|
// JAY: Trying to make a more physical bullet response
|
|
#define BULLET_MASS_GRAINS_TO_LB(grains) (0.002285*(grains)/16.0f)
|
|
#define BULLET_MASS_GRAINS_TO_KG(grains) lbs2kg(BULLET_MASS_GRAINS_TO_LB(grains))
|
|
|
|
// exaggerate all of the forces, but use real numbers to keep them consistent
|
|
#define BULLET_IMPULSE_EXAGGERATION 1
|
|
|
|
// convert a velocity in ft/sec and a mass in grains to an impulse in kg in/s
|
|
#define BULLET_IMPULSE(grains, ftpersec) ((ftpersec)*12*BULLET_MASS_GRAINS_TO_KG(grains)*BULLET_IMPULSE_EXAGGERATION)
|
|
|
|
|
|
static CCSAmmoDef ammoDef;
|
|
CCSAmmoDef* GetCSAmmoDef()
|
|
{
|
|
GetAmmoDef(); // to initialize the ammo info
|
|
return &ammoDef;
|
|
}
|
|
|
|
CAmmoDef* GetAmmoDef()
|
|
{
|
|
static bool bInitted = false;
|
|
|
|
if ( !bInitted )
|
|
{
|
|
bInitted = true;
|
|
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_50AE, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_50AE_max", 2400 * BULLET_IMPULSE_EXAGGERATION, 0, 10, 14 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_762MM, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_762mm_max", 2400 * BULLET_IMPULSE_EXAGGERATION, 0, 10, 14 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_556MM, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_556mm_max", 2400 * BULLET_IMPULSE_EXAGGERATION, 0, 10, 14 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_556MM_BOX, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_556mm_box_max",2400 * BULLET_IMPULSE_EXAGGERATION, 0, 10, 14 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_338MAG, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_338mag_max", 2800 * BULLET_IMPULSE_EXAGGERATION, 0, 12, 16 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_9MM, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_9mm_max", 2000 * BULLET_IMPULSE_EXAGGERATION, 0, 5, 10 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_BUCKSHOT, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_buckshot_max", 600 * BULLET_IMPULSE_EXAGGERATION, 0, 3, 6 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_45ACP, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_45acp_max", 2100 * BULLET_IMPULSE_EXAGGERATION, 0, 6, 10 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_357SIG, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_357sig_max", 2000 * BULLET_IMPULSE_EXAGGERATION, 0, 4, 8 );
|
|
ammoDef.AddAmmoType( BULLET_PLAYER_57MM, DMG_BULLET, TRACER_LINE, 0, 0, "ammo_57mm_max", 2000 * BULLET_IMPULSE_EXAGGERATION, 0, 4, 8 );
|
|
ammoDef.AddAmmoType( AMMO_TYPE_HEGRENADE, DMG_BLAST, TRACER_LINE, 0, 0, "ammo_hegrenade_max", 1, 0 );
|
|
ammoDef.AddAmmoType( AMMO_TYPE_FLASHBANG, 0, TRACER_LINE, 0, 0, "ammo_flashbang_max", 1, 0 );
|
|
ammoDef.AddAmmoType( AMMO_TYPE_SMOKEGRENADE, 0, TRACER_LINE, 0, 0, "ammo_smokegrenade_max", 1, 0 );
|
|
|
|
//Adrian: I set all the prices to 0 just so the rest of the buy code works
|
|
//This should be revisited.
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_50AE, 0, 7 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_762MM, 0, 30 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_556MM, 0, 30 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_556MM_BOX, 0, 30 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_338MAG, 0, 10 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_9MM, 0, 30 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_BUCKSHOT, 0, 8 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_45ACP, 0, 25 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_357SIG, 0, 13 );
|
|
ammoDef.AddAmmoCost( BULLET_PLAYER_57MM, 0, 50 );
|
|
}
|
|
|
|
return &ammoDef;
|
|
}
|
|
|
|
#ifndef CLIENT_DLL
|
|
const char *CCSGameRules::GetChatPrefix( bool bTeamOnly, CBasePlayer *pPlayer )
|
|
{
|
|
const char *pszPrefix = NULL;
|
|
|
|
if ( !pPlayer ) // dedicated server output
|
|
{
|
|
pszPrefix = "";
|
|
}
|
|
else
|
|
{
|
|
// team only
|
|
if ( bTeamOnly == TRUE )
|
|
{
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT )
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
pszPrefix = "(Counter-Terrorist)";
|
|
}
|
|
else
|
|
{
|
|
pszPrefix = "*DEAD*(Counter-Terrorist)";
|
|
}
|
|
}
|
|
else if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST )
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
pszPrefix = "(Terrorist)";
|
|
}
|
|
else
|
|
{
|
|
pszPrefix = "*DEAD*(Terrorist)";
|
|
}
|
|
}
|
|
else if ( pPlayer->GetTeamNumber() == TEAM_SPECTATOR )
|
|
{
|
|
pszPrefix = "(Spectator)";
|
|
}
|
|
}
|
|
// everyone
|
|
else
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
pszPrefix = "";
|
|
}
|
|
else
|
|
{
|
|
if ( pPlayer->GetTeamNumber() != TEAM_SPECTATOR )
|
|
{
|
|
pszPrefix = "*DEAD*";
|
|
}
|
|
else
|
|
{
|
|
pszPrefix = "*SPEC*";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return pszPrefix;
|
|
}
|
|
|
|
const char *CCSGameRules::GetChatLocation( bool bTeamOnly, CBasePlayer *pPlayer )
|
|
{
|
|
if ( !pPlayer ) // dedicated server output
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
// only teammates see locations
|
|
if ( !bTeamOnly )
|
|
return NULL;
|
|
|
|
// only living players have locations
|
|
if ( pPlayer->GetTeamNumber() != TEAM_CT && pPlayer->GetTeamNumber() != TEAM_TERRORIST )
|
|
return NULL;
|
|
|
|
if ( !pPlayer->IsAlive() )
|
|
return NULL;
|
|
|
|
return pPlayer->GetLastKnownPlaceName();
|
|
}
|
|
|
|
const char *CCSGameRules::GetChatFormat( bool bTeamOnly, CBasePlayer *pPlayer )
|
|
{
|
|
if ( !pPlayer ) // dedicated server output
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
const char *pszFormat = NULL;
|
|
|
|
// team only
|
|
if ( bTeamOnly == TRUE )
|
|
{
|
|
if ( pPlayer->GetTeamNumber() == TEAM_CT )
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
const char *chatLocation = GetChatLocation( bTeamOnly, pPlayer );
|
|
if ( chatLocation && *chatLocation )
|
|
{
|
|
pszFormat = "Cstrike_Chat_CT_Loc";
|
|
}
|
|
else
|
|
{
|
|
pszFormat = "Cstrike_Chat_CT";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pszFormat = "Cstrike_Chat_CT_Dead";
|
|
}
|
|
}
|
|
else if ( pPlayer->GetTeamNumber() == TEAM_TERRORIST )
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
const char *chatLocation = GetChatLocation( bTeamOnly, pPlayer );
|
|
if ( chatLocation && *chatLocation )
|
|
{
|
|
pszFormat = "Cstrike_Chat_T_Loc";
|
|
}
|
|
else
|
|
{
|
|
pszFormat = "Cstrike_Chat_T";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pszFormat = "Cstrike_Chat_T_Dead";
|
|
}
|
|
}
|
|
else if ( pPlayer->GetTeamNumber() == TEAM_SPECTATOR )
|
|
{
|
|
pszFormat = "Cstrike_Chat_Spec";
|
|
}
|
|
}
|
|
// everyone
|
|
else
|
|
{
|
|
if ( pPlayer->m_lifeState == LIFE_ALIVE )
|
|
{
|
|
pszFormat = "Cstrike_Chat_All";
|
|
}
|
|
else
|
|
{
|
|
if ( pPlayer->GetTeamNumber() != TEAM_SPECTATOR )
|
|
{
|
|
pszFormat = "Cstrike_Chat_AllDead";
|
|
}
|
|
else
|
|
{
|
|
pszFormat = "Cstrike_Chat_AllSpec";
|
|
}
|
|
}
|
|
}
|
|
|
|
return pszFormat;
|
|
}
|
|
|
|
void CCSGameRules::ClientSettingsChanged( CBasePlayer *pPlayer )
|
|
{
|
|
const char *pszNewName = engine->GetClientConVarValue( pPlayer->entindex(), "name" );
|
|
const char *pszOldName = pPlayer->GetPlayerName();
|
|
CCSPlayer *pCSPlayer = (CCSPlayer*)pPlayer;
|
|
if ( pszOldName[0] != 0 && Q_strncmp( pszOldName, pszNewName, MAX_PLAYER_NAME_LENGTH-1 ) )
|
|
{
|
|
pCSPlayer->ChangeName( pszNewName );
|
|
}
|
|
|
|
pCSPlayer->m_bShowHints = true;
|
|
if ( pCSPlayer->IsNetClient() )
|
|
{
|
|
const char *pShowHints = engine->GetClientConVarValue( engine->IndexOfEdict( pCSPlayer->edict() ), "cl_autohelp" );
|
|
if ( pShowHints && atoi( pShowHints ) <= 0 )
|
|
{
|
|
pCSPlayer->m_bShowHints = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CCSGameRules::FAllowNPCs( void )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool CCSGameRules::IsFriendlyFireOn( void )
|
|
{
|
|
return friendlyfire.GetBool();
|
|
}
|
|
|
|
|
|
CON_COMMAND( map_showspawnpoints, "Shows player spawn points (red=invalid)" )
|
|
{
|
|
CSGameRules()->ShowSpawnPoints();
|
|
}
|
|
|
|
void DrawSphere( const Vector& pos, float radius, int r, int g, int b, float lifetime )
|
|
{
|
|
Vector edge, lastEdge;
|
|
NDebugOverlay::Line( pos, pos + Vector( 0, 0, 50 ), r, g, b, true, lifetime );
|
|
|
|
lastEdge = Vector( radius + pos.x, pos.y, pos.z );
|
|
float angle;
|
|
for( angle=0.0f; angle <= 360.0f; angle += 22.5f )
|
|
{
|
|
edge.x = radius * BotCOS( angle ) + pos.x;
|
|
edge.y = pos.y;
|
|
edge.z = radius * BotSIN( angle ) + pos.z;
|
|
|
|
NDebugOverlay::Line( edge, lastEdge, r, g, b, true, lifetime );
|
|
|
|
lastEdge = edge;
|
|
}
|
|
|
|
lastEdge = Vector( pos.x, radius + pos.y, pos.z );
|
|
for( angle=0.0f; angle <= 360.0f; angle += 22.5f )
|
|
{
|
|
edge.x = pos.x;
|
|
edge.y = radius * BotCOS( angle ) + pos.y;
|
|
edge.z = radius * BotSIN( angle ) + pos.z;
|
|
|
|
NDebugOverlay::Line( edge, lastEdge, r, g, b, true, lifetime );
|
|
|
|
lastEdge = edge;
|
|
}
|
|
|
|
lastEdge = Vector( pos.x, radius + pos.y, pos.z );
|
|
for( angle=0.0f; angle <= 360.0f; angle += 22.5f )
|
|
{
|
|
edge.x = radius * BotCOS( angle ) + pos.x;
|
|
edge.y = radius * BotSIN( angle ) + pos.y;
|
|
edge.z = pos.z;
|
|
|
|
NDebugOverlay::Line( edge, lastEdge, r, g, b, true, lifetime );
|
|
|
|
lastEdge = edge;
|
|
}
|
|
}
|
|
|
|
CON_COMMAND_F( map_showbombradius, "Shows bomb radius from the center of each bomb site and planted bomb.", FCVAR_CHEAT )
|
|
{
|
|
float flBombDamage = 500.0f;
|
|
if ( g_pMapInfo )
|
|
flBombDamage = g_pMapInfo->m_flBombRadius;
|
|
float flBombRadius = flBombDamage * 3.5f;
|
|
Msg( "Bomb Damage is %.0f, Radius is %.0f\n", flBombDamage, flBombRadius );
|
|
|
|
CBaseEntity* ent = NULL;
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "func_bomb_target" ) ) != NULL )
|
|
{
|
|
const Vector &pos = ent->WorldSpaceCenter();
|
|
DrawSphere( pos, flBombRadius, 255, 255, 0, 10 );
|
|
}
|
|
|
|
ent = NULL;
|
|
while ( ( ent = gEntList.FindEntityByClassname( ent, "planted_c4" ) ) != NULL )
|
|
{
|
|
const Vector &pos = ent->WorldSpaceCenter();
|
|
DrawSphere( pos, flBombRadius, 255, 0, 0, 10 );
|
|
}
|
|
}
|
|
|
|
CON_COMMAND_F( map_setbombradius, "Sets the bomb radius for the map.", FCVAR_CHEAT )
|
|
{
|
|
if ( args.ArgC() != 2 )
|
|
return;
|
|
|
|
if ( !UTIL_IsCommandIssuedByServerAdmin() )
|
|
return;
|
|
|
|
if ( !g_pMapInfo )
|
|
CBaseEntity::Create( "info_map_parameters", vec3_origin, vec3_angle );
|
|
|
|
if ( !g_pMapInfo )
|
|
return;
|
|
|
|
g_pMapInfo->m_flBombRadius = atof( args[1] );
|
|
map_showbombradius( args );
|
|
}
|
|
|
|
void CreateBlackMarketString( void )
|
|
{
|
|
g_StringTableBlackMarket = networkstringtable->CreateStringTable( "BlackMarketTable" , 1 );
|
|
}
|
|
|
|
int CCSGameRules::GetStartMoney( void )
|
|
{
|
|
if ( IsBlackMarket() )
|
|
{
|
|
return atoi( mp_startmoney.GetDefault() );
|
|
}
|
|
|
|
return mp_startmoney.GetInt();
|
|
}
|
|
|
|
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [menglish] Set up anything for all players that changes based on new players spawning mid-game
|
|
// Find and return fun fact data
|
|
//=============================================================================
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Called when a player joins the game after it's started yet can still spawn in
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::SpawningLatePlayer( CCSPlayer* pLatePlayer )
|
|
{
|
|
//Reset the round kills number of enemies for the opposite team
|
|
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
|
|
{
|
|
CCSPlayer *pPlayer = (CCSPlayer*) UTIL_PlayerByIndex( i );
|
|
if(pPlayer)
|
|
{
|
|
if(pPlayer->GetTeamNumber() == pLatePlayer->GetTeamNumber())
|
|
{
|
|
continue;
|
|
}
|
|
pPlayer->m_NumEnemiesAtRoundStart++;
|
|
}
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [pfreese] Test for "pistol" round, defined as the default starting round
|
|
// when players cannot purchase anything primary weapons
|
|
//=============================================================================
|
|
|
|
bool CCSGameRules::IsPistolRound()
|
|
{
|
|
return m_iTotalRoundsPlayed == 0 && GetStartMoney() <= 800;
|
|
}
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
|
|
//=============================================================================
|
|
// HPE_BEGIN:
|
|
// [tj] So game rules can react to damage taken
|
|
// [menglish]
|
|
//=============================================================================
|
|
|
|
void CCSGameRules::PlayerTookDamage(CCSPlayer* player, const CTakeDamageInfo &damageInfo)
|
|
{
|
|
CBaseEntity *pInflictor = damageInfo.GetInflictor();
|
|
CBaseEntity *pAttacker = damageInfo.GetAttacker();
|
|
CCSPlayer *pCSScorer = (CCSPlayer *)(GetDeathScorer( pAttacker, pInflictor ));
|
|
|
|
if ( player && pCSScorer )
|
|
{
|
|
if (player->GetTeamNumber() == TEAM_CT)
|
|
{
|
|
m_bNoCTsDamaged = false;
|
|
}
|
|
|
|
if (player->GetTeamNumber() == TEAM_TERRORIST)
|
|
{
|
|
m_bNoTerroristsDamaged = false;
|
|
}
|
|
// set the first blood if this is the first and the victim is on a different team then the player
|
|
if ( m_pFirstBlood == NULL && pCSScorer != player && pCSScorer->GetTeamNumber() != player ->GetTeamNumber() )
|
|
{
|
|
m_pFirstBlood = pCSScorer;
|
|
m_firstBloodTime = gpGlobals->curtime - m_fRoundStartTime;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//=============================================================================
|
|
// HPE_END
|
|
//=============================================================================
|
|
#endif
|
|
|
|
bool CCSGameRules::IsConnectedUserInfoChangeAllowed( CBasePlayer *pPlayer )
|
|
{
|
|
#ifdef GAME_DLL
|
|
if( pPlayer )
|
|
{
|
|
int iPlayerTeam = pPlayer->GetTeamNumber();
|
|
if( ( iPlayerTeam == TEAM_CT ) || ( iPlayerTeam == TEAM_TERRORIST ) )
|
|
return false;
|
|
}
|
|
#else
|
|
int iLocalPlayerTeam = GetLocalPlayerTeam();
|
|
if( ( iLocalPlayerTeam == TEAM_CT ) || ( iLocalPlayerTeam == TEAM_TERRORIST ) )
|
|
return false;
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
#ifdef GAME_DLL
|
|
|
|
struct convar_tags_t
|
|
{
|
|
const char *pszConVar;
|
|
const char *pszTag;
|
|
};
|
|
|
|
// The list of convars that automatically turn on tags when they're changed.
|
|
// Convars in this list need to have the FCVAR_NOTIFY flag set on them, so the
|
|
// tags are recalculated and uploaded to the master server when the convar is changed.
|
|
convar_tags_t convars_to_check_for_tags[] =
|
|
{
|
|
{ "mp_friendlyfire", "friendlyfire" },
|
|
{ "bot_quota", "bots" },
|
|
{ "sv_nostats", "nostats" },
|
|
{ "mp_startmoney", "startmoney" },
|
|
{ "sv_allowminmodels", "nominmodels" },
|
|
{ "sv_enablebunnyhopping", "bunnyhopping" },
|
|
{ "sv_competitive_minspec", "compspec" },
|
|
{ "mp_holiday_nogifts", "nogifts" },
|
|
};
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Engine asks for the list of convars that should tag the server
|
|
//-----------------------------------------------------------------------------
|
|
void CCSGameRules::GetTaggedConVarList( KeyValues *pCvarTagList )
|
|
{
|
|
BaseClass::GetTaggedConVarList( pCvarTagList );
|
|
|
|
for ( int i = 0; i < ARRAYSIZE(convars_to_check_for_tags); i++ )
|
|
{
|
|
KeyValues *pKV = new KeyValues( "tag" );
|
|
pKV->SetString( "convar", convars_to_check_for_tags[i].pszConVar );
|
|
pKV->SetString( "tag", convars_to_check_for_tags[i].pszTag );
|
|
|
|
pCvarTagList->AddSubKey( pKV );
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
int CCSGameRules::GetBlackMarketPriceForWeapon( int iWeaponID )
|
|
{
|
|
if ( m_pPrices == NULL )
|
|
{
|
|
GetBlackMarketPriceList();
|
|
}
|
|
|
|
if ( m_pPrices )
|
|
return m_pPrices->iCurrentPrice[iWeaponID];
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int CCSGameRules::GetBlackMarketPreviousPriceForWeapon( int iWeaponID )
|
|
{
|
|
if ( m_pPrices == NULL )
|
|
{
|
|
GetBlackMarketPriceList();
|
|
}
|
|
|
|
if ( m_pPrices )
|
|
return m_pPrices->iPreviousPrice[iWeaponID];
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
const weeklyprice_t *CCSGameRules::GetBlackMarketPriceList( void )
|
|
{
|
|
if ( m_StringTableBlackMarket == NULL )
|
|
{
|
|
m_StringTableBlackMarket = networkstringtable->FindTable( CS_GAMERULES_BLACKMARKET_TABLE_NAME);
|
|
}
|
|
|
|
if ( m_pPrices == NULL )
|
|
{
|
|
int iSize = 0;
|
|
INetworkStringTable *pTable = m_StringTableBlackMarket;
|
|
if ( pTable && pTable->GetNumStrings() > 0 )
|
|
{
|
|
m_pPrices = (const weeklyprice_t *)pTable->GetStringUserData( 0, &iSize );
|
|
}
|
|
}
|
|
|
|
if ( m_pPrices )
|
|
{
|
|
PrepareEquipmentInfo();
|
|
}
|
|
|
|
return m_pPrices;
|
|
}
|
|
|
|
void CCSGameRules::SetBlackMarketPrices( bool bSetDefaults )
|
|
{
|
|
for ( int i = 1; i < WEAPON_MAX; i++ )
|
|
{
|
|
if ( i == WEAPON_SHIELDGUN )
|
|
continue;
|
|
|
|
CCSWeaponInfo *info = GetWeaponInfo( (CSWeaponID)i );
|
|
|
|
if ( info == NULL )
|
|
continue;
|
|
|
|
if ( bSetDefaults == false )
|
|
{
|
|
info->SetWeaponPrice( GetBlackMarketPriceForWeapon( i ) );
|
|
info->SetPreviousPrice( GetBlackMarketPreviousPriceForWeapon( i ) );
|
|
}
|
|
else
|
|
{
|
|
info->SetWeaponPrice( info->GetDefaultPrice() );
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef CLIENT_DLL
|
|
|
|
CCSGameRules::CCSGameRules()
|
|
{
|
|
CSGameRules()->m_StringTableBlackMarket = NULL;
|
|
m_pPrices = NULL;
|
|
m_bBlackMarket = false;
|
|
}
|
|
|
|
void TestTable( void )
|
|
{
|
|
CSGameRules()->m_StringTableBlackMarket = networkstringtable->FindTable( CS_GAMERULES_BLACKMARKET_TABLE_NAME);
|
|
|
|
if ( CSGameRules()->m_StringTableBlackMarket == NULL )
|
|
return;
|
|
|
|
int iIndex = CSGameRules()->m_StringTableBlackMarket->FindStringIndex( "blackmarket_prices" );
|
|
int iSize = 0;
|
|
|
|
const weeklyprice_t *pPrices = NULL;
|
|
|
|
pPrices = (const weeklyprice_t *)(CSGameRules()->m_StringTableBlackMarket)->GetStringUserData( iIndex, &iSize );
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
ConCommand cs_testtable( "cs_testtable", TestTable );
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Enforce certain values on the specified convar.
|
|
//-----------------------------------------------------------------------------
|
|
void EnforceCompetitiveCVar( const char *szCvarName, float fMinValue, float fMaxValue = FLT_MAX, int iArgs = 0, ... )
|
|
{
|
|
// Doing this check first because OK values might be outside the min/max range
|
|
ConVarRef competitiveConvar(szCvarName);
|
|
float fValue = competitiveConvar.GetFloat();
|
|
va_list vl;
|
|
va_start(vl, iArgs);
|
|
for( int i=0; i< iArgs; ++i )
|
|
{
|
|
if( (int)fValue == (int)va_arg(vl,double) )
|
|
return;
|
|
}
|
|
va_end(vl);
|
|
|
|
if( fValue < fMinValue || fValue > fMaxValue )
|
|
{
|
|
float fNewValue = MAX( MIN( fValue, fMaxValue ), fMinValue );
|
|
competitiveConvar.SetValue( fNewValue );
|
|
DevMsg( "Convar %s enforced by server (see sv_competitive_minspec.) Set to %2f.\n", szCvarName, fNewValue );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// An interface used by ENABLE_COMPETITIVE_CONVAR macro that lets the classes
|
|
// defined in the macro to be stored and acted on.
|
|
//-----------------------------------------------------------------------------
|
|
class ICompetitiveConvar
|
|
{
|
|
public:
|
|
// It is a best practice to always have a virtual destructor in an interface
|
|
// class. Otherwise if the derived classes have destructors they will not be
|
|
// called.
|
|
virtual ~ICompetitiveConvar() {}
|
|
virtual void BackupConvar() = 0;
|
|
virtual void EnforceRestrictions() = 0;
|
|
virtual void RestoreOriginalValue() = 0;
|
|
virtual void InstallChangeCallback() = 0;
|
|
};
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// A manager for all enforced competitive convars.
|
|
//-----------------------------------------------------------------------------
|
|
class CCompetitiveCvarManager : public CAutoGameSystem
|
|
{
|
|
public:
|
|
typedef CUtlVector<ICompetitiveConvar*> CompetitiveConvarList_t;
|
|
static void AddConvarToList( ICompetitiveConvar* pCVar )
|
|
{
|
|
GetConvarList()->AddToTail( pCVar );
|
|
}
|
|
|
|
static void BackupAllConvars()
|
|
{
|
|
FOR_EACH_VEC( *GetConvarList(), i )
|
|
{
|
|
(*GetConvarList())[i]->BackupConvar();
|
|
}
|
|
}
|
|
|
|
static void EnforceRestrictionsOnAllConvars()
|
|
{
|
|
FOR_EACH_VEC( *GetConvarList(), i )
|
|
{
|
|
(*GetConvarList())[i]->EnforceRestrictions();
|
|
}
|
|
}
|
|
|
|
static void RestoreAllOriginalValues()
|
|
{
|
|
FOR_EACH_VEC( *GetConvarList(), i )
|
|
{
|
|
(*GetConvarList())[i]->RestoreOriginalValue();
|
|
}
|
|
}
|
|
|
|
static CompetitiveConvarList_t* GetConvarList()
|
|
{
|
|
if( !s_pCompetitiveConvars )
|
|
{
|
|
s_pCompetitiveConvars = new CompetitiveConvarList_t();
|
|
}
|
|
return s_pCompetitiveConvars;
|
|
}
|
|
|
|
static KeyValues* GetConVarBackupKV()
|
|
{
|
|
if( !s_pConVarBackups )
|
|
{
|
|
s_pConVarBackups = new KeyValues("ConVarBackups");
|
|
}
|
|
return s_pConVarBackups;
|
|
}
|
|
|
|
virtual bool Init()
|
|
{
|
|
FOR_EACH_VEC( *GetConvarList(), i )
|
|
{
|
|
(*GetConvarList())[i]->InstallChangeCallback();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
virtual void Shutdown()
|
|
{
|
|
FOR_EACH_VEC( *GetConvarList(), i )
|
|
{
|
|
delete (*GetConvarList())[i];
|
|
}
|
|
delete s_pCompetitiveConvars;
|
|
s_pCompetitiveConvars = null;
|
|
s_pConVarBackups->deleteThis();
|
|
s_pConVarBackups = null;
|
|
}
|
|
private:
|
|
static CompetitiveConvarList_t* s_pCompetitiveConvars;
|
|
static KeyValues* s_pConVarBackups;
|
|
};
|
|
static CCompetitiveCvarManager *s_pCompetitiveCvarManager = new CCompetitiveCvarManager();
|
|
CCompetitiveCvarManager::CompetitiveConvarList_t* CCompetitiveCvarManager::s_pCompetitiveConvars = null;
|
|
KeyValues* CCompetitiveCvarManager::s_pConVarBackups = null;
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Macro to define restrictions on convars with "sv_competitive_minspec 1"
|
|
// Usage: ENABLE_COMPETITIVE_CONVAR( convarName, minValue, maxValue, optionalValues, opVal1, opVal2, ...
|
|
//-----------------------------------------------------------------------------
|
|
#define ENABLE_COMPETITIVE_CONVAR( convarName, ... ) \
|
|
class CCompetitiveMinspecConvar##convarName : public ICompetitiveConvar { \
|
|
public: \
|
|
CCompetitiveMinspecConvar##convarName(){ CCompetitiveCvarManager::AddConvarToList(this);} \
|
|
static void on_changed_##convarName( IConVar *var, const char *pOldValue, float flOldValue ){ \
|
|
if( sv_competitive_minspec.GetBool() ) { \
|
|
EnforceCompetitiveCVar( #convarName , __VA_ARGS__ ); }\
|
|
else {\
|
|
CCompetitiveCvarManager::GetConVarBackupKV()->SetFloat( #convarName, ConVarRef( #convarName ).GetFloat() ); } } \
|
|
virtual void BackupConvar() { CCompetitiveCvarManager::GetConVarBackupKV()->SetFloat( #convarName, ConVarRef( #convarName ).GetFloat() ); } \
|
|
virtual void EnforceRestrictions() { EnforceCompetitiveCVar( #convarName , __VA_ARGS__ ); } \
|
|
virtual void RestoreOriginalValue() { ConVarRef(#convarName).SetValue(CCompetitiveCvarManager::GetConVarBackupKV()->GetFloat( #convarName ) ); } \
|
|
virtual void InstallChangeCallback() { static_cast<ConVar*>(ConVarRef( #convarName ).GetLinkedConVar())->InstallChangeCallback( CCompetitiveMinspecConvar##convarName::on_changed_##convarName); } \
|
|
}; \
|
|
static CCompetitiveMinspecConvar##convarName *s_pCompetitiveConvar##convarName = new CCompetitiveMinspecConvar##convarName();
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Callback function for sv_competitive_minspec convar value change.
|
|
//-----------------------------------------------------------------------------
|
|
void sv_competitive_minspec_changed_f( IConVar *var, const char *pOldValue, float flOldValue )
|
|
{
|
|
ConVar *pCvar = static_cast<ConVar*>(var);
|
|
|
|
if( pCvar->GetBool() == true && (bool)flOldValue == false )
|
|
{
|
|
// Backup the values of each cvar and enforce new ones
|
|
CCompetitiveCvarManager::BackupAllConvars();
|
|
CCompetitiveCvarManager::EnforceRestrictionsOnAllConvars();
|
|
}
|
|
else if( pCvar->GetBool() == false && (bool)flOldValue == true )
|
|
{
|
|
// If sv_competitive_minspec is disabled, restore old client values
|
|
CCompetitiveCvarManager::RestoreAllOriginalValues();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
static ConVar sv_competitive_minspec( "sv_competitive_minspec",
|
|
"0",
|
|
FCVAR_REPLICATED | FCVAR_NOTIFY,
|
|
"Enable to force certain client convars to minimum/maximum values to help prevent competitive advantages:\n \
|
|
r_drawdetailprops = 1\n \
|
|
r_staticprop_lod = minimum -1 maximum 3\n \
|
|
fps_max minimum 59 (0 works too)\n \
|
|
cl_detailfade minimum 400\n \
|
|
cl_detaildist minimum 1200\n \
|
|
cl_interp_ratio = minimum 1 maximum 2\n \
|
|
cl_interp = minimum 0 maximum 0.031\n \
|
|
"
|
|
#ifdef CLIENT_DLL
|
|
,sv_competitive_minspec_changed_f
|
|
#endif
|
|
);
|
|
|
|
#ifdef CLIENT_DLL
|
|
|
|
ENABLE_COMPETITIVE_CONVAR( r_drawdetailprops, true, true ); // force r_drawdetailprops on
|
|
ENABLE_COMPETITIVE_CONVAR( r_staticprop_lod, -1, 3 ); // force r_staticprop_lod from -1 to 3
|
|
ENABLE_COMPETITIVE_CONVAR( fps_max, 59, FLT_MAX, 1, 0 ); // force fps_max above 59. One additional value (0) works
|
|
ENABLE_COMPETITIVE_CONVAR( cl_detailfade, 400 ); // force cl_detailfade above 400.
|
|
ENABLE_COMPETITIVE_CONVAR( cl_detaildist, 1200 ); // force cl_detaildist above 1200.
|
|
ENABLE_COMPETITIVE_CONVAR( cl_interp_ratio, 1, 2 ); // force cl_interp_ratio from 1 to 2
|
|
ENABLE_COMPETITIVE_CONVAR( cl_interp, 0, 0.031 ); // force cl_interp from 0.0152 to 0.031
|
|
|
|
// Stubs for replay client code
|
|
const char *GetMapDisplayName( const char *pMapName )
|
|
{
|
|
return pMapName;
|
|
}
|
|
|
|
bool IsTakingAFreezecamScreenshot()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#endif
|