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

790 lines
25 KiB
C++

//========= Copyright Valve Corporation, All rights reserved. ============//
//-------------------------------------------------------------
// File: cs_client_gamestats.cpp
// Desc: Manages client side stat storage, accumulation, and access
// Author: Peter Freese <peter@hiddenpath.com>
// Date: 2009/09/11
// Copyright: © 2009 Hidden Path Entertainment
//
// Keywords:
//-------------------------------------------------------------
#include "cbase.h"
#include "cs_client_gamestats.h"
#include "achievementmgr.h"
#include "engine/imatchmaking.h"
#include "ipresence.h"
#include "usermessages.h"
#include "c_cs_player.h"
#include "achievements_cs.h"
#include "vgui/ILocalize.h"
#include "c_team.h"
#include "../shared/steamworks_gamestats.h"
CCSClientGameStats g_CSClientGameStats;
void MsgFunc_PlayerStatsUpdate( bf_read &msg )
{
g_CSClientGameStats.MsgFunc_PlayerStatsUpdate(msg);
}
void MsgFunc_MatchStatsUpdate( bf_read &msg )
{
g_CSClientGameStats.MsgFunc_MatchStatsUpdate(msg);
}
CCSClientGameStats::StatContainerList_t* CCSClientGameStats::s_StatLists = new CCSClientGameStats::StatContainerList_t();
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
CCSClientGameStats::CCSClientGameStats()
{
m_bSteamStatsDownload = false;
}
//-----------------------------------------------------------------------------
// Purpose: called at init time after all systems are init'd. We have to
// do this in PostInit because the Steam app ID is not available earlier
//-----------------------------------------------------------------------------
void CCSClientGameStats::PostInit()
{
// listen for events
ListenForGameEvent( "player_stats_updated" );
ListenForGameEvent( "user_data_downloaded" );
ListenForGameEvent( "round_end" );
ListenForGameEvent( "round_start" );
usermessages->HookMessage( "PlayerStatsUpdate", ::MsgFunc_PlayerStatsUpdate );
usermessages->HookMessage( "MatchStatsUpdate", ::MsgFunc_MatchStatsUpdate );
GetSteamWorksSGameStatsUploader().StartSession();
m_RoundEndReason = Invalid_Round_End_Reason;
}
//-----------------------------------------------------------------------------
// Purpose: called at level shutdown
//-----------------------------------------------------------------------------
void CCSClientGameStats::LevelShutdownPreEntity()
{
// This is a good opportunity to update our last match stats
UpdateLastMatchStats();
// upload user stats to Steam on every map change
UpdateSteamStats();
}
//-----------------------------------------------------------------------------
// Purpose: called at app shutdown
//-----------------------------------------------------------------------------
void CCSClientGameStats::Shutdown()
{
GetSteamWorksSGameStatsUploader().EndSession();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CCSClientGameStats::LevelShutdownPreClearSteamAPIContext( void )
{
UploadRoundData();
}
//-----------------------------------------------------------------------------
// Purpose: called when the stats have changed in-game
//-----------------------------------------------------------------------------
void CCSClientGameStats::FireGameEvent( IGameEvent *event )
{
const char *pEventName = event->GetName();
if ( 0 == Q_strcmp( pEventName, "player_stats_updated" ) )
{
UpdateSteamStats();
}
else if ( 0 == Q_strcmp( pEventName, "user_data_downloaded" ) )
{
RetrieveSteamStats();
}
else if ( Q_strcmp( pEventName, "round_end" ) == 0 )
{
// Store off the reason the round ended. Used with the OGS data.
m_RoundEndReason = event->GetInt( "reason", Invalid_Round_End_Reason );
// update player count for last match stats
int iCurrentPlayerCount = 0;
if ( GetGlobalTeam(TEAM_CT) != NULL )
iCurrentPlayerCount += GetGlobalTeam(TEAM_CT)->Get_Number_Players();
if ( GetGlobalTeam(TEAM_TERRORIST) != NULL )
iCurrentPlayerCount += GetGlobalTeam(TEAM_TERRORIST)->Get_Number_Players();
m_matchMaxPlayerCount = MAX(m_matchMaxPlayerCount, iCurrentPlayerCount);
}
// The user stats for a round get sent piece meal, so we'll wait until a new round starts
// before sending the previous round stats.
else if ( Q_strcmp( pEventName, "round_start" ) == 0 && m_roundStats[CSSTAT_PLAYTIME] > 0 )
{
SRoundData *pRoundStatData = new SRoundData( &m_roundStats);
C_CSPlayer *pPlayer = ToCSPlayer( C_BasePlayer::GetLocalPlayer() );
if ( pPlayer )
{
// Our current money + what we spent is what we started with at the beginning of round
pRoundStatData->nStartingMoney = pPlayer->GetAccount() + m_roundStats[CSSTAT_MONEY_SPENT];
}
m_RoundStatData.AddToTail( pRoundStatData );
UploadRoundData();
m_roundStats.Reset();
}
}
void CCSClientGameStats::RetrieveSteamStats()
{
Assert( steamapicontext->SteamUserStats() );
if ( !steamapicontext->SteamUserStats() )
return;
// we shouldn't be downloading stats more than once
Assert(m_bSteamStatsDownload == false);
if (m_bSteamStatsDownload)
return;
int nStatFailCount = 0;
for ( int i = 0; i < CSSTAT_MAX; ++i )
{
if ( CSStatProperty_Table[i].szSteamName == NULL )
continue;
int iData;
if ( steamapicontext->SteamUserStats()->GetStat( CSStatProperty_Table[i].szSteamName, &iData ) )
{
m_lifetimeStats[CSStatProperty_Table[i].statId] = iData;
}
else
{
++nStatFailCount;
}
}
if ( nStatFailCount > 0 )
{
Msg("RetrieveSteamStats: failed to get %i stats\n", nStatFailCount);
return;
}
IGameEvent * event = gameeventmanager->CreateEvent( "player_stats_updated" );
if ( event )
{
gameeventmanager->FireEventClientSide( event );
}
m_bSteamStatsDownload = true;
}
//-----------------------------------------------------------------------------
// Purpose: Uploads stats for current Steam user to Steam
//-----------------------------------------------------------------------------
void CCSClientGameStats::UpdateSteamStats()
{
// only upload if Steam is running
if ( !steamapicontext->SteamUserStats() )
return;
CAchievementMgr *pAchievementMgr = dynamic_cast<CAchievementMgr *>( engine->GetAchievementMgr() );
Assert(pAchievementMgr != NULL);
if (!pAchievementMgr)
return;
// don't upload any stats if we haven't successfully download stats yet
if ( !m_bSteamStatsDownload )
{
// try to periodically download stats from Steam if we haven't gotten them yet
static float fLastStatsRetrieveTime = 0.0f;
const float kRetrieveInterval = 30.0f;
if ( gpGlobals->curtime > fLastStatsRetrieveTime + kRetrieveInterval )
{
pAchievementMgr->DownloadUserData();
fLastStatsRetrieveTime = gpGlobals->curtime;
}
return;
}
for ( int i = 0; i < CSSTAT_MAX; ++i )
{
if ( CSStatProperty_Table[i].szSteamName == NULL )
continue;
// set the stats locally in Steam client
steamapicontext->SteamUserStats()->SetStat( CSStatProperty_Table[i].szSteamName, m_lifetimeStats[CSStatProperty_Table[i].statId]);
}
// let the achievement manager know the stats have changed
pAchievementMgr->SetDirty(true);
}
CON_COMMAND_F( stats_reset, "Resets all player stats", FCVAR_CLIENTDLL )
{
g_CSClientGameStats.ResetAllStats();
}
CON_COMMAND_F( stats_dump, "Dumps all player stats", FCVAR_DEVELOPMENTONLY )
{
Msg( "Accumulated stats on Steam\n");
const StatsCollection_t& accumulatedStats = g_CSClientGameStats.GetLifetimeStats();
for ( int i = 0; i < CSSTAT_MAX; ++i )
{
if ( CSStatProperty_Table[i].szSteamName == NULL )
continue;
Msg( "%42s: %i\n", CSStatProperty_Table[i].szSteamName, accumulatedStats[CSStatProperty_Table[i].statId]);
}
}
#if defined(_DEBUG)
CON_COMMAND_F( stats_preload, "Load stats with data ripe for getting achievmenets", FCVAR_DEVELOPMENTONLY )
{
struct DataSet
{
CSStatType_t statId;
int value;
};
DataSet statData[] =
{
{ CSSTAT_KILLS, 9999},
{ CSSTAT_ROUNDS_WON, 4999},
{ CSSTAT_PISTOLROUNDS_WON, 249},
{ CSSTAT_MONEY_EARNED, 49999999},
{ CSSTAT_DAMAGE, 999999},
{ CSSTAT_KILLS_DEAGLE, 199},
{ CSSTAT_KILLS_USP, 199},
{ CSSTAT_KILLS_GLOCK, 199},
{ CSSTAT_KILLS_P228, 199},
{ CSSTAT_KILLS_ELITE, 99},
{ CSSTAT_KILLS_FIVESEVEN, 99},
{ CSSTAT_KILLS_AWP, 999},
{ CSSTAT_KILLS_AK47, 999},
{ CSSTAT_KILLS_M4A1, 999},
{ CSSTAT_KILLS_AUG, 499},
{ CSSTAT_KILLS_SG552, 499},
{ CSSTAT_KILLS_SG550, 499},
{ CSSTAT_KILLS_GALIL, 499},
{ CSSTAT_KILLS_FAMAS, 499},
{ CSSTAT_KILLS_SCOUT, 999},
{ CSSTAT_KILLS_G3SG1, 499},
{ CSSTAT_KILLS_P90, 999},
{ CSSTAT_KILLS_MP5NAVY, 999},
{ CSSTAT_KILLS_TMP, 499},
{ CSSTAT_KILLS_MAC10, 499},
{ CSSTAT_KILLS_UMP45, 999},
{ CSSTAT_KILLS_M3, 199},
{ CSSTAT_KILLS_XM1014, 199},
{ CSSTAT_KILLS_M249, 499},
{ CSSTAT_KILLS_KNIFE, 99},
{ CSSTAT_KILLS_HEGRENADE, 499},
{ CSSTAT_KILLS_HEADSHOT, 249},
{ CSSTAT_KILLS_ENEMY_WEAPON, 99},
{ CSSTAT_KILLS_ENEMY_BLINDED, 24},
{ CSSTAT_NUM_BOMBS_DEFUSED, 99},
{ CSSTAT_NUM_BOMBS_PLANTED, 99},
{ CSSTAT_NUM_HOSTAGES_RESCUED, 499},
{ CSSTAT_KILLS_KNIFE_FIGHT, 99},
{ CSSTAT_DECAL_SPRAYS, 99},
{ CSSTAT_NIGHTVISION_DAMAGE, 4999},
{ CSSTAT_KILLS_AGAINST_ZOOMED_SNIPER, 99},
{ CSSTAT_MAP_WINS_CS_ASSAULT, 99},
{ CSSTAT_MAP_WINS_CS_COMPOUND, 99},
{ CSSTAT_MAP_WINS_CS_HAVANA, 99},
{ CSSTAT_MAP_WINS_CS_ITALY, 99},
{ CSSTAT_MAP_WINS_CS_MILITIA, 99},
{ CSSTAT_MAP_WINS_CS_OFFICE, 99},
{ CSSTAT_MAP_WINS_DE_AZTEC, 99},
{ CSSTAT_MAP_WINS_DE_CBBLE, 99},
{ CSSTAT_MAP_WINS_DE_CHATEAU, 99},
{ CSSTAT_MAP_WINS_DE_DUST2, 99},
{ CSSTAT_MAP_WINS_DE_DUST, 99},
{ CSSTAT_MAP_WINS_DE_INFERNO, 99},
{ CSSTAT_MAP_WINS_DE_NUKE, 99},
{ CSSTAT_MAP_WINS_DE_PIRANESI, 99},
{ CSSTAT_MAP_WINS_DE_PORT, 99},
{ CSSTAT_MAP_WINS_DE_PRODIGY, 99},
{ CSSTAT_MAP_WINS_DE_TIDES, 99},
{ CSSTAT_MAP_WINS_DE_TRAIN, 99},
{ CSSTAT_WEAPONS_DONATED, 99},
{ CSSTAT_DOMINATIONS, 9},
{ CSSTAT_DOMINATION_OVERKILLS, 99},
{ CSSTAT_REVENGES, 19},
};
StatsCollection_t& lifetimeStats = const_cast<StatsCollection_t&>(g_CSClientGameStats.GetLifetimeStats());
for ( int i = 0; i < ARRAYSIZE(statData); ++i )
{
CSStatType_t statId = statData[i].statId;
lifetimeStats[statId] = statData[i].value;
}
IGameEvent * event = gameeventmanager->CreateEvent( "player_stats_updated" );
if ( event )
{
gameeventmanager->FireEventClientSide( event );
}
}
#endif
#if defined(_DEBUG)
CON_COMMAND_F( stats_corrupt, "Load stats with corrupt values", FCVAR_DEVELOPMENTONLY )
{
struct DataSet
{
CSStatType_t statId;
int value;
};
DataSet badData[] =
{
{ CSSTAT_SHOTS_HIT, 0x40000089 },
{ CSSTAT_SHOTS_FIRED, 0x400002BE },
{ CSSTAT_KILLS, 0x40000021 },
{ CSSTAT_DEATHS, 0x00000056 },
{ CSSTAT_DAMAGE, 0x00000FE3 },
{ CSSTAT_NUM_BOMBS_PLANTED, 0x00000004 },
{ CSSTAT_NUM_BOMBS_DEFUSED, 0x00000000 },
{ CSSTAT_PLAYTIME, 0x40000F46 },
{ CSSTAT_ROUNDS_WON, 0x40000028 },
{ CSSTAT_ROUNDS_PLAYED, 0x40001019 },
{ CSSTAT_PISTOLROUNDS_WON, 0x00000001 },
{ CSSTAT_MONEY_EARNED, 0x00021E94 },
{ CSSTAT_KILLS_DEAGLE, 0x00000009 },
{ CSSTAT_KILLS_USP, 0x00000000 },
{ CSSTAT_KILLS_GLOCK, 0x00000002 },
{ CSSTAT_KILLS_P228, 0x00000000 },
{ CSSTAT_KILLS_ELITE, 0x00000000 },
{ CSSTAT_KILLS_FIVESEVEN, 0x00000000 },
{ CSSTAT_KILLS_AWP, 0x00000000 },
{ CSSTAT_KILLS_AK47, 0x00000001 },
{ CSSTAT_KILLS_M4A1, 0x00000000 },
{ CSSTAT_KILLS_AUG, 0x00000000 },
{ CSSTAT_KILLS_SG552, 0x00000000 },
{ CSSTAT_KILLS_SG550, 0x00000000 },
{ CSSTAT_KILLS_GALIL, 0x00000000 },
{ CSSTAT_KILLS_FAMAS, 0x00000001 },
{ CSSTAT_KILLS_SCOUT, 0x00000000 },
{ CSSTAT_KILLS_G3SG1, 0x00000000 },
{ CSSTAT_KILLS_P90, 0x00000001 },
{ CSSTAT_KILLS_MP5NAVY, 0x00000000 },
{ CSSTAT_KILLS_TMP, 0x00000002 },
{ CSSTAT_KILLS_MAC10, 0x00000000 },
{ CSSTAT_KILLS_UMP45, 0x00000001 },
{ CSSTAT_KILLS_M3, 0x00000000 },
{ CSSTAT_KILLS_XM1014, 0x0000000A },
{ CSSTAT_KILLS_M249, 0x00000000 },
{ CSSTAT_KILLS_KNIFE, 0x00000000 },
{ CSSTAT_KILLS_HEGRENADE, 0x00000000 },
{ CSSTAT_SHOTS_DEAGLE, 0x0000004C },
{ CSSTAT_SHOTS_USP, 0x00000001 },
{ CSSTAT_SHOTS_GLOCK, 0x00000017 },
{ CSSTAT_SHOTS_P228, 0x00000000 },
{ CSSTAT_SHOTS_ELITE, 0x00000000 },
{ CSSTAT_SHOTS_FIVESEVEN, 0x00000000 },
{ CSSTAT_SHOTS_AWP, 0x00000000 },
{ CSSTAT_SHOTS_AK47, 0x0000000E },
{ CSSTAT_SHOTS_M4A1, 0x00000000 },
{ CSSTAT_SHOTS_AUG, 0x00000000 },
{ CSSTAT_SHOTS_SG552, 0x00000000 },
{ CSSTAT_SHOTS_SG550, 0x00000008 },
{ CSSTAT_SHOTS_GALIL, 0x00000000 },
{ CSSTAT_SHOTS_FAMAS, 0x00000010 },
{ CSSTAT_SHOTS_SCOUT, 0x00000000 },
{ CSSTAT_SHOTS_G3SG1, 0x00000000 },
{ CSSTAT_SHOTS_P90, 0x0000007F },
{ CSSTAT_SHOTS_MP5NAVY, 0x00000000 },
{ CSSTAT_SHOTS_TMP, 0x00000010 },
{ CSSTAT_SHOTS_MAC10, 0x00000000 },
{ CSSTAT_SHOTS_UMP45, 0x00000015 },
{ CSSTAT_SHOTS_M3, 0x00000009 },
{ CSSTAT_SHOTS_XM1014, 0x0000024C },
{ CSSTAT_SHOTS_M249, 0x00000000 },
{ CSSTAT_HITS_DEAGLE, 0x00000019 },
{ CSSTAT_HITS_USP, 0x00000000 },
{ CSSTAT_HITS_GLOCK, 0x0000000A },
{ CSSTAT_HITS_P228, 0x00000000 },
{ CSSTAT_HITS_ELITE, 0x00000000 },
{ CSSTAT_HITS_FIVESEVEN, 0x00000000 },
{ CSSTAT_HITS_AWP, 0x00000000 },
{ CSSTAT_HITS_AK47, 0x00000003 },
{ CSSTAT_HITS_M4A1, 0x00000000 },
{ CSSTAT_HITS_AUG, 0x00000000 },
{ CSSTAT_HITS_SG552, 0x00000000 },
{ CSSTAT_HITS_SG550, 0x00000001 },
{ CSSTAT_HITS_GALIL, 0x00000000 },
{ CSSTAT_HITS_FAMAS, 0x00000007 },
{ CSSTAT_HITS_SCOUT, 0x00000000 },
{ CSSTAT_HITS_G3SG1, 0x00000000 },
{ CSSTAT_HITS_P90, 0x0000000D },
{ CSSTAT_HITS_MP5NAVY, 0x00000000 },
{ CSSTAT_HITS_TMP, 0x00000006 },
{ CSSTAT_HITS_MAC10, 0x00000000 },
{ CSSTAT_HITS_UMP45, 0x00000006 },
{ CSSTAT_HITS_M3, 0x00000000 },
{ CSSTAT_HITS_XM1014, 0x0000004C },
{ CSSTAT_HITS_M249, 0x00000000 },
{ CSSTAT_KILLS_HEADSHOT, 0x00000013 },
{ CSSTAT_KILLS_ENEMY_BLINDED, 0x00000002 },
{ CSSTAT_KILLS_ENEMY_WEAPON, 0x00000002 },
{ CSSTAT_KILLS_KNIFE_FIGHT, 0x00000000 },
{ CSSTAT_DECAL_SPRAYS, 0x00000000 },
{ CSSTAT_NIGHTVISION_DAMAGE, 0x00000000 },
{ CSSTAT_NUM_HOSTAGES_RESCUED, 0x00000000 },
{ CSSTAT_NUM_BROKEN_WINDOWS, 0x00000000 },
{ CSSTAT_KILLS_AGAINST_ZOOMED_SNIPER, 0x00000000 },
{ CSSTAT_WEAPONS_DONATED, 0x00000000 },
{ CSSTAT_DOMINATIONS, 0x00000001 },
{ CSSTAT_DOMINATION_OVERKILLS, 0x00000000 },
{ CSSTAT_REVENGES, 0x00000000 },
{ CSSTAT_MVPS, 0x00000005 },
{ CSSTAT_MAP_WINS_CS_ASSAULT, 0x00000000 },
{ CSSTAT_MAP_WINS_CS_COMPOUND, 0x00000000 },
{ CSSTAT_MAP_WINS_CS_HAVANA, 0x00000000 },
{ CSSTAT_MAP_WINS_CS_ITALY, 0x40000002 },
{ CSSTAT_MAP_WINS_CS_MILITIA, 0x00000000 },
{ CSSTAT_MAP_WINS_CS_OFFICE, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_AZTEC, 0x0000000A },
{ CSSTAT_MAP_WINS_DE_CBBLE, 0x40000000 },
{ CSSTAT_MAP_WINS_DE_CHATEAU, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_DUST2, 0x0000000B },
{ CSSTAT_MAP_WINS_DE_DUST, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_INFERNO, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_NUKE, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_PIRANESI, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_PORT, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_PRODIGY, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_TIDES, 0x00000000 },
{ CSSTAT_MAP_WINS_DE_TRAIN, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_ASSAULT, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_COMPOUND, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_HAVANA, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_ITALY, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_MILITIA, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_CS_OFFICE, 0x00000003 },
{ CSSTAT_MAP_ROUNDS_DE_AZTEC, 0x00000019 },
{ CSSTAT_MAP_ROUNDS_DE_CBBLE, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_CHATEAU, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_DUST2, 0x00000014 },
{ CSSTAT_MAP_ROUNDS_DE_DUST, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_INFERNO, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_NUKE, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_PIRANESI, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_PORT, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_PRODIGY, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_TIDES, 0x00000000 },
{ CSSTAT_MAP_ROUNDS_DE_TRAIN, 0x00000000 },
{ CSSTAT_LASTMATCH_T_ROUNDS_WON, 0x00000000 },
{ CSSTAT_LASTMATCH_CT_ROUNDS_WON, 0x00000000 },
{ CSSTAT_LASTMATCH_ROUNDS_WON, 0x40000000 },
{ CSSTAT_LASTMATCH_KILLS, 0x00000000 },
{ CSSTAT_LASTMATCH_DEATHS, 0x00000000 },
{ CSSTAT_LASTMATCH_MVPS, 0x00000000 },
{ CSSTAT_LASTMATCH_DAMAGE, 0x00000000 },
{ CSSTAT_LASTMATCH_MONEYSPENT, 0x00000000 },
{ CSSTAT_LASTMATCH_DOMINATIONS, 0x00000000 },
{ CSSTAT_LASTMATCH_REVENGES, 0x00000000 },
{ CSSTAT_LASTMATCH_MAX_PLAYERS, 0x0000001B },
{ CSSTAT_LASTMATCH_FAVWEAPON_ID, 0x00000000 },
{ CSSTAT_LASTMATCH_FAVWEAPON_SHOTS, 0x00000000 },
{ CSSTAT_LASTMATCH_FAVWEAPON_HITS, 0x00000000 },
{ CSSTAT_LASTMATCH_FAVWEAPON_KILLS, 0x00000000 },
};
StatsCollection_t& lifetimeStats = const_cast<StatsCollection_t&>(g_CSClientGameStats.GetLifetimeStats());
for ( int i = 0; i < ARRAYSIZE(badData); ++i )
{
CSStatType_t statId = badData[i].statId;
lifetimeStats[statId] = badData[i].value;
}
IGameEvent * event = gameeventmanager->CreateEvent( "player_stats_updated" );
if ( event )
{
gameeventmanager->FireEventClientSide( event );
}
}
#endif
int CCSClientGameStats::GetStatCount()
{
return CSSTAT_MAX;
}
PlayerStatData_t CCSClientGameStats::GetStatByIndex( int index )
{
PlayerStatData_t statData;
statData.iStatId = CSStatProperty_Table[index].statId;
statData.iStatValue = m_lifetimeStats[statData.iStatId];
// we can make this more efficient by caching the localized names
statData.pStatDisplayName = g_pVGuiLocalize->Find( CSStatProperty_Table[index].szLocalizationToken );
return statData;
}
PlayerStatData_t CCSClientGameStats::GetStatById( int id )
{
Assert(id >= 0 && id < CSSTAT_MAX);
if ( id >= 0 && id < CSSTAT_MAX)
{
return GetStatByIndex(id);
}
else
{
PlayerStatData_t dummy;
dummy.pStatDisplayName = NULL;
dummy.iStatId = CSSTAT_UNDEFINED;
dummy.iStatValue = 0;
return dummy;
}
}
void CCSClientGameStats::UpdateStats( const StatsCollection_t &stats )
{
C_CSPlayer *pPlayer = C_CSPlayer::GetLocalCSPlayer();
if ( !pPlayer )
return;
// don't count stats if cheats on, commentary mode, etc
if ( !g_AchievementMgrCS.CheckAchievementsEnabled() )
return;
// Update the accumulated stats
m_lifetimeStats.Aggregate(stats);
m_matchStats.Aggregate(stats);
m_roundStats.Aggregate(stats);
IGameEvent * event = gameeventmanager->CreateEvent( "player_stats_updated" );
if ( event )
{
gameeventmanager->FireEventClientSide( event );
}
}
void CCSClientGameStats::ResetAllStats( void )
{
m_lifetimeStats.Reset();
m_matchStats.Reset();
m_roundStats.Reset();
// reset the stats on Steam
if (steamapicontext && steamapicontext->SteamUserStats())
{
steamapicontext->SteamUserStats()->ResetAllStats(false);
}
IGameEvent * event = gameeventmanager->CreateEvent( "player_stats_updated" );
if ( event )
{
gameeventmanager->FireEventClientSide( event );
}
}
void CCSClientGameStats::MsgFunc_MatchStatsUpdate( bf_read &msg )
{
int firstStat = msg.ReadShort();
for (int iStat = firstStat; iStat < CSSTAT_MAX && msg.GetNumBytesLeft() > 0; iStat++ )
{
m_directTStatAverages.m_fStat[iStat] = msg.ReadFloat();
m_directCTStatAverages.m_fStat[iStat] = msg.ReadFloat();
m_directPlayerStatAverages.m_fStat[iStat] = msg.ReadFloat();
}
// sanity check: the message should contain exactly the # of bytes we expect based on the bit field
Assert( !msg.IsOverflowed() );
Assert( 0 == msg.GetNumBytesLeft() );
}
void CCSClientGameStats::MsgFunc_PlayerStatsUpdate( bf_read &msg )
{
// Note: if any check fails while decoding this message, bail out and disregard this data to avoid
// potentially polluting player stats
StatsCollection_t deltaStats;
CRC32_t crc;
CRC32_Init( &crc );
const uint32 key = 0x82DA9F4C; // this key should match the key in cs_gamestats.cpp
CRC32_ProcessBuffer( &crc, &key, sizeof(key));
const byte version = 0x01;
CRC32_ProcessBuffer( &crc, &version, sizeof(version));
if (msg.ReadByte() != version)
{
Warning("PlayerStatsUpdate message: ignoring unsupported version\n");
return;
}
byte iStatsToRead = msg.ReadByte();
CRC32_ProcessBuffer( &crc, &iStatsToRead, sizeof(iStatsToRead));
for ( int i = 0; i < iStatsToRead; ++i)
{
byte iStat = msg.ReadByte();
CRC32_ProcessBuffer( &crc, &iStat, sizeof(iStat));
if (iStat >= CSSTAT_MAX)
{
Warning("PlayerStatsUpdate: invalid statId encountered; ignoring stats update\n");
return;
}
short delta = msg.ReadShort();
deltaStats[iStat] = delta;
CRC32_ProcessBuffer( &crc, &delta, sizeof(delta));
}
CRC32_Final( &crc );
CRC32_t readCRC = msg.ReadLong();
if (readCRC != crc || msg.IsOverflowed() || ( 0 != msg.GetNumBytesLeft() ) )
{
Warning("PlayerStatsUpdate message from server is corrupt; ignoring\n");
return;
}
// do one additional pass for out of band values
for ( int iStat = CSSTAT_FIRST; iStat < CSSTAT_MAX; ++iStat )
{
if (deltaStats[iStat] < 0 || deltaStats[iStat] >= 0x4000)
{
Warning("PlayerStatsUpdate message from server has out of band values; ignoring\n");
return;
}
}
// everything looks okay at this point; add these stats for the player's round, match, and lifetime stats
UpdateStats(deltaStats);
}
void CCSClientGameStats::UploadRoundData()
{
// If there's nothing to send, don't bother
if ( !m_RoundStatData.Count() )
return;
// Temporary ConVar to disable stats
if ( sv_noroundstats.GetBool() )
{
m_RoundStatData.PurgeAndDeleteElements();
return;
}
// Send off all OGS stats at level shutdown
KeyValues *pKV = new KeyValues( "basedata" );
if ( !pKV )
return;
pKV->SetString( "MapID", MapName() );
// Add all the vector based stats
for ( int k=0 ; k < m_RoundStatData.Count() ; ++k )
{
m_RoundStatData[ k ] ->nRoundEndReason = m_RoundEndReason;
SubmitStat( m_RoundStatData[ k ] );
}
// Perform the actual submission
SubmitGameStats( pKV );
// Clear out the per map stats
m_RoundStatData.Purge();
pKV->deleteThis();
// Reset the last round's ending status.
m_RoundEndReason = Invalid_Round_End_Reason;
}
void CCSClientGameStats::ResetMatchStats()
{
m_roundStats.Reset();
m_matchStats.Reset();
m_matchMaxPlayerCount = 0;
}
void CCSClientGameStats::UpdateLastMatchStats()
{
// only update that last match if we actually have valid data
if ( m_matchStats[CSSTAT_ROUNDS_PLAYED] == 0 )
return;
// check to see if the player materially participate; they could have been spectating or joined just in time for the ending.
int s = 0;
s += m_matchStats[CSSTAT_ROUNDS_WON];
s += m_matchStats[CSSTAT_KILLS];
s += m_matchStats[CSSTAT_DEATHS];
s += m_matchStats[CSSTAT_MVPS];
s += m_matchStats[CSSTAT_DAMAGE];
s += m_matchStats[CSSTAT_MONEY_SPENT];
if ( s == 0 )
return;
m_lifetimeStats[CSSTAT_LASTMATCH_T_ROUNDS_WON] = m_matchStats[CSSTAT_T_ROUNDS_WON];
m_lifetimeStats[CSSTAT_LASTMATCH_CT_ROUNDS_WON] = m_matchStats[CSSTAT_CT_ROUNDS_WON];
m_lifetimeStats[CSSTAT_LASTMATCH_ROUNDS_WON] = m_matchStats[CSSTAT_ROUNDS_WON];
m_lifetimeStats[CSSTAT_LASTMATCH_KILLS] = m_matchStats[CSSTAT_KILLS];
m_lifetimeStats[CSSTAT_LASTMATCH_DEATHS] = m_matchStats[CSSTAT_DEATHS];
m_lifetimeStats[CSSTAT_LASTMATCH_MVPS] = m_matchStats[CSSTAT_MVPS];
m_lifetimeStats[CSSTAT_LASTMATCH_DAMAGE] = m_matchStats[CSSTAT_DAMAGE];
m_lifetimeStats[CSSTAT_LASTMATCH_MONEYSPENT] = m_matchStats[CSSTAT_MONEY_SPENT];
m_lifetimeStats[CSSTAT_LASTMATCH_DOMINATIONS] = m_matchStats[CSSTAT_DOMINATIONS];
m_lifetimeStats[CSSTAT_LASTMATCH_REVENGES] = m_matchStats[CSSTAT_REVENGES];
m_lifetimeStats[CSSTAT_LASTMATCH_MAX_PLAYERS] = m_matchMaxPlayerCount;
CalculateMatchFavoriteWeapons();
}
//-----------------------------------------------------------------------------
// Purpose: Calculate and store the match favorite weapon for each player as only deltaStats for that weapon are stored on Steam
//-----------------------------------------------------------------------------
void CCSClientGameStats::CalculateMatchFavoriteWeapons()
{
int maxKills = 0, maxKillId = -1;
for( int j = CSSTAT_KILLS_DEAGLE; j <= CSSTAT_KILLS_M249; ++j )
{
if ( m_matchStats[j] > maxKills )
{
maxKills = m_matchStats[j];
maxKillId = j;
}
}
if ( maxKillId == -1 )
{
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_ID] = WEAPON_NONE;
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_SHOTS] = 0;
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_HITS] = 0;
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_KILLS] = 0;
}
else
{
int statTableID = -1;
for (int j = 0; WeaponName_StatId_Table[j].killStatId != CSSTAT_UNDEFINED; ++j)
{
if ( WeaponName_StatId_Table[j].killStatId == maxKillId )
{
statTableID = j;
break;
}
}
Assert( statTableID != -1 );
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_ID] = WeaponName_StatId_Table[statTableID].weaponId;
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_SHOTS] = m_matchStats[WeaponName_StatId_Table[statTableID].shotStatId];
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_HITS] = m_matchStats[WeaponName_StatId_Table[statTableID].hitStatId];
m_lifetimeStats[CSSTAT_LASTMATCH_FAVWEAPON_KILLS] = m_matchStats[WeaponName_StatId_Table[statTableID].killStatId];
}
}