mirror of
https://github.com/nillerusr/source-engine.git
synced 2025-01-05 06:56:41 +00:00
602 lines
18 KiB
C++
602 lines
18 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
//=====================================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "filesystem.h"
|
|
#include "gcsdk/webapi_response.h"
|
|
#include "c_tf_player.h"
|
|
#include "tf_weapon_medigun.h"
|
|
#include "tf_demo_support.h"
|
|
#include "tf_gamerules.h"
|
|
#include "tf_hud_chat.h"
|
|
#include "vguicenterprint.h"
|
|
|
|
// Global singleton
|
|
static CTFDemoSupport g_DemoSupport;
|
|
|
|
extern ConVar mp_tournament;
|
|
|
|
ConVar ds_enable( "ds_enable", "0", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - enable automatic .dem file recording and features. 0 - Manual, 1 - Auto-record competitive matches, 2 - Auto-record all matches, 3 - Auto-record tournament (mp_tournament) matches", true, 0, true, 3 );
|
|
ConVar ds_dir( "ds_dir", "demos", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - will put all files into this folder under the gamedir. 24 characters max." );
|
|
ConVar ds_prefix( "ds_prefix", "", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - will prefix files with this string. 24 characters max." );
|
|
ConVar ds_min_streak( "ds_min_streak", "4", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - minimum kill streak count before being recorded.", true, 2, false, 0 );
|
|
ConVar ds_kill_delay( "ds_kill_delay", "15", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - maximum time between kills for tracking kill streaks.", true, 5, false, 0 );
|
|
ConVar ds_log( "ds_log", "1", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - log kill streak and bookmark events to an associated .txt file.", true, 0, true, 1 );
|
|
ConVar ds_sound( "ds_sound", "1", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - play start/stop sound for demo recording.", true, 0, true, 1 );
|
|
ConVar ds_notify( "ds_notify", "0", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - text output when recording start/stop/bookmark events : 0 - console, 1 - console and chat, 2 - console and HUD.", true, 0, true, 2 );
|
|
ConVar ds_screens( "ds_screens", "1", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - take screenshot of the scoreboard for non-competitive matches or the match summary stats for competitive matches. For competitive matches, it will not capture the screenshot if you disconnect from the server before the medal awards have completed.", true, 0, true, 1 );
|
|
ConVar ds_autodelete( "ds_autodelete", "0", FCVAR_CLIENTDLL | FCVAR_DONTRECORD | FCVAR_ARCHIVE, "Demo support - automatically delete .dem files with no associated bookmark or kill streak events.", true, 0, true, 1 );
|
|
|
|
CON_COMMAND_F( ds_mark, "Demo support - bookmark (with optional single-word description) the current tick count for the demo being recorded.", FCVAR_CLIENTDLL | FCVAR_DONTRECORD )
|
|
{
|
|
g_DemoSupport.BookMarkCurrentTick( ( args.ArgC() > 1 ) ? args[1] : NULL );
|
|
}
|
|
|
|
CON_COMMAND_F( ds_record, "Demo support - start recording a demo.", FCVAR_CLIENTDLL | FCVAR_DONTRECORD )
|
|
{
|
|
g_DemoSupport.StartRecording();
|
|
}
|
|
|
|
CON_COMMAND_F( ds_stop, "Demo support - stop recording a demo.", FCVAR_CLIENTDLL | FCVAR_DONTRECORD )
|
|
{
|
|
g_DemoSupport.StopRecording();
|
|
}
|
|
|
|
CON_COMMAND_F( ds_status, "Demo support - show the current recording status.", FCVAR_CLIENTDLL | FCVAR_DONTRECORD )
|
|
{
|
|
g_DemoSupport.Status();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
static const char *g_aDemoEventNames[] =
|
|
{
|
|
"Bookmark",
|
|
"Killstreak",
|
|
};
|
|
COMPILE_TIME_ASSERT( ARRAYSIZE( g_aDemoEventNames ) == eDemoEvent_Last );
|
|
|
|
const char *GetDemoEventName( EDemoEventType eEventType )
|
|
{
|
|
if ( ( eEventType >= ARRAYSIZE( g_aDemoEventNames ) ) || ( eEventType < 0 ) )
|
|
return NULL;
|
|
|
|
return g_aDemoEventNames[eEventType];
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
CTFDemoSupport::CTFDemoSupport() : CAutoGameSystemPerFrame( "CTFDemoSupport" )
|
|
{
|
|
m_nKillCount = 0;
|
|
m_flLastKill = -1.f;
|
|
m_bRecording = false;
|
|
m_hGlobalEventList = FILESYSTEM_INVALID_HANDLE;
|
|
m_DemoSpecificEventList.Clear();
|
|
m_DemoSpecificEventList.SetStatusCode( k_EHTTPStatusCode200OK );
|
|
m_pRoot = NULL;
|
|
m_pChildArray = NULL;
|
|
m_flScreenshotTime = -1.f;
|
|
m_bAlreadyAutoRecordedOnce = false;
|
|
m_flNextRecordStartCheckTime = -1.f;
|
|
m_bFirstEvent = false;
|
|
m_nStartingTickCount = 0;
|
|
m_bHasAtLeastOneEvent = false;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
bool CTFDemoSupport::Init()
|
|
{
|
|
ListenForGameEvent( "localplayer_respawn" );
|
|
ListenForGameEvent( "player_death" );
|
|
ListenForGameEvent( "client_disconnect" );
|
|
ListenForGameEvent( "ds_screenshot" );
|
|
ListenForGameEvent( "ds_stop" );
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::LevelInitPostEntity()
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
m_bAlreadyAutoRecordedOnce = false;
|
|
m_flNextRecordStartCheckTime = -1.f;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::LevelShutdownPostEntity()
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
StopRecording();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::Update( float frametime )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
if ( ds_enable.GetInt() > 0 )
|
|
{
|
|
if ( !m_bRecording && !m_bAlreadyAutoRecordedOnce )
|
|
{
|
|
if ( ds_enable.GetInt() == 1 )
|
|
{
|
|
if ( TFGameRules() && !TFGameRules()->IsCompetitiveMode() )
|
|
return;
|
|
}
|
|
else if ( ds_enable.GetInt() == 3 )
|
|
{
|
|
if ( !mp_tournament.GetBool() )
|
|
return;
|
|
}
|
|
|
|
if ( ( m_flNextRecordStartCheckTime < 0 ) || ( m_flNextRecordStartCheckTime < gpGlobals->curtime ) )
|
|
{
|
|
CTFPlayer *pLocalPlayer = CTFPlayer::GetLocalTFPlayer();
|
|
if ( pLocalPlayer )
|
|
{
|
|
// if the local player is on team spectator or is on a game team and has picked a player class
|
|
if ( ( pLocalPlayer->GetTeamNumber() == TEAM_SPECTATOR ) ||
|
|
( ( pLocalPlayer->GetTeamNumber() >= FIRST_GAME_TEAM ) && pLocalPlayer->GetPlayerClass() && ( pLocalPlayer->GetPlayerClass()->GetClassIndex() >= TF_FIRST_NORMAL_CLASS ) && ( pLocalPlayer->GetPlayerClass()->GetClassIndex() < TF_LAST_NORMAL_CLASS ) ) )
|
|
{
|
|
if ( !StartRecording() )
|
|
{
|
|
// we'll try again in 5 seconds
|
|
m_flNextRecordStartCheckTime = gpGlobals->curtime + 5.f;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( !m_bRecording )
|
|
return;
|
|
|
|
if ( ( m_flScreenshotTime > 0 ) && ( m_flScreenshotTime < gpGlobals->curtime ) )
|
|
{
|
|
m_flScreenshotTime = -1.f;
|
|
|
|
if ( ds_screens.GetBool() )
|
|
{
|
|
engine->TakeScreenshot( m_szFilename, m_szFolder );
|
|
Notify( "(Demo Support) Screenshot saved\n" );
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::Status( void )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
char szStatus[64] = {0};
|
|
|
|
if ( m_bRecording )
|
|
{
|
|
V_sprintf_safe( szStatus, "(Demo Support) Currently recording to %s\n", m_szFolderAndFilename );
|
|
}
|
|
else
|
|
{
|
|
V_strcpy_safe( szStatus, "(Demo Support) Not currently recording\n" );
|
|
}
|
|
|
|
Notify( szStatus );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::Notify( char *pszMessage )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
if ( pszMessage && pszMessage[0] )
|
|
{
|
|
// we'll always put the message in the console
|
|
Msg( "%s", pszMessage );
|
|
|
|
switch ( ds_notify.GetInt() )
|
|
{
|
|
default: // console
|
|
case 0:
|
|
break;
|
|
case 1: // chat window
|
|
{
|
|
CBaseHudChat *pHUDChat = (CBaseHudChat *)GET_HUDELEMENT( CHudChat );
|
|
if ( pHUDChat )
|
|
{
|
|
pHUDChat->Printf( CHAT_FILTER_NONE, "%s", pszMessage );
|
|
}
|
|
}
|
|
break;
|
|
case 2: // hud center print
|
|
internalCenterPrint->Print( pszMessage );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::LogEvent( EDemoEventType eType, int nValue /* = 0 */, const char *pszValue /* = NULL */ )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
if ( !m_bRecording )
|
|
return;
|
|
|
|
if ( !ds_log.GetBool() )
|
|
return;
|
|
|
|
char szArg[32] = {0};
|
|
|
|
switch ( eType )
|
|
{
|
|
case eDemoEvent_Bookmark:
|
|
{
|
|
const char *pszTemp = "General";
|
|
if ( pszValue && pszValue[0] )
|
|
{
|
|
pszTemp = pszValue;
|
|
}
|
|
|
|
V_strcpy_safe( szArg, pszTemp );
|
|
}
|
|
break;
|
|
case eDemoEvent_Killstreak:
|
|
V_sprintf_safe( szArg, "%d", nValue );
|
|
break;
|
|
default:
|
|
// don't continue if this is an unknown type
|
|
return;
|
|
}
|
|
|
|
time_t tTime = CRTime::RTime32TimeCur();
|
|
struct tm tmStruct;
|
|
struct tm *ptm = Plat_localtime( &tTime, &tmStruct );
|
|
|
|
int nTickCount = gpGlobals->tickcount - m_nStartingTickCount;
|
|
|
|
if ( m_hGlobalEventList != FILESYSTEM_INVALID_HANDLE )
|
|
{
|
|
if ( m_bFirstEvent )
|
|
{
|
|
m_bFirstEvent = false;
|
|
g_pFullFileSystem->FPrintf( m_hGlobalEventList, ">\n" );
|
|
}
|
|
|
|
g_pFullFileSystem->FPrintf( m_hGlobalEventList, "[%04u/%02u/%02u %02u:%02u] %s %s (\"%s\" at %d)\n",
|
|
ptm->tm_year + 1900, ptm->tm_mon + 1, ptm->tm_mday, ptm->tm_hour, ptm->tm_min, g_aDemoEventNames[eType], szArg, m_szFilename, nTickCount );
|
|
}
|
|
|
|
if ( m_pChildArray )
|
|
{
|
|
GCSDK::CWebAPIValues *pChildObject = m_pChildArray->AddChildObjectToArray();
|
|
pChildObject->CreateChildObject( "name" )->SetStringValue( g_aDemoEventNames[eType] );
|
|
pChildObject->CreateChildObject( "value" )->SetStringValue( szArg );
|
|
pChildObject->CreateChildObject( "tick" )->SetInt32Value( nTickCount );
|
|
m_bHasAtLeastOneEvent = true;
|
|
}
|
|
|
|
char szMessage[MAX_PATH] = {0};
|
|
V_sprintf_safe( szMessage, "(Demo Support) Event recorded: %s %s\n", g_aDemoEventNames[eType], szArg );
|
|
Notify( szMessage );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::FireGameEvent( IGameEvent * event )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
if ( !m_bRecording )
|
|
return;
|
|
|
|
const char *pszEvent = event->GetName();
|
|
|
|
if ( FStrEq( pszEvent, "localplayer_respawn" ) )
|
|
{
|
|
m_nKillCount = 0;
|
|
m_flLastKill = -1.f;
|
|
}
|
|
else if ( FStrEq( pszEvent, "player_death" ) )
|
|
{
|
|
if ( m_bRecording )
|
|
{
|
|
CBasePlayer *pLocalPlayer = CBasePlayer::GetLocalPlayer();
|
|
if ( !pLocalPlayer )
|
|
return;
|
|
|
|
int nOldKillCount = m_nKillCount;
|
|
int nLocalPlayerUserID = pLocalPlayer->GetUserID();
|
|
|
|
if ( nLocalPlayerUserID == event->GetInt( "userid" ) )
|
|
{
|
|
// local player was the victim
|
|
m_nKillCount = 0;
|
|
}
|
|
else if ( nLocalPlayerUserID == event->GetInt( "attacker" ) )
|
|
{
|
|
// local player was the killer
|
|
if ( ( m_flLastKill < 0 ) || ( gpGlobals->curtime - m_flLastKill > ds_kill_delay.GetFloat() ) )
|
|
{
|
|
m_nKillCount = 1;
|
|
}
|
|
else
|
|
{
|
|
m_nKillCount++;
|
|
}
|
|
|
|
m_flLastKill = gpGlobals->curtime;
|
|
}
|
|
else if ( nLocalPlayerUserID == event->GetInt( "assister" ) )
|
|
{
|
|
CTFPlayer *pTFPlayer = ToTFPlayer( pLocalPlayer );
|
|
if ( pTFPlayer )
|
|
{
|
|
if ( pTFPlayer->IsPlayerClass( TF_CLASS_MEDIC ) && pLocalPlayer->GetActiveWeapon() )
|
|
{
|
|
CWeaponMedigun *pMedigun = dynamic_cast<CWeaponMedigun*>( pLocalPlayer->GetActiveWeapon() );
|
|
if ( pMedigun )
|
|
{
|
|
// local player was a medic healing the attacker so give them credit
|
|
if ( ( m_flLastKill < 0 ) || ( gpGlobals->curtime - m_flLastKill > ds_kill_delay.GetFloat() ) )
|
|
{
|
|
m_nKillCount = 1;
|
|
}
|
|
else
|
|
{
|
|
m_nKillCount++;
|
|
}
|
|
|
|
m_flLastKill = gpGlobals->curtime;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// if our kill-streak has increased, make an event entry
|
|
if ( ( nOldKillCount != m_nKillCount ) && ( m_nKillCount > 0 ) && ( m_nKillCount >= ds_min_streak.GetInt() ) )
|
|
{
|
|
LogEvent( eDemoEvent_Killstreak, m_nKillCount );
|
|
}
|
|
}
|
|
}
|
|
else if ( FStrEq( pszEvent, "client_disconnect" ) )
|
|
{
|
|
StopRecording();
|
|
}
|
|
else if ( FStrEq( pszEvent, "ds_stop" ) )
|
|
{
|
|
StopRecording( true );
|
|
}
|
|
else if ( FStrEq( pszEvent, "ds_screenshot" ) )
|
|
{
|
|
if ( ds_screens.GetBool() )
|
|
{
|
|
float flDelay = event->GetFloat( "delay" );
|
|
m_flScreenshotTime = gpGlobals->curtime + flDelay;
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::BookMarkCurrentTick( const char *pszValue /* = NULL */ )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
LogEvent( eDemoEvent_Bookmark, 0, pszValue );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
bool CTFDemoSupport::IsValidPath( const char *pszFolder )
|
|
{
|
|
if ( !pszFolder )
|
|
return false;
|
|
|
|
if ( Q_strlen( pszFolder ) <= 0 ||
|
|
Q_strstr( pszFolder, "\\\\" ) || // to protect network paths
|
|
Q_strstr( pszFolder, ":" ) || // to protect absolute paths
|
|
Q_strstr( pszFolder, ".." ) || // to protect relative paths
|
|
Q_strstr( pszFolder, "\n" ) || // CFileSystem_Stdio::FS_fopen doesn't allow this
|
|
Q_strstr( pszFolder, "\r" ) ) // CFileSystem_Stdio::FS_fopen doesn't allow this
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
bool CTFDemoSupport::StartRecording( void )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return false;
|
|
|
|
// are we already recording?
|
|
if ( m_bRecording )
|
|
{
|
|
Notify( "(Demo Support) Already recording\n" );
|
|
return false;
|
|
}
|
|
|
|
// start recording the demo
|
|
char szTime[k_RTimeRenderBufferSize] = {0};
|
|
|
|
time_t tTime = CRTime::RTime32TimeCur();
|
|
struct tm tmStruct;
|
|
struct tm *ptm = Plat_localtime( &tTime, &tmStruct );
|
|
|
|
V_sprintf_safe( szTime, "%04u-%02u-%02u_%02u-%02u-%02u",
|
|
ptm->tm_year + 1900, ptm->tm_mon + 1, ptm->tm_mday,
|
|
ptm->tm_hour, ptm->tm_min, ptm->tm_sec );
|
|
|
|
char szPrefix[24] = {0};
|
|
V_sprintf_safe( szPrefix, "%s", ds_prefix.GetString() );
|
|
V_sprintf_safe( m_szFilename, "%s%s", szPrefix, szTime );
|
|
|
|
if ( Q_strlen( ds_dir.GetString() ) > 0 )
|
|
{
|
|
// check folder
|
|
if ( !IsValidPath( ds_dir.GetString() ) )
|
|
{
|
|
Msg( "DemoSupport: invalid folder.\n" );
|
|
return false;
|
|
}
|
|
|
|
V_sprintf_safe( m_szFolder, "%s", ds_dir.GetString() );
|
|
|
|
// make sure the folder exists
|
|
g_pFullFileSystem->CreateDirHierarchy( m_szFolder, "GAME" );
|
|
|
|
V_sprintf_safe( m_szFolderAndFilename, "%s%c%s", m_szFolder, CORRECT_PATH_SEPARATOR, m_szFilename );
|
|
}
|
|
else
|
|
{
|
|
m_szFolder[0] = '\0';
|
|
V_sprintf_safe( m_szFolderAndFilename, "%s", m_szFilename );
|
|
}
|
|
|
|
if ( !engine->StartDemoRecording( m_szFilename, m_szFolder ) )
|
|
{
|
|
Notify( "(Demo Support) Unable to start recording\n" );
|
|
return false;
|
|
}
|
|
|
|
char szGobalFile[MAX_PATH] = { 0 };
|
|
V_sprintf_safe( szGobalFile, "%s%c%s", m_szFolder, CORRECT_PATH_SEPARATOR, EVENTS_FILENAME );
|
|
m_hGlobalEventList = g_pFullFileSystem->Open( szGobalFile, "at", "GAME" );
|
|
m_bFirstEvent = true;
|
|
|
|
m_DemoSpecificEventList.Clear();
|
|
m_DemoSpecificEventList.SetStatusCode( k_EHTTPStatusCode200OK );
|
|
m_pRoot = m_DemoSpecificEventList.CreateRootValue( "summary" );
|
|
m_DemoSpecificEventList.SetJSONAnonymousRootNode( true );
|
|
m_pChildArray = m_pRoot->CreateChildArray( "events", "event" );
|
|
|
|
m_bRecording = true;
|
|
m_bAlreadyAutoRecordedOnce = true;
|
|
m_nStartingTickCount = gpGlobals->tickcount;
|
|
m_bHasAtLeastOneEvent = false;
|
|
|
|
if ( ds_sound.GetBool() )
|
|
{
|
|
CBasePlayer *pLocalPlayer = CBasePlayer::GetLocalPlayer();
|
|
if ( pLocalPlayer )
|
|
{
|
|
pLocalPlayer->EmitSound( "DemoSupport.StartRecording" );
|
|
}
|
|
}
|
|
|
|
char szMessage[MAX_PATH] = { 0 };
|
|
V_sprintf_safe( szMessage, "(Demo Support) Start recording %s\n", m_szFolderAndFilename );
|
|
Notify( szMessage );
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CTFDemoSupport::StopRecording( bool bFromEngine /* = false */ )
|
|
{
|
|
if ( engine->IsPlayingDemo() )
|
|
return;
|
|
|
|
if ( !m_bRecording )
|
|
return;
|
|
|
|
m_bRecording = false;
|
|
m_nStartingTickCount = 0;
|
|
|
|
// stop recording the demo
|
|
if ( !bFromEngine )
|
|
{
|
|
engine->StopDemoRecording();
|
|
}
|
|
|
|
if ( ds_sound.GetBool() )
|
|
{
|
|
CBasePlayer *pLocalPlayer = CBasePlayer::GetLocalPlayer();
|
|
if ( pLocalPlayer )
|
|
{
|
|
pLocalPlayer->EmitSound( "DemoSupport.EndRecording" );
|
|
}
|
|
}
|
|
|
|
char szMessage[MAX_PATH] = { 0 };
|
|
V_sprintf_safe( szMessage, "(Demo Support) End recording %s\n", m_szFolderAndFilename );
|
|
Notify( szMessage );
|
|
|
|
if ( m_hGlobalEventList != FILESYSTEM_INVALID_HANDLE )
|
|
{
|
|
g_pFullFileSystem->Close( m_hGlobalEventList );
|
|
m_bFirstEvent = true;
|
|
}
|
|
|
|
if ( ds_autodelete.GetBool() && !m_bHasAtLeastOneEvent )
|
|
{
|
|
char szTemp[MAX_PATH] = { 0 };
|
|
V_sprintf_safe( szTemp, "%s.dem", m_szFolderAndFilename );
|
|
|
|
g_pFullFileSystem->RemoveFile( szTemp, "GAME" );
|
|
V_sprintf_safe( szMessage, "(Demo Support) Auto-delete recording %s\n", m_szFolderAndFilename );
|
|
Notify( szMessage );
|
|
}
|
|
else
|
|
{
|
|
if ( ds_log.GetBool() )
|
|
{
|
|
// write out the associated bookmark and kill-streak data file
|
|
char szTempFilename[MAX_PATH] = {0};
|
|
V_sprintf_safe( szTempFilename, "%s.json", m_szFolderAndFilename );
|
|
|
|
CUtlBuffer buffer( 0, 0, CUtlBuffer::TEXT_BUFFER | CUtlBuffer::EXTERNAL_GROWABLE );
|
|
m_DemoSpecificEventList.BEmitFormattedOutput( GCSDK::k_EWebAPIOutputFormat_JSON, buffer, 0 );
|
|
g_pFullFileSystem->WriteFile( szTempFilename, "GAME", buffer );
|
|
}
|
|
}
|
|
|
|
m_DemoSpecificEventList.Clear();
|
|
m_pRoot = NULL;
|
|
m_pChildArray = NULL;
|
|
m_bHasAtLeastOneEvent = false;
|
|
}
|
|
|
|
|