source-engine/hammer/mapanimator.cpp

501 lines
14 KiB
C++
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#include "stdafx.h"
#include "GlobalFunctions.h"
#include "fgdlib/HelperInfo.h"
#include "MapAnimator.h"
#include "MapDoc.h"
#include "MapEntity.h"
#include "MapWorld.h"
#include "KeyFrame/KeyFrame.h"
// memdbgon must be the last include file in a .cpp file!!!
#include <tier0/memdbgon.h>
IMPLEMENT_MAPCLASS( CMapAnimator );
//-----------------------------------------------------------------------------
// Purpose: Factory function. Used for creating a CMapKeyFrame from a set
// of string parameters from the FGD file.
// Input : *pInfo - Pointer to helper info class which gives us information
// about how to create the class.
// Output : Returns a pointer to the class, NULL if an error occurs.
//-----------------------------------------------------------------------------
CMapClass *CMapAnimator::CreateMapAnimator(CHelperInfo *pHelperInfo, CMapEntity *pParent)
{
return(new CMapAnimator);
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CMapAnimator::CMapAnimator()
{
m_CoordFrame.Identity();
m_bCurrentlyAnimating = false;
m_pCurrentKeyFrame = this;
m_iPositionInterpolator = m_iRotationInterpolator = m_iTimeModifier = 0;
m_nKeysChanged = 0;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
CMapAnimator::~CMapAnimator()
{
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CMapClass *
//-----------------------------------------------------------------------------
CMapClass *CMapAnimator::Copy(bool bUpdateDependencies)
{
CMapAnimator *pNew = new CMapAnimator;
pNew->CopyFrom(this, bUpdateDependencies);
return pNew;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *pObj -
// Output : CMapClass
//-----------------------------------------------------------------------------
CMapClass *CMapAnimator::CopyFrom(CMapClass *pObj, bool bUpdateDependencies)
{
CMapKeyFrame::CopyFrom(pObj, bUpdateDependencies);
CMapAnimator *pFrom = dynamic_cast<CMapAnimator*>( pObj );
Assert( pFrom != NULL );
memcpy( m_CoordFrame.Base(), pFrom->m_CoordFrame.Base(), sizeof(m_CoordFrame) );
m_bCurrentlyAnimating = false;
m_pCurrentKeyFrame = NULL; // keyframe it's currently at
m_iTimeModifier = pFrom->m_iTimeModifier;
m_iPositionInterpolator = pFrom->m_iPositionInterpolator;
m_iRotationInterpolator = pFrom->m_iRotationInterpolator;
return this;
}
//-----------------------------------------------------------------------------
// Purpose: Returns a coordinate frame to render in, if the entity is animating
// Input : matrix -
// Output : returns true if a new matrix is returned, false if it is invalid
//-----------------------------------------------------------------------------
bool CMapAnimator::GetTransformMatrix( VMatrix& matrix )
{
// are we currently animating?
if ( m_bCurrentlyAnimating )
{
matrix = m_CoordFrame;
return true;
}
return false;
}
//-----------------------------------------------------------------------------
// Purpose: Notifies that the entity this is attached to has had a key change
// Input : key -
// value -
//-----------------------------------------------------------------------------
void CMapAnimator::OnParentKeyChanged( const char* key, const char* value )
{
if ( !stricmp(key, "TimeModifier") )
{
m_iTimeModifier = atoi( value );
}
else if ( !stricmp(key, "PositionInterpolator") )
{
m_iPositionInterpolator = atoi( value );
// HACK: Force everything in the path to update. Better to follow our path and update only it.
UpdateAllDependencies(this);
}
else if ( !stricmp(key, "RotationInterpolator") )
{
m_iRotationInterpolator = atoi( value );
}
m_nKeysChanged++;
CMapKeyFrame::OnParentKeyChanged( key, value );
}
//-----------------------------------------------------------------------------
// Purpose: Gets the current and previous keyframes for a given time.
// Input : time - time into sequence
// pKeyFrame - receives current keyframe pointer
// pPrevKeyFrame - receives previous keyframe pointer
// Output : time remaining after the thing has reached this key
//-----------------------------------------------------------------------------
float CMapAnimator::GetKeyFramesAtTime( float time, CMapKeyFrame *&pKeyFrame, CMapKeyFrame *&pPrevKeyFrame )
{
pKeyFrame = this;
pPrevKeyFrame = this;
float outTime = time;
while ( pKeyFrame )
{
if ( pKeyFrame->MoveTime() > outTime )
{
break;
}
// make sure this anim has enough time
if ( pKeyFrame->MoveTime() < 0.01f )
{
outTime = 0.0f;
break;
}
outTime -= pKeyFrame->MoveTime();
pPrevKeyFrame = pKeyFrame;
pKeyFrame = pKeyFrame->NextKeyFrame();
}
return outTime;
}
//-----------------------------------------------------------------------------
// Purpose: creates a new keyframe at the specified time
// Input : time -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
CMapEntity *CMapAnimator::CreateNewKeyFrame( float time )
{
// work out where we are in the animation
CMapKeyFrame *key;
CMapKeyFrame *pPrevKey;
float partialTime = GetKeyFramesAtTime( time, key, pPrevKey );
CMapEntity *pCurrentEnt = dynamic_cast<CMapEntity*>( key->m_pParent );
// check to see if we're direction on a key frame
Vector posOffset( 0, 0, 0 );
if ( partialTime == 0 )
{
// create this new key frame slightly after the current one, and offset
posOffset[0] = 64;
}
// get our orientation and position at this time
Vector vOrigin;
QAngle angles;
Quaternion qAngles;
GetAnimationAtTime( key, pPrevKey, partialTime, vOrigin, qAngles, m_iPositionInterpolator, m_iRotationInterpolator );
QuaternionAngles( qAngles, angles );
// create the new map entity
CMapEntity *pNewEntity = new CMapEntity;
Vector newPos;
VectorAdd( vOrigin, posOffset, newPos );
pNewEntity->SetPlaceholder( TRUE );
pNewEntity->SetOrigin( newPos );
pNewEntity->SetClass( "keyframe_track" );
char buf[128];
sprintf( buf, "%f %f %f", angles[0], angles[1], angles[2] );
pNewEntity->SetKeyValue( "angles", buf );
// link it into the keyframe list
// take over this existing next keyframe pointer
const char *nextKeyName = pCurrentEnt->GetKeyValue( "NextKey" );
if ( nextKeyName )
{
pNewEntity->SetKeyValue( "NextKey", nextKeyName );
}
// create a new unique name for this ent
char newName[128];
const char *oldName = pCurrentEnt->GetKeyValue( "targetname" );
if ( !oldName || oldName[0] == 0 )
oldName = "keyframe";
CMapWorld *pWorld = GetWorldObject( this );
if ( pWorld )
{
pWorld->GenerateNewTargetname( oldName, newName, sizeof( newName ), true, NULL );
pNewEntity->SetKeyValue( "targetname", newName );
// point the current entity at the newly created one
pCurrentEnt->SetKeyValue( "NextKey", newName );
// copy any relevant values
const char *keyValue = pCurrentEnt->GetKeyValue( "parentname" );
if ( keyValue )
pNewEntity->SetKeyValue( "parentname", keyValue );
keyValue = pCurrentEnt->GetKeyValue( "MoveSpeed" );
if ( keyValue )
pNewEntity->SetKeyValue( "MoveSpeed", keyValue );
}
return(pNewEntity);
}
//-----------------------------------------------------------------------------
// Purpose: stops CMapKeyframe from doing it's auto-connect behavior when cloning
// Input : pClone -
//-----------------------------------------------------------------------------
void CMapAnimator::OnClone( CMapClass *pClone, CMapWorld *pWorld, const CMapObjectList &OriginalList, CMapObjectList &NewList )
{
CMapClass::OnClone( pClone, pWorld, OriginalList, NewList );
}
//-----------------------------------------------------------------------------
// Purpose: calculates the position of an animating object at a given time in
// it's animation sequence
// Input : animTime -
// *newOrigin -
// *newAngles -
//-----------------------------------------------------------------------------
void CMapAnimator::GetAnimationAtTime( float animTime, Vector& newOrigin, Quaternion &newAngles )
{
// setup the animation, given the time
// get our new position & orientation
float newTime, totalAnimTime = GetRemainingTime();
animTime /= totalAnimTime;
// don't use time modifier until we work out what we're going to do with it
//Motion_CalculateModifiedTime( animTime, m_iTimeModifier, &newTime );
newTime = animTime;
// find out where we are in the keyframe sequence based on the time
CMapKeyFrame *pPrevKeyFrame;
float posTime = GetKeyFramesAtTime( newTime * totalAnimTime, m_pCurrentKeyFrame, pPrevKeyFrame );
// find the position from that keyframe
GetAnimationAtTime( m_pCurrentKeyFrame, pPrevKeyFrame, posTime, newOrigin, newAngles, m_iPositionInterpolator, m_iRotationInterpolator );
}
//-----------------------------------------------------------------------------
// Purpose: calculates the position of the animating object between two keyframes
// Input : currentKey -
// pPrevKey -
// partialTime -
// newOrigin -
// newAngles -
// posInterpolator -
// rotInterpolator -
//-----------------------------------------------------------------------------
void CMapAnimator::GetAnimationAtTime( CMapKeyFrame *currentKey, CMapKeyFrame *pPrevKey, float partialTime, Vector& newOrigin, Quaternion &newAngles, int posInterpolator, int rotInterpolator )
{
// calculate the proportion of time to be spent on this keyframe
float animTime;
if ( currentKey->MoveTime() < 0.01 )
{
animTime = 1.0f;
}
else
{
animTime = partialTime / currentKey->MoveTime();
}
Assert( animTime >= 0.0f && animTime <= 1.0f );
IPositionInterpolator *pInterp = currentKey->SetupPositionInterpolator( posInterpolator );
// setup interpolation keyframes
Vector keyOrigin;
Quaternion keyAngles;
pPrevKey->GetOrigin( keyOrigin );
pPrevKey->GetQuatAngles( keyAngles );
pInterp->SetKeyPosition( -1, keyOrigin );
Motion_SetKeyAngles ( -1, keyAngles );
currentKey->GetOrigin( keyOrigin );
currentKey->GetQuatAngles( keyAngles );
pInterp->SetKeyPosition( 0, keyOrigin );
Motion_SetKeyAngles ( 0, keyAngles );
currentKey->NextKeyFrame()->GetOrigin( keyOrigin );
currentKey->NextKeyFrame()->GetQuatAngles( keyAngles );
pInterp->SetKeyPosition( 1, keyOrigin );
Motion_SetKeyAngles ( 1, keyAngles );
currentKey->NextKeyFrame()->NextKeyFrame()->GetOrigin( keyOrigin );
currentKey->NextKeyFrame()->NextKeyFrame()->GetQuatAngles( keyAngles );
pInterp->SetKeyPosition( 2, keyOrigin );
Motion_SetKeyAngles ( 2, keyAngles );
// get our new interpolated position
// HACK HACK - Hey Brian, look here!!!!
Vector hackOrigin;
pInterp->InterpolatePosition( animTime, hackOrigin );
newOrigin[0] = hackOrigin[0];
newOrigin[1] = hackOrigin[1];
newOrigin[2] = hackOrigin[2];
Motion_InterpolateRotation( animTime, rotInterpolator, newAngles );
}
//-----------------------------------------------------------------------------
// Purpose: Builds the animation transformation matrix for the entity if it's animating
//-----------------------------------------------------------------------------
void CMapAnimator::UpdateAnimation( float animTime )
{
// only animate if the doc is animating and we're selected
if ( !CMapDoc::GetActiveMapDoc()->IsAnimating() || !m_pParent->IsSelected() )
{
// we're not animating
m_bCurrentlyAnimating = false;
return;
}
m_bCurrentlyAnimating = true;
Vector newOrigin;
Quaternion newAngles;
GetAnimationAtTime( animTime, newOrigin, newAngles );
VMatrix mat, tmpMat;
Vector ourOrigin;
GetOrigin( ourOrigin );
// build us a matrix
// T(newOrigin)R(angle)T(-ourOrigin)
m_CoordFrame.Identity() ;
// transform back to the origin
for ( int i = 0; i < 3; i++ )
{
m_CoordFrame[i][3] = -ourOrigin[i];
}
// Apply interpolated Rotation
mat.Identity();
QuaternionMatrix( newAngles, const_cast< matrix3x4_t & > ( mat.As3x4() ) );
m_CoordFrame = m_CoordFrame * mat;
// transform back to our new position
mat.Identity();
for ( int i = 0; i < 3; i++ )
{
mat[i][3] = newOrigin[i];
}
m_CoordFrame = m_CoordFrame * mat;
}
//-----------------------------------------------------------------------------
// Purpose: Rebuilds the line path between keyframe entities
// samples the interpolator function to get an approximation of the curve
//-----------------------------------------------------------------------------
void CMapAnimator::RebuildPath( void )
{
CMapWorld *pWorld = GetWorldObject( this );
if ( !pWorld )
{
// Sometimes the object isn't linked back into the world yet when RebuildPath()
// is called... we will be get called again when needed, but may cause an incorrect
// (linear-only) path to get drawn temporarily.
return;
}
//
// Build the path forward from the head. Keep a list of nodes we've visited to
// use in detecting circularities.
//
CMapObjectList VisitedList;
CMapKeyFrame *pCurKey = this;
while ( pCurKey != NULL )
{
VisitedList.AddToTail( pCurKey );
//
// Attach ourselves as this keyframe's animator.
//
pCurKey->SetAnimator( this );
//
// Get the entity parent of this keyframe so we can query keyvalues.
//
CMapEntity *pCurEnt = dynamic_cast<CMapEntity *>( pCurKey->GetParent() );
if ( !pCurEnt )
{
return;
}
//
// Find the next keyframe in the path.
//
CMapEntity *pNextEnt = pWorld->FindEntityByName( pCurEnt->GetKeyValue( "NextKey" ) );
CMapKeyFrame *pNextKey = NULL;
if ( pNextEnt )
{
pNextKey = pNextEnt->GetChildOfType( ( CMapKeyFrame * )NULL );
pCurKey->SetNextKeyFrame(pNextKey);
}
else
{
pCurKey->SetNextKeyFrame( NULL );
}
pCurKey = pNextKey;
//
// If we detect a circularity, stop.
//
if ( VisitedList.Find( pCurKey ) != -1 )
{
break;
}
}
//
// Now traverse the path again building the spline points, once again checking
// the visited list for circularities.
//
VisitedList.RemoveAll();
pCurKey = this;
CMapKeyFrame *pPrevKey = this;
while ( pCurKey != NULL )
{
VisitedList.AddToTail( pCurKey );
pCurKey->BuildPathSegment(pPrevKey);
pPrevKey = pCurKey;
pCurKey = pCurKey->m_pNextKeyFrame;
//
// If we detect a circularity, stop.
//
if ( VisitedList.Find( pCurKey ) != -1 )
{
break;
}
}
}