665 lines
18 KiB
C++
665 lines
18 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
//=======================================================================================//
|
|
|
|
#include "cl_replaymanager.h"
|
|
#include "replay/ienginereplay.h"
|
|
#include "replay/iclientreplay.h"
|
|
#include "replay/ireplaymoviemanager.h"
|
|
#include "replay/ireplayfactory.h"
|
|
#include "replay/replayutils.h"
|
|
#include "replay/ireplaymovierenderer.h"
|
|
#include "replay/shared_defs.h"
|
|
#include "baserecordingsession.h"
|
|
#include "cl_screenshotmanager.h"
|
|
#include "cl_recordingsession.h"
|
|
#include "cl_recordingsessionblock.h"
|
|
#include "replaysystem.h"
|
|
#include "cl_replaymoviemanager.h"
|
|
#include "replay_dbg.h"
|
|
#include "inetchannel.h"
|
|
#include "cl_replaycontext.h"
|
|
#include <time.h>
|
|
#include "vprof.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
//----------------------------------------------------------------------------------------
|
|
|
|
extern IEngineClientReplay *g_pEngineClient;
|
|
extern ConVar replay_postdeathrecordtime;
|
|
|
|
//----------------------------------------------------------------------------------------
|
|
|
|
#define REPLAY_INDEX_VERSION 0
|
|
|
|
//----------------------------------------------------------------------------------------
|
|
|
|
CReplayManager::CReplayManager()
|
|
: m_pPendingReplay( NULL ),
|
|
m_pReplayLastLife( NULL ),
|
|
m_pReplayThisLife( NULL ),
|
|
m_flPlayerSpawnCreateReplayFailTime( 0.0f )
|
|
{
|
|
}
|
|
|
|
CReplayManager::~CReplayManager()
|
|
{
|
|
}
|
|
|
|
bool CReplayManager::Init( CreateInterfaceFn fnCreateFactory )
|
|
{
|
|
// Get out if the user is running an unsupported mod or platform
|
|
if ( !g_pEngine->IsSupportedModAndPlatform() )
|
|
return false;
|
|
|
|
// Clear anything already loaded (since we reuse the same instance)
|
|
Clear();
|
|
|
|
// Register replay factory
|
|
m_pReplayFactory = GetReplayFactory( fnCreateFactory ); Assert( m_pReplayFactory );
|
|
|
|
// Load all replays from disk
|
|
if ( !BaseClass::Init() )
|
|
{
|
|
Warning( "Failed to load replay history!\n" );
|
|
}
|
|
|
|
// Session manager init'd by this point - go through and link up replays to sessions
|
|
CL_GetRecordingSessionManager()->OnReplaysLoaded();
|
|
|
|
return true;
|
|
}
|
|
|
|
void CReplayManager::Shutdown()
|
|
{
|
|
// Get out if the user is running an unsupported mod or platform
|
|
if ( !g_pEngine->IsSupportedModAndPlatform() )
|
|
return;
|
|
|
|
// Make sure we aren't waiting to write
|
|
BaseClass::Shutdown(); // Saves
|
|
}
|
|
|
|
IReplayFactory *CReplayManager::GetReplayFactory( CreateInterfaceFn fnCreateFactory )
|
|
{
|
|
return (IReplayFactory *)fnCreateFactory( INTERFACE_VERSION_REPLAY_FACTORY, NULL );
|
|
}
|
|
|
|
void CReplayManager::OnSessionStart()
|
|
{
|
|
// The pending replay doesn't exist yet at this point as far as I've seen, since the "replay_sessioninfo"
|
|
// event comes down a frame or more after the "replay_recording" replicated cvar is set to 1, which is
|
|
// what triggers AttemptToSetupNewReplay().
|
|
if ( !m_pPendingReplay )
|
|
{
|
|
AttemptToSetupNewReplay();
|
|
}
|
|
|
|
if ( m_pPendingReplay )
|
|
{
|
|
// Link up the pending replay to the recording session in progress
|
|
if ( m_pPendingReplay->m_hSession == REPLAY_HANDLE_INVALID )
|
|
{
|
|
ReplayHandle_t hSessionInProgress = CL_GetRecordingSessionManager()->GetRecordingSessionInProgress()->GetHandle();
|
|
m_pPendingReplay->m_hSession = hSessionInProgress;
|
|
}
|
|
|
|
// Make sure the spawn tick has the proper server start tick subtracted out
|
|
if ( m_pPendingReplay->m_nSpawnTick < 0 )
|
|
{
|
|
const int nServerStartTick = CL_GetRecordingSessionManager()->m_ServerRecordingState.m_nStartTick; Assert( nServerStartTick > 0 );
|
|
m_pPendingReplay->m_nSpawnTick = MAX( 0, -m_pPendingReplay->m_nSpawnTick - nServerStartTick );
|
|
}
|
|
}
|
|
}
|
|
|
|
void CReplayManager::OnSessionEnd()
|
|
{
|
|
// Complete the pending replay, if there is one
|
|
CompletePendingReplay();
|
|
}
|
|
|
|
const char *CReplayManager::GetRelativeIndexPath() const
|
|
{
|
|
return Replay_va( "%s%c", SUBDIR_REPLAYS, CORRECT_PATH_SEPARATOR );
|
|
}
|
|
|
|
CReplay *CReplayManager::Create()
|
|
{
|
|
return m_pReplayFactory->Create();
|
|
}
|
|
|
|
IReplayContext *CReplayManager::GetReplayContext() const
|
|
{
|
|
return g_pClientReplayContextInternal;
|
|
}
|
|
|
|
bool CReplayManager::ShouldLoadObj( const CReplay *pReplay ) const
|
|
{
|
|
return pReplay && pReplay->m_bComplete;
|
|
}
|
|
|
|
void CReplayManager::OnObjLoaded( CReplay *pReplay )
|
|
{
|
|
if ( !pReplay )
|
|
return;
|
|
|
|
pReplay->m_bSavedDuringThisSession = false;
|
|
}
|
|
|
|
int CReplayManager::GetVersion() const
|
|
{
|
|
return REPLAY_INDEX_VERSION;
|
|
}
|
|
|
|
void CReplayManager::ClearPendingReplay()
|
|
{
|
|
m_pPendingReplay = NULL;
|
|
}
|
|
|
|
void CReplayManager::SanityCheckReplay( CReplay *pReplay )
|
|
{
|
|
if ( !pReplay )
|
|
return;
|
|
|
|
// DEBUG: Make sure this replay does not already exist in the list
|
|
FOR_EACH_VEC( Replays(), i )
|
|
{
|
|
if ( Replays()[ i ]->GetHandle() == pReplay->GetHandle() )
|
|
{
|
|
IF_REPLAY_DBG( Warning( "Replay %i already found in history!\n", pReplay->GetHandle() ) );
|
|
}
|
|
}
|
|
|
|
if ( pReplay->m_nDeathTick < pReplay->m_nSpawnTick )
|
|
{
|
|
IF_REPLAY_DBG( Warning( "Spawn tick (%i) is greater than death tick (%i)!\n", pReplay->m_nSpawnTick, pReplay->m_nDeathTick ) );
|
|
}
|
|
}
|
|
|
|
void CReplayManager::SaveDanglingReplay()
|
|
{
|
|
if ( !m_pReplayThisLife )
|
|
return;
|
|
|
|
if ( m_pReplayThisLife->m_bRequestedByUser )
|
|
{
|
|
CompletePendingReplay();
|
|
FlagReplayForFlush( m_pReplayThisLife, false );
|
|
}
|
|
}
|
|
|
|
void CReplayManager::FreeLifeIfNotSaved( CReplay *&pReplay )
|
|
{
|
|
if ( pReplay )
|
|
{
|
|
if ( !pReplay->m_bSaved && !IsDirty( pReplay ) )
|
|
{
|
|
CleanupReplay( pReplay );
|
|
}
|
|
else
|
|
{
|
|
// If it's been saved, don't free the memory, just clear the pointer
|
|
pReplay = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CReplayManager::CleanupReplay( CReplay *&pReplay )
|
|
{
|
|
if ( !pReplay )
|
|
return;
|
|
|
|
// Get rid of a replay that was never committed:
|
|
// Remove screenshots taken
|
|
CL_GetScreenshotManager()->DeleteScreenshotsForReplay( pReplay );
|
|
|
|
// Free
|
|
delete pReplay;
|
|
pReplay = NULL;
|
|
}
|
|
|
|
void CReplayManager::OnReplayRecordingCvarChanged()
|
|
{
|
|
DBG( "OnReplayRecordingCvarChanged()\n" );
|
|
|
|
// If set to 0, get out - we don't care
|
|
extern ConVar replay_recording;
|
|
if ( !replay_recording.GetBool() )
|
|
{
|
|
DBG( " replay_recording is false...aborting\n" );
|
|
return;
|
|
}
|
|
|
|
// If OnPlayerSpawn() hasn't failed to create the scratch replay, get out
|
|
if ( m_flPlayerSpawnCreateReplayFailTime == 0.0f )
|
|
{
|
|
DBG( " m_flPlayerSpawnCreateReplayFailTime == 0.0f...aborting.\n" );
|
|
return;
|
|
}
|
|
|
|
DBG( " Calling AttemptToSetupNewReplay()\n" );
|
|
|
|
// Try to create & setup again
|
|
AttemptToSetupNewReplay();
|
|
|
|
// Reset
|
|
m_flPlayerSpawnCreateReplayFailTime = 0.0f;
|
|
}
|
|
|
|
void CReplayManager::OnClientSideDisconnect()
|
|
{
|
|
SaveDanglingReplay();
|
|
ClearPendingReplay();
|
|
|
|
FreeLifeIfNotSaved( m_pReplayLastLife );
|
|
FreeLifeIfNotSaved( m_pReplayThisLife );
|
|
|
|
m_flPlayerSpawnCreateReplayFailTime = 0.0f;
|
|
}
|
|
|
|
void CReplayManager::CommitPendingReplayAndBeginDownload()
|
|
{
|
|
// Update the last session block we should download
|
|
CClientRecordingSession *pSession = CL_CastSession( CL_GetRecordingSessionManager()->FindSession( m_pReplayThisLife->m_hSession ) );
|
|
const int iPostDeathBlockIndex = pSession->UpdateLastBlockToDownload();
|
|
|
|
// Update the # of blocks required to reconstruct the replay
|
|
m_pReplayThisLife->m_iMaxSessionBlockRequired = iPostDeathBlockIndex;
|
|
|
|
Commit( m_pReplayThisLife );
|
|
}
|
|
|
|
void CReplayManager::CompletePendingReplay()
|
|
{
|
|
// Get out if no pending replay
|
|
if ( !m_pPendingReplay )
|
|
return;
|
|
|
|
// Get session associated w/ the replay
|
|
CBaseRecordingSession *pSession = CL_GetRecordingSessionManager()->FindSession( m_pPendingReplay->m_hSession );
|
|
|
|
// Sometimes the session isn't valid here, like when we're first joining a server
|
|
if ( !pSession )
|
|
return;
|
|
|
|
Assert( pSession->m_nServerStartRecordTick >= 0 );
|
|
|
|
// Cache death tick
|
|
m_pPendingReplay->m_nDeathTick = g_pEngineClient->GetLastServerTickTime() - pSession->m_nServerStartRecordTick;
|
|
|
|
SanityCheckReplay( m_pPendingReplay );
|
|
|
|
// Calc replay length
|
|
m_pPendingReplay->m_flLength = g_pEngine->TicksToTime(
|
|
m_pPendingReplay->m_nDeathTick -
|
|
m_pPendingReplay->m_nSpawnTick +
|
|
g_pEngine->TimeToTicks( replay_postdeathrecordtime.GetFloat() )
|
|
);
|
|
|
|
// Cache player slot so we can start playback of replays from recorder player's perspective
|
|
m_pPendingReplay->m_nPlayerSlot = g_pEngineClient->GetPlayerSlot() + 1;
|
|
|
|
// Setup status
|
|
m_pPendingReplay->m_nStatus = CReplay::REPLAYSTATUS_DOWNLOADPHASE;
|
|
|
|
// The replay is now "complete," ie has all the data needed
|
|
m_pPendingReplay->m_bComplete = true;
|
|
|
|
// Let derived classes do whatever it wants
|
|
m_pPendingReplay->OnComplete();
|
|
|
|
// If the replay was requested by the user already, update the # of blocks we should download & commit the replay
|
|
if ( m_pPendingReplay->m_bRequestedByUser )
|
|
{
|
|
#ifdef DBGFLAG_ASSERT
|
|
Assert( m_pReplayThisLife->m_bComplete );
|
|
#endif
|
|
CommitPendingReplayAndBeginDownload();
|
|
}
|
|
|
|
// Before we copy the pointer to "this life," end recording so the replay can do any cleanup (eg listening for game events)
|
|
m_pPendingReplay->OnEndRecording();
|
|
|
|
// Cache off scratch replay to "this life"
|
|
m_pReplayThisLife = m_pPendingReplay;
|
|
|
|
ClearPendingReplay();
|
|
}
|
|
|
|
bool CReplayManager::Commit( CReplay *pNewReplay )
|
|
{
|
|
if ( !g_pClientReplayContextInternal->IsInitialized() || !pNewReplay )
|
|
return false;
|
|
|
|
SanityCheckReplay( pNewReplay );
|
|
|
|
// NOTE: Marks index as dirty, as well as pNewReplay
|
|
Add( pNewReplay );
|
|
|
|
// Save now
|
|
Save();
|
|
|
|
return true;
|
|
}
|
|
|
|
//
|
|
// IReplayManager implementation
|
|
//
|
|
CReplay *CReplayManager::GetReplay( ReplayHandle_t hReplay )
|
|
{
|
|
if ( m_pReplayThisLife && m_pReplayThisLife->GetHandle() == hReplay )
|
|
return m_pReplayThisLife;
|
|
|
|
return Find( hReplay );
|
|
}
|
|
|
|
void CReplayManager::DeleteReplay( ReplayHandle_t hReplay, bool bNotifyUI )
|
|
{
|
|
CReplay *pReplay = GetReplay( hReplay ); Assert( pReplay );
|
|
|
|
// The session manager will delete the .dem, the session .dmx and remove the session
|
|
// item itself if this is the last replay associated with it.
|
|
CL_GetRecordingSessionManager()->OnReplayDeleted( pReplay );
|
|
|
|
// Notify the replay browser if necessary
|
|
if ( bNotifyUI )
|
|
{
|
|
extern IClientReplay *g_pClient;
|
|
g_pClient->OnDeleteReplay( hReplay );
|
|
}
|
|
|
|
// Remove it
|
|
Remove( pReplay );
|
|
|
|
// If the replay deleted was just saved and we haven't respawned yet,
|
|
// we need to clear out some stuff so GetReplay() doesn't crash.
|
|
if ( m_pReplayThisLife == pReplay )
|
|
{
|
|
m_pReplayThisLife = NULL;
|
|
m_pPendingReplay = NULL;
|
|
}
|
|
|
|
if ( m_pReplayLastLife == pReplay )
|
|
{
|
|
m_pReplayLastLife = NULL;
|
|
}
|
|
}
|
|
|
|
void CReplayManager::FlagReplayForFlush( CReplay *pReplay, bool bForceImmediate )
|
|
{
|
|
FlagForFlush( pReplay, bForceImmediate );
|
|
}
|
|
|
|
int CReplayManager::GetUnrenderedReplayCount()
|
|
{
|
|
int nCount = 0;
|
|
FOR_EACH_VEC( m_vecObjs, i )
|
|
{
|
|
CReplay *pCurReplay = m_vecObjs[ i ];
|
|
if ( !pCurReplay->m_bRendered &&
|
|
pCurReplay->m_nStatus == CReplay::REPLAYSTATUS_READYTOCONVERT )
|
|
{
|
|
++nCount;
|
|
}
|
|
}
|
|
return nCount;
|
|
}
|
|
|
|
void CReplayManager::InitReplay( CReplay *pReplay )
|
|
{
|
|
// Setup record time right now
|
|
pReplay->m_RecordTime.InitDateAndTimeToNow();
|
|
|
|
// Store start time
|
|
pReplay->m_flStartTime = g_pEngine->GetHostTime();
|
|
|
|
// Get map name (w/o the path)
|
|
V_FileBase( g_pEngineClient->GetLevelName(), m_pPendingReplay->m_szMapName, sizeof( m_pPendingReplay->m_szMapName ) );
|
|
|
|
// Give the replay a default name
|
|
pReplay->AutoNameTitleIfEmpty();
|
|
}
|
|
|
|
CReplay *CReplayManager::CreatePendingReplay()
|
|
{
|
|
Assert( m_pPendingReplay == NULL );
|
|
m_pPendingReplay = CreateAndGenerateHandle();
|
|
|
|
// If we've already begun recording, link to the session now, otherwise link once
|
|
// we start recording.
|
|
CBaseRecordingSession *pSessionInProgress = CL_GetRecordingSessionInProgress();
|
|
if ( pSessionInProgress )
|
|
{
|
|
m_pPendingReplay->m_hSession = pSessionInProgress->GetHandle();
|
|
}
|
|
|
|
InitReplay( m_pPendingReplay );
|
|
|
|
// Setup replay handle for screenshots
|
|
CL_GetScreenshotManager()->SetScreenshotReplay( m_pPendingReplay->GetHandle() );
|
|
|
|
return m_pPendingReplay;
|
|
}
|
|
|
|
void CReplayManager::AttemptToSetupNewReplay()
|
|
{
|
|
DBG( "AttemptToSetupNewReplay()\n" );
|
|
|
|
if ( !g_pReplay->IsRecording() || g_pEngineClient->IsPlayingReplayDemo() )
|
|
{
|
|
DBG( " Aborting...not recording, or playing back replay.\n" );
|
|
m_flPlayerSpawnCreateReplayFailTime = g_pEngine->GetHostTime();
|
|
return;
|
|
}
|
|
|
|
// Create the replay if necessary - we only do setup if we're creating
|
|
// a new replay, because on a full update this function may be called
|
|
// even though we're not actually spawning.
|
|
if ( !m_pPendingReplay )
|
|
{
|
|
DBG( " Creating new replay...\n" );
|
|
|
|
// If there is a "last life" replay that was not saved already, delete it
|
|
FreeLifeIfNotSaved( m_pReplayLastLife );
|
|
|
|
// Cache last life
|
|
m_pReplayLastLife = m_pReplayThisLife;
|
|
|
|
// Create the scratch replay (sets m_pPendingReplay and returns it)
|
|
CReplay *pPendingReplay = CreatePendingReplay();
|
|
|
|
SanityCheckReplay( pPendingReplay );
|
|
|
|
// "This life" is the scratch replay
|
|
m_pReplayThisLife = pPendingReplay;
|
|
|
|
// Setup spawn tick
|
|
const int nServerStartTick = CL_GetRecordingSessionManager()->m_ServerRecordingState.m_nStartTick;
|
|
pPendingReplay->m_nSpawnTick = g_pEngineClient->GetLastServerTickTime() - nServerStartTick;
|
|
if ( nServerStartTick == 0 )
|
|
{
|
|
// Didn't receive the replay_sessioninfo event yet - make spawn tick negative so when the
|
|
// event IS received, we can detect that the server start tick still needs to be subtracted.
|
|
pPendingReplay->m_nSpawnTick *= -1;
|
|
}
|
|
|
|
// Setup post-death record time
|
|
extern ConVar replay_postdeathrecordtime;
|
|
pPendingReplay->m_nPostDeathRecordTime = replay_postdeathrecordtime.GetFloat();
|
|
|
|
// Let the replay know we're recording
|
|
pPendingReplay->OnBeginRecording();
|
|
}
|
|
else
|
|
{
|
|
DBG( " NOT creating new replay.\n" );
|
|
}
|
|
|
|
// Served its purpose
|
|
m_flPlayerSpawnCreateReplayFailTime = 0.0f;
|
|
}
|
|
|
|
void CReplayManager::Think()
|
|
{
|
|
VPROF_BUDGET( "CReplayManager::Think", VPROF_BUDGETGROUP_REPLAY );
|
|
|
|
BaseClass::Think();
|
|
|
|
DebugThink();
|
|
|
|
// Only update pending replay, since it's recording
|
|
// NOTE: we use Sys_FloatTime() here, since the client sets the next update time with engine->Time(),
|
|
// which also uses Sys_FloatTime().
|
|
if ( m_pPendingReplay && m_pPendingReplay->m_flNextUpdateTime <= Sys_FloatTime() )
|
|
{
|
|
m_pPendingReplay->Update(); // Allow the replay's Update() function to set the next update time
|
|
}
|
|
}
|
|
|
|
void CReplayManager::DebugThink()
|
|
{
|
|
// Debugging
|
|
if ( replay_debug.GetBool() )
|
|
{
|
|
const char *pReplayNames[] = { "Scratch", "This life", "Last life" };
|
|
CReplay *pReplays[] = { m_pPendingReplay, m_pReplayThisLife, m_pReplayLastLife };
|
|
for ( int i = 0; i < 3; ++i )
|
|
{
|
|
CReplay *pCurReplay = pReplays[ i ];
|
|
if ( !pCurReplay )
|
|
{
|
|
g_pEngineClient->Con_NPrintf( i, "%s: NULL", pReplayNames[ i ] );
|
|
continue;
|
|
}
|
|
|
|
g_pEngineClient->Con_NPrintf( i, "%s: handle=%i [%i, %i] C? %s R? %s MaxBlock: %i", pReplayNames[ i ],
|
|
pCurReplay->GetHandle(), pCurReplay->m_nSpawnTick,
|
|
pCurReplay->m_nDeathTick, pCurReplay->m_bComplete ? "YES" : "NO",
|
|
pCurReplay->m_bRequestedByUser ? "YES" : "NO",
|
|
pCurReplay->m_iMaxSessionBlockRequired
|
|
);
|
|
|
|
// Screenshot handle
|
|
int nCurLine = 5;
|
|
g_pEngineClient->Con_NPrintf( nCurLine, "Screenshot replay: handle=%i", CL_GetScreenshotManager()->GetScreenshotReplay() );
|
|
nCurLine += 2;
|
|
|
|
// Saved replay handles
|
|
g_pEngineClient->Con_NPrintf( nCurLine++, "REPLAYS:" );
|
|
FOR_EACH_REPLAY( j )
|
|
{
|
|
CReplay *pReplay = GET_REPLAY_AT( j );
|
|
g_pEngineClient->Con_NPrintf( nCurLine++, "%i: handle=%i ticks=[%i %i]", i, pReplay->GetHandle(),
|
|
pReplay->m_nSpawnTick, pReplay->m_nDeathTick );
|
|
}
|
|
|
|
// Current tick:
|
|
g_pEngineClient->Con_NPrintf( ++nCurLine, "MAIN tick: %f", g_pEngineClient->GetLastServerTickTime() );
|
|
g_pEngineClient->Con_NPrintf( ++nCurLine, " server tick: %f", g_pEngineClient->GetLastServerTickTime() );
|
|
nCurLine += 2;
|
|
}
|
|
}
|
|
}
|
|
|
|
float CReplayManager::GetNextThinkTime() const
|
|
{
|
|
return g_pEngine->GetHostTime() + 0.1f;
|
|
}
|
|
|
|
CReplay *CReplayManager::GetPlayingReplay()
|
|
{
|
|
return g_pReplayDemoPlayer->GetCurrentReplay();
|
|
}
|
|
|
|
CReplay *CReplayManager::GetReplayForCurrentLife()
|
|
{
|
|
return m_pReplayThisLife;
|
|
}
|
|
|
|
void CReplayManager::GetReplays( CUtlLinkedList< CReplay *, int > &lstReplays )
|
|
{
|
|
lstReplays.RemoveAll();
|
|
FOR_EACH_REPLAY( i )
|
|
{
|
|
lstReplays.AddToTail( GET_REPLAY_AT( i ) );
|
|
}
|
|
}
|
|
|
|
void CReplayManager::GetReplaysAsQueryableItems( CUtlLinkedList< IQueryableReplayItem *, int > &lstReplays )
|
|
{
|
|
lstReplays.RemoveAll();
|
|
FOR_EACH_REPLAY( i )
|
|
{
|
|
lstReplays.AddToHead( dynamic_cast< IQueryableReplayItem * >( GET_REPLAY_AT( i ) ) );
|
|
}
|
|
|
|
if ( m_pPendingReplay &&
|
|
!m_pPendingReplay->m_bComplete &&
|
|
m_pPendingReplay->m_bRequestedByUser )
|
|
{
|
|
Assert( lstReplays.Find( m_pPendingReplay ) == lstReplays.InvalidIndex() );
|
|
lstReplays.AddToHead( m_pPendingReplay );
|
|
}
|
|
}
|
|
|
|
int CReplayManager::GetNumReplaysDependentOnSession( ReplayHandle_t hSession )
|
|
{
|
|
int nResult = 0;
|
|
FOR_EACH_REPLAY( i )
|
|
{
|
|
CReplay *pCurReplay = GET_REPLAY_AT( i );
|
|
if ( pCurReplay->m_hSession == hSession )
|
|
{
|
|
++nResult;
|
|
}
|
|
}
|
|
return nResult;
|
|
}
|
|
|
|
const char *CReplayManager::GetReplaysDir() const
|
|
{
|
|
return GetIndexPath();
|
|
}
|
|
|
|
float CReplayManager::GetDownloadProgress( const CReplay *pReplay )
|
|
{
|
|
// Give each downloadable session block equal weight since we won't know the size of blocks that
|
|
// have not been created/written yet on the server.
|
|
|
|
// Go through all blocks in the replay and figure out how many bytes have been downloaded
|
|
float flSum = 0.0f;
|
|
|
|
CClientRecordingSession *pSession = CL_CastSession( CL_GetRecordingSessionManager()->FindSession( pReplay->m_hSession ) ); Assert( pSession );
|
|
if ( !pSession )
|
|
return 0.0f;
|
|
|
|
const CBaseRecordingSession::BlockContainer_t &vecBlocks = pSession->GetBlocks();
|
|
FOR_EACH_VEC( vecBlocks, i )
|
|
{
|
|
CClientRecordingSessionBlock *pCurBlock = CL_CastBlock( vecBlocks[ i ] );
|
|
if ( !pReplay->IsSignificantBlock( pCurBlock->m_iReconstruction ) )
|
|
continue;
|
|
|
|
// Calculate progress for this block
|
|
Assert( pCurBlock->m_uFileSize > 0 );
|
|
const float flSubProgress = pCurBlock->m_uFileSize == 0 ? 0.0f : clamp( (float)pCurBlock->m_uBytesDownloaded / pCurBlock->m_uFileSize, 0.0f, 1.0f );
|
|
|
|
flSum += flSubProgress;
|
|
}
|
|
|
|
// Account for blocks that haven't been created yet
|
|
// NOTE: This will cause a bug in download progress if the round ends and cuts the number of
|
|
// expected blocks down - but that situation is probably less likely to occur than the situation
|
|
// where a client is expecting more blocks that *will* be created. To avoid pops in the latter
|
|
// situation, we account for those blocks here.
|
|
const int nTotalSubBlocks = pReplay->m_iMaxSessionBlockRequired + 1;
|
|
|
|
// Calculate mean
|
|
Assert( nTotalSubBlocks > 0 );
|
|
return nTotalSubBlocks == 0 ? 0.0f : ( flSum / (float)nTotalSubBlocks );
|
|
}
|
|
|
|
//----------------------------------------------------------------------------------------
|