source-engine/game/server/cstrike/funfact_cs.cpp
2022-03-02 11:45:17 +03:00

794 lines
29 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
#include "cbase.h"
#include "cs_gamerules.h"
#include "cs_gamestats.h"
#include "funfactmgr_cs.h"
#include "funfact_cs.h"
#include "../../game/shared/cstrike/weapon_csbase.h"
#include "cs_achievement_constants.h"
#define FIRST_BLOOD_TIME 45.0f
#define FIRST_KILL_TIME 45.0f
#define SHORT_ROUND_TIME 30.0f
#define MIN_SHOTS_FOR_ACCURACY 10
enum FunFactId
{
FUNFACT_CT_WIN_NO_KILLS,
FUNFACT_T_WIN_NO_KILLS,
FUNFACT_KILL_DEFUSER,
FUNFACT_KILL_RESCUER,
FUNFACT_T_WIN_NO_CASUALTIES,
FUNFACT_CT_WIN_NO_CASUALTIES,
FUNFACT_DAMAGE_WITH_GRENADES,
FUNFACT_KILLS_WITH_GRENADES,
FUNFACT_KILLS_WITH_SINGLE_GRENADE,
FUNFACT_DAMAGE_NO_KILLS,
FUNFACT_KILLED_ENEMIES,
FUNFACT_FIRST_KILL,
FUNFACT_FIRST_BLOOD,
FUNFACT_SHORT_ROUND,
FUNFACT_BEST_ACCURACY,
FUNFACT_KNIFE_KILLS,
FUNFACT_BLIND_KILLS,
FUNFACT_KILLS_WITH_LAST_ROUND,
FUNFACT_DONATED_WEAPONS,
FUNFACT_POSTHUMOUS_KILLS_WITH_GRENADE,
FUNFACT_KNIFE_IN_GUNFIGHT,
FUNFACT_NUM_TIMES_JUMPED,
FUNFACT_FALL_DAMAGE,
FUNFACT_ITEMS_PURCHASED,
FUNFACT_WON_AS_LAST_MEMBER,
FUNFACT_NUMBER_OF_OVERKILLS,
FUNFACT_SHOTS_FIRED,
FUNFACT_MONEY_SPENT,
FUNFACT_SURVIVED_MULTIPLE_ATTACKERS,
FUNFACT_DIED_FROM_MULTIPLE_ATTACKERS,
FUNFACT_DAMAGE_MULTIPLE_ENEMIES,
FUNFACT_GRENADES_THROWN,
FUNFACT_USED_ALL_AMMO,
FUNFACT_DEFENDED_BOMB,
FUNFACT_ITEMS_DROPPED_VALUE,
FUNFACT_KILL_WOUNDED_ENEMIES,
FUNFACT_USED_MULTIPLE_WEAPONS,
FUNFACT_TERRORIST_ACCURACY,
FUNFACT_CT_ACCURACY,
FUNFACT_SAME_UNIFORM_TERRORIST,
FUNFACT_SAME_UNIFORM_CT,
FUNFACT_BEST_TERRORIST_ACCURACY,
FUNFACT_BEST_COUNTERTERRORIST_ACCURACY,
FUNFACT_FALLBACK1,
FUNFACT_FALLBACK2,
FUNFACT_KILLS_HEADSHOTS,
FUNFACT_BROKE_WINDOWS,
FUNFACT_NIGHTVISION_DAMAGE,
FUNFACT_DEFUSED_WITH_DROPPED_KIT,
FUNFACT_KILLED_HALF_OF_ENEMIES,
};
CFunFactHelper *CFunFactHelper::s_pFirst = NULL;
//=============================================================================
// Generic evaluation Fun Fact
// This fun fact will evaluate the specified function to determine when it is
// valid. This is basically just a glue class for simple evaluation functions.
//=============================================================================
// Function type that we use to evaluate our fun facts. The data is returned as ints then floats that are passed in as reference parameters
typedef bool (*fFunFactEval)( int &iPlayer, int &data1, int &data2, int &data3 );
class CFunFact_GenericEvalFunction : public FunFactEvaluator
{
public:
CFunFact_GenericEvalFunction(FunFactId id, const char* szLocalizationToken, float fCoolness, fFunFactEval pfnEval ) :
FunFactEvaluator(id, szLocalizationToken, fCoolness),
m_pfnEval(pfnEval)
{}
virtual bool Evaluate( FunFactVector& results ) const
{
FunFact funfact;
if (m_pfnEval(funfact.iPlayer, funfact.iData1, funfact.iData2, funfact.iData3))
{
funfact.id = GetId();
funfact.szLocalizationToken = GetLocalizationToken();
funfact.fMagnitude = 0.0f;
results.AddToTail(funfact);
return true;
}
else
return false;
}
private:
fFunFactEval m_pfnEval;
};
#define DECLARE_FUNFACT_EVALFUNC(funfactId, szLocalizationToken, fCoolness, pfnEval) \
static FunFactEvaluator *CreateFunFact_##funfactId( void ) \
{ \
return new CFunFact_GenericEvalFunction(funfactId, szLocalizationToken, fCoolness, pfnEval);\
}; \
static CFunFactHelper g_##funfactId##_Helper( CreateFunFact_##funfactId );
//=============================================================================
// Per-player evaluation Fun Fact
// Evaluate the function per player and generate a fun fact for each valid or
// highest valid player
//=============================================================================
namespace EvalFlags
{
enum Type
{
All = 0x00,
TeamCT = 0x01,
TeamTerrorist = 0x02,
HighestOnly = 0x04, // when not set, generates fun facts for all valid testees
Alive = 0x08,
Dead = 0x10,
WinningTeam = 0x20,
LosingTeam = 0x40,
};
};
bool PlayerQualifies( const CBasePlayer* pPlayer, int flags )
{
if ( (flags & EvalFlags::TeamCT) && pPlayer->GetTeamNumber() != TEAM_CT )
return false;
if ( (flags & EvalFlags::TeamTerrorist) && pPlayer->GetTeamNumber() != TEAM_TERRORIST )
return false;
if ( (flags & EvalFlags::Dead) && const_cast<CBasePlayer*>(pPlayer)->IsAlive() ) // IsAlive() really isn't const correct
return false;
if ( (flags & EvalFlags::Alive) && !const_cast<CBasePlayer*>(pPlayer)->IsAlive() )
return false;
if ( (flags & EvalFlags::WinningTeam) && pPlayer->GetTeamNumber() != CSGameRules()->m_iRoundWinStatus )
return false;
if ( (flags & EvalFlags::LosingTeam) && pPlayer->GetTeamNumber() == CSGameRules()->m_iRoundWinStatus )
return false;
return true;
}
typedef int (*PlayerEvalFunction)(CCSPlayer* pPlayer);
class CFunFact_PlayerEvalFunction : public FunFactEvaluator
{
public:
CFunFact_PlayerEvalFunction(FunFactId id, const char* szLocalizationToken, float fCoolness, PlayerEvalFunction pfnEval,
int iMin, int flags ) :
FunFactEvaluator(id, szLocalizationToken, fCoolness),
m_pfnEval(pfnEval),
m_min(iMin),
m_flags(flags)
{}
virtual bool Evaluate( FunFactVector& results ) const
{
int iBestValue = 0;
int iBestPlayer = 0;
bool bResult = false;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
CCSPlayer* pPlayer = ToCSPlayer(UTIL_PlayerByIndex( i ) );
if ( pPlayer )
{
if (!PlayerQualifies(pPlayer, m_flags))
continue;
int iValue = m_pfnEval(pPlayer);
if (m_flags & EvalFlags::HighestOnly)
{
if ( iValue > iBestValue )
{
iBestValue = iValue;
iBestPlayer = i;
}
}
else
{
// generate fun facts for any player who meets the validation requirement
if ( iValue >= m_min )
{
FunFact funfact;
funfact.id = GetId();
funfact.szLocalizationToken = GetLocalizationToken();
funfact.iPlayer = i;
funfact.iData1 = iValue;
funfact.fMagnitude = 1.0f - ((float)m_min / iValue);
results.AddToTail(funfact);
bResult = true;
}
}
}
}
if ( (m_flags & EvalFlags::HighestOnly) && iBestValue >= m_min )
{
FunFact funfact;
funfact.id = GetId();
funfact.szLocalizationToken = GetLocalizationToken();
funfact.iPlayer = iBestPlayer;
funfact.iData1 = iBestValue;
funfact.fMagnitude = 1.0f - ((float)m_min / iBestValue);
results.AddToTail(funfact);
bResult = true;
}
return bResult;
}
private:
PlayerEvalFunction m_pfnEval;
int m_min;
int m_flags;
};
#define DECLARE_FUNFACT_PLAYERFUNC(funfactId, szLocalizationToken, fCoolness, pfnEval, iMin, iFlags) \
static FunFactEvaluator *CreateFunFact_##funfactId( void ) \
{ \
return new CFunFact_PlayerEvalFunction(funfactId, szLocalizationToken, fCoolness, pfnEval, iMin, iFlags); \
}; \
static CFunFactHelper g_##funfactId##_Helper( CreateFunFact_##funfactId );
//=============================================================================
// Per-team evaluation Fun Fact
//=============================================================================
typedef bool (*TeamEvalFunction)(int iTeam, int &data1, int &data2, int &data3);
class CFunFact_TeamEvalFunction : public FunFactEvaluator
{
public:
CFunFact_TeamEvalFunction(FunFactId id, const char* szLocalizationToken, float fCoolness, TeamEvalFunction pfnEval, int iTeam ) :
FunFactEvaluator(id, szLocalizationToken, fCoolness),
m_pfnEval(pfnEval),
m_team(iTeam)
{}
virtual bool Evaluate( FunFactVector& results ) const
{
int iData1, iData2, iData3;
if ( m_pfnEval(m_team, iData1, iData2, iData3) )
{
FunFact funfact;
funfact.id = GetId();
funfact.szLocalizationToken = GetLocalizationToken();
funfact.fMagnitude = 0.0f;
results.AddToTail(funfact);
return true;
}
return false;
}
private:
TeamEvalFunction m_pfnEval;
int m_team;
};
#define DECLARE_FUNFACT_TEAMFUNC(funfactId, szLocalizationToken, fCoolness, pfnEval, iTeam) \
static FunFactEvaluator *CreateFunFact_##funfactId( void ) \
{ \
return new CFunFact_TeamEvalFunction(funfactId, szLocalizationToken, fCoolness, pfnEval, iTeam);\
}; \
static CFunFactHelper g_##funfactId##_Helper( CreateFunFact_##funfactId );
//=============================================================================
// High Stat-based Fun Fact
// This fun fact will find the player with the highest value for a particular
// stat, and validate when that stat exceeds a specified minimum
//=============================================================================
class CFunFact_StatBest : public FunFactEvaluator
{
public:
CFunFact_StatBest(FunFactId id, const char* szLocalizationToken, float fCoolness, CSStatType_t statId, int iMin, int flags ) :
FunFactEvaluator(id, szLocalizationToken, fCoolness),
m_statId(statId),
m_min(iMin),
m_flags(flags)
{
V_strncpy(m_singularLocalizationToken, szLocalizationToken, sizeof(m_singularLocalizationToken));
if (m_min == 1)
{
V_strncat(m_singularLocalizationToken, "_singular", sizeof(m_singularLocalizationToken));
}
}
virtual bool Evaluate( FunFactVector& results ) const
{
int iBestValue = 0;
int iBestPlayer = 0;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
if ( pPlayer )
{
if (!PlayerQualifies(pPlayer, m_flags))
continue;
int iValue = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[m_statId];
if ( iValue > iBestValue )
{
iBestValue = iValue;
iBestPlayer = i;
}
}
}
if ( iBestValue >= m_min )
{
FunFact funfact;
funfact.id = GetId();
funfact.szLocalizationToken = iBestValue == 1 ? m_singularLocalizationToken : GetLocalizationToken();
funfact.iPlayer = iBestPlayer;
funfact.iData1 = iBestValue;
funfact.fMagnitude = 1.0f - ((float)m_min / iBestValue);
results.AddToTail(funfact);
return true;
}
return false;
}
private:
CSStatType_t m_statId;
int m_min;
char m_singularLocalizationToken[128];
int m_flags;
};
#define DECLARE_FUNFACT_STATBEST(funfactId, szLocalizationToken, fCoolness, statId, iMin, flags) \
static FunFactEvaluator *CreateFunFact_##funfactId( void ) \
{ \
return new CFunFact_StatBest(funfactId, szLocalizationToken, fCoolness, statId, iMin, flags); \
}; \
static CFunFactHelper g_##funfactId##_Helper( CreateFunFact_##funfactId );
//=============================================================================
// Sum-based Fun Fact
// This fun fact will add up a stat for all players, and is valid when the
// sum exceeds a threshold
//=============================================================================
class CFunFact_StatSum : public FunFactEvaluator
{
public:
CFunFact_StatSum(FunFactId id, const char* szLocalizationToken, float fCoolness, CSStatType_t statId, int iMin, EvalFlags::Type flags ) :
FunFactEvaluator(id, szLocalizationToken, fCoolness),
m_statId(statId),
m_min(iMin),
m_flags(flags)
{}
virtual bool Evaluate( FunFactVector& results ) const
{
int iSum = 0;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
CBasePlayer *pPlayer = UTIL_PlayerByIndex( i );
if ( pPlayer )
{
if (!PlayerQualifies(pPlayer, m_flags))
continue;
iSum += CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[m_statId];
}
}
if ( iSum >= m_min )
{
FunFact funfact;
funfact.id = GetId();
funfact.szLocalizationToken = GetLocalizationToken();
funfact.iPlayer = 0;
funfact.iData1 = iSum;
funfact.fMagnitude = 1.0f - ((float)m_min / iSum);
results.AddToTail(funfact);
return true;
}
return false;
}
private:
CSStatType_t m_statId;
int m_min;
int m_flags;
};
#define DECLARE_FUNFACT_STATSUM(funfactId, szLocalizationToken, fCoolness, statId, iMin, flags) \
static FunFactEvaluator *CreateFunFact_##funfactId( void ) \
{ \
return new CFunFact_StatSum(funfactId, szLocalizationToken, fCoolness, statId, iMin, flags); \
}; \
static CFunFactHelper g_##funfactId##_Helper( CreateFunFact_##funfactId );
//=============================================================================
// Helper function to calculate team accuracy
//=============================================================================
float GetTeamAccuracy( int teamNumber )
{
int teamShots = 0;
int teamHits = 0;
//Add up hits and shots
CBasePlayer *pPlayer = NULL;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
pPlayer = UTIL_PlayerByIndex( i );
if (pPlayer)
{
if (pPlayer->GetTeamNumber() == teamNumber)
{
teamShots += CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_FIRED];;
teamHits += CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_HIT];;
}
}
}
if (teamShots > MIN_SHOTS_FOR_ACCURACY)
return (float)teamHits / teamShots;
return 0.0f;
}
//=============================================================================
// fun fact explicit evaluation functions
//=============================================================================
bool FFEVAL_ALWAYS_TRUE( int &iPlayer, int &data1, int &data2, int &data3 )
{
return true;
}
bool FFEVAL_CT_WIN_NO_KILLS( int &iPlayer, int &data1, int &data2, int &data3 )
{
return ( CSGameRules()->m_iRoundWinStatus == WINNER_CT && CSGameRules()->m_bNoTerroristsKilled );
}
bool FFEVAL_T_WIN_NO_KILLS( int &iPlayer, int &data1, int &data2, int &data3 )
{
return ( CSGameRules()->m_iRoundWinStatus == WINNER_TER && CSGameRules()->m_bNoCTsKilled );
}
bool FFEVAL_T_WIN_NO_CASUALTIES( int &iPlayer, int &data1, int &data2, int &data3 )
{
return ( CSGameRules()->m_iRoundWinStatus == WINNER_TER && CSGameRules()->m_bNoTerroristsKilled );
}
bool FFEVAL_CT_WIN_NO_CASUALTIES( int &iPlayer, int &data1, int &data2, int &data3 )
{
return ( CSGameRules()->m_iRoundWinStatus == WINNER_CT && CSGameRules()->m_bNoCTsKilled );
}
int FFEVAL_KILLED_DEFUSER( CCSPlayer* pPlayer )
{
return pPlayer->GetKilledDefuser() ? 1 : 0;
}
int FFEVAL_KILLED_RESCUER( CCSPlayer* pPlayer )
{
return pPlayer->GetKilledRescuer() ? 1 : 0;
}
int FFEVAL_KILLS_WITH_GRENADE( CCSPlayer* pPlayer )
{
return pPlayer->GetMaxGrenadeKills();
}
int FFEVAL_DAMAGE_NO_KILLS( CCSPlayer* pPlayer )
{
if (CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_KILLS] == 0)
return CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_DAMAGE];
else
return 0;
}
int FFEVAL_FIRST_KILL( CCSPlayer* pPlayer )
{
if ( pPlayer == CSGameRules()->m_pFirstKill && CSGameRules()->m_firstKillTime < FIRST_KILL_TIME )
return CSGameRules()->m_firstKillTime;
else
return 0;
}
int FFEVAL_FIRST_BLOOD( CCSPlayer* pPlayer )
{
if ( pPlayer == CSGameRules()->m_pFirstBlood && CSGameRules()->m_firstBloodTime < FIRST_BLOOD_TIME )
return CSGameRules()->m_firstBloodTime;
else
return 0;
}
bool FFEVAL_SHORT_ROUND( int &iPlayer, int &data1, int &data2, int &data3 )
{
if ( CSGameRules()->GetRoundLength() - CSGameRules()->GetRoundRemainingTime() < SHORT_ROUND_TIME )
{
data1 = CSGameRules()->GetRoundLength() - CSGameRules()->GetRoundRemainingTime();
return true;
}
return false;
}
int FFEVAL_ACCURACY( CCSPlayer* pPlayer )
{
float shots = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_FIRED];
float hits = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_HIT];
if (shots >= MIN_SHOTS_FOR_ACCURACY)
return RoundFloatToInt(100.0f * hits / shots);
return 0;
}
int FFEVAL_KILLED_HALF_OF_ENEMIES( CCSPlayer* pPlayer )
{
return pPlayer->GetPercentageOfEnemyTeamKilled();
}
bool FFEVAL_WON_AS_LAST_MEMBER( int &iPlayer, int &data1, int &data2, int &data3 )
{
CCSPlayer *pCSPlayer = NULL;
int winningTeam = CSGameRules()->m_iRoundWinStatus;
if (winningTeam != TEAM_TERRORIST && winningTeam != TEAM_CT)
{
return false;
}
int losingTeam = (winningTeam == TEAM_TERRORIST) ? TEAM_CT : TEAM_TERRORIST;
CCSGameRules::TeamPlayerCounts playerCounts[TEAM_MAXCOUNT];
CSGameRules()->GetPlayerCounts(playerCounts);
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
pCSPlayer = ToCSPlayer(UTIL_PlayerByIndex( i ) );
if( pCSPlayer && pCSPlayer->GetTeamNumber() == winningTeam && pCSPlayer->IsAlive())
{
//Check if the player is still the only living member of his team ( on the off chance that a player joins late)
//This check is a little hacky. We make sure that there are no enemies alive. Since the bomb causes the round to end before exploding,
//the only way for only 1 person to be alive at round win time is extermination or defuse (in both cases, the last living player caused the win)
if (playerCounts[winningTeam].totalAlivePlayers == 1 && playerCounts[losingTeam].totalAlivePlayers == 0)
{
const PlayerStats_t& playerStats = CCS_GameStats.FindPlayerStats( pCSPlayer );
iPlayer = i;
data1 = playerStats.statsCurrentRound[CSSTAT_KILLS_WHILE_LAST_PLAYER_ALIVE];
if (data1 >= 2)
{
return true;
}
}
}
}
return false;
}
int FFEVAL_KNIFE_IN_GUNFIGHT( CCSPlayer* pPlayer )
{
return pPlayer->WasWieldingKnifeAndKilledByGun() ? 1 : 0;
}
int FFEVAL_MULTIPLE_ATTACKER_COUNT( CCSPlayer* pPlayer )
{
return pPlayer->GetNumEnemyDamagers();
}
int FFEVAL_USED_ALL_AMMO( CCSPlayer* pPlayer )
{
CWeaponCSBase *pRifleWeapon = dynamic_cast< CWeaponCSBase * >(pPlayer->Weapon_GetSlot( WEAPON_SLOT_RIFLE ));
CWeaponCSBase *pHandgunWeapon = dynamic_cast< CWeaponCSBase * >(pPlayer->Weapon_GetSlot( WEAPON_SLOT_PISTOL ));
if ( pRifleWeapon && !pRifleWeapon->HasAmmo() && pHandgunWeapon && !pHandgunWeapon->HasAmmo() )
return 1;
else
return 0;
}
int FFEVAL_DAMAGE_MULTIPLE_ENEMIES( CCSPlayer* pPlayer )
{
return pPlayer->GetNumEnemiesDamaged();
}
int FFEVAL_USED_MULTIPLE_WEAPONS( CCSPlayer* pPlayer )
{
return pPlayer->GetNumFirearmsUsed();
}
int FFEVAL_DEFUSED_WITH_DROPPED_KIT( CCSPlayer* pPlayer )
{
return pPlayer->GetDefusedWithPickedUpKit() ? 1 : 0;
}
bool FFEVAL_TERRORIST_ACCURACY( int &iPlayer, int &data1, int &data2, int &data3 )
{
float terroristAccuracy = GetTeamAccuracy(TEAM_TERRORIST);
float ctAccuracy = GetTeamAccuracy(TEAM_CT);
if (terroristAccuracy > 0.2f && terroristAccuracy > ctAccuracy)
{
data1 = RoundFloatToInt(terroristAccuracy * 100.0f);
return true;
}
return false;
}
bool FFEVAL_CT_ACCURACY( int &iPlayer, int &data1, int &data2, int &data3 )
{
float terroristAccuracy = GetTeamAccuracy(TEAM_TERRORIST);
float ctAccuracy = GetTeamAccuracy(TEAM_CT);
if (ctAccuracy > 0.2f && ctAccuracy > terroristAccuracy)
{
data1 = RoundFloatToInt(ctAccuracy * 100.0f);
return true;
}
return false;
}
bool FFEVAL_SAME_UNIFORM( int iTeam, int &iData1, int &iData2, int &iData3 )
{
int numberInUniform = 0;
int iUniform = -1;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
CCSPlayer *pCSPlayer = ToCSPlayer(UTIL_PlayerByIndex( i ) );
if ( pCSPlayer && pCSPlayer->GetTeamNumber() == iTeam && pCSPlayer->State_Get() != STATE_PICKINGCLASS)
{
if (iUniform == -1)
{
iUniform = pCSPlayer->PlayerClass();
}
else if (pCSPlayer->PlayerClass() != iUniform)
{
return false;
}
++numberInUniform;
}
}
return numberInUniform >= 3;
}
bool FFEVAL_BEST_TERRORIST_ACCURACY( int &iPlayer, int &data1, int &data2, int &data3 )
{
float fAccuracy = 0.0f, fBestAccuracy = 0.0f;
CBasePlayer *pPlayer = NULL;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
pPlayer = UTIL_PlayerByIndex( i );
// Look only at terrorist players
if ( pPlayer && pPlayer->GetTeamNumber() == TEAM_TERRORIST )
{
// Calculate accuracy the terrorist
float shots = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_FIRED];
float hits = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_HIT];
if (shots > MIN_SHOTS_FOR_ACCURACY)
{
fAccuracy = (float)hits / shots;
}
// Track the most accurate terrorist
if ( fAccuracy > fBestAccuracy )
{
fBestAccuracy = fAccuracy;
iPlayer = i;
}
}
}
if ( fBestAccuracy - GetTeamAccuracy( TEAM_TERRORIST ) >= 0.10f )
{
data1 = RoundFloatToInt(fBestAccuracy * 100.0f);
data2 = RoundFloatToInt(GetTeamAccuracy( TEAM_TERRORIST ) * 100.0f);
return true;
}
return false;
}
bool FFEVAL_BEST_COUNTERTERRORIST_ACCURACY( int &iPlayer, int &data1, int &data2, int &data3 )
{
float fAccuracy = 0.0f, fBestAccuracy = 0.0f;
CBasePlayer *pPlayer = NULL;
for ( int i = 1; i <= gpGlobals->maxClients; i++ )
{
pPlayer = UTIL_PlayerByIndex( i );
// Look only at counter-terrorist players
if ( pPlayer && pPlayer->GetTeamNumber() == TEAM_CT )
{
// Calculate accuracy the counter-terrorist
float shots = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_FIRED];
float hits = CCS_GameStats.FindPlayerStats(pPlayer).statsCurrentRound[CSSTAT_SHOTS_HIT];
if (shots > MIN_SHOTS_FOR_ACCURACY)
{
fAccuracy = (float)hits / shots;
}
// Track the most accurate counter-terrorist
if ( fAccuracy > fBestAccuracy )
{
fBestAccuracy = fAccuracy;
iPlayer = i;
}
}
}
if ( fBestAccuracy - GetTeamAccuracy( TEAM_CT ) >= 0.10f )
{
data1 = RoundFloatToInt(fBestAccuracy * 100.0f);
data2 = RoundFloatToInt(GetTeamAccuracy( TEAM_CT ) * 100.0f);
return true;
}
return false;
}
//=============================================================================
// Fun Fact Declarations
//=============================================================================
DECLARE_FUNFACT_STATBEST( FUNFACT_DAMAGE_WITH_GRENADES, "#funfact_damage_with_grenade", 0.5f, CSSTAT_GRENADE_DAMAGE, 200, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KNIFE_KILLS, "#funfact_knife_kills", 0.5f, CSSTAT_KILLS_KNIFE, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KILLS_WITH_GRENADES, "#funfact_kills_grenades", 0.7f, CSSTAT_KILLS_HEGRENADE, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_BLIND_KILLS, "#funfact_blind_kills", 0.9f, CSSTAT_KILLS_WHILE_BLINDED, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KILLED_ENEMIES, "#funfact_killed_enemies", 0.6f, CSSTAT_KILLS, 3, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KILLS_WITH_LAST_ROUND, "#funfact_kills_with_last_round", 0.6f, CSSTAT_KILLS_WITH_LAST_ROUND, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_DONATED_WEAPONS, "#funfact_donated_weapons", 0.3f, CSSTAT_WEAPONS_DONATED, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_NUM_TIMES_JUMPED, "#funfact_num_times_jumped", 0.2f, CSSTAT_TOTAL_JUMPS, 10, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_FALL_DAMAGE, "#funfact_fall_damage", 0.2f, CSSTAT_FALL_DAMAGE, 50, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_POSTHUMOUS_KILLS_WITH_GRENADE, "#funfact_posthumous_kills_with_grenade", 1.0f, CSSTAT_GRENADE_POSTHUMOUSKILLS, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_ITEMS_PURCHASED, "#funfact_items_purchased", 0.2f, CSSTAT_ITEMS_PURCHASED, 5, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_NUMBER_OF_OVERKILLS, "#funfact_number_of_overkills", 0.5f, CSSTAT_DOMINATION_OVERKILLS, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_MONEY_SPENT, "#funfact_money_spent", 0.2f, CSSTAT_MONEY_SPENT, 5000, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_GRENADES_THROWN, "#funfact_grenades_thrown", 0.3f, CSSTAT_GRENADES_THROWN, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_DEFENDED_BOMB, "#funfact_defended_bomb", 0.5f, CSSTAT_KILLS_WHILE_DEFENDING_BOMB, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_ITEMS_DROPPED_VALUE, "#funfact_items_dropped_value", 0.5f, CSTAT_ITEMS_DROPPED_VALUE, 10000, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KILL_WOUNDED_ENEMIES, "#funfact_kill_wounded_enemies", 0.4f, CSSTAT_KILLS_ENEMY_WOUNDED, 3, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_KILLS_HEADSHOTS, "#funfact_kills_headshots", 0.7f, CSSTAT_KILLS_HEADSHOT, 3, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_BROKE_WINDOWS, "#funfact_broke_windows", 0.3f, CSSTAT_NUM_BROKEN_WINDOWS, 5, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATBEST( FUNFACT_NIGHTVISION_DAMAGE, "#funfact_nightvision_damage", 0.5f, CSSTAT_NIGHTVISION_DAMAGE, 100, EvalFlags::HighestOnly);
DECLARE_FUNFACT_STATSUM( FUNFACT_SHOTS_FIRED, "#funfact_shots_fired", 0.1f, CSSTAT_SHOTS_FIRED, 200, EvalFlags::All);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_KILL_DEFUSER, "#funfact_kill_defuser", 0.6f, FFEVAL_KILLED_DEFUSER, 1, EvalFlags::TeamTerrorist | EvalFlags::WinningTeam);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_KILL_RESCUER, "#funfact_kill_rescuer", 0.6f, FFEVAL_KILLED_RESCUER, 1, EvalFlags::TeamTerrorist);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_KILLS_WITH_SINGLE_GRENADE, "#funfact_kills_with_single_grenade", 0.8f, FFEVAL_KILLS_WITH_GRENADE, 2, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_DAMAGE_NO_KILLS, "#funfact_damage_no_kills", 0.4f, FFEVAL_DAMAGE_NO_KILLS, 200, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_FIRST_KILL, "#funfact_first_kill", 0.2f, FFEVAL_FIRST_KILL, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_FIRST_BLOOD, "#funfact_first_blood", 0.2f, FFEVAL_FIRST_BLOOD, 1, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_BEST_ACCURACY, "#funfact_best_accuracy", 0.4f, FFEVAL_ACCURACY, 20, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_KNIFE_IN_GUNFIGHT, "#funfact_knife_in_gunfight", 0.6f, FFEVAL_KNIFE_IN_GUNFIGHT , 1, EvalFlags::All);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_SURVIVED_MULTIPLE_ATTACKERS, "#funfact_survived_multiple_attackers", 0.3f, FFEVAL_MULTIPLE_ATTACKER_COUNT, 3, EvalFlags::HighestOnly | EvalFlags::Alive);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_DIED_FROM_MULTIPLE_ATTACKERS, "#funfact_died_from_multiple_attackers", 0.5f, FFEVAL_MULTIPLE_ATTACKER_COUNT, 3, EvalFlags::HighestOnly | EvalFlags::Dead);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_USED_ALL_AMMO, "#funfact_used_all_ammo", 0.5f, FFEVAL_USED_ALL_AMMO, 1, EvalFlags::All );
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_DAMAGE_MULTIPLE_ENEMIES, "#funfact_damage_multiple_enemies", 0.5f, FFEVAL_DAMAGE_MULTIPLE_ENEMIES, 3, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_USED_MULTIPLE_WEAPONS, "#funfact_used_multiple_weapons", 0.5f, FFEVAL_USED_MULTIPLE_WEAPONS, 4, EvalFlags::HighestOnly);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_DEFUSED_WITH_DROPPED_KIT, "#funfact_defused_with_dropped_kit", 0.4f, FFEVAL_DEFUSED_WITH_DROPPED_KIT, 1, EvalFlags::TeamCT);
DECLARE_FUNFACT_PLAYERFUNC( FUNFACT_KILLED_HALF_OF_ENEMIES, "#funfact_killed_half_of_enemies", 0.5f, FFEVAL_KILLED_HALF_OF_ENEMIES, 50, EvalFlags::WinningTeam | EvalFlags::HighestOnly);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_CT_WIN_NO_KILLS, "#funfact_ct_win_no_kills", 0.4f, FFEVAL_CT_WIN_NO_KILLS);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_T_WIN_NO_KILLS, "#funfact_t_win_no_kills", 0.4f, FFEVAL_T_WIN_NO_KILLS );
DECLARE_FUNFACT_EVALFUNC( FUNFACT_T_WIN_NO_CASUALTIES, "#funfact_t_win_no_casualties", 0.2f, FFEVAL_T_WIN_NO_CASUALTIES );
DECLARE_FUNFACT_EVALFUNC( FUNFACT_CT_WIN_NO_CASUALTIES, "#funfact_ct_win_no_casualties", 0.2f, FFEVAL_CT_WIN_NO_CASUALTIES );
DECLARE_FUNFACT_EVALFUNC( FUNFACT_SHORT_ROUND, "#funfact_short_round", 0.3f, FFEVAL_SHORT_ROUND );
DECLARE_FUNFACT_EVALFUNC( FUNFACT_WON_AS_LAST_MEMBER, "#funfact_won_as_last_member", 0.6f, FFEVAL_WON_AS_LAST_MEMBER );
DECLARE_FUNFACT_EVALFUNC( FUNFACT_TERRORIST_ACCURACY, "#funfact_terrorist_accuracy", 0.2f, FFEVAL_TERRORIST_ACCURACY);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_CT_ACCURACY, "#funfact_ct_accuracy", 0.2f, FFEVAL_CT_ACCURACY);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_BEST_TERRORIST_ACCURACY, "#funfact_best_terrorist_accuracy", 0.3f, FFEVAL_BEST_TERRORIST_ACCURACY);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_BEST_COUNTERTERRORIST_ACCURACY, "#funfact_best_counterterrorist_accuracy", 0.3f, FFEVAL_BEST_COUNTERTERRORIST_ACCURACY);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_FALLBACK1, "#funfact_fallback1", 0.0f, FFEVAL_ALWAYS_TRUE);
DECLARE_FUNFACT_EVALFUNC( FUNFACT_FALLBACK2, "#funfact_fallback2", 0.0f, FFEVAL_ALWAYS_TRUE);
DECLARE_FUNFACT_TEAMFUNC( FUNFACT_SAME_UNIFORM_TERRORIST, "#funfact_same_uniform_terrorist", 0.5f, FFEVAL_SAME_UNIFORM, TEAM_TERRORIST);
DECLARE_FUNFACT_TEAMFUNC( FUNFACT_SAME_UNIFORM_CT, "#funfact_same_uniform_ct", 0.5f, FFEVAL_SAME_UNIFORM, TEAM_CT);