source-engine/engine/cl_demosmootherpanel.cpp

2730 lines
61 KiB
C++
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//===========================================================================//
#include "client_pch.h"
#include "cl_demosmootherpanel.h"
#include <vgui_controls/Button.h>
#include <vgui_controls/CheckButton.h>
#include <vgui_controls/Label.h>
#include <vgui_controls/Controls.h>
#include <vgui/ISystem.h>
#include <vgui/ISurface.h>
#include <vgui_controls/PropertySheet.h>
#include <vgui/IVGui.h>
#include <vgui_controls/FileOpenDialog.h>
#include <vgui_controls/ProgressBar.h>
#include <vgui_controls/ListPanel.h>
#include <vgui_controls/MenuButton.h>
#include <vgui_controls/Menu.h>
#include <vgui_controls/TextEntry.h>
#include <vgui/IInput.h>
#include "cl_demouipanel.h"
#include "demofile/demoformat.h"
#include "cl_demoactionmanager.h"
#include "tier2/renderutils.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
using namespace vgui;
static float Ease_In( float t )
{
float out = sqrt( t );
return out;
}
static float Ease_Out( float t )
{
float out = t * t;
return out;
}
static float Ease_Both( float t )
{
return SimpleSpline( t );
}
//-----------------------------------------------------------------------------
// Purpose: A menu button that knows how to parse cvar/command menu data from gamedir\scripts\debugmenu.txt
//-----------------------------------------------------------------------------
class CSmoothingTypeButton : public vgui::MenuButton
{
typedef vgui::MenuButton BaseClass;
public:
// Construction
CSmoothingTypeButton( vgui::Panel *parent, const char *panelName, const char *text );
private:
// Menu associated with this button
Menu *m_pMenu;
};
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
CSmoothingTypeButton::CSmoothingTypeButton(Panel *parent, const char *panelName, const char *text)
: BaseClass( parent, panelName, text )
{
// Assume no menu
m_pMenu = new Menu( this, "DemoSmootherTypeMenu" );
m_pMenu->AddMenuItem( "Smooth Selection Angles", "smoothselectionangles", parent );
m_pMenu->AddMenuItem( "Smooth Selection Origin", "smoothselectionorigin", parent );
m_pMenu->AddMenuItem( "Linear Interp Angles", "smoothlinearinterpolateangles", parent );
m_pMenu->AddMenuItem( "Linear Interp Origin", "smoothlinearinterpolateorigin", parent );
m_pMenu->AddMenuItem( "Spline Angles", "splineangles", parent );
m_pMenu->AddMenuItem( "Spline Origin", "splineorigin", parent );
m_pMenu->AddMenuItem( "Look At Points", "lookatpoints", parent );
m_pMenu->AddMenuItem( "Look At Points Spline", "lookatpointsspline", parent );
m_pMenu->AddMenuItem( "Two Point Origin Ease Out", "origineaseout", parent );
m_pMenu->AddMenuItem( "Two Point Origin Ease In", "origineasein", parent );
m_pMenu->AddMenuItem( "Two Point Origin Ease In/Out", "origineaseboth", parent );
m_pMenu->AddMenuItem( "Auto-setup keys 1/2 second", "keyshalf", parent );
m_pMenu->AddMenuItem( "Auto-setup keys 1 second", "keys1", parent );
m_pMenu->AddMenuItem( "Auto-setup keys 2 second", "keys2", parent );
m_pMenu->AddMenuItem( "Auto-setup keys 4 second", "keys4", parent );
m_pMenu->MakePopup();
MenuButton::SetMenu(m_pMenu);
SetOpenDirection(Menu::UP);
}
//-----------------------------------------------------------------------------
// Purpose: A menu button that knows how to parse cvar/command menu data from gamedir\scripts\debugmenu.txt
//-----------------------------------------------------------------------------
class CFixEdgeButton : public vgui::MenuButton
{
typedef vgui::MenuButton BaseClass;
public:
// Construction
CFixEdgeButton( vgui::Panel *parent, const char *panelName, const char *text );
private:
// Menu associated with this button
Menu *m_pMenu;
};
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
CFixEdgeButton::CFixEdgeButton(Panel *parent, const char *panelName, const char *text)
: BaseClass( parent, panelName, text )
{
// Assume no menu
m_pMenu = new Menu( this, "DemoSmootherEdgeFixType" );
m_pMenu->AddMenuItem( "Smooth Left", "smoothleft", parent );
m_pMenu->AddMenuItem( "Smooth Right", "smoothright", parent );
m_pMenu->AddMenuItem( "Smooth Both", "smoothboth", parent );
m_pMenu->MakePopup();
MenuButton::SetMenu(m_pMenu);
SetOpenDirection(Menu::UP);
}
//-----------------------------------------------------------------------------
// Purpose: Basic help dialog
//-----------------------------------------------------------------------------
CDemoSmootherPanel::CDemoSmootherPanel( vgui::Panel *parent ) : Frame( parent, "DemoSmootherPanel")
{
int w = 440;
int h = 300;
SetSize( w, h );
SetTitle("Demo Smoother", true);
m_pType = new CSmoothingTypeButton( this, "DemoSmootherType", "Process->" );
m_pRevert = new vgui::Button( this, "DemoSmoothRevert", "Revert" );;
m_pOK = new vgui::Button( this, "DemoSmoothOk", "OK" );
m_pCancel = new vgui::Button( this, "DemoSmoothCancel", "Cancel" );
m_pSave = new vgui::Button( this, "DemoSmoothSave", "Save" );
m_pReloadFromDisk = new vgui::Button( this, "DemoSmoothReload", "Reload" );
m_pStartFrame = new vgui::TextEntry( this, "DemoSmoothStartFrame" );
m_pEndFrame = new vgui::TextEntry( this, "DemoSmoothEndFrame" );
m_pPreviewOriginal = new vgui::Button( this, "DemoSmoothPreviewOriginal", "Show Original" );
m_pPreviewProcessed = new vgui::Button( this, "DemoSmoothPreviewProcessed", "Show Processed" );
m_pBackOff = new vgui::CheckButton( this, "DemoSmoothBackoff", "Back off" );
m_pHideLegend = new vgui::CheckButton( this, "DemoSmoothHideLegend", "Hide legend" );
m_pHideOriginal = new vgui::CheckButton( this, "DemoSmoothHideOriginal", "Hide original" );
m_pHideProcessed = new vgui::CheckButton( this, "DemoSmoothHideProcessed", "Hide processed" );
m_pSelectionInfo = new vgui::Label( this, "DemoSmoothSelectionInfo", "" );
m_pShowAllSamples = new vgui::CheckButton( this, "DemoSmoothShowAll", "Show All" );
m_pSelectSamples = new vgui::Button( this, "DemoSmoothSelect", "Select" );
m_pPauseResume = new vgui::Button( this, "DemoSmoothPauseResume", "Pause" );
m_pStepForward = new vgui::Button( this, "DemoSmoothStepForward", ">>" );
m_pStepBackward = new vgui::Button( this, "DemoSmoothStepBackward", "<<" );
m_pRevertPoint = new vgui::Button( this, "DemoSmoothRevertPoint", "Revert Pt." );
m_pToggleKeyFrame = new vgui::Button( this, "DemoSmoothSetKeyFrame", "Mark Keyframe" );
m_pToggleLookTarget = new vgui::Button( this, "DemoSmoothSetLookTarget", "Mark Look Target" );
m_pUndo = new vgui::Button( this, "DemoSmoothUndo", "Undo" );
m_pRedo = new vgui::Button( this, "DemoSmoothRedo", "Redo" );
m_pNextKey = new vgui::Button( this, "DemoSmoothNextKey", "+Key" );
m_pPrevKey = new vgui::Button( this, "DemoSmoothPrevKey", "-Key" );
m_pNextTarget = new vgui::Button( this, "DemoSmoothNextTarget", "+Target" );
m_pPrevTarget = new vgui::Button( this, "DemoSmoothPrevTarget", "-Target" );
m_pMoveCameraToPoint = new vgui::Button( this, "DemoSmoothCameraAtPoint", "Set View" );
m_pFixEdges = new CFixEdgeButton( this, "DemoSmoothFixFrameButton", "Edge->" );
m_pFixEdgeFrames = new vgui::TextEntry( this, "DemoSmoothFixFrames" );
m_pProcessKey = new vgui::Button( this, "DemoSmoothSaveKey", "Save Key" );
m_pGotoFrame = new vgui::TextEntry( this, "DemoSmoothGotoFrame" );
m_pGoto = new vgui::Button( this, "DemoSmoothGoto", "Jump To" );
//m_pCurrentDemo = new vgui::Label( this, "DemoName", "" );
vgui::ivgui()->AddTickSignal( GetVPanel(), 0 );
LoadControlSettings("Resource\\DemoSmootherPanel.res");
/*
int xpos, ypos;
parent->GetPos( xpos, ypos );
ypos += parent->GetTall();
SetPos( xpos, ypos );
*/
OnRefresh();
SetVisible( true );
SetSizeable( false );
SetMoveable( true );
Reset();
m_vecEyeOffset = Vector( 0, 0, 64 );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CDemoSmootherPanel::~CDemoSmootherPanel()
{
}
void CDemoSmootherPanel::Reset( void )
{
ClearSmoothingInfo( m_Smoothing );
m_bPreviewing = false;
m_bPreviewPaused = false;
m_bPreviewOriginal = false;
m_iPreviewStartTick = 0;
m_fPreviewCurrentTime = 0.0f;
m_nPreviewLastFrame = 0;
m_bHasSelection = false;
memset( m_nSelection, 0, sizeof( m_nSelection ) );
m_iSelectionTicksSpan = 0;
m_bInputActive = false;
memset( m_nOldCursor, 0, sizeof( m_nOldCursor ) );
WipeUndo();
WipeRedo();
m_bRedoPending = false;
m_nUndoLevel = 0;
m_bDirty = false;
}
void CDemoSmootherPanel::OnTick()
{
BaseClass::OnTick();
m_pUndo->SetEnabled( CanUndo() );
m_pRedo->SetEnabled( CanRedo() );
m_pPauseResume->SetEnabled( m_bPreviewing );
m_pStepForward->SetEnabled( m_bPreviewing );
m_pStepBackward->SetEnabled( m_bPreviewing );
m_pSave->SetEnabled( m_bDirty );
demosmoothing_t *p = GetCurrent();
if ( p )
{
m_pToggleKeyFrame->SetEnabled( true );
m_pToggleLookTarget->SetEnabled( true );
m_pToggleKeyFrame->SetText( p->samplepoint ? "Delete Key" : "Make Key" );
m_pToggleLookTarget->SetText( p->targetpoint ? "Delete Target" : "Make Target" );
m_pProcessKey->SetEnabled( p->samplepoint );
}
else
{
m_pToggleKeyFrame->SetEnabled( false );
m_pToggleLookTarget->SetEnabled( false );
m_pProcessKey->SetEnabled( false );
}
if ( m_bPreviewing )
{
m_pPauseResume->SetText( m_bPreviewPaused ? "Resume" : "Pause" );
}
if ( !m_Smoothing.active )
{
m_pSelectionInfo->SetText( "No smoothing info loaded" );
return;
}
if ( !demoplayer->IsPlayingBack() )
{
m_pSelectionInfo->SetText( "Not playing back .dem" );
return;
}
if ( !m_bHasSelection )
{
m_pSelectionInfo->SetText( "No selection." );
return;
}
char sz[ 512 ];
if ( m_bPreviewing )
{
Q_snprintf( sz, sizeof( sz ), "%.3f at tick %i (%.3f s)",
m_fPreviewCurrentTime,
GetTickForFrame( m_nPreviewLastFrame ),
TICKS_TO_TIME( m_iSelectionTicksSpan ) );
}
else
{
Q_snprintf( sz, sizeof( sz ), "%i to %i (%.3f s)",
m_Smoothing.smooth[ m_nSelection[ 0 ] ].frametick,
m_Smoothing.smooth[ m_nSelection[ 1 ] ].frametick,
TICKS_TO_TIME( m_iSelectionTicksSpan ) );
}
m_pSelectionInfo->SetText( sz );
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CDemoSmootherPanel::CanEdit()
{
if ( !m_Smoothing.active )
return false;
if ( !demoplayer->IsPlayingBack() )
return false;
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *command -
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnCommand(const char *command)
{
if ( !Q_strcasecmp( command, "cancel" ) )
{
OnRevert();
MarkForDeletion();
Reset();
OnClose();
}
else if ( !Q_strcasecmp( command, "close" ) )
{
OnSave();
MarkForDeletion();
Reset();
OnClose();
}
else if ( !Q_strcasecmp( command, "gotoframe" ) )
{
OnGotoFrame();
}
else if ( !Q_strcasecmp( command, "undo" ) )
{
Undo();
}
else if ( !Q_strcasecmp( command, "redo" ) )
{
Redo();
}
else if ( !Q_strcasecmp( command, "revert" ) )
{
OnRevert();
}
else if ( !Q_strcasecmp( command, "original" ) )
{
OnPreview( true );
}
else if ( !Q_strcasecmp( command, "processed" ) )
{
OnPreview( false );
}
else if ( !Q_strcasecmp( command, "save" ) )
{
OnSave();
}
else if ( !Q_strcasecmp( command, "reload" ) )
{
OnReload();
}
else if ( !Q_strcasecmp( command, "select" ) )
{
OnSelect();
}
else if ( !Q_strcasecmp( command, "togglepause" ) )
{
OnTogglePause();
}
else if ( !Q_strcasecmp( command, "stepforward" ) )
{
OnStep( true );
}
else if ( !Q_strcasecmp( command, "stepbackward" ) )
{
OnStep( false );
}
else if ( !Q_strcasecmp( command, "revertpoint" ) )
{
OnRevertPoint();
}
else if ( !Q_strcasecmp( command, "keyframe" ) )
{
OnToggleKeyFrame();
}
else if ( !Q_strcasecmp( command, "looktarget" ) )
{
OnToggleLookTarget();
}
else if ( !Q_strcasecmp( command, "nextkey" ) )
{
OnNextKey();
}
else if ( !Q_strcasecmp( command, "prevkey" ) )
{
OnPrevKey();
}
else if ( !Q_strcasecmp( command, "nexttarget" ) )
{
OnNextTarget();
}
else if ( !Q_strcasecmp( command, "prevtarget" ) )
{
OnPrevTarget();
}
else if ( !Q_strcasecmp( command, "smoothselectionangles" ) )
{
OnSmoothSelectionAngles();
}
else if ( !Q_strcasecmp( command, "keyshalf" ) )
{
OnSetKeys( 0.5f );
}
else if ( !Q_strcasecmp( command, "keys1" ) )
{
OnSetKeys( 1.0f );
}
else if ( !Q_strcasecmp( command, "keys2" ) )
{
OnSetKeys( 2.0f );
}
else if ( !Q_strcasecmp( command, "keys4" ) )
{
OnSetKeys( 4.0f );
}
else if ( !Q_strcasecmp( command, "smoothselectionorigin" ) )
{
OnSmoothSelectionOrigin();
}
else if ( !Q_strcasecmp( command, "smoothlinearinterpolateangles" ) )
{
OnLinearInterpolateAnglesBasedOnEndpoints();
}
else if ( !Q_strcasecmp( command, "smoothlinearinterpolateorigin" ) )
{
OnLinearInterpolateOriginBasedOnEndpoints();
}
else if ( !Q_strcasecmp( command, "splineorigin" ) )
{
OnSplineSampleOrigin();
}
else if ( !Q_strcasecmp( command, "splineangles" ) )
{
OnSplineSampleAngles();
}
else if ( !Q_strcasecmp( command, "lookatpoints" ) )
{
OnLookAtPoints( false );
}
else if ( !Q_strcasecmp( command, "lookatpointsspline" ) )
{
OnLookAtPoints( true );
}
else if ( !Q_strcasecmp( command, "smoothleft" ) )
{
OnSmoothEdges( true, false );
}
else if ( !Q_strcasecmp( command, "smoothright" ) )
{
OnSmoothEdges( false, true );
}
else if ( !Q_strcasecmp( command, "smoothboth" ) )
{
OnSmoothEdges( true, true );
}
else if ( !Q_strcasecmp( command, "origineasein" ) )
{
OnOriginEaseCurve( Ease_In );
}
else if ( !Q_strcasecmp( command, "origineaseout" ) )
{
OnOriginEaseCurve( Ease_Out );
}
else if ( !Q_strcasecmp( command, "origineaseboth" ) )
{
OnOriginEaseCurve( Ease_Both );
}
else if ( !Q_strcasecmp( command, "processkey" ) )
{
OnSaveKey();
}
else if ( !Q_strcasecmp( command, "setview" ) )
{
OnSetView();
}
else
{
BaseClass::OnCommand( command );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnSave()
{
if ( !m_Smoothing.active )
return;
SaveSmoothingInfo( demoaction->GetCurrentDemoFile(), m_Smoothing );
WipeUndo();
m_bDirty = false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnReload()
{
WipeUndo();
WipeRedo();
LoadSmoothingInfo( demoaction->GetCurrentDemoFile(), m_Smoothing );
m_bDirty = false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnVDMChanged( void )
{
if ( IsVisible() )
{
OnReload();
}
else
{
Reset();
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnRevert()
{
OnRefresh();
if ( !m_Smoothing.active )
{
LoadSmoothingInfo( demoaction->GetCurrentDemoFile(), m_Smoothing );
WipeUndo();
WipeRedo();
}
else
{
ClearSmoothingInfo( m_Smoothing );
WipeUndo();
WipeRedo();
}
m_bDirty = false;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnRefresh()
{
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pScheme -
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::ApplySchemeSettings( vgui::IScheme *pScheme )
{
BaseClass::ApplySchemeSettings( pScheme );
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CDemoSmootherPanel::GetStartFrame()
{
char text[ 32 ];
m_pStartFrame->GetText( text, sizeof( text ) );
int tick = atoi( text );
return GetFrameForTick( tick );
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int CDemoSmootherPanel::GetEndFrame()
{
char text[ 32 ];
m_pEndFrame->GetText( text, sizeof( text ) );
int tick = atoi( text );
return GetFrameForTick( tick );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : original -
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnPreview( bool original )
{
if ( !CanEdit() )
return;
if ( !m_bHasSelection )
{
ConMsg( "Must have smoothing selection active\n" );
return;
}
m_bPreviewing = true;
m_bPreviewPaused = false;
m_bPreviewOriginal = original;
SetLastFrame( false, max( 0, m_nSelection[0] - 10 ) );
m_iPreviewStartTick = GetTickForFrame( m_nPreviewLastFrame );
m_fPreviewCurrentTime = TICKS_TO_TIME( m_iPreviewStartTick );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : frame -
// elapsed -
// info -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CDemoSmootherPanel::OverrideView( democmdinfo_t& info, int tick )
{
if ( !CanEdit() )
return false;
if ( !demoplayer->IsPlaybackPaused() )
return false;
if ( m_bPreviewing )
{
if ( m_bPreviewPaused && GetCurrent() && GetCurrent()->samplepoint )
{
info.viewOrigin = GetCurrent()->vecmoved;
info.viewAngles = GetCurrent()->angmoved;
info.localViewAngles = info.viewAngles;
bool back_off = m_pBackOff->IsSelected();
if ( back_off )
{
Vector fwd;
AngleVectors( info.viewAngles, &fwd, NULL, NULL );
info.viewOrigin -= fwd * 75.0f;
}
return true;
}
// TODO: Hook up previewing view
if ( !m_bPreviewPaused )
{
m_fPreviewCurrentTime += host_frametime;
}
if ( GetInterpolatedViewPoint( info.viewOrigin, info.viewAngles ) )
{
info.localViewAngles = info.viewAngles;
return true;
}
else
{
return false;
}
}
bool back_off = m_pBackOff->IsSelected();
if ( back_off )
{
int useframe = GetFrameForTick( tick );
if ( useframe < m_Smoothing.smooth.Count() && useframe >= 0 )
{
demosmoothing_t *p = &m_Smoothing.smooth[ useframe ];
Vector fwd;
AngleVectors( p->info.viewAngles, &fwd, NULL, NULL );
info.viewOrigin = p->info.viewOrigin - fwd * 75.0f;
}
}
return false;
}
void DrawVecForward( bool active, const Vector& origin, const QAngle& angles, int r, int g, int b )
{
Vector fwd;
AngleVectors( angles, &fwd, NULL, NULL );
Vector end;
end = origin + fwd * ( active ? 64 : 16 );
RenderLine( origin, end, Color( r, g, b, 255 ), true );
}
void GetColorForSample( bool original, bool samplepoint, bool targetpoint, demosmoothing_t *sample, int& r, int& g, int& b )
{
if ( samplepoint && sample->samplepoint )
{
r = 0;
g = 255;
b = 0;
return;
}
if ( targetpoint && sample->targetpoint )
{
r = 255;
g = 0;
b = 0;
return;
}
if ( sample->selected )
{
if( original )
{
r = 255;
g = 200;
b = 100;
}
else
{
r = 200;
g = 100;
b = 255;
}
if ( sample->samplepoint || sample->targetpoint )
{
r = 255;
g = 255;
b = 0;
}
return;
}
if ( original )
{
r = g = b = 255;
}
else
{
r = 150;
g = 255;
b = 100;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : origin -
// mins -
// maxs -
// angles -
// r -
// g -
// b -
// a -
//-----------------------------------------------------------------------------
void Draw_Box( const Vector& origin, const Vector& mins, const Vector& maxs, const QAngle& angles, int r, int g, int b, int a )
{
RenderBox( origin, angles, mins, maxs, Color( r, g, b, a ), false );
RenderWireframeBox( origin, angles, mins, maxs, Color( r, g, b, a ), true );
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *sample -
// *next -
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::DrawSmoothingSample( bool original, bool processed, int samplenumber, demosmoothing_t *sample, demosmoothing_t *next )
{
int r, g, b;
if ( original )
{
RenderLine( sample->info.viewOrigin + m_vecEyeOffset, next->info.viewOrigin + m_vecEyeOffset,
Color( 180, 180, 180, 255 ), true );
GetColorForSample( true, false, false, sample, r, g, b );
DrawVecForward( false, sample->info.viewOrigin + m_vecEyeOffset, sample->info.viewAngles, r, g, b );
}
if ( processed && sample->info.flags != 0 )
{
RenderLine( sample->info.GetViewOrigin() + m_vecEyeOffset, next->info.GetViewOrigin() + m_vecEyeOffset,
Color( 255, 255, 180, 255 ), true );
GetColorForSample( false, false, false, sample, r, g, b );
DrawVecForward( false, sample->info.GetViewOrigin() + m_vecEyeOffset, sample->info.GetViewAngles(), r, g, b );
}
if ( sample->samplepoint )
{
GetColorForSample( false, true, false, sample, r, g, b );
RenderBox( sample->vecmoved + m_vecEyeOffset, sample->angmoved, Vector( -2, -2, -2 ), Vector( 2, 2, 2 ), Color( r, g, b, 127 ), false );
DrawVecForward( false, sample->vecmoved + m_vecEyeOffset, sample->angmoved, r, g, b );
}
if ( sample->targetpoint )
{
GetColorForSample( false, false, true, sample, r, g, b );
RenderBox( sample->vectarget, vec3_angle, Vector( -2, -2, -2 ), Vector( 2, 2, 2 ), Color( r, g, b, 127 ), false );
}
if ( samplenumber == m_nPreviewLastFrame + 1 )
{
r = 50;
g = 100;
b = 250;
RenderBox( sample->info.GetViewOrigin() + m_vecEyeOffset, sample->info.GetViewAngles(), Vector( -2, -2, -2 ), Vector( 2, 2, 2 ), Color( r, g, b, 92 ), false );
}
if ( sample->targetpoint )
{
r = 200;
g = 200;
b = 220;
RenderLine( sample->info.GetViewOrigin() + m_vecEyeOffset, sample->vectarget, Color( r, g, b, 255 ), true );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::DrawDebuggingInfo( int frame, float elapsed )
{
if ( !CanEdit() )
return;
if ( !IsVisible() )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
int start = 0;
int end = c - 1;
bool showall = m_pShowAllSamples->IsSelected();
if ( !showall )
{
start = max( frame - 200, 0 );
end = min( frame + 200, c - 1 );
}
if ( m_bHasSelection && !showall )
{
start = max( m_nSelection[ 0 ] - 10, 0 );
end = min( m_nSelection[ 1 ] + 10, c - 1 );
}
bool draworiginal = !m_pHideOriginal->IsSelected();
bool drawprocessed = !m_pHideProcessed->IsSelected();
int i;
demosmoothing_t *p = NULL;
demosmoothing_t *prev = NULL;
for ( i = start; i < end; i++ )
{
p = &m_Smoothing.smooth[ i ];
if ( prev && p )
{
DrawSmoothingSample( draworiginal, drawprocessed, i, prev, p );
}
prev = p;
}
Vector org;
QAngle ang;
if ( m_bPreviewing )
{
if ( GetInterpolatedOriginAndAngles( true, org, ang ) )
{
DrawVecForward( true, org + m_vecEyeOffset, ang, 200, 10, 50 );
}
}
int useframe = frame;
useframe = clamp( useframe, 0, c - 1 );
if ( useframe < c )
{
p = &m_Smoothing.smooth[ useframe ];
org = p->info.GetViewOrigin();
ang = p->info.GetViewAngles();
DrawVecForward( true, org + m_vecEyeOffset, ang, 100, 220, 250 );
Draw_Box( org + m_vecEyeOffset, Vector( -1, -1, -1 ), Vector( 1, 1, 1 ), ang, 100, 220, 250, 127 );
}
DrawKeySpline();
DrawTargetSpline();
if ( !m_pHideLegend->IsSelected() )
{
DrawLegend( start, end );
}
}
void CDemoSmootherPanel::OnSelect()
{
if ( !CanEdit() )
return;
m_bHasSelection = false;
m_iSelectionTicksSpan = 0;
memset( m_nSelection, 0, sizeof( m_nSelection ) );
int start, end;
start = GetStartFrame();
end = GetEndFrame();
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
start = clamp( start, 0, c - 1 );
end = clamp( end, 0, c - 1 );
if ( start >= end )
return;
m_nSelection[ 0 ] = start;
m_nSelection[ 1 ] = end;
m_bHasSelection = true;
demosmoothing_t *startsample = &m_Smoothing.smooth[ start ];
demosmoothing_t *endsample = &m_Smoothing.smooth[ end ];
m_bDirty = true;
PushUndo( "select" );
int i = 0;
for ( i = 0; i < c; i++ )
{
if ( i >= start && i <= end )
{
m_Smoothing.smooth[ i ].selected = true;
}
else
{
m_Smoothing.smooth[ i ].selected = false;
}
}
PushRedo( "select" );
m_iSelectionTicksSpan = endsample->frametick - startsample->frametick;
}
int CDemoSmootherPanel::GetFrameForTick( int tick )
{
int count = m_Smoothing.smooth.Count();
int last = count - 1;
int first = m_Smoothing.m_nFirstSelectableSample;
if ( first > last )
return -1;
if ( count <= 0 )
{
return -1; // no valid index
}
else if ( count == 1 )
{
return 0; // return the one and only frame we have
}
if ( tick <= m_Smoothing.smooth[ first ].frametick )
return first;
if ( tick >= m_Smoothing.smooth[ last ].frametick )
return last;
// binary search
int middle;
while ( true )
{
middle = (first+last)/2;
int middleTick = m_Smoothing.smooth[ middle ].frametick;
if ( tick == middleTick )
return middle;
if ( tick > middleTick )
{
if ( first == middle )
return first;
first = middle;
}
else
{
if ( last == middle )
return last;
last = middle;
}
}
}
int CDemoSmootherPanel::GetTickForFrame( int frame )
{
if ( !CanEdit() )
return -1;
int c = m_Smoothing.smooth.Count();
if ( c < 1 )
return -1;
if ( frame < 0 )
return m_Smoothing.smooth[ 0 ].frametick;
if ( frame >= c )
return m_Smoothing.smooth[ c - 1 ].frametick;
return m_Smoothing.smooth[ frame ].frametick;
}
//-----------------------------------------------------------------------------
// Purpose: Interpolate Euler angles using quaternions to avoid singularities
// Input : start -
// end -
// output -
// frac -
//-----------------------------------------------------------------------------
static void InterpolateAngles( const QAngle& start, const QAngle& end, QAngle& output, float frac )
{
Quaternion src, dest;
// Convert to quaternions
AngleQuaternion( start, src );
AngleQuaternion( end, dest );
Quaternion result;
// Slerp
QuaternionSlerp( src, dest, frac, result );
// Convert to euler
QuaternionAngles( result, output );
}
bool CDemoSmootherPanel::GetInterpolatedOriginAndAngles( bool readonly, Vector& origin, QAngle& angles )
{
origin.Init();
angles.Init();
Assert( m_bPreviewing );
// Figure out the best samples
int startframe = m_nPreviewLastFrame;
int nextframe = startframe + 1;
float time = m_fPreviewCurrentTime;
int c = m_Smoothing.smooth.Count();
do
{
if ( startframe >= c || nextframe >= c )
{
if ( !readonly )
{
//m_bPreviewing = false;
}
return false;
}
demosmoothing_t *startsample = &m_Smoothing.smooth[ startframe ];
demosmoothing_t *endsample = &m_Smoothing.smooth[ nextframe ];
if ( nextframe >= min( m_nSelection[1] + 10, c - 1 ) )
{
if ( !readonly )
{
OnPreview( m_bPreviewOriginal );
}
return false;
}
// If large dt, then jump ahead quickly in time
float dt = TICKS_TO_TIME( endsample->frametick - startsample->frametick );
if ( dt > 1.0f )
{
startframe++;
nextframe++;
continue;
}
if ( TICKS_TO_TIME( endsample->frametick ) >= time )
{
// Found a spot
dt = TICKS_TO_TIME( endsample->frametick - startsample->frametick );
// Should never occur!!!
if ( dt <= 0.0f )
{
return false;
}
float frac = (float)( time - TICKS_TO_TIME(startsample->frametick) ) / dt;
frac = clamp( frac, 0.0f, 1.0f );
// Compute render origin/angles
Vector renderOrigin;
QAngle renderAngles;
if ( m_bPreviewOriginal )
{
VectorLerp( startsample->info.viewOrigin, endsample->info.viewOrigin, frac, renderOrigin );
InterpolateAngles( startsample->info.viewAngles, endsample->info.viewAngles, renderAngles, frac );
}
else
{
VectorLerp( startsample->info.GetViewOrigin(), endsample->info.GetViewOrigin(), frac, renderOrigin );
InterpolateAngles( startsample->info.GetViewAngles(), endsample->info.GetViewAngles(), renderAngles, frac );
}
origin = renderOrigin;
angles = renderAngles;
if ( !readonly )
{
SetLastFrame( false, startframe );
}
break;
}
startframe++;
nextframe++;
} while ( true );
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : t -
//-----------------------------------------------------------------------------
bool CDemoSmootherPanel::GetInterpolatedViewPoint( Vector& origin, QAngle& angles )
{
Assert( m_bPreviewing );
if ( !GetInterpolatedOriginAndAngles( false, origin, angles ) )
return false;
bool back_off = m_pBackOff->IsSelected();
if ( back_off )
{
Vector fwd;
AngleVectors( angles, &fwd, NULL, NULL );
origin = origin - fwd * 75.0f;
}
return true;
}
void CDemoSmootherPanel::OnTogglePause()
{
if ( !m_bPreviewing )
return;
m_bPreviewPaused = !m_bPreviewPaused;
}
void CDemoSmootherPanel::OnStep( bool forward )
{
if ( !m_bPreviewing )
return;
if ( !m_bPreviewPaused )
return;
int c = m_Smoothing.smooth.Count();
SetLastFrame( false, m_nPreviewLastFrame + ( forward ? 1 : -1 ) );
SetLastFrame( false, clamp( m_nPreviewLastFrame, max( m_nSelection[ 0 ] - 10, 0 ), min( m_nSelection[ 1 ] + 10, c - 1 ) ) );
m_fPreviewCurrentTime = TICKS_TO_TIME( GetTickForFrame( m_nPreviewLastFrame ) );
}
void CDemoSmootherPanel::DrawLegend( int startframe, int endframe )
{
int i;
int skip = 20;
bool back_off = m_pBackOff->IsSelected();
for ( i = startframe; i <= endframe; i++ )
{
bool show = ( i % skip ) == 0;
demosmoothing_t *sample = &m_Smoothing.smooth[ i ];
if ( sample->samplepoint || sample->targetpoint )
show = true;
if ( !show )
continue;
char sz[ 512 ];
Q_snprintf( sz, sizeof( sz ), "%.3f", TICKS_TO_TIME(sample->frametick) );
Vector fwd;
AngleVectors( sample->info.GetViewAngles(), &fwd, NULL, NULL );
CDebugOverlay::AddTextOverlay( sample->info.GetViewOrigin() + m_vecEyeOffset + fwd * ( back_off ? 5.0f : 50.0f ), 0, -1.0f, sz );
}
}
#define EASE_TIME 0.2f
Quaternion SmoothAngles( CUtlVector< Quaternion >& stack )
{
int c = stack.Count();
Assert( c >= 1 );
float weight = 1.0f / (float)c;
Quaternion output;
output.Init();
int i;
for ( i = 0; i < c; i++ )
{
Quaternion t = stack[ i ];
QuaternionBlend( output, t, weight, output );
}
return output;
}
Vector SmoothOrigin( CUtlVector< Vector >& stack )
{
int c = stack.Count();
Assert( c >= 1 );
Vector output;
output.Init();
int i;
for ( i = 0; i < c; i++ )
{
Vector t = stack[ i ];
VectorAdd( output, t, output );
}
VectorScale( output, 1.0f / (float)c, output );
return output;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnSetKeys(float interval)
{
if ( !m_bHasSelection )
return;
m_bDirty = true;
PushUndo( "OnSetKeys" );
int c = m_Smoothing.smooth.Count();
int i;
demosmoothing_t *lastkey = NULL;
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
if ( !p->selected )
continue;
p->angmoved = p->info.GetViewAngles();;
p->vecmoved = p->info.GetViewOrigin();
p->samplepoint = false;
if ( !lastkey ||
TICKS_TO_TIME( p->frametick - lastkey->frametick ) >= interval )
{
lastkey = p;
p->samplepoint = true;
}
}
PushRedo( "OnSetKeys" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnSmoothSelectionAngles( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
int i;
CUtlVector< Quaternion > stack;
m_bDirty = true;
PushUndo( "smooth angles" );
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
if ( !p->selected )
continue;
while ( stack.Count() > 10 )
{
stack.Remove( 0 );
}
Quaternion q;
AngleQuaternion( p->info.GetViewAngles(), q );
stack.AddToTail( q );
p->info.flags |= FDEMO_USE_ANGLES2;
Quaternion aveq = SmoothAngles( stack );
QAngle outangles;
QuaternionAngles( aveq, outangles );
p->info.viewAngles2 = outangles;
p->info.localViewAngles2 = outangles;
}
PushRedo( "smooth angles" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnSmoothSelectionOrigin( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
int i;
CUtlVector< Vector > stack;
m_bDirty = true;
PushUndo( "smooth origin" );
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
if ( !p->selected )
continue;
if ( i < 2 )
continue;
if ( i >= c - 2 )
continue;
stack.RemoveAll();
for ( int j = -2; j <= 2; j++ )
{
stack.AddToTail( m_Smoothing.smooth[ i + j ].info.GetViewOrigin() );
}
p->info.flags |= FDEMO_USE_ORIGIN2;
Vector org = SmoothOrigin( stack );
p->info.viewOrigin2 = org;
}
PushRedo( "smooth origin" );
}
void CDemoSmootherPanel::PerformLinearInterpolatedAngleSmoothing( int startframe, int endframe )
{
demosmoothing_t *pstart = &m_Smoothing.smooth[ startframe ];
demosmoothing_t *pend = &m_Smoothing.smooth[ endframe ];
int dt = pend->frametick - pstart->frametick;
if ( dt <= 0 )
{
dt = 1;
}
CUtlVector< Quaternion > stack;
Quaternion qstart, qend;
AngleQuaternion( pstart->info.GetViewAngles(), qstart );
AngleQuaternion( pend->info.GetViewAngles(), qend );
for ( int i = startframe; i <= endframe; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
int elapsed = p->frametick - pstart->frametick;
float frac = (float)elapsed / (float)dt;
frac = clamp( frac, 0.0f, 1.0f );
p->info.flags |= FDEMO_USE_ANGLES2;
Quaternion interpolated;
QuaternionSlerp( qstart, qend, frac, interpolated );
QAngle outangles;
QuaternionAngles( interpolated, outangles );
p->info.viewAngles2 = outangles;
p->info.localViewAngles2 = outangles;
}
}
void CDemoSmootherPanel::OnLinearInterpolateAnglesBasedOnEndpoints( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
m_bDirty = true;
PushUndo( "linear interp angles" );
PerformLinearInterpolatedAngleSmoothing( m_nSelection[ 0 ], m_nSelection[ 1 ] );
PushRedo( "linear interp angles" );
}
void CDemoSmootherPanel::OnLinearInterpolateOriginBasedOnEndpoints( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
int dt = pend->frametick - pstart->frametick;
if ( dt <= 0 )
return;
m_bDirty = true;
PushUndo( "linear interp origin" );
Vector vstart, vend;
vstart = pstart->info.GetViewOrigin();
vend = pend->info.GetViewOrigin();
for ( int i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
float elapsed = p->frametick - pstart->frametick;
float frac = elapsed / (float)dt;
frac = clamp( frac, 0.0f, 1.0f );
p->info.flags |= FDEMO_USE_ORIGIN2;
Vector interpolated;
VectorLerp( vstart, vend, frac, interpolated );
p->info.viewOrigin2 = interpolated;
}
PushRedo( "linear interp origin" );
}
void CDemoSmootherPanel::OnRevertPoint( void )
{
demosmoothing_t *p = GetCurrent();
if ( !p )
return;
m_bDirty = true;
PushUndo( "revert point" );
p->angmoved = p->info.GetViewAngles();
p->vecmoved = p->info.GetViewOrigin();
p->samplepoint = false;
p->vectarget = p->info.GetViewOrigin();
p->targetpoint = false;
// m_ViewOrigin = p->info.viewOrigin;
// m_ViewAngles = p->info.viewAngles;
PushRedo( "revert point" );
}
demosmoothing_t *CDemoSmootherPanel::GetCurrent( void )
{
if ( !CanEdit() )
return NULL;
int c = m_Smoothing.smooth.Count();
if ( c < 1 )
return NULL;
int frame = clamp( m_nPreviewLastFrame, 0, c - 1 );
return &m_Smoothing.smooth[ frame ];
}
void CDemoSmootherPanel::AddSamplePoints( bool usetarget, bool includeboundaries, CUtlVector< demosmoothing_t * >& points, int start, int end )
{
points.RemoveAll();
int i;
for ( i = start; i <= end; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
if ( includeboundaries )
{
if ( i == start )
{
// Add it twice
points.AddToTail( p );
continue;
}
else if ( i == end )
{
// Add twice
points.AddToTail( p );
continue;
}
}
if ( usetarget && p->targetpoint )
{
points.AddToTail( p );
}
if ( !usetarget && p->samplepoint )
{
points.AddToTail( p );
}
}
}
demosmoothing_t *CDemoSmootherPanel::GetBoundedSample( CUtlVector< demosmoothing_t * >& points, int sample )
{
int c = points.Count();
if ( sample < 0 )
return points[ 0 ];
else if ( sample >= c )
return points[ c - 1 ];
return points[ sample ];
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : t -
// points -
// prev -
// next -
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::FindSpanningPoints( int tick, CUtlVector< demosmoothing_t * >& points, int& prev, int& next )
{
prev = -1;
next = 0;
int c = points.Count();
int i;
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = points[ i ];
if ( tick < p->frametick )
break;
}
next = i;
prev = i - 1;
next = clamp( next, 0, c - 1 );
prev = clamp( prev, 0, c - 1 );
}
void CDemoSmootherPanel::OnSplineSampleOrigin( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
if ( pend->frametick - pstart->frametick <= 0 )
return;
CUtlVector< demosmoothing_t * > points;
AddSamplePoints( false, false, points, m_nSelection[ 0 ], m_nSelection[ 1 ] );
if ( points.Count() <= 0 )
return;
m_bDirty = true;
PushUndo( "spline origin" );
for ( int i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
demosmoothing_t *earliest;
demosmoothing_t *current;
demosmoothing_t *next;
demosmoothing_t *latest;
int cur;
int cur2;
FindSpanningPoints( p->frametick, points, cur, cur2 );
earliest = GetBoundedSample( points, cur - 1 );
current = GetBoundedSample( points, cur );
next = GetBoundedSample( points, cur2 );
latest = GetBoundedSample( points, cur2 + 1 );
float frac = 0.0f;
float dt = next->frametick - current->frametick;
if ( dt > 0.0f )
{
frac = (float)( p->frametick - current->frametick ) / dt;
}
frac = clamp( frac, 0.0f, 1.0f );
Vector splined;
Catmull_Rom_Spline_Normalize( earliest->vecmoved, current->vecmoved, next->vecmoved, latest->vecmoved, frac, splined );
p->info.flags |= FDEMO_USE_ORIGIN2;
p->info.viewOrigin2 = splined;
}
PushRedo( "spline origin" );
}
void CDemoSmootherPanel::OnSplineSampleAngles( void )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
if ( pend->frametick - pstart->frametick <= 0 )
return;
CUtlVector< demosmoothing_t * > points;
AddSamplePoints( false, false, points, m_nSelection[ 0 ], m_nSelection[ 1 ] );
if ( points.Count() <= 0 )
return;
m_bDirty = true;
PushUndo( "spline angles" );
for ( int i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
demosmoothing_t *current;
demosmoothing_t *next;
int cur;
int cur2;
FindSpanningPoints( p->frametick, points, cur, cur2 );
current = GetBoundedSample( points, cur );
next = GetBoundedSample( points, cur2 );
float frac = 0.0f;
float dt = next->frametick - current->frametick;
if ( dt > 0.0f )
{
frac = (float)( p->frametick - current->frametick ) / dt;
}
frac = clamp( frac, 0.0f, 1.0f );
frac = SimpleSpline( frac );
QAngle splined;
InterpolateAngles( current->angmoved, next->angmoved, splined, frac );
p->info.flags |= FDEMO_USE_ANGLES2;
p->info.viewAngles2 = splined;
p->info.localViewAngles2 = splined;
}
PushRedo( "spline angles" );
}
void CDemoSmootherPanel::OnLookAtPoints( bool spline )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
int i;
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
if ( pend->frametick - pstart->frametick <= 0 )
return;
CUtlVector< demosmoothing_t * > points;
AddSamplePoints( true, false, points, m_nSelection[ 0 ], m_nSelection[ 1 ] );
if ( points.Count() < 1 )
return;
m_bDirty = true;
PushUndo( "lookat points" );
for ( i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
demosmoothing_t *earliest;
demosmoothing_t *current;
demosmoothing_t *next;
demosmoothing_t *latest;
int cur;
int cur2;
FindSpanningPoints( p->frametick, points, cur, cur2 );
earliest = GetBoundedSample( points, cur - 1 );
current = GetBoundedSample( points, cur );
next = GetBoundedSample( points, cur2 );
latest = GetBoundedSample( points, cur2 + 1 );
float frac = 0.0f;
float dt = next->frametick - current->frametick;
if ( dt > 0.0f )
{
frac = (float)( p->frametick - current->frametick ) / dt;
}
frac = clamp( frac, 0.0f, 1.0f );
Vector splined;
if ( spline )
{
Catmull_Rom_Spline_Normalize( earliest->vectarget, current->vectarget, next->vectarget, latest->vectarget, frac, splined );
}
else
{
Vector d = next->vectarget - current->vectarget;
VectorMA( current->vectarget, frac, d, splined );
}
Vector vecToTarget = splined - ( p->info.GetViewOrigin() + m_vecEyeOffset );
VectorNormalize( vecToTarget );
QAngle angles;
VectorAngles( vecToTarget, angles );
p->info.flags |= FDEMO_USE_ANGLES2;
p->info.viewAngles2 = angles;
p->info.localViewAngles2 = angles;
}
PushRedo( "lookat points" );
}
void CDemoSmootherPanel::SetLastFrame( bool jumptotarget, int frame )
{
// bool changed = frame != m_nPreviewLastFrame;
int useFrame = max( m_Smoothing.m_nFirstSelectableSample, frame );
m_nPreviewLastFrame = useFrame;
/* if ( changed && !m_pLockCamera->IsSelected() )
{
// Reset default view/angles
demosmoothing_t *p = GetCurrent();
if ( p )
{
if ( p->samplepoint && !jumptotarget )
{
m_ViewOrigin = p->vecmoved;
m_ViewAngles = p->angmoved;
}
else if ( p->targetpoint && jumptotarget )
{
m_ViewOrigin = p->vectarget - m_vecEyeOffset;
}
else
{
if ( m_bPreviewing && m_bPreviewOriginal )
{
m_ViewOrigin = p->info.viewOrigin;
m_ViewAngles = p->info.viewAngles;
}
else
{
m_ViewOrigin = p->info.GetViewOrigin();
m_ViewAngles = p->info.GetViewAngles();
}
}
}
} */
}
// Undo/Redo
void CDemoSmootherPanel::Undo( void )
{
if ( m_UndoStack.Size() > 0 && m_nUndoLevel > 0 )
{
m_nUndoLevel--;
DemoSmoothUndo *u = m_UndoStack[ m_nUndoLevel ];
Assert( u->undo );
m_Smoothing = *(u->undo);
}
InvalidateLayout();
}
void CDemoSmootherPanel::Redo( void )
{
if ( m_UndoStack.Size() > 0 && m_nUndoLevel <= m_UndoStack.Size() - 1 )
{
DemoSmoothUndo *u = m_UndoStack[ m_nUndoLevel ];
Assert( u->redo );
m_Smoothing = *(u->redo);
m_nUndoLevel++;
}
InvalidateLayout();
}
void CDemoSmootherPanel::PushUndo( const char *description )
{
Assert( !m_bRedoPending );
m_bRedoPending = true;
WipeRedo();
// Copy current data
CSmoothingContext *u = new CSmoothingContext;
*u = m_Smoothing;
DemoSmoothUndo *undo = new DemoSmoothUndo;
undo->undo = u;
undo->redo = NULL;
undo->udescription = COM_StringCopy( description );
undo->rdescription = NULL;
m_UndoStack.AddToTail( undo );
m_nUndoLevel++;
}
void CDemoSmootherPanel::PushRedo( const char *description )
{
Assert( m_bRedoPending );
m_bRedoPending = false;
// Copy current data
CSmoothingContext *r = new CSmoothingContext;
*r = m_Smoothing;
DemoSmoothUndo *undo = m_UndoStack[ m_nUndoLevel - 1 ];
undo->redo = r;
undo->rdescription = COM_StringCopy( description );
}
void CDemoSmootherPanel::WipeUndo( void )
{
while ( m_UndoStack.Size() > 0 )
{
DemoSmoothUndo *u = m_UndoStack[ 0 ];
delete u->undo;
delete u->redo;
delete[] u->udescription;
delete[] u->rdescription;
delete u;
m_UndoStack.Remove( 0 );
}
m_nUndoLevel = 0;
}
void CDemoSmootherPanel::WipeRedo( void )
{
// Wipe everything above level
while ( m_UndoStack.Size() > m_nUndoLevel )
{
DemoSmoothUndo *u = m_UndoStack[ m_nUndoLevel ];
delete u->undo;
delete u->redo;
delete[] u->udescription;
delete[] u->rdescription;
delete u;
m_UndoStack.Remove( m_nUndoLevel );
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : const char
//-----------------------------------------------------------------------------
const char *CDemoSmootherPanel::GetUndoDescription( void )
{
if ( m_nUndoLevel != 0 )
{
DemoSmoothUndo *u = m_UndoStack[ m_nUndoLevel - 1 ];
return u->udescription;
}
return "???undo";
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : const char
//-----------------------------------------------------------------------------
const char *CDemoSmootherPanel::GetRedoDescription( void )
{
if ( m_nUndoLevel != m_UndoStack.Size() )
{
DemoSmoothUndo *u = m_UndoStack[ m_nUndoLevel ];
return u->rdescription;
}
return "???redo";
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CDemoSmootherPanel::CanRedo( void )
{
if ( !m_UndoStack.Count() )
return false;
if ( m_nUndoLevel == m_UndoStack.Count() )
return false;
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CDemoSmootherPanel::CanUndo( void )
{
if ( !m_UndoStack.Count() )
return false;
if ( m_nUndoLevel == 0 )
return false;
return true;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnToggleKeyFrame( void )
{
demosmoothing_t *p = GetCurrent();
if ( !p )
return;
m_bDirty = true;
PushUndo( "toggle keyframe" );
// use orginal data by default
p->angmoved = p->info.GetViewAngles();
p->vecmoved = p->info.GetViewOrigin();
if ( !p->samplepoint )
{
if ( g_pDemoUI->IsInDriveMode() )
{
g_pDemoUI->GetDriveViewPoint( p->vecmoved, p->angmoved );
}
if ( g_pDemoUI2->IsInDriveMode() )
{
g_pDemoUI2->GetDriveViewPoint( p->vecmoved, p->angmoved );
}
p->samplepoint = true;
}
else
{
p->samplepoint = false;
}
PushRedo( "toggle keyframe" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnToggleLookTarget( void )
{
demosmoothing_t *p = GetCurrent();
if ( !p )
return;
m_bDirty = true;
PushUndo( "toggle look target" );
// use orginal data by default
p->vectarget = p->info.GetViewOrigin();
if ( !p->targetpoint )
{
QAngle angles;
g_pDemoUI->GetDriveViewPoint( p->vectarget, angles );
g_pDemoUI2->GetDriveViewPoint( p->vectarget, angles );
p->targetpoint = true;
}
else
{
p->targetpoint = false;
}
PushRedo( "toggle look target" );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnNextKey()
{
if( !m_bHasSelection )
return;
int start = m_nPreviewLastFrame + 1;
int maxmove = m_nSelection[1] - m_nSelection[0] + 1;
int moved = 0;
while ( moved < maxmove )
{
demosmoothing_t *p = &m_Smoothing.smooth[ start ];
if ( p->samplepoint )
{
SetLastFrame( false, start );
break;
}
start++;
if ( start > m_nSelection[1] )
start = m_nSelection[0];
moved++;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnPrevKey()
{
if( !m_bHasSelection )
return;
int start = m_nPreviewLastFrame - 1;
int maxmove = m_nSelection[1] - m_nSelection[0] + 1;
int moved = 0;
while ( moved < maxmove && start >= 0 )
{
demosmoothing_t *p = &m_Smoothing.smooth[ start ];
if ( p->samplepoint )
{
SetLastFrame( false, start );
break;
}
start--;
if ( start < m_nSelection[0] )
start = m_nSelection[1];
moved++;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnNextTarget()
{
if( !m_bHasSelection )
return;
int start = m_nPreviewLastFrame + 1;
int maxmove = m_nSelection[1] - m_nSelection[0] + 1;
int moved = 0;
while ( moved < maxmove )
{
demosmoothing_t *p = &m_Smoothing.smooth[ start ];
if ( p->targetpoint )
{
SetLastFrame( true, start );
break;
}
start++;
if ( start > m_nSelection[1] )
start = m_nSelection[0];
moved++;
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnPrevTarget()
{
if( !m_bHasSelection )
return;
int start = m_nPreviewLastFrame - 1;
int maxmove = m_nSelection[1] - m_nSelection[0] + 1;
int moved = 0;
while ( moved < maxmove )
{
demosmoothing_t *p = &m_Smoothing.smooth[ start ];
if ( p->targetpoint )
{
SetLastFrame( true, start );
break;
}
start--;
if ( start < m_nSelection[0] )
start = m_nSelection[1];
moved++;
}
}
void CDemoSmootherPanel::DrawTargetSpline()
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
int i;
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
if ( pend->frametick - pstart->frametick <= 0 )
return;
CUtlVector< demosmoothing_t * > points;
AddSamplePoints( true, false, points, m_nSelection[ 0 ], m_nSelection[ 1 ] );
if ( points.Count() < 1 )
return;
Vector previous(0,0,0);
for ( i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
demosmoothing_t *earliest;
demosmoothing_t *current;
demosmoothing_t *next;
demosmoothing_t *latest;
int cur;
int cur2;
FindSpanningPoints( p->frametick, points, cur, cur2 );
earliest = GetBoundedSample( points, cur - 1 );
current = GetBoundedSample( points, cur );
next = GetBoundedSample( points, cur2 );
latest = GetBoundedSample( points, cur2 + 1 );
float frac = 0.0f;
float dt = next->frametick - current->frametick;
if ( dt > 0.0f )
{
frac = (float)( p->frametick - current->frametick ) / dt;
}
frac = clamp( frac, 0.0f, 1.0f );
Vector splined;
Catmull_Rom_Spline_Normalize( earliest->vectarget, current->vectarget, next->vectarget, latest->vectarget, frac, splined );
if ( i > m_nSelection[0] )
{
RenderLine( previous, splined, Color( 0, 255, 0, 255 ), true );
}
previous = splined;
}
}
void CDemoSmootherPanel::DrawKeySpline()
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
int i;
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
if ( pend->frametick - pstart->frametick <= 0 )
return;
CUtlVector< demosmoothing_t * > points;
AddSamplePoints( false, false, points, m_nSelection[ 0 ], m_nSelection[ 1 ] );
if ( points.Count() < 1 )
return;
Vector previous(0,0,0);
for ( i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
demosmoothing_t *earliest;
demosmoothing_t *current;
demosmoothing_t *next;
demosmoothing_t *latest;
int cur;
int cur2;
FindSpanningPoints( p->frametick, points, cur, cur2 );
earliest = GetBoundedSample( points, cur - 1 );
current = GetBoundedSample( points, cur );
next = GetBoundedSample( points, cur2 );
latest = GetBoundedSample( points, cur2 + 1 );
float frac = 0.0f;
float dt = next->frametick - current->frametick;
if ( dt > 0.0f )
{
frac = (float)( p->frametick - current->frametick ) / dt;
}
frac = clamp( frac, 0.0f, 1.0f );
Vector splined;
Catmull_Rom_Spline_Normalize( earliest->vecmoved, current->vecmoved, next->vecmoved, latest->vecmoved, frac, splined );
splined += m_vecEyeOffset;
if ( i > m_nSelection[0] )
{
RenderLine( previous, splined, Color( 0, 255, 0, 255 ), true );
}
previous = splined;
}
}
void CDemoSmootherPanel::OnSmoothEdges( bool left, bool right )
{
if ( !m_bHasSelection )
return;
if ( !left && !right )
return;
int c = m_Smoothing.smooth.Count();
// Get number of frames
char sz[ 512 ];
m_pFixEdgeFrames->GetText( sz, sizeof( sz ) );
int frames = atoi( sz );
if ( frames <= 2 )
return;
m_bDirty = true;
PushUndo( "smooth edges" );
if ( left && m_nSelection[0] > 0 )
{
PerformLinearInterpolatedAngleSmoothing( m_nSelection[ 0 ] - 1, m_nSelection[ 0 ] + frames );
}
if ( right && m_nSelection[1] < c - 1 )
{
PerformLinearInterpolatedAngleSmoothing( m_nSelection[ 1 ] - frames, m_nSelection[ 1 ] + 1 );
}
PushRedo( "smooth edges" );
}
void CDemoSmootherPanel::OnSaveKey()
{
if ( !m_bHasSelection )
return;
demosmoothing_t *p = GetCurrent();
if ( !p )
return;
if ( !p->samplepoint )
return;
m_bDirty = true;
PushUndo( "save key" );
p->info.viewAngles2 = p->angmoved;
p->info.localViewAngles2 = p->angmoved;
p->info.viewOrigin2 = p->vecmoved;
p->info.flags |= FDEMO_USE_ORIGIN2;
p->info.flags |= FDEMO_USE_ANGLES2;
PushRedo( "save key" );
}
void CDemoSmootherPanel::OnSetView()
{
if ( !m_bHasSelection )
return;
demosmoothing_t *p = GetCurrent();
if ( !p )
return;
Vector origin = p->info.GetViewOrigin();
QAngle angle = p->info.GetViewAngles();
g_pDemoUI->SetDriveViewPoint( origin, angle );
g_pDemoUI2->SetDriveViewPoint( origin, angle );
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CDemoSmootherPanel::OnGotoFrame()
{
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
char sz[ 256 ];
m_pGotoFrame->GetText( sz, sizeof( sz ) );
int frame = atoi( sz );
if ( !m_bPreviewing )
{
if ( !m_bHasSelection )
{
m_pStartFrame->SetText( va( "%i", 0 ) );
m_pEndFrame->SetText( va( "%i", c - 1 ) );
OnSelect();
}
OnPreview( false );
OnTogglePause();
}
if ( !m_bPreviewing )
return;
SetLastFrame( false, frame );
m_iPreviewStartTick = GetTickForFrame( m_nPreviewLastFrame );
m_fPreviewCurrentTime = TICKS_TO_TIME( m_iPreviewStartTick );
}
void CDemoSmootherPanel::OnOriginEaseCurve( EASEFUNC easefunc )
{
if ( !m_bHasSelection )
return;
int c = m_Smoothing.smooth.Count();
if ( c < 2 )
return;
demosmoothing_t *pstart = &m_Smoothing.smooth[ m_nSelection[ 0 ] ];
demosmoothing_t *pend = &m_Smoothing.smooth[ m_nSelection[ 1 ] ];
float dt = pend->frametick - pstart->frametick;
if ( dt <= 0.0f )
return;
m_bDirty = true;
PushUndo( "ease origin" );
Vector vstart, vend;
vstart = pstart->info.GetViewOrigin();
vend = pend->info.GetViewOrigin();
for ( int i = m_nSelection[0]; i <= m_nSelection[1]; i++ )
{
demosmoothing_t *p = &m_Smoothing.smooth[ i ];
float elapsed = p->frametick - pstart->frametick;
float frac = elapsed / dt;
// Apply ease function
frac = (*easefunc)( frac );
frac = clamp( frac, 0.0f, 1.0f );
p->info.flags |= FDEMO_USE_ORIGIN2;
Vector interpolated;
VectorLerp( vstart, vend, frac, interpolated );
p->info.viewOrigin2 = interpolated;
}
PushRedo( "ease origin" );
}
void CDemoSmootherPanel::ParseSmoothingInfo( CDemoFile &demoFile, CSmoothingContext& smoothing )
{
democmdinfo_t info;
int dummy;
bool foundFirstSelectable = false;
bool demofinished = false;
while ( !demofinished )
{
int tick = 0;
byte cmd;
bool swallowmessages = true;
do
{
demoFile.ReadCmdHeader( cmd, tick );
// COMMAND HANDLERS
switch ( cmd )
{
case dem_synctick:
break;
case dem_stop:
{
swallowmessages = false;
demofinished = true;
}
break;
case dem_consolecmd:
{
demoFile.ReadConsoleCommand();
}
break;
case dem_datatables:
{
demoFile.ReadNetworkDataTables( NULL );
}
break;
case dem_stringtables:
{
demoFile.ReadStringTables( NULL );
}
break;
case dem_usercmd:
{
demoFile.ReadUserCmd( NULL, dummy );
}
break;
default:
{
swallowmessages = false;
}
break;
}
}
while ( swallowmessages );
if ( demofinished )
{
// StopPlayback();
return;
}
int curpos = demoFile.GetCurPos( true );
demoFile.ReadCmdInfo( info );
demoFile.ReadSequenceInfo( dummy, dummy );
demoFile.ReadRawData( NULL, 0 );
// Add to end of list
demosmoothing_t smoothing_entry;
smoothing_entry.file_offset = curpos;
smoothing_entry.frametick = tick;
smoothing_entry.info = info;
smoothing_entry.samplepoint = false;
smoothing_entry.vecmoved = info.GetViewOrigin();
smoothing_entry.angmoved = info.GetViewAngles();
smoothing_entry.targetpoint = false;
smoothing_entry.vectarget = info.GetViewOrigin();
int sampleIndex = smoothing.smooth.AddToTail( smoothing_entry );
if ( !foundFirstSelectable &&
smoothing_entry.vecmoved.LengthSqr() > 0.0f )
{
foundFirstSelectable = true;
smoothing.m_nFirstSelectableSample = sampleIndex;
}
}
}
void CDemoSmootherPanel::LoadSmoothingInfo( const char *filename, CSmoothingContext& smoothing )
{
char name[ MAX_OSPATH ];
Q_strncpy (name, filename, sizeof(name) );
Q_DefaultExtension( name, ".dem", sizeof( name ) );
CDemoFile demoFile;
if ( !demoFile.Open( filename, true ) )
{
ConMsg( "ERROR: couldn't open %s.\n", name );
return;
}
demoheader_t * header = demoFile.ReadDemoHeader();
if ( !header )
{
demoFile.Close();
return;
}
ConMsg ("Smoothing demo from %s ...", name );
smoothing.active = true;
Q_strncpy( smoothing.filename, name, sizeof(smoothing.filename) );
smoothing.smooth.RemoveAll();
ClearSmoothingInfo( smoothing );
ParseSmoothingInfo( demoFile, smoothing );
demoFile.Close();
//Performsmoothing( smooth );
//SaveSmoothedDemo( name, smooth );
ConMsg ( " done.\n" );
}
void CDemoSmootherPanel::ClearSmoothingInfo( CSmoothingContext& smoothing )
{
int c = smoothing.smooth.Count();
int i;
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = &smoothing.smooth[ i ];
p->info.Reset();
p->vecmoved = p->info.GetViewOrigin();
p->angmoved = p->info.GetViewAngles();
p->samplepoint = false;
p->vectarget = p->info.GetViewOrigin();
p->targetpoint = false;
}
}
void CDemoSmootherPanel::SaveSmoothingInfo( char const *filename, CSmoothingContext& smoothing )
{
// Nothing to do
int c = smoothing.smooth.Count();
if ( !c )
return;
IFileSystem *fs = g_pFileSystem;
FileHandle_t infile, outfile;
COM_OpenFile( filename, &infile );
if ( infile == FILESYSTEM_INVALID_HANDLE )
return;
int filesize = fs->Size( infile );
char outfilename[ 512 ];
Q_StripExtension( filename, outfilename, sizeof( outfilename ) );
Q_strncat( outfilename, "_smooth", sizeof(outfilename), COPY_ALL_CHARACTERS );
Q_DefaultExtension( outfilename, ".dem", sizeof( outfilename ) );
outfile = fs->Open( outfilename, "wb" );
if ( outfile == FILESYSTEM_INVALID_HANDLE )
{
fs->Close( infile );
return;
}
int i;
int lastwritepos = 0;
for ( i = 0; i < c; i++ )
{
demosmoothing_t *p = &smoothing.smooth[ i ];
int copyamount = p->file_offset - lastwritepos;
COM_CopyFileChunk( outfile, infile, copyamount );
fs->Seek( infile, p->file_offset, FILESYSTEM_SEEK_HEAD );
// wacky hacky overwriting
fs->Write( &p->info, sizeof( democmdinfo_t ), outfile );
lastwritepos = fs->Tell( outfile );
fs->Seek( infile, p->file_offset + sizeof( democmdinfo_t ), FILESYSTEM_SEEK_HEAD );
}
int final = filesize - lastwritepos;
COM_CopyFileChunk( outfile, infile, final );
fs->Close( outfile );
fs->Close( infile );
}