1
0
mirror of https://github.com/alliedmodders/hl2sdk.git synced 2025-01-05 17:13:36 +08:00
hl2sdk/choreoobjects/choreoscene.cpp
2008-09-15 01:33:59 -05:00

3988 lines
101 KiB
C++

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
#if defined(_WIN32) && !defined(_XBOX)
#include <windows.h>
#endif
#include "basetypes.h"
#include <stdio.h>
#include "choreoscene.h"
#include "choreoevent.h"
#include "choreochannel.h"
#include "choreoactor.h"
#include "ichoreoeventcallback.h"
#include "iscenetokenprocessor.h"
#include "utlbuffer.h"
#include "filesystem.h"
#include "utlrbtree.h"
#include "mathlib.h"
#include "vstdlib/strtools.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
extern IFileSystem *SceneFileSystem();
#ifdef _MSC_VER
#pragma warning( disable : 4127 )
#endif
// Let scene linger for 1/4 second so blends can finish
#define SCENE_LINGER_TIME 0.25f
// The engine turns this to true in dlls/sceneentity.cpp at bool SceneCacheInit()!!!
bool CChoreoScene::s_bEditingDisabled = false;
//-----------------------------------------------------------------------------
// Purpose: Debug printout
// Input : level -
// *fmt -
// ... -
//-----------------------------------------------------------------------------
void CChoreoScene::choreoprintf( int level, const char *fmt, ... )
{
char string[ 2048 ];
va_list argptr;
va_start( argptr, fmt );
Q_vsnprintf( string, sizeof(string), fmt, argptr );
va_end( argptr );
while ( level-- > 0 )
{
if (m_pfnPrint )
{
(*m_pfnPrint)( " " );
}
else
{
printf( " " );
}
Msg( " " );
}
if ( m_pfnPrint )
{
(*m_pfnPrint)( string );
}
else
{
printf( string );
}
Msg( "%s", string );
}
//-----------------------------------------------------------------------------
// Purpose: Creates scene from a file
// Input : *filename -
// *pfn -
// Output : CChoreoScene
//-----------------------------------------------------------------------------
CChoreoScene *ChoreoLoadScene
(
char const *filename,
IChoreoEventCallback *callback,
ISceneTokenProcessor *tokenizer,
void ( *pfn ) ( const char *fmt, ... )
)
{
MEM_ALLOC_CREDIT_CLASS();
CChoreoScene *scene = new CChoreoScene( callback );
Assert( scene );
scene->ParseFromBuffer( filename, tokenizer );
scene->SetPrintFunc( pfn );
return scene;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CChoreoScene::CChoreoScene( IChoreoEventCallback *callback )
{
Init( callback );
}
//-----------------------------------------------------------------------------
// Purpose: // Assignment
// Input : src -
// Output : CChoreoScene&
//-----------------------------------------------------------------------------
CChoreoScene& CChoreoScene::operator=( const CChoreoScene& src )
{
Init( src.m_pIChoreoEventCallback );
// Delete existing
int i;
for ( i = 0; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
Assert( a );
delete a;
}
m_Actors.RemoveAll();
for ( i = 0; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
Assert( e );
delete e;
}
m_Events.RemoveAll();
for ( i = 0 ; i < m_Channels.Size(); i++ )
{
CChoreoChannel *c = m_Channels[ i ];
Assert( c );
delete c;
}
m_Channels.RemoveAll();
m_pTokenizer = src.m_pTokenizer;
m_flCurrentTime = src.m_flCurrentTime;
m_flStartTime = src.m_flStartTime;
m_flEndTime = src.m_flEndTime;
m_flSoundSystemLatency = src.m_flSoundSystemLatency;
m_pfnPrint = src.m_pfnPrint;
m_flLastActiveTime = src.m_flLastActiveTime;
m_pTokenizer = src.m_pTokenizer;
m_bSubScene = src.m_bSubScene;
m_nSceneFPS = src.m_nSceneFPS;
m_bUseFrameSnap = src.m_bUseFrameSnap;
// Now copy the object tree
// First copy the global events
for ( i = 0; i < src.m_Events.Size(); i++ )
{
CChoreoEvent *event = src.m_Events[ i ];
if ( event->GetActor() == NULL )
{
MEM_ALLOC_CREDIT();
// Copy it
CChoreoEvent *newEvent = AllocEvent();
*newEvent = *event;
}
}
// Finally, push actors, channels, events onto global stacks
for ( i = 0; i < src.m_Actors.Size(); i++ )
{
CChoreoActor *actor = src.m_Actors[ i ];
CChoreoActor *newActor = AllocActor();
*newActor = *actor;
for ( int j = 0; j < newActor->GetNumChannels() ; j++ )
{
CChoreoChannel *ch = newActor->GetChannel( j );
m_Channels.AddToTail( ch );
for ( int k = 0; k < ch->GetNumEvents(); k++ )
{
CChoreoEvent *ev = ch->GetEvent( k );
m_Events.AddToTail( ev );
ev->SetScene( this );
}
}
}
Q_strncpy( m_szMapname, src.m_szMapname, sizeof( m_szMapname ) );
// Copy ramp over
m_SceneRamp.RemoveAll();
for ( i = 0; i < src.m_SceneRamp.Count(); i++ )
{
CExpressionSample sample = src.m_SceneRamp[ i ];
AddSceneRamp( sample.time, sample.value, sample.selected );
}
m_SceneRampEdgeInfo[ 0 ] = src.m_SceneRampEdgeInfo[ 0 ];
m_SceneRampEdgeInfo[ 1 ] = src.m_SceneRampEdgeInfo[ 1 ];
m_TimeZoomLookup.RemoveAll();
for ( i = 0; i < (int)src.m_TimeZoomLookup.Count(); i++ )
{
m_TimeZoomLookup.Insert( src.m_TimeZoomLookup.GetElementName( i ), src.m_TimeZoomLookup[ i ] );
}
Q_strncpy( m_szFileName, src.m_szFileName, sizeof( m_szFileName ) );
m_nLastPauseEvent = src.m_nLastPauseEvent;
m_flPrecomputedStopTime = src.m_flPrecomputedStopTime;
m_bitvecHasEventOfType = src.m_bitvecHasEventOfType;
return *this;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::Init( IChoreoEventCallback *callback )
{
m_flPrecomputedStopTime = 0.0f;
m_pTokenizer = NULL;
m_szMapname[ 0 ] = 0;
m_flCurrentTime = 0.0f;
m_flStartTime = 0.0f;
m_flEndTime = 0.0f;
m_flSoundSystemLatency = 0.0f;
m_pfnPrint = NULL;
m_flLastActiveTime = 0.0f;
m_flEarliestTime = 0.0f;
m_flLatestTime = 0.0f;
m_nActiveEvents = 0;
m_pIChoreoEventCallback = callback;
m_bSubScene = false;
m_nSceneFPS = DEFAULT_SCENE_FPS;
m_bUseFrameSnap = false;
m_szFileName[0] = 0;
m_bIsBackground = false;
m_bitvecHasEventOfType.ClearAll();
m_nLastPauseEvent = -1;
}
//-----------------------------------------------------------------------------
// Purpose: Destroy objects and queues
//-----------------------------------------------------------------------------
CChoreoScene::~CChoreoScene( void )
{
int i;
for ( i = 0; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
Assert( a );
delete a;
}
m_Actors.RemoveAll();
for ( i = 0; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
Assert( e );
delete e;
}
m_Events.RemoveAll();
for ( i = 0 ; i < m_Channels.Size(); i++ )
{
CChoreoChannel *c = m_Channels[ i ];
Assert( c );
delete c;
}
m_Channels.RemoveAll();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *callback -
//-----------------------------------------------------------------------------
void CChoreoScene::SetEventCallbackInterface( IChoreoEventCallback *callback )
{
m_pIChoreoEventCallback = callback;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : level -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::PrintEvent( int level, CChoreoEvent *e )
{
choreoprintf( level, "event %s \"%s\"\n", CChoreoEvent::NameForType( e->GetType() ), e->GetName() );
choreoprintf( level, "{\n" );
choreoprintf( level + 1, "time %f %f\n", e->GetStartTime(), e->GetEndTime() );
choreoprintf( level + 1, "param \"%s\"\n", e->GetParameters() );
if ( strlen( e->GetParameters2() ) > 0 )
{
choreoprintf( level + 1, "param2 \"%s\"\n", e->GetParameters2() );
}
choreoprintf( level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : level -
// *c -
//-----------------------------------------------------------------------------
void CChoreoScene::PrintChannel( int level, CChoreoChannel *c )
{
choreoprintf( level, "channel \"%s\"\n", c->GetName() );
choreoprintf( level, "{\n" );
for ( int i = 0; i < c->GetNumEvents(); i++ )
{
CChoreoEvent *e = c->GetEvent( i );
if ( e )
{
PrintEvent( level + 1, e );
}
}
choreoprintf( level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : level -
// *a -
//-----------------------------------------------------------------------------
void CChoreoScene::PrintActor( int level, CChoreoActor *a )
{
choreoprintf( level, "actor \"%s\"\n", a->GetName() );
choreoprintf( level, "{\n" );
for ( int i = 0; i < a->GetNumChannels(); i++ )
{
CChoreoChannel *c = a->GetChannel( i );
if ( c )
{
PrintChannel( level + 1, c );
}
}
choreoprintf( level, "}\n\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::Print( void )
{
// Look for events that don't have actor/channel set
int i;
for ( i = 0 ; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetActor() )
continue;
PrintEvent( 0, e );
}
for ( i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
PrintActor( 0, a );
}
}
//-----------------------------------------------------------------------------
// Purpose: prints if m_pfnPrint is active
// Output :
//-----------------------------------------------------------------------------
void CChoreoScene::SceneMsg( const char *pFormat, ... )
{
char string[ 2048 ];
va_list argptr;
va_start( argptr, pFormat );
Q_vsnprintf( string, sizeof(string), pFormat, argptr );
va_end( argptr );
if ( m_pfnPrint )
{
(*m_pfnPrint)( string );
}
else
{
Msg( "%s", string );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CChoreoEvent
//-----------------------------------------------------------------------------
CChoreoEvent *CChoreoScene::AllocEvent( void )
{
MEM_ALLOC_CREDIT_CLASS();
CChoreoEvent *e = new CChoreoEvent( this );
Assert( e );
m_Events.AddToTail( e );
return e;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CChoreoChannel
//-----------------------------------------------------------------------------
CChoreoChannel *CChoreoScene::AllocChannel( void )
{
MEM_ALLOC_CREDIT_CLASS();
CChoreoChannel *c = new CChoreoChannel();
Assert( c );
m_Channels.AddToTail( c );
return c;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CChoreoActor
//-----------------------------------------------------------------------------
CChoreoActor *CChoreoScene::AllocActor( void )
{
MEM_ALLOC_CREDIT_CLASS();
CChoreoActor *a = new CChoreoActor;
Assert( a );
m_Actors.AddToTail( a );
return a;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *name -
// Output : CChoreoActor
//-----------------------------------------------------------------------------
CChoreoActor *CChoreoScene::FindActor( const char *name )
{
for ( int i = 0; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
if ( !Q_stricmp( a->GetName(), name ) )
return a;
}
return NULL;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::GetNumEvents( void )
{
return m_Events.Size();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : event -
// Output : CChoreoEvent
//-----------------------------------------------------------------------------
CChoreoEvent *CChoreoScene::GetEvent( int event )
{
if ( event < 0 || event >= m_Events.Size() )
return NULL;
return m_Events[ event ];
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::GetNumActors( void )
{
return m_Actors.Size();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : actor -
// Output : CChoreoActor
//-----------------------------------------------------------------------------
CChoreoActor *CChoreoScene::GetActor( int actor )
{
if ( actor < 0 || actor >= GetNumActors() )
return NULL;
return m_Actors[ actor ];
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::GetNumChannels( void )
{
return m_Channels.Size();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : channel -
// Output : CChoreoChannel
//-----------------------------------------------------------------------------
CChoreoChannel *CChoreoScene::GetChannel( int channel )
{
if ( channel < 0 || channel >= GetNumChannels() )
return NULL;
return m_Channels[ channel ];
}
void CChoreoScene::ParseRamp( ISceneTokenProcessor *tokenizer, CChoreoEvent *e )
{
e->ClearRamp();
tokenizer->GetToken( true );
if ( !Q_stricmp( tokenizer->CurrentToken(), "leftedge" ) )
{
ParseEdgeInfo( tokenizer, e->GetRampEdgeInfo( 0 ) );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "rightedge" ) )
{
ParseEdgeInfo( tokenizer, e->GetRampEdgeInfo( 1 ) );
}
if ( stricmp( tokenizer->CurrentToken(), "{" ) )
tokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
tokenizer->GetToken( true );
if ( strlen( tokenizer->CurrentToken() ) <= 0 )
{
tokenizer->Error( "expecting ramp data\n" );
break;
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "}" ) )
break;
CUtlVector< CExpressionSample > samples;
float time = (float)atof( tokenizer->CurrentToken() );
tokenizer->GetToken( false );
float value = (float)atof( tokenizer->CurrentToken() );
// Add to counter
int idx = samples.AddToTail();
CExpressionSample *s = &samples[ idx ];
s->time = time;
s->value = value;
// If there are more tokens on this line, then it's a new format curve name
if ( tokenizer->TokenAvailable() )
{
tokenizer->GetToken( false );
int curveType = Interpolator_CurveTypeForName( tokenizer->CurrentToken() );
s->SetCurveType( curveType );
}
if ( samples.Size() >= 1 )
{
for ( int i = 0; i < samples.Size(); i++ )
{
CExpressionSample sample = samples[ i ];
CExpressionSample *newSample = e->AddRamp( sample.time, sample.value, false );
newSample->SetCurveType( sample.GetCurveType() );
}
}
}
e->ResortRamp();
}
void CChoreoScene::ParseSceneRamp( ISceneTokenProcessor *tokenizer, CChoreoScene *scene )
{
scene->ClearSceneRamp();
tokenizer->GetToken( true );
if ( !Q_stricmp( tokenizer->CurrentToken(), "leftedge" ) )
{
ParseEdgeInfo( tokenizer, scene->GetSceneRampEdgeInfo( 0 ) );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "rightedge" ) )
{
ParseEdgeInfo( tokenizer, scene->GetSceneRampEdgeInfo( 1 ) );
}
if ( stricmp( tokenizer->CurrentToken(), "{" ) )
tokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
tokenizer->GetToken( true );
if ( strlen( tokenizer->CurrentToken() ) <= 0 )
{
tokenizer->Error( "expecting ramp data\n" );
break;
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "}" ) )
break;
CUtlVector< CExpressionSample > samples;
float time = (float)atof( tokenizer->CurrentToken() );
tokenizer->GetToken( false );
float value = (float)atof( tokenizer->CurrentToken() );
// Add to counter
int idx = samples.AddToTail();
CExpressionSample *s = &samples[ idx ];
s->time = time;
s->value = value;
// If there are more tokens on this line, then it's a new format curve name
if ( tokenizer->TokenAvailable() )
{
tokenizer->GetToken( false );
int curveType = Interpolator_CurveTypeForName( tokenizer->CurrentToken() );
s->SetCurveType( curveType );
}
if ( samples.Size() >= 1 )
{
for ( int i = 0; i < samples.Size(); i++ )
{
CExpressionSample sample = samples[ i ];
CExpressionSample *newSample = scene->AddSceneRamp( sample.time, sample.value, false );
newSample->SetCurveType( sample.GetCurveType() );
}
}
}
scene->ResortSceneRamp();
}
//-----------------------------------------------------------------------------
// Purpose: Helper for restoring edge info
// Input : *edgeinfo -
//-----------------------------------------------------------------------------
void CChoreoScene::ParseEdgeInfo( ISceneTokenProcessor *tokenizer, EdgeInfo_t *edgeinfo )
{
Assert( edgeinfo );
Assert( tokenizer );
tokenizer->GetToken( false );
edgeinfo->m_bActive = true;
edgeinfo->m_CurveType = Interpolator_CurveTypeForName( tokenizer->CurrentToken() );
tokenizer->GetToken( false );
edgeinfo->m_flZeroPos = atof( tokenizer->CurrentToken() );
tokenizer->GetToken( true );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *tokenizer -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::ParseFlexAnimations( ISceneTokenProcessor *tokenizer, CChoreoEvent *e, bool removeold /*= true*/ )
{
Assert( e );
if ( removeold )
{
// Make sure there's nothing already there...
e->RemoveAllTracks();
// Make it re-index
e->SetTrackLookupSet( false );
}
// BACKWARD COMPATABILITY
// in the old system the samples were 0.0 to 1.0 mapped to endtime - starttime
// if samples_use_time is true, then samples are actually offsets of time from starttime
bool samples_use_realtime = false;
// Parse tags between { }
//
tokenizer->GetToken( true );
Assert( e->HasEndTime() );
float endtime = e->GetEndTime();
float starttime = e->GetStartTime();
float event_time = endtime - starttime;
// Is it the new file format?
if ( !Q_stricmp( tokenizer->CurrentToken(), "samples_use_time" ) )
{
samples_use_realtime = true;
tokenizer->GetToken( true );
}
if ( stricmp( tokenizer->CurrentToken(), "{" ) )
tokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
tokenizer->GetToken( true );
if ( strlen( tokenizer->CurrentToken() ) <= 0 )
{
tokenizer->Error( "expecting flex animation data\n" );
break;
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "}" ) )
break;
char flexcontroller[ CFlexAnimationTrack::MAX_CONTROLLER_NAME ];
Q_strncpy( flexcontroller, tokenizer->CurrentToken(), sizeof( flexcontroller ) );
// Animations default to active
bool active = true;
bool combo = false;
float range_min = 0.0f;
float range_max = 1.0f;
tokenizer->GetToken( true );
EdgeInfo_t edgeinfo[ 2 ];
if ( !Q_stricmp( tokenizer->CurrentToken(), "disabled" ) )
{
active = false;
tokenizer->GetToken( true );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "combo" ) )
{
combo = true;
tokenizer->GetToken( true );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "range" ) )
{
tokenizer->GetToken( false );
range_min = atof( tokenizer->CurrentToken() );
tokenizer->GetToken( false );
range_max = atof( tokenizer->CurrentToken() );
tokenizer->GetToken( true );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "leftedge" ) )
{
ParseEdgeInfo( tokenizer, &edgeinfo[ 0 ] );
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "rightedge" ) )
{
ParseEdgeInfo( tokenizer, &edgeinfo[ 1 ] );
}
CUtlVector< CExpressionSample > samples[2];
for ( int samplecount = 0; samplecount < ( combo ? 2 : 1 ); samplecount++ )
{
if ( stricmp( tokenizer->CurrentToken(), "{" ) )
{
tokenizer->Error( "expecting {\n" );
}
while ( 1 )
{
tokenizer->GetToken( true );
if ( strlen( tokenizer->CurrentToken() ) <= 0 )
{
tokenizer->Error( "expecting flex animation data\n" );
break;
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "}" ) )
break;
float time = (float)atof( tokenizer->CurrentToken() );
tokenizer->GetToken( false );
float value = (float)atof( tokenizer->CurrentToken() );
// Add to counter
int idx = samples[ samplecount ].AddToTail();
CExpressionSample *s = &samples[ samplecount ][ idx ];
if ( samples_use_realtime )
{
s->time = time;
}
else
{
// Time is an old style fraction (0 to 1) map into real time
s->time = time * event_time;
}
s->value = value;
// If there are more tokens on this line, then it's a new format curve name
if ( tokenizer->TokenAvailable() )
{
tokenizer->GetToken( false );
int curveType = Interpolator_CurveTypeForName( tokenizer->CurrentToken() );
s->SetCurveType( curveType );
}
}
if ( combo && samplecount == 0 )
{
tokenizer->GetToken( true );
}
}
if ( active || samples[ 0 ].Size() >= 1 )
{
// Add it in
CFlexAnimationTrack *track = e->AddTrack( flexcontroller );
Assert( track );
track->SetTrackActive( active );
track->SetComboType( combo );
track->SetMin( range_min );
track->SetMax( range_max );
for ( int t = 0; t < ( combo ? 2 : 1 ); t++ )
{
for ( int i = 0; i < samples[ t ].Size(); i++ )
{
CExpressionSample *sample = &samples[ t ][ i ];
CExpressionSample *added = track->AddSample( sample->time, sample->value, t );
Assert( added );
added->SetCurveType( sample->GetCurveType() );
}
}
for ( int edge = 0; edge < 2; ++edge )
{
if ( !edgeinfo[ edge ].m_bActive )
continue;
track->SetEdgeActive( edge == 0 ? true : false, true );
track->SetEdgeInfo( edge == 0 ? true : false, edgeinfo[ edge ].m_CurveType, edgeinfo[ edge ].m_flZeroPos );
}
track->Resort( 0 );
track->Resort( 1 );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
// *channel -
// Output : CChoreoEvent
//-----------------------------------------------------------------------------
CChoreoEvent *CChoreoScene::ParseEvent( CChoreoActor *actor, CChoreoChannel *channel )
{
// For conversion of old style attack/sustain/decay ramps
bool hadramp = false;
float attack = 1.0f, sustain = 1.0f, decay = 1.0f;
CChoreoEvent *e;
{
MEM_ALLOC_CREDIT();
e = AllocEvent();
}
MEM_ALLOC_CREDIT();
Assert( e );
// read event type
m_pTokenizer->GetToken( false );
e->SetType( CChoreoEvent::TypeForName( m_pTokenizer->CurrentToken() ) );
m_pTokenizer->GetToken( false );
e->SetName( m_pTokenizer->CurrentToken() );
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {\n" );
while ( 1 )
{
m_pTokenizer->GetToken( true );
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
{
m_pTokenizer->Error( "expecting more tokens!" );
break;
}
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "time" ) )
{
float start, end = 1.0f;
m_pTokenizer->GetToken( false );
start = (float)atof( m_pTokenizer->CurrentToken() );
if ( m_pTokenizer->TokenAvailable() )
{
m_pTokenizer->GetToken( false );
end = (float)atof( m_pTokenizer->CurrentToken() );
}
e->SetStartTime( start );
e->SetEndTime( end );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "ramp" ) )
{
hadramp = true;
m_pTokenizer->GetToken( false );
attack = (float)atof( m_pTokenizer->CurrentToken() );
if ( m_pTokenizer->TokenAvailable() )
{
m_pTokenizer->GetToken( false );
sustain = (float)atof( m_pTokenizer->CurrentToken() );
}
if ( m_pTokenizer->TokenAvailable() )
{
m_pTokenizer->GetToken( false );
decay = (float)atof( m_pTokenizer->CurrentToken() );
}
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "param" ) )
{
m_pTokenizer->GetToken( false );
e->SetParameters( m_pTokenizer->CurrentToken() );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "param2" ) )
{
m_pTokenizer->GetToken( false );
e->SetParameters2( m_pTokenizer->CurrentToken() );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "pitch" ) )
{
m_pTokenizer->GetToken( false );
e->SetPitch( atoi( m_pTokenizer->CurrentToken() ) );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "yaw" ) )
{
m_pTokenizer->GetToken( false );
e->SetYaw( atoi( m_pTokenizer->CurrentToken() ) );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "loopcount" ) )
{
m_pTokenizer->GetToken( false );
e->SetLoopCount( atoi( m_pTokenizer->CurrentToken() ) );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "resumecondition" ) )
{
e->SetResumeCondition( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "fixedlength" ) )
{
e->SetFixedLength( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "lockbodyfacing" ) )
{
e->SetLockBodyFacing( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "distancetotarget" ) )
{
m_pTokenizer->GetToken( false );
e->SetDistanceToTarget( atof( m_pTokenizer->CurrentToken() ) );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "forceshortmovement" ) )
{
e->SetForceShortMovement( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "synctofollowinggesture" ) )
{
e->SetSyncToFollowingGesture( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "tags" ) )
{
// Parse tags between { }
//
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
m_pTokenizer->GetToken( true );
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
{
m_pTokenizer->Error( "expecting relative tag\n" );
break;
}
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
char tagname[ CEventRelativeTag::MAX_EVENTTAG_LENGTH ];
float percentage;
Q_strncpy( tagname, m_pTokenizer->CurrentToken(), sizeof( tagname ) );
m_pTokenizer->GetToken( false );
percentage = (float)atof( m_pTokenizer->CurrentToken() );
e->AddRelativeTag( tagname, percentage );
}
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "sequenceduration" ) )
{
float duration = 0.0f;
m_pTokenizer->GetToken( false );
duration = (float)atof( m_pTokenizer->CurrentToken() );
e->SetGestureSequenceDuration( duration );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "absolutetags" ) )
{
m_pTokenizer->GetToken( true );
CChoreoEvent::AbsTagType tagtype;
tagtype = CChoreoEvent::TypeForAbsoluteTagName( m_pTokenizer->CurrentToken() );
if ( tagtype == (CChoreoEvent::AbsTagType) -1 )
{
m_pTokenizer->Error( "expecting valid tag type!!!" );
}
// Parse tags between { }
//
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
m_pTokenizer->GetToken( true );
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
{
m_pTokenizer->Error( "expecting relative tag\n" );
break;
}
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
char tagname[ CFlexTimingTag::MAX_EVENTTAG_LENGTH ];
float t;
Q_strncpy( tagname, m_pTokenizer->CurrentToken(), sizeof( tagname ) );
m_pTokenizer->GetToken( false );
t = (float)atof( m_pTokenizer->CurrentToken() );
e->AddAbsoluteTag( tagtype, tagname, t );
}
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "flextimingtags" ) )
{
// Parse tags between { }
//
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
m_pTokenizer->GetToken( true );
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
{
m_pTokenizer->Error( "expecting relative tag\n" );
break;
}
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
char tagname[ CFlexTimingTag::MAX_EVENTTAG_LENGTH ];
float percentage;
bool locked;
Q_strncpy( tagname, m_pTokenizer->CurrentToken(), sizeof( tagname ) );
m_pTokenizer->GetToken( false );
percentage = (float)atof( m_pTokenizer->CurrentToken() );
m_pTokenizer->GetToken( false );
locked = atoi( m_pTokenizer->CurrentToken() ) ? true : false;
e->AddTimingTag( tagname, percentage, locked );
}
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "relativetag" ) )
{
char tagname[ CChoreoEvent::MAX_TAGNAME_STRING ];
char wavname[ CChoreoEvent::MAX_TAGNAME_STRING ];
m_pTokenizer->GetToken( false );
Q_strncpy( tagname, m_pTokenizer->CurrentToken(), sizeof( tagname ) );
m_pTokenizer->GetToken( false );
Q_strncpy( wavname, m_pTokenizer->CurrentToken(), sizeof( wavname ) );
e->SetUsingRelativeTag( true, tagname, wavname );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "flexanimations" ) )
{
ParseFlexAnimations( m_pTokenizer, e );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "event_ramp" ) )
{
ParseRamp( m_pTokenizer, e );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "cctype" ) )
{
m_pTokenizer->GetToken( false );
e->SetCloseCaptionType( CChoreoEvent::CCTypeForName( m_pTokenizer->CurrentToken() ) );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "cctoken" ) )
{
m_pTokenizer->GetToken( false );
e->SetCloseCaptionToken( m_pTokenizer->CurrentToken() );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "cc_usingcombinedfile" ) )
{
e->SetUsingCombinedFile( true );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "cc_combinedusesgender" ) )
{
e->SetCombinedUsingGenderToken( true );
}
else if( !Q_stricmp( m_pTokenizer->CurrentToken(), "cc_noattenuate" ) )
{
e->SetSuppressingCaptionAttenuation( true );
}
}
if ( channel )
{
channel->AddEvent( e );
}
e->SetActor( actor );
e->SetChannel( channel );
// It had old sytle ramp and none of the new style stuff
// Convert it
if ( hadramp && !e->GetRampCount() )
{
// Only retrofit if something was changed by user
if ( attack != 1.0f ||
sustain != 1.0f ||
decay != 1.0f )
{
float attacktime = ( 1.0f - attack ) * e->GetDuration();
float decaytime = decay * e->GetDuration();
float midpoint = ( attacktime + decaytime ) * 0.5f;
e->AddRamp( attacktime, sustain, false );
e->AddRamp( midpoint, sustain, false );
e->AddRamp( decaytime, sustain, false );
e->ResortRamp();
}
}
return e;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CChoreoActor
//-----------------------------------------------------------------------------
CChoreoActor *CChoreoScene::ParseActor( void )
{
CChoreoActor *a = AllocActor();
Assert( a );
m_pTokenizer->GetToken( false );
a->SetName( m_pTokenizer->CurrentToken() );
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {" );
// Parse channels
while ( 1 )
{
m_pTokenizer->GetToken( true );
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "channel" ) )
{
ParseChannel( a );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "faceposermodel" ) )
{
ParseFacePoserModel( a );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "active" ) )
{
m_pTokenizer->GetToken( true );
a->SetActive( atoi( m_pTokenizer->CurrentToken() ) ? true : false );
}
else
{
m_pTokenizer->Error( "expecting channel got %s\n", m_pTokenizer->CurrentToken() );
}
}
return a;
}
//-----------------------------------------------------------------------------
// Output : char const
//-----------------------------------------------------------------------------
const char *CChoreoScene::GetMapname( void )
{
return m_szMapname;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *name -
//-----------------------------------------------------------------------------
void CChoreoScene::SetMapname( const char *name )
{
Q_strncpy( m_szMapname, name, sizeof( m_szMapname ) );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ParseMapname( void )
{
m_szMapname[ 0 ] = 0;
m_pTokenizer->GetToken( true );
Q_strncpy( m_szMapname, m_pTokenizer->CurrentToken(), sizeof( m_szMapname ) );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ParseFPS( void )
{
m_pTokenizer->GetToken( true );
m_nSceneFPS = atoi( m_pTokenizer->CurrentToken() );
// Clamp to valid range
m_nSceneFPS = clamp( m_nSceneFPS, MIN_SCENE_FPS, MAX_SCENE_FPS);
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ParseSnap( void )
{
m_pTokenizer->GetToken( true );
m_bUseFrameSnap = !Q_stricmp( m_pTokenizer->CurrentToken(), "on" ) ? true : false;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
//-----------------------------------------------------------------------------
void CChoreoScene::ParseFacePoserModel( CChoreoActor *actor )
{
m_pTokenizer->GetToken( true );
actor->SetFacePoserModelName( m_pTokenizer->CurrentToken() );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
// Output : CChoreoChannel
//-----------------------------------------------------------------------------
CChoreoChannel *CChoreoScene::ParseChannel( CChoreoActor *actor )
{
CChoreoChannel *c = AllocChannel();
Assert( c );
m_pTokenizer->GetToken( false );
c->SetName( m_pTokenizer->CurrentToken() );
m_pTokenizer->GetToken( true );
if ( stricmp( m_pTokenizer->CurrentToken(), "{" ) )
m_pTokenizer->Error( "expecting {" );
// Parse channels
while ( 1 )
{
m_pTokenizer->GetToken( true );
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "}" ) )
break;
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "event" ) )
{
ParseEvent( actor, c );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "active" ) )
{
m_pTokenizer->GetToken( true );
c->SetActive( atoi( m_pTokenizer->CurrentToken() ) ? true : false );
}
else
{
m_pTokenizer->Error( "expecting event got %s\n", m_pTokenizer->CurrentToken() );
}
}
Assert( actor );
if ( actor )
{
actor->AddChannel( c );
c->SetActor( actor );
}
return c;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CChoreoScene::ParseFromBuffer( char const *filenae, ISceneTokenProcessor *tokenizer )
{
Q_strncpy( m_szFileName, filenae, sizeof( m_szFileName ) );
m_pTokenizer = tokenizer;
while ( 1 )
{
if ( !m_pTokenizer->GetToken( true ) )
{
break;
}
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
break;
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "event" ) )
{
ParseEvent( NULL, NULL );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "actor" ) )
{
ParseActor();
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "mapname" ) )
{
ParseMapname();
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "fps" ) )
{
ParseFPS();
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "snap" ) )
{
ParseSnap();
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "scene_ramp" ) )
{
ParseSceneRamp( m_pTokenizer, this );
}
else if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "scalesettings" ) )
{
ParseScaleSettings( m_pTokenizer, this );
}
else
{
m_pTokenizer->Error( "unexpected token %s\n", m_pTokenizer->CurrentToken() );
break;
}
}
// Fixup time tags
ReconcileTags();
ReconcileGestureTimes();
ReconcileCloseCaption();
InternalDetermineEventTypes();
if ( CChoreoScene::s_bEditingDisabled )
{
m_flPrecomputedStopTime = FindStopTime();
}
return true;
}
void CChoreoScene::RemoveEventsExceptTypes( int* typeList, int count )
{
int i;
for ( i = 0 ; i < m_Actors.Count(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
int num = c->GetNumEvents();
for ( int k = num - 1 ; k >= 0; --k )
{
CChoreoEvent *e = c->GetEvent( k );
if ( !e )
continue;
bool found = false;
for ( int idx = 0; idx < count; ++idx )
{
if ( e->GetType() == ( CChoreoEvent::EVENTTYPE )typeList[ idx ] )
{
found = true;
break;
}
}
if ( !found )
{
c->RemoveEvent( e );
DeleteReferencedObjects( e );
}
}
}
}
// Remvoe non-matching global events, too
for ( i = m_Events.Count() - 1 ; i >= 0; --i )
{
CChoreoEvent *e = m_Events[ i ];
// This was already dealt with above...
if ( e->GetActor() )
continue;
bool found = false;
for ( int idx = 0; idx < count; ++idx )
{
if ( e->GetType() == ( CChoreoEvent::EVENTTYPE )typeList[ idx ] )
{
found = true;
break;
}
}
if ( !found )
{
DeleteReferencedObjects( e );
}
}
}
void CChoreoScene::InternalDetermineEventTypes()
{
m_bitvecHasEventOfType.ClearAll();
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
for ( int k = 0 ; k < c->GetNumEvents(); k++ )
{
CChoreoEvent *e = c->GetEvent( k );
if ( !e )
continue;
m_bitvecHasEventOfType.Set( e->GetType(), true );
}
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : float
//-----------------------------------------------------------------------------
float CChoreoScene::FindStopTime( void )
{
if ( m_flPrecomputedStopTime != 0.0f )
{
return m_flPrecomputedStopTime;
}
float lasttime = 0.0f;
int c = m_Events.Count();
for ( int i = 0; i < c ; i++ )
{
CChoreoEvent *e = m_Events[ i ];
Assert( e );
float checktime = e->HasEndTime() ? e->GetEndTime() : e->GetStartTime();
if ( checktime > lasttime )
{
lasttime = checktime;
}
}
return lasttime;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fp -
// level -
// *fmt -
// ... -
//-----------------------------------------------------------------------------
void CChoreoScene::FilePrintf( CUtlBuffer& buf, int level, const char *fmt, ... )
{
va_list argptr;
va_start( argptr, fmt );
while ( level-- > 0 )
{
buf.Printf( " " );
}
buf.VaPrintf( fmt, argptr );
va_end( argptr );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fp -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveHeader( CUtlBuffer& buf )
{
FilePrintf( buf, 0, "// Choreo version 1\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : mark -
//-----------------------------------------------------------------------------
void CChoreoScene::MarkForSaveAll( bool mark )
{
int i;
// Mark global events
for ( i = 0 ; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetActor() )
continue;
e->SetMarkedForSave( mark );
}
// Recursively mark everything else
for ( i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
a->MarkForSaveAll( mark );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *filename -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CChoreoScene::ExportMarkedToFile( const char *filename )
{
// Create a serialization buffer
CUtlBuffer buf( 0, 0, true );
FileSaveHeader( buf );
// Look for events that don't have actor/channel set
int i;
for ( i = 0 ; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetActor() )
continue;
FileSaveEvent( buf, 0, e );
}
for ( i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
FileSaveActor( buf, 0, a );
}
// Write it out baby
FileHandle_t fh = SceneFileSystem()->Open( filename, "wt" );
if (fh)
{
SceneFileSystem()->Write( buf.Base(), buf.TellPut(), fh );
SceneFileSystem()->Close(fh);
return true;
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *filename -
//-----------------------------------------------------------------------------
bool CChoreoScene::SaveToFile( const char *filename )
{
// Create a serialization buffer
CUtlBuffer buf( 0, 0, true );
FileSaveHeader( buf );
MarkForSaveAll( true );
// Look for events that don't have actor/channel set
int i;
for ( i = 0 ; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetActor() )
continue;
FileSaveEvent( buf, 0, e );
}
for ( i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
FileSaveActor( buf, 0, a );
}
if ( m_szMapname[ 0 ] )
{
FilePrintf( buf, 0, "mapname \"%s\"\n", m_szMapname );
}
FileSaveSceneRamp( buf, 0, this );
FileSaveScaleSettings( buf, 0, this );
FilePrintf( buf, 0, "fps %i\n", m_nSceneFPS );
FilePrintf( buf, 0, "snap %s\n", m_bUseFrameSnap ? "on" : "off" );
// Write it out baby
FileHandle_t fh = SceneFileSystem()->Open( filename, "wt" );
if (fh)
{
SceneFileSystem()->Write( buf.Base(), buf.TellPut(), fh );
SceneFileSystem()->Close(fh);
return true;
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : buf -
// level -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveRamp( CUtlBuffer& buf, int level, CChoreoEvent *e )
{
// Nothing to save?
int c = e->GetRampCount();
if ( c <= 0 &&
!e->RampIsEdgeActive( true ) &&
!e->RampIsEdgeActive( false ) )
return;
char line[ 1024 ];
Q_strncpy( line, "event_ramp ", sizeof( line ) );
if ( e->RampIsEdgeActive( true ) || e->RampIsEdgeActive( false ) )
{
if ( e->RampIsEdgeActive( true ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "leftedge %s %.3f ", Interpolator_NameForCurveType( e->RampGetEdgeCurveType( true ), false ), e->RampGetEdgeZeroValue( true ) );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
if ( e->RampIsEdgeActive( false ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "rightedge %s %.3f", Interpolator_NameForCurveType( e->RampGetEdgeCurveType( false ), false ), e->RampGetEdgeZeroValue( false ) );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
}
FilePrintf( buf, level, "%s\n", line );
FilePrintf( buf, level, "{\n" );
for ( int i = 0; i < c; i++ )
{
CExpressionSample *sample = e->GetRamp( i );
if ( sample->GetCurveType() != CURVE_DEFAULT )
{
FilePrintf( buf, level + 1, "%.4f %.4f \"%s\"\n",
sample->time,
sample->value,
Interpolator_NameForCurveType( sample->GetCurveType(), false ) );
}
else
{
FilePrintf( buf, level + 1, "%.4f %.4f\n",
sample->time,
sample->value );
}
}
FilePrintf( buf, level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : buf -
// level -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveSceneRamp( CUtlBuffer& buf, int level, CChoreoScene *scene )
{
// Nothing to save?
int c = scene->GetSceneRampCount();
if ( c <= 0 &&
!scene->SceneRampIsEdgeActive( true ) &&
!scene->SceneRampIsEdgeActive( false ) )
return;
char line[ 1024 ];
Q_strncpy( line, "scene_ramp ", sizeof( line ) );
if ( scene->SceneRampIsEdgeActive( true ) || scene->SceneRampIsEdgeActive( false ) )
{
if ( scene->SceneRampIsEdgeActive( true ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "leftedge %s %.3f ", Interpolator_NameForCurveType( scene->SceneRampGetEdgeCurveType( true ), false ), scene->SceneRampGetEdgeZeroValue( true ) );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
if ( scene->SceneRampIsEdgeActive( false ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ),"rightedge %s %.3f", Interpolator_NameForCurveType( scene->SceneRampGetEdgeCurveType( false ), false ), scene->SceneRampGetEdgeZeroValue( false ) );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
}
FilePrintf( buf, level, "%s\n", line );
FilePrintf( buf, level, "{\n" );
for ( int i = 0; i < c; i++ )
{
CExpressionSample *sample = scene->GetSceneRamp( i );
if ( sample->GetCurveType() != CURVE_DEFAULT )
{
FilePrintf( buf, level + 1, "%.4f %.4f \"%s\"\n",
sample->time,
sample->value,
Interpolator_NameForCurveType( sample->GetCurveType(), false ) );
}
else
{
FilePrintf( buf, level + 1, "%.4f %.4f\n",
sample->time,
sample->value );
}
}
FilePrintf( buf, level, "}\n" );
}
void CChoreoScene::FileSaveScaleSettings( CUtlBuffer& buf, int level, CChoreoScene *scene )
{
// Nothing to save?
int c = scene->m_TimeZoomLookup.Count();
if ( c <= 0 )
return;
FilePrintf( buf, level, "scalesettings\n" );
FilePrintf( buf, level, "{\n" );
for ( int i = 0; i < c; i++ )
{
int value = scene->m_TimeZoomLookup[ i ];
FilePrintf( buf, level + 1, "\"%s\" \"%i\"\n",
scene->m_TimeZoomLookup.GetElementName( i ),
value );
}
FilePrintf( buf, level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : buf -
// level -
// *track -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveFlexAnimationTrack( CUtlBuffer& buf, int level, CFlexAnimationTrack *track )
{
if ( !track )
return;
if ( !track->IsTrackActive() && track->GetNumSamples() <= 0 )
return;
char line[ 1024 ];
Q_snprintf( line, sizeof( line ), "\"%s\" ", track->GetFlexControllerName() );
if ( !track->IsTrackActive() )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "disabled " );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
if ( track->IsComboType() )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "combo " );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
if ( track->GetMin() != 0.0f || track->GetMax() != 1.0f)
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "range %.1f %.1f ", track->GetMin(), track->GetMax() );
Q_strncat( line, sz, sizeof( line ), COPY_ALL_CHARACTERS );
}
if ( track->IsEdgeActive( true ) || track->IsEdgeActive( false ) )
{
char edgestr[ 512 ];
edgestr[ 0 ] = 0;
if ( track->IsEdgeActive( true ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "leftedge %s %.3f ", Interpolator_NameForCurveType( track->GetEdgeCurveType( true ), false ), track->GetEdgeZeroValue( true ) );
Q_strncat( edgestr, sz, sizeof( edgestr ), COPY_ALL_CHARACTERS );
}
if ( track->IsEdgeActive( false ) )
{
char sz[ 256 ];
Q_snprintf( sz, sizeof( sz ), "rightedge %s %.3f ", Interpolator_NameForCurveType( track->GetEdgeCurveType( false ), false ), track->GetEdgeZeroValue( false ) );
Q_strncat( edgestr, sz, sizeof( edgestr ), COPY_ALL_CHARACTERS );
}
Q_strncat( line, edgestr, sizeof( line ), COPY_ALL_CHARACTERS );
}
FilePrintf( buf, level + 2, "%s\n", line );
// Write out samples
FilePrintf( buf, level + 2, "{\n" );
for ( int j = 0 ; j < track->GetNumSamples( 0 ) ; j++ )
{
CExpressionSample *s = track->GetSample( j, 0 );
if ( !s )
continue;
if ( s->GetCurveType() != CURVE_DEFAULT )
{
FilePrintf( buf, level + 3, "%.4f %.4f \"%s\"\n",
s->time,
s->value,
Interpolator_NameForCurveType( s->GetCurveType(), false ) );
}
else
{
FilePrintf( buf, level + 3, "%.4f %.4f\n",
s->time,
s->value );
}
}
FilePrintf( buf, level + 2, "}\n" );
// Write out combo samples
if ( track->IsComboType() )
{
FilePrintf( buf, level + 2, "{\n" );
for ( int j = 0 ; j < track->GetNumSamples( 1) ; j++ )
{
CExpressionSample *s = track->GetSample( j, 1 );
if ( !s )
continue;
if ( s->GetCurveType() != CURVE_DEFAULT )
{
FilePrintf( buf, level + 3, "%.4f %.4f \"%s\"\n",
s->time,
s->value,
Interpolator_NameForCurveType( s->GetCurveType(), false ) );
}
else
{
FilePrintf( buf, level + 3, "%.4f %.4f\n",
s->time,
s->value );
}
}
FilePrintf( buf, level + 2, "}\n" );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : buf -
// level -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveFlexAnimations( CUtlBuffer& buf, int level, CChoreoEvent *e )
{
// Nothing to save
if ( e->GetNumFlexAnimationTracks() <= 0 )
return;
FilePrintf( buf, level + 1, "flexanimations samples_use_time\n" );
FilePrintf( buf, level + 1, "{\n" );
for ( int i = 0; i < e->GetNumFlexAnimationTracks(); i++ )
{
CFlexAnimationTrack *track = e->GetFlexAnimationTrack( i );
FileSaveFlexAnimationTrack( buf, level, track );
}
FilePrintf( buf, level + 1, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fp -
// level -
// *e -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveEvent( CUtlBuffer& buf, int level, CChoreoEvent *e )
{
if ( !e->IsMarkedForSave() )
return;
FilePrintf( buf, level, "event %s \"%s\"\n", CChoreoEvent::NameForType( e->GetType() ), e->GetName() );
FilePrintf( buf, level, "{\n" );
float st, et;
st = e->GetStartTime();
et = e->GetEndTime();
FilePrintf( buf, level + 1, "time %f %f\n", st, et );
FilePrintf( buf, level + 1, "param \"%s\"\n", e->GetParameters() );
if ( strlen( e->GetParameters2() ) > 0 )
{
FilePrintf( buf, level + 1, "param2 \"%s\"\n", e->GetParameters2() );
}
if ( e->GetRampCount() > 0 )
{
FileSaveRamp( buf, level + 1, e );
}
if ( e->GetPitch() != 0 )
{
FilePrintf( buf, level + 1, "pitch \"%i\"\n", e->GetPitch() );
}
if ( e->GetYaw() != 0 )
{
FilePrintf( buf, level + 1, "yaw \"%i\"\n", e->GetYaw() );
}
if ( e->IsResumeCondition() )
{
FilePrintf( buf, level + 1, "resumecondition\n" );
}
if ( e->IsLockBodyFacing() )
{
FilePrintf( buf, level + 1, "lockbodyfacing\n" );
}
if ( e->GetDistanceToTarget() > 0.0f )
{
FilePrintf( buf, level + 1, "distancetotarget %.2f\n", e->GetDistanceToTarget() );
}
if ( e->GetForceShortMovement() )
{
FilePrintf( buf, level + 1, "forceshortmovement\n" );
}
if ( e->GetSyncToFollowingGesture() )
{
FilePrintf( buf, level + 1, "synctofollowinggesture\n" );
}
if ( e->IsFixedLength() )
{
FilePrintf( buf, level + 1, "fixedlength\n" );
}
if ( e->GetNumRelativeTags() > 0 )
{
FilePrintf( buf, level + 1, "tags\n" );
FilePrintf( buf, level + 1, "{\n" );
for ( int t = 0; t < e->GetNumRelativeTags(); t++ )
{
CEventRelativeTag *rt = e->GetRelativeTag( t );
Assert( rt );
FilePrintf( buf, level + 2, "\"%s\" %f\n", rt->GetName(), rt->GetPercentage() );
}
FilePrintf( buf, level + 1, "}\n" );
}
if ( e->GetNumTimingTags() > 0 )
{
FilePrintf( buf, level + 1, "flextimingtags\n" );
FilePrintf( buf, level + 1, "{\n" );
for ( int t = 0; t < e->GetNumTimingTags(); t++ )
{
CFlexTimingTag *tt = e->GetTimingTag( t );
Assert( tt );
FilePrintf( buf, level + 2, "\"%s\" %f %i\n", tt->GetName(), tt->GetPercentage(), tt->GetLocked() ? 1 : 0 );
}
FilePrintf( buf, level + 1, "}\n" );
}
int tagtype;
for ( tagtype = 0; tagtype < CChoreoEvent::NUM_ABS_TAG_TYPES; tagtype++ )
{
if ( e->GetNumAbsoluteTags( (CChoreoEvent::AbsTagType)tagtype ) > 0 )
{
FilePrintf( buf, level + 1, "absolutetags %s\n", CChoreoEvent::NameForAbsoluteTagType( (CChoreoEvent::AbsTagType)tagtype ) );
FilePrintf( buf, level + 1, "{\n" );
for ( int t = 0; t < e->GetNumAbsoluteTags( (CChoreoEvent::AbsTagType)tagtype ); t++ )
{
CEventAbsoluteTag *abstag = e->GetAbsoluteTag( (CChoreoEvent::AbsTagType)tagtype, t );
Assert( abstag );
FilePrintf( buf, level + 2, "\"%s\" %f\n", abstag->GetName(), abstag->GetPercentage() );
}
FilePrintf( buf, level + 1, "}\n" );
}
}
if ( e->GetType() == CChoreoEvent::GESTURE )
{
float duration;
if ( e->GetGestureSequenceDuration( duration ) )
{
FilePrintf( buf, level + 1, "sequenceduration %f\n", duration );
}
}
if ( e->IsUsingRelativeTag() )
{
FilePrintf( buf, level + 1, "relativetag \"%s\" \"%s\"\n",
e->GetRelativeTagName(), e->GetRelativeWavName() );
}
if ( e->GetNumFlexAnimationTracks() > 0 )
{
FileSaveFlexAnimations( buf, level, e );
}
if ( e->GetType() == CChoreoEvent::LOOP )
{
FilePrintf( buf, level + 1, "loopcount \"%i\"\n", e->GetLoopCount() );
}
if ( e->GetType() == CChoreoEvent::SPEAK )
{
FilePrintf( buf, level + 1, "cctype \"%s\"\n", CChoreoEvent::NameForCCType( e->GetCloseCaptionType() ) );
FilePrintf( buf, level + 1, "cctoken \"%s\"\n", e->GetCloseCaptionToken() );
if ( e->GetCloseCaptionType() != CChoreoEvent::CC_DISABLED &&
e->IsUsingCombinedFile() )
{
FilePrintf( buf, level + 1, "cc_usingcombinedfile\n" );
}
if ( e->IsCombinedUsingGenderToken() )
{
FilePrintf( buf, level + 1, "cc_combinedusesgender\n" );
}
if ( e->IsSuppressingCaptionAttenuation() )
{
FilePrintf( buf, level + 1, "cc_noattenuate\n" );
}
}
FilePrintf( buf, level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fp -
// level -
// *c -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveChannel( CUtlBuffer& buf, int level, CChoreoChannel *c )
{
if ( !c->IsMarkedForSave() )
return;
FilePrintf( buf, level, "channel \"%s\"\n", c->GetName() );
FilePrintf( buf, level, "{\n" );
for ( int i = 0; i < c->GetNumEvents(); i++ )
{
CChoreoEvent *e = c->GetEvent( i );
if ( e )
{
FileSaveEvent( buf, level + 1, e );
}
}
if ( !c->GetActive() )
{
// Only write out inactive
FilePrintf( buf, level + 1, "active \"0\"\n" );
}
FilePrintf( buf, level, "}\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *fp -
// level -
// *a -
//-----------------------------------------------------------------------------
void CChoreoScene::FileSaveActor( CUtlBuffer& buf, int level, CChoreoActor *a )
{
if ( !a->IsMarkedForSave() )
return;
FilePrintf( buf, level, "actor \"%s\"\n", a->GetName() );
FilePrintf( buf, level, "{\n" );
for ( int i = 0; i < a->GetNumChannels(); i++ )
{
CChoreoChannel *c = a->GetChannel( i );
if ( c )
{
FileSaveChannel( buf, level + 1, c );
}
}
if ( Q_strlen( a->GetFacePoserModelName() ) > 0 )
{
FilePrintf( buf, level + 1, "faceposermodel \"%s\"\n", a->GetFacePoserModelName() );
}
if ( !a->GetActive() )
{
// Only write out inactive
FilePrintf( buf, level + 1, "active \"0\"\n" );
}
FilePrintf( buf, level, "}\n\n" );
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : float
//-----------------------------------------------------------------------------
float CChoreoScene::FindAdjustedStartTime( void )
{
float earliest_time = 0.0f;
CChoreoEvent *e;
for ( int i = 0; i < m_Events.Size(); i++ )
{
e = m_Events[ i ];
float starttime = e->GetStartTime();
// If it's a wav file, pre-queue the starting time by the sound system's
// current latency
if ( e->GetType() == CChoreoEvent::SPEAK )
{
starttime -= m_flSoundSystemLatency;
}
if ( starttime < earliest_time )
{
earliest_time = starttime;
}
}
return earliest_time;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : float
//-----------------------------------------------------------------------------
float CChoreoScene::FindAdjustedEndTime( void )
{
float latest_time = 0.0f;
CChoreoEvent *e;
for ( int i = 0; i < m_Events.Size(); i++ )
{
e = m_Events[ i ];
float endtime = e->GetStartTime();
if ( e->HasEndTime() )
{
endtime = e->GetEndTime();
}
// If it's a wav file, pre-queue the starting time by the sound system's
// current latency
if ( e->GetType() == CChoreoEvent::SPEAK )
{
endtime += m_flSoundSystemLatency;
}
if ( endtime > latest_time )
{
latest_time = endtime;
}
}
return latest_time;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ResetSimulation( bool forward /*= true*/, float starttime /*= 0.0f*/, float endtime /*= 0.0f*/ )
{
CChoreoEvent *e;
m_ActiveResumeConditions.RemoveAll();
m_ResumeConditions.RemoveAll();
m_PauseEvents.RemoveAll();
// Put all items into the pending queue
for ( int i = 0; i < m_Events.Size(); i++ )
{
e = m_Events[ i ];
e->ResetProcessing();
if ( e->GetType() == CChoreoEvent::SECTION )
{
m_PauseEvents.AddToTail( e );
continue;
}
if ( e->IsResumeCondition() )
{
m_ResumeConditions.AddToTail( e );
continue;
}
}
// Find earliest adjusted start time
m_flEarliestTime = FindAdjustedStartTime();
m_flLatestTime = FindAdjustedEndTime();
m_flCurrentTime = forward ? m_flEarliestTime : m_flLatestTime;
// choreoprintf( 0, "Start time %f\n", m_flCurrentTime );
m_flLastActiveTime = 0.0f;
m_nActiveEvents = m_Events.Size();
m_flStartTime = starttime;
m_flEndTime = endtime;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CChoreoScene::CheckEventCompletion( void )
{
CChoreoEvent *e;
bool bAllCompleted = true;
// check all items in the active pending queue
for ( int i = 0; i < m_ActiveResumeConditions.Size(); i++ )
{
e = m_ActiveResumeConditions[ i ];
bAllCompleted = bAllCompleted && e->CheckProcessing( m_pIChoreoEventCallback, this, m_flCurrentTime );
}
return bAllCompleted;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CChoreoScene::SimulationFinished( void )
{
// Scene's linger for a little bit to allow things to settle
// check for events that are still active...
if ( m_flCurrentTime > m_flLatestTime )
{
if ( m_nActiveEvents != 0 )
{
return false;
}
return true;
}
if ( m_flCurrentTime < m_flEarliestTime )
{
return true;
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CChoreoEvent *CChoreoScene::FindPauseBetweenTimes( float starttime, float endtime )
{
CChoreoEvent *e;
// Iterate through all events in the scene
for ( int i = 0; i < m_PauseEvents.Size(); i++ )
{
e = m_PauseEvents[ i ];
if ( !e )
continue;
Assert( e->GetType() == CChoreoEvent::SECTION );
int time_is = IsTimeInRange( e->GetStartTime(), starttime, endtime );
if ( IN_RANGE != time_is )
continue;
// Found a pause in between start and end time
return e;
}
// No pause inside the specified time span
return NULL;
}
int CChoreoScene::IsTimeInRange( float t, float starttime, float endtime )
{
if ( t > endtime )
{
return AFTER_RANGE;
}
else if ( t < starttime )
{
return BEFORE_RANGE;
}
return IN_RANGE;
}
int CChoreoScene::EventThink( CChoreoEvent *e, float frame_start_time, float frame_end_time, bool playing_forward, PROCESSING_TYPE& disposition )
{
disposition = PROCESSING_TYPE_IGNORE;
int iret = 0;
bool hasend = e->HasEndTime();
float starttime, endtime;
starttime = e->GetStartTime();
endtime = hasend ? e->GetEndTime() : e->GetStartTime();
if ( !playing_forward )
{
// Swap intervals
float temp = frame_start_time;
frame_start_time = frame_end_time;
frame_end_time = temp;
}
bool suppressed = false;
// Special processing
switch ( e->GetType() )
{
default:
break;
case CChoreoEvent::SPEAK:
// If it's a wav file, pre-queue the starting/endtime time by the sound system's
// current latency
{
if ( playing_forward )
{
starttime -= m_flSoundSystemLatency;
// Search for pause condition in between the original time and the
// adjusted start time, but make sure that the pause event hasn't already triggered...
CChoreoEvent *pauseEvent = FindPauseBetweenTimes( starttime, starttime + m_flSoundSystemLatency );
if ( pauseEvent &&
( frame_start_time <= pauseEvent->GetStartTime() ) )
{
pauseEvent->AddEventDependency( e );
suppressed = true;
}
}
/*
else
// Don't bother if playing backward!!!
{
endtime += m_flSoundSystemLatency;
// Search for pause condition in between the original time and the
// adjusted start time
CChoreoEvent *pauseEvent = FindPauseBetweenTimes( endtime - m_flSoundSystemLatency, endtime );
if ( pauseEvent )
{
pauseEvent->AddEventDependency( e );
suppressed = true;
}
}
*/
}
break;
case CChoreoEvent::SUBSCENE:
{
if ( IsSubScene() )
{
suppressed = true;
}
}
break;
}
if ( suppressed )
{
if ( e->IsProcessing() )
{
disposition = PROCESSING_TYPE_STOP;
}
return iret;
}
int where_is_event;
if ( e->IsProcessing() )
{
where_is_event = IsTimeInRange( frame_start_time, starttime, endtime );
if ( IN_RANGE == where_is_event )
{
disposition = PROCESSING_TYPE_CONTINUE;
iret = 1;
}
else
{
disposition = PROCESSING_TYPE_STOP;
}
}
else
{
// Is the event supposed to be active at this time
where_is_event = IsTimeInRange( frame_start_time, starttime, endtime );
if ( IN_RANGE == where_is_event )
{
if ( e->IsResumeCondition() )
{
disposition = PROCESSING_TYPE_START_RESUMECONDITION;
}
else
{
disposition = PROCESSING_TYPE_START;
}
iret = 1;
}
// See if it's a single fire event which should occur during this frame
else if ( !hasend )
{
where_is_event = IsTimeInRange( starttime, frame_start_time, frame_end_time );
if ( IN_RANGE == where_is_event )
{
disposition = PROCESSING_TYPE_START;
iret = 1;
}
}
}
return iret;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : &e0 -
// &e1 -
// Output : static bool
//-----------------------------------------------------------------------------
bool CChoreoScene::EventLess( const CChoreoScene::ActiveList &al0, const CChoreoScene::ActiveList &al1 )
{
CChoreoEvent *event0, *event1;
event0 = const_cast< CChoreoEvent * >( al0.e );
event1 = const_cast< CChoreoEvent * >( al1.e );
if ( event0->GetStartTime() < event1->GetStartTime() )
{
return true;
}
if ( event0->GetStartTime() > event1->GetStartTime() )
{
return false;
}
// Check for end time overlap
if ( event0->HasEndTime() && event1->HasEndTime() )
{
if ( event0->GetEndTime() > event1->GetEndTime() )
return true;
else if ( event0->GetEndTime() < event1->GetEndTime() )
return false;
}
CChoreoActor *a0, *a1;
a0 = event0->GetActor();
a1 = event1->GetActor();
// Start time equal, go to order in channel
if ( !a0 || !a1 || a0 != a1 )
{
return strcmp( event0->GetName(), event1->GetName() ) == -1;
}
CChoreoChannel *c0 = event0->GetChannel();
CChoreoChannel *c1 = event1->GetChannel();
if ( !c0 || !c1 || c0 != c1 )
{
return strcmp( event0->GetName(), event1->GetName() ) == -1;
}
// Go by slot within channel
int index0 = a0->FindChannelIndex( c0 );
int index1 = a1->FindChannelIndex( c1 );
return ( index0 < index1 );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ClearPauseEventDependencies()
{
int c = m_PauseEvents.Count();
for ( int i = 0 ; i < c; ++i )
{
CChoreoEvent *pause = m_PauseEvents[ i ];
Assert( pause );
pause->ClearEventDependencies();
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pauseEvent -
// *suppressed -
//-----------------------------------------------------------------------------
void CChoreoScene::AddPauseEventDependency( CChoreoEvent *pauseEvent, CChoreoEvent *suppressed )
{
Assert( pauseEvent );
Assert( pauseEvent != suppressed );
pauseEvent->AddEventDependency( suppressed );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : dt -
//-----------------------------------------------------------------------------
void CChoreoScene::Think( float curtime )
{
CChoreoEvent *e;
float oldt = m_flCurrentTime;
float dt = curtime - oldt;
bool playing_forward = ( dt >= 0.0f ) ? true : false;
m_nActiveEvents = 0;
ClearPauseEventDependencies();
CUtlRBTree< ActiveList, int > pending(0,0,EventLess);
// Iterate through all events in the scene
int i;
for ( i = 0; i < m_Events.Size(); i++ )
{
e = m_Events[ i ];
if ( !e )
continue;
PROCESSING_TYPE disposition;
m_nActiveEvents += EventThink( e, m_flCurrentTime, curtime, playing_forward, disposition );
if ( disposition != PROCESSING_TYPE_IGNORE )
{
ActiveList entry;
entry.e = e;
entry.pt = disposition;
pending.Insert( entry );
}
}
// Events are sorted start time and then by channel and actor slot or by name if those aren't equal
bool dump = false;
i = pending.FirstInorder();
while ( i != pending.InvalidIndex() )
{
ActiveList *entry = &pending[ i ];
Assert( entry->e );
if ( dump )
{
Msg( "%f == %s starting at %f (actor %p channel %p)\n",
m_flCurrentTime, entry->e->GetName(), entry->e->GetStartTime(),
entry->e->GetActor(), entry->e->GetChannel() );
}
switch ( entry->pt )
{
default:
case PROCESSING_TYPE_IGNORE:
{
Assert( 0 );
}
break;
case PROCESSING_TYPE_START:
case PROCESSING_TYPE_START_RESUMECONDITION:
{
entry->e->StartProcessing( m_pIChoreoEventCallback, this, m_flCurrentTime );
if ( entry->pt == PROCESSING_TYPE_START_RESUMECONDITION )
{
Assert( entry->e->IsResumeCondition() );
m_ActiveResumeConditions.AddToTail( entry->e );
}
// This event can "pause" the scene, so we need to remember who "paused" the scene so that
// when we resume we can resume any suppressed events dependent on this pauser...
if ( entry->e->GetType() == CChoreoEvent::SECTION )
{
// So this event should be in the pauseevents list, otherwise this'll be -1
m_nLastPauseEvent = m_PauseEvents.Find( entry->e );
}
}
break;
case PROCESSING_TYPE_CONTINUE:
{
entry->e->ContinueProcessing( m_pIChoreoEventCallback, this, m_flCurrentTime );
}
break;
case PROCESSING_TYPE_STOP:
{
entry->e->StopProcessing( m_pIChoreoEventCallback, this, m_flCurrentTime );
}
break;
}
i = pending.NextInorder( i );
}
if ( dump )
{
Msg( "\n" );
}
// If a Process call slams this time, don't override it!!!
if ( oldt == m_flCurrentTime )
{
m_flCurrentTime = curtime;
}
// Still processing?
if ( m_nActiveEvents )
{
m_flLastActiveTime = m_flCurrentTime;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : float
//-----------------------------------------------------------------------------
float CChoreoScene::GetTime( void )
{
return m_flCurrentTime;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : t -
//-----------------------------------------------------------------------------
void CChoreoScene::SetTime( float t )
{
m_flCurrentTime = t;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : t -
//-----------------------------------------------------------------------------
void CChoreoScene::LoopToTime( float t )
{
m_flCurrentTime = t;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pfn -
//-----------------------------------------------------------------------------
void CChoreoScene::SetPrintFunc( void ( *pfn ) ( const char *fmt, ... ) )
{
m_pfnPrint = pfn;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
//-----------------------------------------------------------------------------
void CChoreoScene::RemoveActor( CChoreoActor *actor )
{
int idx = FindActorIndex( actor );
if ( idx == -1 )
return;
m_Actors.Remove( idx );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::FindActorIndex( CChoreoActor *actor )
{
for ( int i = 0; i < m_Actors.Size(); i++ )
{
if ( actor == m_Actors[ i ] )
{
return i;
}
}
return -1;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : a1 -
// a2 -
//-----------------------------------------------------------------------------
void CChoreoScene::SwapActors( int a1, int a2 )
{
CChoreoActor *temp;
temp = m_Actors[ a1 ];
m_Actors[ a1 ] = m_Actors[ a2 ];
m_Actors[ a2 ] = temp;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
//-----------------------------------------------------------------------------
void CChoreoScene::DeleteReferencedObjects( CChoreoActor *actor )
{
for ( int i = 0; i < actor->GetNumChannels(); i++ )
{
CChoreoChannel *channel = actor->GetChannel( i );
actor->RemoveChannel( channel );
DeleteReferencedObjects( channel );
}
DestroyActor( actor );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *channel -
//-----------------------------------------------------------------------------
void CChoreoScene::DeleteReferencedObjects( CChoreoChannel *channel )
{
for ( int i = 0; i < channel->GetNumEvents(); i++ )
{
CChoreoEvent *event = channel->GetEvent( i );
channel->RemoveEvent( event );
DeleteReferencedObjects( event );
}
DestroyChannel( channel );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *event -
//-----------------------------------------------------------------------------
void CChoreoScene::DeleteReferencedObjects( CChoreoEvent *event )
{
int idx = m_PauseEvents.Find( event );
if ( idx != m_PauseEvents.InvalidIndex() )
{
m_PauseEvents.Remove( idx );
}
// Events don't reference anything lower
DestroyEvent( event );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
//-----------------------------------------------------------------------------
void CChoreoScene::DestroyActor( CChoreoActor *actor )
{
int size = m_Actors.Size();
for ( int i = size - 1; i >= 0; i-- )
{
CChoreoActor *a = m_Actors[ i ];
if ( a == actor )
{
m_Actors.Remove( i );
}
}
delete actor;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *channel -
//-----------------------------------------------------------------------------
void CChoreoScene::DestroyChannel( CChoreoChannel *channel )
{
int size = m_Channels.Size();
for ( int i = size - 1; i >= 0; i-- )
{
CChoreoChannel *c = m_Channels[ i ];
if ( c == channel )
{
m_Channels.Remove( i );
}
}
delete channel;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *event -
//-----------------------------------------------------------------------------
void CChoreoScene::DestroyEvent( CChoreoEvent *event )
{
int size = m_Events.Size();
for ( int i = size - 1; i >= 0; i-- )
{
CChoreoEvent *e = m_Events[ i ];
if ( e == event )
{
m_Events.Remove( i );
}
}
delete event;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ResumeSimulation( void )
{
// If the thing that paused us was a SECTION pause event, then this will be set
if ( m_nLastPauseEvent >= 0 &&
m_nLastPauseEvent < m_PauseEvents.Count() )
{
// Start any suppressed dependencies immediately, should only be .wav files!!!
// These are .wav files which are placed at or just after the SECTION pause event
// in the .vcd, but due to the user's sound system latency, they would have triggered before the
// pause (we pre-queue sounds). Since we suppressed that, we need to unsupress / start these sounds
// now that the SECTION pause is being resumed from
CUtlVector< CChoreoEvent * > deps;
CChoreoEvent *pauseEvent = m_PauseEvents[ m_nLastPauseEvent ];
Assert( pauseEvent );
// Sanity check ( this should be about 1 tick usually 15 msec)
float timeSincePaused = m_flCurrentTime - pauseEvent->GetStartTime();
if ( fabs( timeSincePaused ) > 1.0f )
{
Assert( !"Resume simulation with unexpected pause event" );
}
// Snag any sounds which were suppressed by this issue
pauseEvent->GetEventDependencies( deps );
for ( int j = 0; j < deps.Count(); ++j )
{
CChoreoEvent *startEvent = deps[ j ];
Assert( startEvent );
// Start them now. Yes, they won't pre-queue, but it's better than totally skipping the sound!!!
startEvent->StartProcessing( m_pIChoreoEventCallback, this, m_flCurrentTime );
}
}
// Reset section pause signal
m_nLastPauseEvent = -1;
m_ActiveResumeConditions.RemoveAll();
}
// Sound system needs to have sounds pre-queued by this much time
void CChoreoScene::SetSoundFileStartupLatency( float time )
{
Assert( time >= 0 );
m_flSoundSystemLatency = time;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : start -
// end -
//-----------------------------------------------------------------------------
void CChoreoScene::GetSceneTimes( float& start, float& end )
{
start = m_flStartTime;
end = m_flEndTime;
}
//-----------------------------------------------------------------------------
// Purpose: Do housekeeping on times that are relative to tags
//-----------------------------------------------------------------------------
void CChoreoScene::ReconcileTags( void )
{
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
for ( int k = 0 ; k < c->GetNumEvents(); k++ )
{
CChoreoEvent *e = c->GetEvent( k );
if ( !e )
continue;
if ( !e->IsUsingRelativeTag() )
continue;
CEventRelativeTag *tag = FindTagByName(
e->GetRelativeWavName(),
e->GetRelativeTagName() );
if ( tag )
{
// Determine correct starting time based on tag
float starttime = tag->GetStartTime();
// Figure out delta
float dt = starttime - e->GetStartTime();
// Fix up start and possible end time
e->OffsetTime( dt );
}
else
{
// The tag was missing!!! unflag it
choreoprintf( 0, "Event %s was missing tag %s for wav %s\n",
e->GetName(), e->GetRelativeWavName(), e->GetRelativeTagName() );
e->SetUsingRelativeTag( false, "", "" );
}
}
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *wavname -
// *name -
// Output : CChoreoEvent
//-----------------------------------------------------------------------------
CChoreoEvent *CChoreoScene::FindTargetingEvent( const char *wavname, const char *name )
{
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
for ( int k = 0 ; k < c->GetNumEvents(); k++ )
{
CChoreoEvent *e = c->GetEvent( k );
if ( !e )
continue;
if ( !e->IsUsingRelativeTag() )
continue;
if ( stricmp( wavname, e->GetRelativeWavName() ) )
continue;
if ( stricmp( name, e->GetRelativeTagName() ) )
continue;
return e;
}
}
}
return NULL;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *wavname -
// *name -
// Output : CEventRelativeTag
//-----------------------------------------------------------------------------
CEventRelativeTag *CChoreoScene::FindTagByName( const char *wavname, const char *name )
{
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
for ( int k = 0 ; k < c->GetNumEvents(); k++ )
{
CChoreoEvent *e = c->GetEvent( k );
if ( !e )
continue;
if ( e->GetType() != CChoreoEvent::SPEAK )
continue;
// Search for tag by name
if ( !strstr( e->GetParameters(), wavname ) )
continue;
CEventRelativeTag *tag = e->FindRelativeTag( name );
if ( !tag )
continue;
return tag;
}
}
}
return NULL;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *filename -
//-----------------------------------------------------------------------------
void CChoreoScene::ExportEvents( const char *filename, CUtlVector< CChoreoEvent * >& events )
{
if ( events.Size() <= 0 )
return;
// Create a serialization buffer
CUtlBuffer buf( 0, 0, true );
FilePrintf( buf, 0, "// Choreo version 1: <%i> Exported Events\n", events.Size() );
// Save out the selected events.
int i;
for ( i = 0 ; i < events.Size(); i++ )
{
CChoreoEvent *e = events[ i ];
if ( !e->GetActor() )
continue;
FileSaveEvent( buf, 0, e );
}
// Write it out baby
FileHandle_t fh = SceneFileSystem()->Open( filename, "wt" );
if (fh)
{
SceneFileSystem()->Write( buf.Base(), buf.TellPut(), fh );
SceneFileSystem()->Close(fh);
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *actor -
// *channel -
// starttime -
//-----------------------------------------------------------------------------
void CChoreoScene::ImportEvents( ISceneTokenProcessor *tokenizer, CChoreoActor *actor, CChoreoChannel *channel )
{
m_pTokenizer = tokenizer;
while ( 1 )
{
if ( !m_pTokenizer->GetToken( true ) )
{
break;
}
if ( strlen( m_pTokenizer->CurrentToken() ) <= 0 )
break;
if ( !Q_stricmp( m_pTokenizer->CurrentToken(), "event" ) )
{
ParseEvent( actor, channel );
}
else
{
m_pTokenizer->Error( "unexpected token %s\n", m_pTokenizer->CurrentToken() );
break;
}
}
// Fixup time tags
ReconcileTags();
}
void CChoreoScene::SetSubScene( bool sub )
{
m_bSubScene = sub;
}
bool CChoreoScene::IsSubScene( void ) const
{
return m_bSubScene;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::GetSceneFPS( void ) const
{
return m_nSceneFPS;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : fps -
//-----------------------------------------------------------------------------
void CChoreoScene::SetSceneFPS( int fps )
{
m_nSceneFPS = fps;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CChoreoScene::IsUsingFrameSnap( void ) const
{
return m_bUseFrameSnap;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : snap -
//-----------------------------------------------------------------------------
void CChoreoScene::SetUsingFrameSnap( bool snap )
{
m_bUseFrameSnap = snap;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : t -
// Output : float
//-----------------------------------------------------------------------------
float CChoreoScene::SnapTime( float t )
{
if ( !IsUsingFrameSnap() )
return t;
float fps = (float)GetSceneFPS();
Assert( fps > 0 );
int itime = (int)( t * fps + 0.5f );
t = (float)itime / fps;
// FIXME: If FPS is set and "using grid", snap to proper fractional time value
return t;
}
EdgeInfo_t *CChoreoScene::GetSceneRampEdgeInfo( int idx )
{
return &m_SceneRampEdgeInfo[ idx ];
}
float CChoreoScene::GetSceneRampIntensity( float time )
{
return CChoreoEvent::GetRampIntensity( this, time );
}
int CChoreoScene::GetSceneRampCount( void )
{
return m_SceneRamp.Count();
}
CExpressionSample *CChoreoScene::GetSceneRamp( int index )
{
if ( index < 0 || index >= GetSceneRampCount() )
return NULL;
return &m_SceneRamp[ index ];
}
CExpressionSample *CChoreoScene::AddSceneRamp( float time, float value, bool selected )
{
CExpressionSample sample;
sample.time = time;
sample.value = value;
sample.selected = selected;
int idx = m_SceneRamp.AddToTail( sample );
return &m_SceneRamp[ idx ];
}
void CChoreoScene::DeleteSceneRamp( int index )
{
if ( index < 0 || index >= GetSceneRampCount() )
return;
m_SceneRamp.Remove( index );
}
void CChoreoScene::ClearSceneRamp( void )
{
m_SceneRamp.RemoveAll();
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ResortSceneRamp( void )
{
for ( int i = 0; i < m_SceneRamp.Size(); i++ )
{
for ( int j = i + 1; j < m_SceneRamp.Size(); j++ )
{
CExpressionSample src = m_SceneRamp[ i ];
CExpressionSample dest = m_SceneRamp[ j ];
if ( src.time > dest.time )
{
m_SceneRamp[ i ] = dest;
m_SceneRamp[ j ] = src;
}
}
}
RemoveOutOfRangeSceneRampSamples();
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : number -
// Output : CExpressionSample
//-----------------------------------------------------------------------------
CExpressionSample *CChoreoScene::GetBoundedSceneRamp( int number, bool& bClamped )
{
if ( number < 0 )
{
// Search for two samples which span time f
static CExpressionSample nullstart;
nullstart.time = 0.0f;
nullstart.value = SceneRampGetEdgeZeroValue( true );
nullstart.SetCurveType( SceneRampGetEdgeCurveType( true ) );
bClamped = true;
return &nullstart;
}
else if ( number >= GetSceneRampCount() )
{
static CExpressionSample nullend;
nullend.time = FindStopTime();
nullend.value = SceneRampGetEdgeZeroValue( false );
nullend.SetCurveType( SceneRampGetEdgeCurveType( false ) );
bClamped = true;
return &nullend;
}
bClamped = false;
return GetSceneRamp( number );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::RemoveOutOfRangeSceneRampSamples( void )
{
float duration = FindStopTime();
int c = GetSceneRampCount();
for ( int i = c-1; i >= 0; i-- )
{
CExpressionSample src = m_SceneRamp[ i ];
if ( src.time < 0 ||
src.time > duration + 0.01 )
{
m_SceneRamp.Remove( i );
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::ReconcileGestureTimes()
{
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
c->ReconcileGestureTimes();
}
}
}
int CChoreoScene::TimeZoomFirst()
{
return m_TimeZoomLookup.First();
}
int CChoreoScene::TimeZoomNext( int i )
{
return m_TimeZoomLookup.Next( i );
}
int CChoreoScene::TimeZoomInvalid() const
{
return m_TimeZoomLookup.InvalidIndex();
}
char const *CChoreoScene::TimeZoomName( int i )
{
return m_TimeZoomLookup.GetElementName( i );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *tool -
// Output : int
//-----------------------------------------------------------------------------
int CChoreoScene::GetTimeZoom( char const *tool )
{
// If not present add it
int idx = m_TimeZoomLookup.Find( tool );
if ( idx == m_TimeZoomLookup.InvalidIndex() )
{
idx = m_TimeZoomLookup.Insert( tool, 100 );
}
return m_TimeZoomLookup[ idx ];
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *tool -
// tz -
//-----------------------------------------------------------------------------
void CChoreoScene::SetTimeZoom( char const *tool, int tz )
{
// If not present add it
int idx = m_TimeZoomLookup.Find( tool );
if ( idx == m_TimeZoomLookup.InvalidIndex() )
{
idx = m_TimeZoomLookup.Insert( tool, 100 );
}
m_TimeZoomLookup[ idx ] = tz;
}
void CChoreoScene::ParseScaleSettings( ISceneTokenProcessor *tokenizer, CChoreoScene *scene )
{
tokenizer->GetToken( true );
if ( stricmp( tokenizer->CurrentToken(), "{" ) )
tokenizer->Error( "expecting {\n" );
while ( 1 )
{
// Parse until }
tokenizer->GetToken( true );
if ( strlen( tokenizer->CurrentToken() ) <= 0 )
{
tokenizer->Error( "expecting scalesettings data\n" );
break;
}
if ( !Q_stricmp( tokenizer->CurrentToken(), "}" ) )
break;
char tool[ 256 ];
Q_strncpy( tool, tokenizer->CurrentToken(), sizeof( tool ) );
tokenizer->GetToken( false );
int tz = Q_atoi( tokenizer->CurrentToken() );
if ( tz <= 0 )
tz = 100;
scene->SetTimeZoom( tool, tz );
}
}
// Merges two .vcd's together
bool CChoreoScene::Merge( CChoreoScene *other )
{
int acount = 0;
int ccount = 0;
int ecount = 0;
// Look for events that don't have actor/channel set
int i;
for ( i = 0 ; i < other->m_Events.Size(); i++ )
{
CChoreoEvent *e = other->m_Events[ i ];
if ( e->GetActor() )
continue;
MEM_ALLOC_CREDIT();
// Make a copy of the other event and add it to this scene
CChoreoEvent *newEvent = AllocEvent();
*newEvent = *e;
newEvent->SetScene( this );
ecount++;
}
for ( i = 0 ; i < other->m_Actors.Size(); i++ )
{
CChoreoActor *a = other->m_Actors[ i ];
// See if that actor already exists
bool newActor = false;
CChoreoActor *destActor = FindActor( a->GetName() );
if ( !destActor )
{
newActor = true;
destActor = AllocActor();
*destActor = *a;
destActor->RemoveAllChannels();
acount++;
}
// Now we have a destination actor, work on channels
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *ch = a->GetChannel( j );
bool newChannel = false;
CChoreoChannel *destChannel = NULL;
destChannel = destActor->FindChannel( ch->GetName() );
if ( !destChannel )
{
destChannel = AllocChannel();
*destChannel = *ch;
destChannel->RemoveAllEvents();
newChannel = true;
ccount++;
}
if ( newChannel )
{
destActor->AddChannel( destChannel );
destChannel->SetActor( destActor );
}
// Now we have a destination channel, work on events themselves
for ( int k = 0 ; k < ch->GetNumEvents(); k++ )
{
CChoreoEvent *e = ch->GetEvent( k );
// Just import them wholesale, no checking
MEM_ALLOC_CREDIT();
CChoreoEvent *newEvent = AllocEvent();
*newEvent = *e;
newEvent->SetScene( this );
destChannel->AddEvent( newEvent );
newEvent->SetChannel( destChannel );
newEvent->SetActor( destActor );
ecount++;
}
}
}
Msg( "Merged in (%i) actors, (%i) channels, and (%i) events\n",
acount, ccount, ecount );
return ( ecount || acount || ccount );
}
//-----------------------------------------------------------------------------
// Purpose: Updates master/slave status info per channel
//-----------------------------------------------------------------------------
void CChoreoScene::ReconcileCloseCaption()
{
for ( int i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
for ( int j = 0; j < a->GetNumChannels(); j++ )
{
CChoreoChannel *c = a->GetChannel( j );
if ( !c )
continue;
c->ReconcileCloseCaption();
}
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : char const
//-----------------------------------------------------------------------------
char const *CChoreoScene::GetFilename() const
{
return m_szFileName;
}
void CChoreoScene::SetFileName( char const *fn )
{
Q_strncpy( m_szFileName, fn, sizeof( m_szFileName ) );
}
//-----------------------------------------------------------------------------
// Purpose: Returns true if this scene has speech events that haven't played yet
//-----------------------------------------------------------------------------
bool CChoreoScene::HasUnplayedSpeech()
{
for ( int i = 0; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetType() == CChoreoEvent::SPEAK )
{
// Have we played it yet?
if ( m_flCurrentTime < e->GetStartTime() )
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose: Returns true if this scene has flex animation events that are playing
//-----------------------------------------------------------------------------
bool CChoreoScene::HasFlexAnimation()
{
for ( int i = 0; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetType() == CChoreoEvent::FLEXANIMATION )
{
// Have we played it yet?
if ( m_flCurrentTime >= e->GetStartTime() && m_flCurrentTime <= e->GetEndTime() )
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CChoreoScene::SetBackground( bool bIsBackground )
{
m_bIsBackground = bIsBackground;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
bool CChoreoScene::IsBackground( )
{
return m_bIsBackground;
}
bool CChoreoScene::HasEventsOfType( CChoreoEvent::EVENTTYPE type ) const
{
return m_bitvecHasEventOfType.IsBitSet( type );
}
void CChoreoScene::SceneRampSetEdgeInfo( bool leftEdge, int curveType, float zero )
{
int idx = leftEdge ? 0 : 1;
m_SceneRampEdgeInfo[ idx ].m_CurveType = curveType;
m_SceneRampEdgeInfo[ idx ].m_flZeroPos = zero;
}
void CChoreoScene::SceneRampGetEdgeInfo( bool leftEdge, int& curveType, float& zero ) const
{
int idx = leftEdge ? 0 : 1;
curveType = m_SceneRampEdgeInfo[ idx ].m_CurveType;
zero = m_SceneRampEdgeInfo[ idx ].m_flZeroPos;
}
void CChoreoScene::SceneRampSetEdgeActive( bool leftEdge, bool state )
{
int idx = leftEdge ? 0 : 1;
m_SceneRampEdgeInfo[ idx ].m_bActive = state;
}
bool CChoreoScene::SceneRampIsEdgeActive( bool leftEdge ) const
{
int idx = leftEdge ? 0 : 1;
return m_SceneRampEdgeInfo[ idx ].m_bActive;
}
int CChoreoScene::SceneRampGetEdgeCurveType( bool leftEdge ) const
{
if ( !SceneRampIsEdgeActive( leftEdge ) )
{
return CURVE_DEFAULT;
}
int idx = leftEdge ? 0 : 1;
return m_SceneRampEdgeInfo[ idx ].m_CurveType;
}
float CChoreoScene::SceneRampGetEdgeZeroValue( bool leftEdge ) const
{
if ( !SceneRampIsEdgeActive( leftEdge ) )
{
return 0.0f;
}
int idx = leftEdge ? 0 : 1;
return m_SceneRampEdgeInfo[ idx ].m_flZeroPos;
}
// ICurveDataAccessor method
bool CChoreoScene::CurveHasEndTime()
{
return true;
}
// ICurveDataAccessor method
int CChoreoScene::CurveGetSampleCount()
{
return GetSceneRampCount();
}
// ICurveDataAccessor method
CExpressionSample *CChoreoScene::CurveGetBoundedSample( int idx, bool& bClamped )
{
return GetBoundedSceneRamp( idx, bClamped );
}
int CChoreoScene::GetDefaultCurveType()
{
return CURVE_CATMULL_ROM_TO_CATMULL_ROM;
}
#define SCENE_BINARY_VERSION 0x01
#define SCENE_TAG MAKEID( 'x', 'v', 'c', 'd' )
bool CChoreoScene::SaveBinary( char const *pszBinaryFileName, char const *pPathID, unsigned int nTextVersionCRC )
{
bool bret = false;
CUtlBuffer buf;
SaveToBuffer( buf, nTextVersionCRC );
if ( SceneFileSystem()->FileExists( pszBinaryFileName, pPathID ) &&
!SceneFileSystem()->IsFileWritable( pszBinaryFileName, pPathID ) )
{
Warning( "Forcing '%s' to be writable!!!\n", pszBinaryFileName );
SceneFileSystem()->SetFileWritable( pszBinaryFileName, true, pPathID );
}
FileHandle_t fh = SceneFileSystem()->Open( pszBinaryFileName, "wb", pPathID );
if ( FILESYSTEM_INVALID_HANDLE != fh )
{
SceneFileSystem()->Write( buf.Base(), buf.TellPut(), fh );
// pad to 512 byte sector size
int align = (buf.TellPut() + 511) & ~511;
int padLength = align - buf.TellPut();
char padByte = 0;
for ( int i=0; i<padLength; i++ )
{
SceneFileSystem()->Write( &padByte, 1, fh );
}
SceneFileSystem()->Close( fh );
// Success
bret = true;
}
else
{
Warning( "Unable to open '%s' for writing!!!\n", pszBinaryFileName );
}
return bret;
}
void CChoreoScene::SaveToBuffer( CUtlBuffer& buf, unsigned int nTextVersionCRC )
{
buf.PutInt( SCENE_TAG );
buf.PutChar( SCENE_BINARY_VERSION );
buf.PutInt( nTextVersionCRC );
CUtlVector< CChoreoEvent * > eventList;
// Look for events that don't have actor/channel set
int i;
for ( i = 0 ; i < m_Events.Size(); i++ )
{
CChoreoEvent *e = m_Events[ i ];
if ( e->GetActor() )
continue;
eventList.AddToTail( e );
}
int c = eventList.Count();
buf.PutShort( c );
for ( i = 0; i < c; ++i )
{
CChoreoEvent *e = eventList[ i ];
e->SaveToBuffer( buf, this );
}
// Now serialize the actors themselves
CUtlVector< CChoreoActor * > actorList;
for ( i = 0 ; i < m_Actors.Size(); i++ )
{
CChoreoActor *a = m_Actors[ i ];
if ( !a )
continue;
actorList.AddToTail( a );
}
c = actorList.Count();
buf.PutShort( c );
for ( i = 0; i < c; ++i )
{
CChoreoActor *a = actorList[ i ];
a->SaveToBuffer( buf, this );
}
/*
// compiled version strips out map name, only used by editor
if ( m_szMapname[ 0 ] )
{
FilePrintf( buf, 0, "mapname \"%s\"\n", m_szMapname );
}
*/
SaveSceneRampToBuffer( buf );
/*
// compiled version strips out scale settings fps and snap, only used by editor
FileSaveScaleSettings( buf, 0, this );
FilePrintf( buf, 0, "fps %i\n", m_nSceneFPS );
FilePrintf( buf, 0, "snap %s\n", m_bUseFrameSnap ? "on" : "off" );
*/
}
//-----------------------------------------------------------------------------
// Purpose: Static method to extract just the CRC from a binary .xcd file
// Input : buf -
// crc -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CChoreoScene::GetCRCFromBuffer( CUtlBuffer& buf, unsigned int& crc )
{
bool bret = false;
int pos = buf.TellGet();
int tag = buf.GetInt();
if ( tag == SCENE_TAG )
{
byte ver = buf.GetChar();
if ( ver == SCENE_BINARY_VERSION )
{
bret = true;
crc = (unsigned int)buf.GetInt();
}
}
buf.SeekGet( CUtlBuffer::SEEK_HEAD, pos );
return bret;
}
bool CChoreoScene::RestoreFromBuffer( CUtlBuffer& buf, char const *filename )
{
Q_strncpy( m_szFileName, filename, sizeof( m_szFileName ) );
int tag = buf.GetInt();
if ( tag != SCENE_TAG )
return false;
byte ver = buf.GetChar();
if ( ver != SCENE_BINARY_VERSION )
return false;
// Skip the CRC
buf.GetInt();
int i;
int eventCount = buf.GetShort();
for ( i = 0; i < eventCount; ++i )
{
MEM_ALLOC_CREDIT();
CChoreoEvent *e = AllocEvent();
Assert( e );
if ( e->RestoreFromBuffer( buf, this ) )
{
continue;
}
return false;
}
int actorCount = buf.GetShort();
for ( i = 0; i < actorCount; ++i )
{
CChoreoActor *a = AllocActor();
Assert( a );
if ( a->RestoreFromBuffer( buf, this ) )
{
continue;
}
return false;
}
if ( !ParseSceneRampFromBuffer( buf ) )
{
return false;
}
// FIXME: Are these ever needed on restore?
// ReconcileTags();
// ReconcileGestureTimes();
ReconcileCloseCaption();
if ( CChoreoScene::s_bEditingDisabled )
{
m_flPrecomputedStopTime = FindStopTime();
}
return true;
}
void CChoreoScene::SaveSceneRampToBuffer( CUtlBuffer& buf )
{
// Nothing to save?
int c = GetSceneRampCount();
buf.PutInt( c );
if ( c <= 0 )
return;
for ( int i = 0; i < c; i++ )
{
CExpressionSample *sample = GetSceneRamp( i );
buf.PutFloat( sample->time );
buf.PutFloat( sample->value );
}
}
bool CChoreoScene::ParseSceneRampFromBuffer( CUtlBuffer& buf )
{
int c = buf.GetInt();
if ( c <= 0 )
return true;
for ( int i = 0; i < c; i++ )
{
float t = buf.GetFloat();
float v = buf.GetFloat();
AddSceneRamp( t, v, false );
}
return true;
}