|
|
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: This will measure the movement of a target entity and move
// another entity to match the movement of the first.
//
//=============================================================================//
#include "cbase.h"
#include "baseentity.h"
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
//-----------------------------------------------------------------------------
// This will measure the movement of a target entity and move
// another entity to match the movement of the first.
//-----------------------------------------------------------------------------
class CLogicMeasureMovement : public CLogicalEntity { DECLARE_DATADESC(); DECLARE_CLASS( CLogicMeasureMovement, CLogicalEntity );
public: virtual void Activate();
private: void SetMeasureTarget( const char *pName ); void SetMeasureReference( const char *pName ); void SetTarget( const char *pName ); void SetTargetReference( const char *pName );
void InputSetMeasureTarget( inputdata_t &inputdata ); void InputSetMeasureReference( inputdata_t &inputdata ); void InputSetTarget( inputdata_t &inputdata ); void InputSetTargetReference( inputdata_t &inputdata ); void InputSetTargetScale( inputdata_t &inputdata ); void InputEnable( inputdata_t &inputdata ); void InputDisable( inputdata_t &inputdata );
void MeasureThink();
private: enum { MEASURE_POSITION = 0, MEASURE_EYE_POSITION, };
string_t m_strMeasureTarget; string_t m_strMeasureReference; string_t m_strTargetReference;
EHANDLE m_hMeasureTarget; EHANDLE m_hMeasureReference; EHANDLE m_hTarget; EHANDLE m_hTargetReference;
float m_flScale; int m_nMeasureType; };
LINK_ENTITY_TO_CLASS( logic_measure_movement, CLogicMeasureMovement );
BEGIN_DATADESC( CLogicMeasureMovement )
DEFINE_KEYFIELD( m_strMeasureTarget, FIELD_STRING, "MeasureTarget" ), DEFINE_KEYFIELD( m_strMeasureReference, FIELD_STRING, "MeasureReference" ), DEFINE_KEYFIELD( m_strTargetReference, FIELD_STRING, "TargetReference" ), DEFINE_KEYFIELD( m_flScale, FIELD_FLOAT, "TargetScale" ), DEFINE_KEYFIELD( m_nMeasureType, FIELD_INTEGER, "MeasureType" ),
DEFINE_FIELD( m_hMeasureTarget, FIELD_EHANDLE ), DEFINE_FIELD( m_hMeasureReference, FIELD_EHANDLE ), DEFINE_FIELD( m_hTarget, FIELD_EHANDLE ), DEFINE_FIELD( m_hTargetReference, FIELD_EHANDLE ),
DEFINE_INPUTFUNC( FIELD_STRING, "SetMeasureTarget", InputSetMeasureTarget ), DEFINE_INPUTFUNC( FIELD_STRING, "SetMeasureReference", InputSetMeasureReference ), DEFINE_INPUTFUNC( FIELD_STRING, "SetTarget", InputSetTarget ), DEFINE_INPUTFUNC( FIELD_STRING, "SetTargetReference", InputSetTargetReference ), DEFINE_INPUTFUNC( FIELD_FLOAT, "SetTargetScale", InputSetTargetScale ),
DEFINE_INPUTFUNC( FIELD_VOID, "Enable", InputEnable ), DEFINE_INPUTFUNC( FIELD_VOID, "Disable", InputDisable ),
DEFINE_THINKFUNC( MeasureThink ),
END_DATADESC()
//-----------------------------------------------------------------------------
// Methods to change various targets
//-----------------------------------------------------------------------------
void CLogicMeasureMovement::Activate() { BaseClass::Activate();
SetMeasureTarget( STRING(m_strMeasureTarget) ); SetMeasureReference( STRING(m_strMeasureReference) ); SetTarget( STRING(m_target) ); SetTargetReference( STRING(m_strTargetReference) ); SetThink( &CLogicMeasureMovement::MeasureThink ); SetNextThink( gpGlobals->curtime + TICK_INTERVAL ); }
//-----------------------------------------------------------------------------
// Sets the name
//-----------------------------------------------------------------------------
void CLogicMeasureMovement::SetMeasureTarget( const char *pName ) { m_hMeasureTarget = gEntList.FindEntityByName( NULL, pName ); if ( !m_hMeasureTarget ) { if ( Q_strnicmp( STRING(m_strMeasureTarget), "!player", 8 ) ) { Warning("logic_measure_movement: Unable to find measure target entity %s\n", pName ); } } }
void CLogicMeasureMovement::SetMeasureReference( const char *pName ) { m_hMeasureReference = gEntList.FindEntityByName( NULL, pName ); if ( !m_hMeasureReference ) { Warning("logic_measure_movement: Unable to find measure reference entity %s\n", pName ); } }
void CLogicMeasureMovement::SetTarget( const char *pName ) { m_hTarget = gEntList.FindEntityByName( NULL, pName ); if ( !m_hTarget ) { Warning("logic_measure_movement: Unable to find movement target entity %s\n", pName ); } }
void CLogicMeasureMovement::SetTargetReference( const char *pName ) { m_hTargetReference = gEntList.FindEntityByName( NULL, pName ); if ( !m_hTargetReference ) { Warning("logic_measure_movement: Unable to find movement reference entity %s\n", pName ); } }
//-----------------------------------------------------------------------------
// Apply movement
//-----------------------------------------------------------------------------
void CLogicMeasureMovement::MeasureThink( ) { // FIXME: This is a hack to make measuring !player simpler. The player isn't
// created at Activate time, so m_hMeasureTarget may be NULL because of that.
if ( !m_hMeasureTarget.Get() && !Q_strnicmp( STRING(m_strMeasureTarget), "!player", 8 ) ) { SetMeasureTarget( STRING(m_strMeasureTarget) ); }
// Make sure all entities are valid
if ( m_hMeasureTarget.Get() && m_hMeasureReference.Get() && m_hTarget.Get() && m_hTargetReference.Get() ) { matrix3x4_t matRefToMeasure, matWorldToMeasure; switch( m_nMeasureType ) { case MEASURE_POSITION: MatrixInvert( m_hMeasureTarget->EntityToWorldTransform(), matWorldToMeasure ); break;
case MEASURE_EYE_POSITION: AngleIMatrix( m_hMeasureTarget->EyeAngles(), m_hMeasureTarget->EyePosition(), matWorldToMeasure ); break;
// FIXME: Could add attachment point measurement here easily
}
ConcatTransforms( matWorldToMeasure, m_hMeasureReference->EntityToWorldTransform(), matRefToMeasure ); // Apply the scale factor
if ( ( m_flScale != 0.0f ) && ( m_flScale != 1.0f ) ) { Vector vecTranslation; MatrixGetColumn( matRefToMeasure, 3, vecTranslation ); vecTranslation /= m_flScale; MatrixSetColumn( vecTranslation, 3, matRefToMeasure ); }
// Now apply the new matrix to the new reference point
matrix3x4_t matMeasureToRef, matNewTargetToWorld; MatrixInvert( matRefToMeasure, matMeasureToRef );
ConcatTransforms( m_hTargetReference->EntityToWorldTransform(), matMeasureToRef, matNewTargetToWorld );
Vector vecNewOrigin; QAngle vecNewAngles; MatrixAngles( matNewTargetToWorld, vecNewAngles, vecNewOrigin ); m_hTarget->SetAbsOrigin( vecNewOrigin ); m_hTarget->SetAbsAngles( vecNewAngles ); }
SetNextThink( gpGlobals->curtime + TICK_INTERVAL ); }
//-----------------------------------------------------------------------------
// Enable, disable
//-----------------------------------------------------------------------------
void CLogicMeasureMovement::InputEnable( inputdata_t &inputdata ) { SetThink( &CLogicMeasureMovement::MeasureThink ); SetNextThink( gpGlobals->curtime + TICK_INTERVAL ); }
void CLogicMeasureMovement::InputDisable( inputdata_t &inputdata ) { SetThink( NULL ); }
//-----------------------------------------------------------------------------
// Methods to change various targets
//-----------------------------------------------------------------------------
void CLogicMeasureMovement::InputSetMeasureTarget( inputdata_t &inputdata ) { m_strMeasureTarget = MAKE_STRING( inputdata.value.String() ); SetMeasureTarget( inputdata.value.String() ); SetTarget( STRING(m_target) ); SetTargetReference( STRING(m_strTargetReference) ); }
void CLogicMeasureMovement::InputSetMeasureReference( inputdata_t &inputdata ) { m_strMeasureReference = MAKE_STRING( inputdata.value.String() ); SetMeasureReference( inputdata.value.String() ); }
void CLogicMeasureMovement::InputSetTarget( inputdata_t &inputdata ) { m_target = MAKE_STRING( inputdata.value.String() ); SetTarget( inputdata.value.String() ); }
void CLogicMeasureMovement::InputSetTargetReference( inputdata_t &inputdata ) { m_strTargetReference = MAKE_STRING( inputdata.value.String() ); SetTargetReference( inputdata.value.String() ); }
void CLogicMeasureMovement::InputSetTargetScale( inputdata_t &inputdata ) { m_flScale = inputdata.value.Float(); }
|