Counter Strike : Global Offensive Source Code

705 lines
21 KiB

  1. //========= Copyright � 1996-2005, Valve Corporation, All rights reserved. ============//
  2. //
  3. // Purpose: Entity that propagates general data needed by clients for every player.
  4. //
  5. // $NoKeywords: $
  6. //=============================================================================//
  7. #include "cbase.h"
  8. #include "c_playerresource.h"
  9. #include "c_team.h"
  10. #include "gamestringpool.h"
  11. #include "hltvreplaysystem.h"
  12. #if !defined( _X360 )
  13. #include "xbox/xboxstubs.h"
  14. #endif
  15. // memdbgon must be the last include file in a .cpp file!!!
  16. #include "tier0/memdbgon.h"
  17. const float PLAYER_RESOURCE_THINK_INTERVAL = 0.2f;
  18. #define PLAYER_DEBUG_NAME "WWWWWWWWWWWWWWW"
  19. ConVar cl_names_debug( "cl_names_debug", "0", FCVAR_DEVELOPMENTONLY );
  20. void RecvProxy_ChangedTeam( const CRecvProxyData *pData, void *pStruct, void *pOut )
  21. {
  22. // Have the regular proxy store the data.
  23. RecvProxy_Int32ToInt32( pData, pStruct, pOut );
  24. if ( g_PR )
  25. {
  26. g_PR->TeamChanged();
  27. }
  28. }
  29. IMPLEMENT_CLIENTCLASS_DT_NOBASE(C_PlayerResource, DT_PlayerResource, CPlayerResource)
  30. RecvPropArray3( RECVINFO_ARRAY(m_iPing), RecvPropInt( RECVINFO(m_iPing[0]))),
  31. RecvPropArray3( RECVINFO_ARRAY(m_iKills), RecvPropInt( RECVINFO(m_iKills[0]))),
  32. RecvPropArray3( RECVINFO_ARRAY(m_iAssists), RecvPropInt( RECVINFO(m_iAssists[0]))),
  33. RecvPropArray3( RECVINFO_ARRAY(m_iDeaths), RecvPropInt( RECVINFO(m_iDeaths[0]))),
  34. RecvPropArray3( RECVINFO_ARRAY(m_bConnected), RecvPropInt( RECVINFO(m_bConnected[0]))),
  35. RecvPropArray3( RECVINFO_ARRAY(m_iTeam), RecvPropInt( RECVINFO(m_iTeam[0]), 0, RecvProxy_ChangedTeam )),
  36. RecvPropArray3( RECVINFO_ARRAY(m_iPendingTeam), RecvPropInt( RECVINFO(m_iPendingTeam[0]), 0, RecvProxy_ChangedTeam )),
  37. RecvPropArray3( RECVINFO_ARRAY(m_bAlive), RecvPropInt( RECVINFO(m_bAlive[0]))),
  38. RecvPropArray3( RECVINFO_ARRAY(m_iHealth), RecvPropInt( RECVINFO(m_iHealth[0]))),
  39. RecvPropArray3( RECVINFO_ARRAY(m_iCoachingTeam), RecvPropInt( RECVINFO(m_iCoachingTeam[0]))),
  40. END_RECV_TABLE()
  41. BEGIN_PREDICTION_DATA( C_PlayerResource )
  42. DEFINE_PRED_ARRAY( m_szName, FIELD_STRING, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  43. DEFINE_PRED_ARRAY( m_iPing, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  44. DEFINE_PRED_ARRAY( m_iKills, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  45. DEFINE_PRED_ARRAY( m_iAssists, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  46. DEFINE_PRED_ARRAY( m_iDeaths, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  47. DEFINE_PRED_ARRAY( m_bConnected, FIELD_BOOLEAN, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  48. DEFINE_PRED_ARRAY( m_iTeam, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  49. DEFINE_PRED_ARRAY( m_iPendingTeam, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  50. DEFINE_PRED_ARRAY( m_bAlive, FIELD_BOOLEAN, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  51. DEFINE_PRED_ARRAY( m_iHealth, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  52. DEFINE_PRED_ARRAY( m_iCoachingTeam, FIELD_INTEGER, MAX_PLAYERS+1, FTYPEDESC_PRIVATE ),
  53. END_PREDICTION_DATA()
  54. C_PlayerResource *g_PR;
  55. IGameResources * GameResources( void ) { return g_PR; }
  56. //-----------------------------------------------------------------------------
  57. // Purpose:
  58. //-----------------------------------------------------------------------------
  59. C_PlayerResource::C_PlayerResource()
  60. {
  61. for ( int i=0; i<ARRAYSIZE(m_szName); ++i )
  62. {
  63. m_szName[i] = AllocPooledString( "unconnected" );
  64. }
  65. memset( m_iPing, 0, sizeof( m_iPing ) );
  66. // memset( m_iPacketloss, 0, sizeof( m_iPacketloss ) );
  67. memset( m_iKills, 0, sizeof( m_iKills ) );
  68. memset( m_iAssists, 0, sizeof( m_iAssists ) );
  69. memset( m_iDeaths, 0, sizeof( m_iDeaths ) );
  70. memset( m_bConnected, 0, sizeof( m_bConnected ) );
  71. memset( m_iTeam, 0, sizeof( m_iTeam ) );
  72. memset( m_iPendingTeam, 0, sizeof( m_iTeam ) );
  73. memset( m_bAlive, 0, sizeof( m_bAlive ) );
  74. memset( m_iHealth, 0, sizeof( m_iHealth ) );
  75. memset( m_Xuids, 0, sizeof( m_Xuids ) );
  76. memset( m_iCoachingTeam, 0, sizeof( m_iCoachingTeam) );
  77. for ( int i=0; i<MAX_TEAMS; i++ )
  78. {
  79. m_Colors[i] = COLOR_GREY;
  80. }
  81. g_PR = this;
  82. g_pScaleformUI->AddDeviceDependentObject( this );
  83. }
  84. //-----------------------------------------------------------------------------
  85. // Purpose:
  86. //-----------------------------------------------------------------------------
  87. C_PlayerResource::~C_PlayerResource()
  88. {
  89. for ( int i = 1; i <= MAX_PLAYERS; i++ )
  90. {
  91. if ( m_Xuids[i] != INVALID_XUID )
  92. {
  93. g_pScaleformUI->AvatarImageRelease( m_Xuids[i] );
  94. }
  95. }
  96. g_PR = NULL;
  97. g_pScaleformUI->RemoveDeviceDependentObject( this );
  98. }
  99. void C_PlayerResource::OnDataChanged(DataUpdateType_t updateType)
  100. {
  101. UpdateXuids();
  102. BaseClass::OnDataChanged( updateType );
  103. if ( updateType == DATA_UPDATE_CREATED )
  104. {
  105. SetNextClientThink( gpGlobals->curtime + PLAYER_RESOURCE_THINK_INTERVAL );
  106. }
  107. }
  108. void C_PlayerResource::UpdateXuids( void )
  109. {
  110. for ( int i = 1; i <= MAX_PLAYERS; i++ )
  111. {
  112. XUID newXuid = INVALID_XUID;
  113. player_info_t sPlayerInfo;
  114. if ( m_bConnected[ i ] && engine->GetPlayerInfo( i, &sPlayerInfo ) )
  115. {
  116. newXuid = sPlayerInfo.xuid;
  117. // When running in anonymous mode wipe out all XUIDs
  118. if ( newXuid != INVALID_XUID )
  119. {
  120. if ( CDemoPlaybackParameters_t const *pParameters = engine->GetDemoPlaybackParameters() )
  121. {
  122. if ( pParameters->m_bAnonymousPlayerIdentity )
  123. {
  124. // CSteamID steamid( newXuid );
  125. // steamid.SetAccountID( ( ~uint32(0) ) -1 - i );
  126. // newXuid = steamid.ConvertToUint64();
  127. newXuid = INVALID_XUID;
  128. }
  129. }
  130. }
  131. }
  132. if ( newXuid != m_Xuids[i] )
  133. {
  134. bool bAddRefSuccess = false;
  135. if ( m_Xuids[i] != INVALID_XUID )
  136. {
  137. g_pScaleformUI->AvatarImageRelease( m_Xuids[i] );
  138. }
  139. if ( newXuid != INVALID_XUID )
  140. {
  141. bAddRefSuccess = g_pScaleformUI->AvatarImageAddRef( newXuid );
  142. }
  143. if ( bAddRefSuccess || ( newXuid == INVALID_XUID ) )
  144. {
  145. m_Xuids[i] = newXuid;
  146. }
  147. }
  148. }
  149. }
  150. void C_PlayerResource::UpdateAsLocalizedFakePlayerName( int slot, char const *pchPlayerName )
  151. {
  152. static CUtlStringMap< CUtlString > s_mapLocalizedNames;
  153. UtlSymId_t symName = s_mapLocalizedNames.Find( pchPlayerName );
  154. if ( symName == UTL_INVAL_SYMBOL )
  155. {
  156. // Need to localize it from scratch
  157. CUtlVector< char * > arrLocTokens;
  158. CFmtStr1024 strComposite;
  159. V_SplitString( pchPlayerName, " ", arrLocTokens );
  160. FOR_EACH_VEC( arrLocTokens, i )
  161. {
  162. if ( i ) strComposite.Append( " " );
  163. if ( wchar_t const * const kwszLocalizedToken = g_pVGuiLocalize->Find( CFmtStr( "#CSGO_FakePlayer_%s", arrLocTokens[ i ] ) ) )
  164. {
  165. char chUtf8token[ MAX_PLAYER_NAME_LENGTH ] = {};
  166. V_UnicodeToUTF8( kwszLocalizedToken, chUtf8token, sizeof( chUtf8token ) );
  167. strComposite.Append( chUtf8token );
  168. }
  169. else
  170. {
  171. strComposite.Append( arrLocTokens[ i ] );
  172. Warning( "Failed to localize fake player name '#CSGO_FakePlayer_%s'\n", arrLocTokens[ i ] );
  173. }
  174. }
  175. symName = s_mapLocalizedNames.Insert( pchPlayerName, strComposite.Access() );
  176. arrLocTokens.PurgeAndDeleteElements();
  177. }
  178. Assert( symName != UTL_INVAL_SYMBOL );
  179. m_szName[ slot ] = s_mapLocalizedNames[ symName ];
  180. }
  181. void C_PlayerResource::UpdatePlayerName( int slot )
  182. {
  183. if ( slot < 1 || slot > MAX_PLAYERS )
  184. {
  185. Error( "UpdatePlayerName with bogus slot %d\n", slot );
  186. return;
  187. }
  188. player_info_t sPlayerInfo;
  189. char const *pchPlayerName = PLAYER_UNCONNECTED_NAME;
  190. if ( IsConnected( slot ) &&
  191. engine->GetPlayerInfo( slot, &sPlayerInfo ) )
  192. {
  193. pchPlayerName = sPlayerInfo.name;
  194. if ( sPlayerInfo.fakeplayer && *pchPlayerName )
  195. {
  196. UpdateAsLocalizedFakePlayerName( slot, pchPlayerName );
  197. return;
  198. }
  199. }
  200. if ( !m_szName[slot] || Q_stricmp( m_szName[slot], pchPlayerName ) )
  201. {
  202. m_szName[slot] = AllocPooledString( pchPlayerName );
  203. }
  204. }
  205. void C_PlayerResource::ClientThink()
  206. {
  207. BaseClass::ClientThink();
  208. for ( int i = 1; i <= gpGlobals->maxClients; ++i )
  209. {
  210. UpdatePlayerName( i );
  211. }
  212. SetNextClientThink( gpGlobals->curtime + PLAYER_RESOURCE_THINK_INTERVAL );
  213. }
  214. #define STATIC_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( s_chStaticArrayVarName, szLocToken, szDefaultText ) \
  215. static char s_chStaticArrayVarName[MAX_PLAYER_NAME_LENGTH] = {}; \
  216. if ( !s_chStaticArrayVarName[ 0 ] ) \
  217. { \
  218. wchar_t const * const kwszTheSuspect = g_pVGuiLocalize->Find( szLocToken ); \
  219. Assert( kwszTheSuspect ); \
  220. V_UnicodeToUTF8( kwszTheSuspect, s_chStaticArrayVarName, sizeof( s_chStaticArrayVarName ) ); \
  221. Assert( s_chStaticArrayVarName[ 0 ] ); \
  222. if ( !s_chStaticArrayVarName[ 0 ] ) \
  223. V_strcpy_safe( s_chStaticArrayVarName, szDefaultText ); \
  224. }
  225. class CStaticPlayerNamesSet
  226. {
  227. public:
  228. CStaticPlayerNamesSet()
  229. {
  230. m_bInitialized = false;
  231. Q_memset( m_szNames, 0, sizeof( m_szNames ) );
  232. }
  233. char const * GetName( int idx )
  234. {
  235. if ( ( idx < 0 ) || ( idx > MAX_PLAYERS ) )
  236. {
  237. STATIC_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( s_utf8LocalPlayer, "#SFUI_LocalPlayer", "Player" );
  238. return s_utf8LocalPlayer;
  239. }
  240. else
  241. {
  242. if ( !m_bInitialized )
  243. InitializeNames();
  244. return m_szNames[idx];
  245. }
  246. }
  247. protected:
  248. bool m_bInitialized;
  249. char const * m_szNames[MAX_PLAYERS+1];
  250. public:
  251. void InitializeNames()
  252. {
  253. CUtlVector< const char * > arrNamesOptions;
  254. #define RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( englishName ) { \
  255. STATIC_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( s_utf8Name##englishName, "#CSGO_FakePlayer_" #englishName, #englishName ); \
  256. arrNamesOptions.AddToTail( s_utf8Name##englishName ); }
  257. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Albatross );
  258. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Alpha );
  259. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Anchor );
  260. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Banjo );
  261. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Bell );
  262. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Beta );
  263. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Blackbird );
  264. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Bulldog );
  265. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Canary );
  266. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Cat );
  267. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Calf );
  268. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Cyclone );
  269. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Daisy );
  270. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Dalmatian );
  271. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Dart );
  272. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Delta );
  273. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Diamond );
  274. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Donkey );
  275. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Duck );
  276. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Emu );
  277. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Eclipse );
  278. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Flamingo );
  279. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Flute );
  280. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Frog );
  281. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Goose );
  282. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Hatchet );
  283. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Heron );
  284. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Husky );
  285. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Hurricane );
  286. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Iceberg );
  287. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Iguana );
  288. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Kiwi );
  289. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Kite );
  290. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Lamb );
  291. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Lily );
  292. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Macaw );
  293. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Manatee );
  294. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Maple );
  295. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Mask );
  296. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Nautilus );
  297. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Ostrich );
  298. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Octopus );
  299. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Pelican );
  300. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Puffin );
  301. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Pyramid );
  302. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Rattle );
  303. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Robin );
  304. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Rose );
  305. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Salmon );
  306. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Seal );
  307. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Shark );
  308. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Sheep );
  309. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Snake );
  310. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Sonar );
  311. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Stump );
  312. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Sparrow );
  313. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Toaster );
  314. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Toucan );
  315. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Torus );
  316. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Violet );
  317. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Vortex );
  318. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Vulture );
  319. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Wagon );
  320. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Whale );
  321. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Woodpecker );
  322. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Zebra );
  323. RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( Zigzag );
  324. #undef RANDOM_FAKE_PLAYER_NAME_UTF8_ARRAY_LOCALIZED
  325. for ( int k = 0; k < MAX_PLAYERS + 1; ++ k )
  326. {
  327. if ( !arrNamesOptions.Count() )
  328. Error( "Insufficient random names pool!\n" );
  329. int iRandomChoice = RandomInt( 0, arrNamesOptions.Count() - 1 );
  330. m_szNames[k] = arrNamesOptions.Element( iRandomChoice );
  331. arrNamesOptions.Remove( iRandomChoice );
  332. }
  333. m_bInitialized = true;
  334. }
  335. } g_staticPlayerNames;
  336. void EnsureStaticPlayerNamesReinitialized()
  337. {
  338. g_staticPlayerNames.InitializeNames();
  339. }
  340. //-----------------------------------------------------------------------------
  341. // Purpose:
  342. //-----------------------------------------------------------------------------
  343. const char *C_PlayerResource::GetPlayerName( int iIndex )
  344. {
  345. if ( cl_names_debug.GetInt() )
  346. return PLAYER_DEBUG_NAME;
  347. if ( iIndex < 1 || iIndex > MAX_PLAYERS )
  348. {
  349. Assert( false );
  350. return PLAYER_ERROR_NAME;
  351. }
  352. if ( !IsConnected( iIndex ) )
  353. return PLAYER_UNCONNECTED_NAME;
  354. // X360TBD: Network - figure out why the name isn't set
  355. if ( !m_szName[ iIndex ] || !Q_stricmp( m_szName[ iIndex ], PLAYER_UNCONNECTED_NAME ) )
  356. {
  357. // If you get a full "reset" uncompressed update from server, then you can have NULLNAME show up in the scoreboard
  358. UpdatePlayerName( iIndex );
  359. }
  360. if ( CDemoPlaybackParameters_t const *pParameters = engine->GetDemoPlaybackParameters() )
  361. {
  362. if ( pParameters->m_bAnonymousPlayerIdentity )
  363. {
  364. player_info_t sPlayerInfo;
  365. if ( pParameters->m_uiLockFirstPersonAccountID && engine->GetPlayerInfo( iIndex, &sPlayerInfo ) &&
  366. ( CSteamID( sPlayerInfo.xuid ).GetAccountID() == pParameters->m_uiLockFirstPersonAccountID ) )
  367. {
  368. STATIC_PLAYER_NAME_UTF8_ARRAY_LOCALIZED( s_utf8TheSuspect, "#CSGO_Overwatch_TheSuspect", "The Suspect" );
  369. return s_utf8TheSuspect;
  370. }
  371. else
  372. {
  373. return g_staticPlayerNames.GetName( iIndex );
  374. }
  375. }
  376. }
  377. // This gets updated in ClientThink, so it could be up to 1 second out of date, oh well.
  378. return m_szName[iIndex];
  379. }
  380. bool C_PlayerResource::IsAlive(int iIndex )
  381. {
  382. return m_bAlive[iIndex];
  383. }
  384. int C_PlayerResource::GetTeam(int iIndex )
  385. {
  386. if ( iIndex < 1 || iIndex > MAX_PLAYERS )
  387. {
  388. Assert( false );
  389. return 0;
  390. }
  391. else
  392. {
  393. return m_iTeam[iIndex];
  394. }
  395. }
  396. int C_PlayerResource::GetPendingTeam(int iIndex )
  397. {
  398. if ( iIndex < 1 || iIndex > MAX_PLAYERS )
  399. {
  400. Assert( false );
  401. return 0;
  402. }
  403. else
  404. {
  405. return m_iPendingTeam[iIndex];
  406. }
  407. }
  408. const char * C_PlayerResource::GetTeamName(int index)
  409. {
  410. C_Team *team = GetGlobalTeam( index );
  411. if ( !team )
  412. return "Unknown";
  413. return team->Get_Name();
  414. }
  415. int C_PlayerResource::GetCoachingTeam(int index)
  416. {
  417. if ( index < 1 || index > MAX_PLAYERS )
  418. {
  419. Assert( false );
  420. return 0;
  421. }
  422. else
  423. {
  424. return m_iCoachingTeam[index];
  425. }
  426. }
  427. int C_PlayerResource::GetTeamScore(int index)
  428. {
  429. C_Team *team = GetGlobalTeam( index );
  430. if ( !team )
  431. return 0;
  432. return team->Get_Score();
  433. }
  434. int C_PlayerResource::GetFrags(int index )
  435. {
  436. return 666;
  437. }
  438. bool C_PlayerResource::IsLocalPlayer(int index)
  439. {
  440. C_BasePlayer *pPlayer = C_BasePlayer::GetLocalPlayer();
  441. if ( !pPlayer )
  442. return false;
  443. // HLTV replay will not set m_bLocalPlayer flag, in a sense there's no selected local player, we're observing everyone
  444. if ( g_HltvReplaySystem.GetHltvReplayDelay() )
  445. return false;
  446. return ( index == pPlayer->entindex() );
  447. }
  448. bool C_PlayerResource::IsHLTV(int index)
  449. {
  450. if ( !IsConnected( index ) )
  451. return false;
  452. // HLTV replay will not set m_bLocalPlayer flag, in a sense there's no selected local player, we're observing everyone
  453. if ( g_HltvReplaySystem.GetHltvReplayDelay() && C_BasePlayer::GetLocalPlayer()->index == index )
  454. return true; // local player is always HLTV in HLTV replay mode, even though the hltv property isn't set because we are in the past and replaying everything as it was (including no hltv flag set)
  455. player_info_t sPlayerInfo;
  456. if ( engine->GetPlayerInfo( index, &sPlayerInfo ) )
  457. {
  458. return sPlayerInfo.ishltv;
  459. }
  460. return false;
  461. }
  462. #if defined( REPLAY_ENABLED )
  463. bool C_PlayerResource::IsReplay(int index)
  464. {
  465. if ( !IsConnected( index ) )
  466. return false;
  467. #if defined( REPLAY_ENABLED )
  468. player_info_t sPlayerInfo;
  469. if ( engine->GetPlayerInfo( index, &sPlayerInfo ) )
  470. {
  471. return sPlayerInfo.isreplay;
  472. }
  473. #endif
  474. return false;
  475. }
  476. #endif
  477. //-----------------------------------------------------------------------------
  478. // Purpose:
  479. //-----------------------------------------------------------------------------
  480. bool C_PlayerResource::IsFakePlayer( int iIndex )
  481. {
  482. if ( !IsConnected( iIndex ) )
  483. return false;
  484. // Yuck, make sure it's up to date
  485. player_info_t sPlayerInfo;
  486. if ( engine->GetPlayerInfo( iIndex, &sPlayerInfo ) )
  487. {
  488. return sPlayerInfo.fakeplayer;
  489. }
  490. return false;
  491. }
  492. //-----------------------------------------------------------------------------
  493. // Purpose:
  494. //-----------------------------------------------------------------------------
  495. int C_PlayerResource::GetPing( int iIndex )
  496. {
  497. if ( !IsConnected( iIndex ) )
  498. return 0;
  499. return m_iPing[iIndex];
  500. }
  501. //-----------------------------------------------------------------------------
  502. // Purpose:
  503. /*-----------------------------------------------------------------------------
  504. int C_PlayerResource::GetPacketloss( int iIndex )
  505. {
  506. if ( !IsConnected( iIndex ) )
  507. return 0;
  508. return m_iPacketloss[iIndex];
  509. }*/
  510. //-----------------------------------------------------------------------------
  511. // Purpose:
  512. //-----------------------------------------------------------------------------
  513. int C_PlayerResource::GetKills( int iIndex )
  514. {
  515. if ( !IsConnected( iIndex ) )
  516. return 0;
  517. return m_iKills[iIndex];
  518. }
  519. //-----------------------------------------------------------------------------
  520. // Purpose:
  521. //-----------------------------------------------------------------------------
  522. int C_PlayerResource::GetAssists( int iIndex )
  523. {
  524. if ( !IsConnected( iIndex ) )
  525. return 0;
  526. return m_iAssists[iIndex];
  527. }
  528. //-----------------------------------------------------------------------------
  529. // Purpose:
  530. //-----------------------------------------------------------------------------
  531. int C_PlayerResource::GetDeaths( int iIndex )
  532. {
  533. if ( !IsConnected( iIndex ) )
  534. return 0;
  535. return m_iDeaths[iIndex];
  536. }
  537. //-----------------------------------------------------------------------------
  538. // Purpose:
  539. //-----------------------------------------------------------------------------
  540. int C_PlayerResource::GetHealth( int iIndex )
  541. {
  542. if ( !IsConnected( iIndex ) )
  543. return 0;
  544. return m_iHealth[iIndex];
  545. }
  546. const Color &C_PlayerResource::GetTeamColor(int index )
  547. {
  548. if ( index < 0 || index >= MAX_TEAMS )
  549. {
  550. Assert( false );
  551. static Color blah;
  552. return blah;
  553. }
  554. else
  555. {
  556. return m_Colors[index];
  557. }
  558. }
  559. //-----------------------------------------------------------------------------
  560. // Purpose:
  561. //-----------------------------------------------------------------------------
  562. bool C_PlayerResource::IsConnected( int iIndex )
  563. {
  564. if ( iIndex < 1 || iIndex > MAX_PLAYERS )
  565. return false;
  566. else
  567. return m_bConnected[iIndex];
  568. }
  569. //-----------------------------------------------------------------------------
  570. // Purpose:
  571. //-----------------------------------------------------------------------------
  572. XUID C_PlayerResource::GetXuid( int iIndex )
  573. {
  574. if ( iIndex < 1 || iIndex > MAX_PLAYERS )
  575. return INVALID_XUID;
  576. else
  577. return m_Xuids[iIndex];
  578. }
  579. //-----------------------------------------------------------------------------
  580. // Purpose: Fills the given string with the player xuid.
  581. //-----------------------------------------------------------------------------
  582. void C_PlayerResource::FillXuidText( int iIndex, char *buf, int bufSize )
  583. {
  584. Assert( buf && bufSize );
  585. if ( buf && bufSize )
  586. {
  587. XUID xuid = GetXuid( iIndex );
  588. buf[0] = '\0';
  589. V_snprintf( buf, bufSize, "%llu", xuid );
  590. }
  591. }
  592. void C_PlayerResource::DeviceLost( void )
  593. {
  594. for ( int i = 1; i <= MAX_PLAYERS; i++ )
  595. {
  596. if ( m_Xuids[i] != INVALID_XUID )
  597. {
  598. g_pScaleformUI->AvatarImageRelease( m_Xuids[i] );
  599. m_Xuids[i] = INVALID_XUID;
  600. }
  601. }
  602. }
  603. void C_PlayerResource::DeviceReset( void *pDevice, void *pPresentParameters, void *pHWnd )
  604. {
  605. UpdateXuids();
  606. }