dhewm3-sdk/game/Actor.cpp
dhewg afebd7e1e5 Untangle the epic precompiled.h mess
Don't include the lazy precompiled.h everywhere, only what's
required for the compilation unit.
platform.h needs to be included instead to provide all essential
defines and types.
All includes use the relative path to the neo or the game
specific root.
Move all idlib related includes from idlib/Lib.h to precompiled.h.
precompiled.h still exists for the MFC stuff in tools/.
Add some missing header guards.
2018-08-20 01:46:28 +02:00

3278 lines
78 KiB
C++

/*
===========================================================================
Doom 3 GPL Source Code
Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
This file is part of the Doom 3 GPL Source Code ("Doom 3 Source Code").
Doom 3 Source Code is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Doom 3 Source Code is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#include "sys/platform.h"
#include "gamesys/SysCvar.h"
#include "script/Script_Thread.h"
#include "Item.h"
#include "Light.h"
#include "Projectile.h"
#include "WorldSpawn.h"
#include "Actor.h"
/***********************************************************************
idAnimState
***********************************************************************/
/*
=====================
idAnimState::idAnimState
=====================
*/
idAnimState::idAnimState() {
self = NULL;
animator = NULL;
thread = NULL;
idleAnim = true;
disabled = true;
channel = ANIMCHANNEL_ALL;
animBlendFrames = 0;
lastAnimBlendFrames = 0;
}
/*
=====================
idAnimState::~idAnimState
=====================
*/
idAnimState::~idAnimState() {
delete thread;
}
/*
=====================
idAnimState::Save
=====================
*/
void idAnimState::Save( idSaveGame *savefile ) const {
savefile->WriteObject( self );
// Save the entity owner of the animator
savefile->WriteObject( animator->GetEntity() );
savefile->WriteObject( thread );
savefile->WriteString( state );
savefile->WriteInt( animBlendFrames );
savefile->WriteInt( lastAnimBlendFrames );
savefile->WriteInt( channel );
savefile->WriteBool( idleAnim );
savefile->WriteBool( disabled );
}
/*
=====================
idAnimState::Restore
=====================
*/
void idAnimState::Restore( idRestoreGame *savefile ) {
savefile->ReadObject( reinterpret_cast<idClass *&>( self ) );
idEntity *animowner;
savefile->ReadObject( reinterpret_cast<idClass *&>( animowner ) );
if ( animowner ) {
animator = animowner->GetAnimator();
}
savefile->ReadObject( reinterpret_cast<idClass *&>( thread ) );
savefile->ReadString( state );
savefile->ReadInt( animBlendFrames );
savefile->ReadInt( lastAnimBlendFrames );
savefile->ReadInt( channel );
savefile->ReadBool( idleAnim );
savefile->ReadBool( disabled );
}
/*
=====================
idAnimState::Init
=====================
*/
void idAnimState::Init( idActor *owner, idAnimator *_animator, int animchannel ) {
assert( owner );
assert( _animator );
self = owner;
animator = _animator;
channel = animchannel;
if ( !thread ) {
thread = new idThread();
thread->ManualDelete();
}
thread->EndThread();
thread->ManualControl();
}
/*
=====================
idAnimState::Shutdown
=====================
*/
void idAnimState::Shutdown( void ) {
delete thread;
thread = NULL;
}
/*
=====================
idAnimState::SetState
=====================
*/
void idAnimState::SetState( const char *statename, int blendFrames ) {
const function_t *func;
func = self->scriptObject.GetFunction( statename );
if ( !func ) {
assert( 0 );
gameLocal.Error( "Can't find function '%s' in object '%s'", statename, self->scriptObject.GetTypeName() );
}
state = statename;
disabled = false;
animBlendFrames = blendFrames;
lastAnimBlendFrames = blendFrames;
thread->CallFunction( self, func, true );
animBlendFrames = blendFrames;
lastAnimBlendFrames = blendFrames;
disabled = false;
idleAnim = false;
if ( ai_debugScript.GetInteger() == self->entityNumber ) {
gameLocal.Printf( "%d: %s: Animstate: %s\n", gameLocal.time, self->name.c_str(), state.c_str() );
}
}
/*
=====================
idAnimState::StopAnim
=====================
*/
void idAnimState::StopAnim( int frames ) {
animBlendFrames = 0;
animator->Clear( channel, gameLocal.time, FRAME2MS( frames ) );
}
/*
=====================
idAnimState::PlayAnim
=====================
*/
void idAnimState::PlayAnim( int anim ) {
if ( anim ) {
animator->PlayAnim( channel, anim, gameLocal.time, FRAME2MS( animBlendFrames ) );
}
animBlendFrames = 0;
}
/*
=====================
idAnimState::CycleAnim
=====================
*/
void idAnimState::CycleAnim( int anim ) {
if ( anim ) {
animator->CycleAnim( channel, anim, gameLocal.time, FRAME2MS( animBlendFrames ) );
}
animBlendFrames = 0;
}
/*
=====================
idAnimState::BecomeIdle
=====================
*/
void idAnimState::BecomeIdle( void ) {
idleAnim = true;
}
/*
=====================
idAnimState::Disabled
=====================
*/
bool idAnimState::Disabled( void ) const {
return disabled;
}
/*
=====================
idAnimState::AnimDone
=====================
*/
bool idAnimState::AnimDone( int blendFrames ) const {
int animDoneTime;
animDoneTime = animator->CurrentAnim( channel )->GetEndTime();
if ( animDoneTime < 0 ) {
// playing a cycle
return false;
} else if ( animDoneTime - FRAME2MS( blendFrames ) <= gameLocal.time ) {
return true;
} else {
return false;
}
}
/*
=====================
idAnimState::IsIdle
=====================
*/
bool idAnimState::IsIdle( void ) const {
return disabled || idleAnim;
}
/*
=====================
idAnimState::GetAnimFlags
=====================
*/
animFlags_t idAnimState::GetAnimFlags( void ) const {
animFlags_t flags;
memset( &flags, 0, sizeof( flags ) );
if ( !disabled && !AnimDone( 0 ) ) {
flags = animator->GetAnimFlags( animator->CurrentAnim( channel )->AnimNum() );
}
return flags;
}
/*
=====================
idAnimState::Enable
=====================
*/
void idAnimState::Enable( int blendFrames ) {
if ( disabled ) {
disabled = false;
animBlendFrames = blendFrames;
lastAnimBlendFrames = blendFrames;
if ( state.Length() ) {
SetState( state.c_str(), blendFrames );
}
}
}
/*
=====================
idAnimState::Disable
=====================
*/
void idAnimState::Disable( void ) {
disabled = true;
idleAnim = false;
}
/*
=====================
idAnimState::UpdateState
=====================
*/
bool idAnimState::UpdateState( void ) {
if ( disabled ) {
return false;
}
if ( ai_debugScript.GetInteger() == self->entityNumber ) {
thread->EnableDebugInfo();
} else {
thread->DisableDebugInfo();
}
thread->Execute();
return true;
}
/***********************************************************************
idActor
***********************************************************************/
const idEventDef AI_EnableEyeFocus( "enableEyeFocus" );
const idEventDef AI_DisableEyeFocus( "disableEyeFocus" );
const idEventDef EV_Footstep( "footstep" );
const idEventDef EV_FootstepLeft( "leftFoot" );
const idEventDef EV_FootstepRight( "rightFoot" );
const idEventDef EV_EnableWalkIK( "EnableWalkIK" );
const idEventDef EV_DisableWalkIK( "DisableWalkIK" );
const idEventDef EV_EnableLegIK( "EnableLegIK", "d" );
const idEventDef EV_DisableLegIK( "DisableLegIK", "d" );
const idEventDef AI_StopAnim( "stopAnim", "dd" );
const idEventDef AI_PlayAnim( "playAnim", "ds", 'd' );
const idEventDef AI_PlayCycle( "playCycle", "ds", 'd' );
const idEventDef AI_IdleAnim( "idleAnim", "ds", 'd' );
const idEventDef AI_SetSyncedAnimWeight( "setSyncedAnimWeight", "ddf" );
const idEventDef AI_SetBlendFrames( "setBlendFrames", "dd" );
const idEventDef AI_GetBlendFrames( "getBlendFrames", "d", 'd' );
const idEventDef AI_AnimState( "animState", "dsd" );
const idEventDef AI_GetAnimState( "getAnimState", "d", 's' );
const idEventDef AI_InAnimState( "inAnimState", "ds", 'd' );
const idEventDef AI_FinishAction( "finishAction", "s" );
const idEventDef AI_AnimDone( "animDone", "dd", 'd' );
const idEventDef AI_OverrideAnim( "overrideAnim", "d" );
const idEventDef AI_EnableAnim( "enableAnim", "dd" );
const idEventDef AI_PreventPain( "preventPain", "f" );
const idEventDef AI_DisablePain( "disablePain" );
const idEventDef AI_EnablePain( "enablePain" );
const idEventDef AI_GetPainAnim( "getPainAnim", NULL, 's' );
const idEventDef AI_SetAnimPrefix( "setAnimPrefix", "s" );
const idEventDef AI_HasAnim( "hasAnim", "ds", 'f' );
const idEventDef AI_CheckAnim( "checkAnim", "ds" );
const idEventDef AI_ChooseAnim( "chooseAnim", "ds", 's' );
const idEventDef AI_AnimLength( "animLength", "ds", 'f' );
const idEventDef AI_AnimDistance( "animDistance", "ds", 'f' );
const idEventDef AI_HasEnemies( "hasEnemies", NULL, 'd' );
const idEventDef AI_NextEnemy( "nextEnemy", "E", 'e' );
const idEventDef AI_ClosestEnemyToPoint( "closestEnemyToPoint", "v", 'e' );
const idEventDef AI_SetNextState( "setNextState", "s" );
const idEventDef AI_SetState( "setState", "s" );
const idEventDef AI_GetState( "getState", NULL, 's' );
const idEventDef AI_GetHead( "getHead", NULL, 'e' );
CLASS_DECLARATION( idAFEntity_Gibbable, idActor )
EVENT( AI_EnableEyeFocus, idActor::Event_EnableEyeFocus )
EVENT( AI_DisableEyeFocus, idActor::Event_DisableEyeFocus )
EVENT( EV_Footstep, idActor::Event_Footstep )
EVENT( EV_FootstepLeft, idActor::Event_Footstep )
EVENT( EV_FootstepRight, idActor::Event_Footstep )
EVENT( EV_EnableWalkIK, idActor::Event_EnableWalkIK )
EVENT( EV_DisableWalkIK, idActor::Event_DisableWalkIK )
EVENT( EV_EnableLegIK, idActor::Event_EnableLegIK )
EVENT( EV_DisableLegIK, idActor::Event_DisableLegIK )
EVENT( AI_PreventPain, idActor::Event_PreventPain )
EVENT( AI_DisablePain, idActor::Event_DisablePain )
EVENT( AI_EnablePain, idActor::Event_EnablePain )
EVENT( AI_GetPainAnim, idActor::Event_GetPainAnim )
EVENT( AI_SetAnimPrefix, idActor::Event_SetAnimPrefix )
EVENT( AI_StopAnim, idActor::Event_StopAnim )
EVENT( AI_PlayAnim, idActor::Event_PlayAnim )
EVENT( AI_PlayCycle, idActor::Event_PlayCycle )
EVENT( AI_IdleAnim, idActor::Event_IdleAnim )
EVENT( AI_SetSyncedAnimWeight, idActor::Event_SetSyncedAnimWeight )
EVENT( AI_SetBlendFrames, idActor::Event_SetBlendFrames )
EVENT( AI_GetBlendFrames, idActor::Event_GetBlendFrames )
EVENT( AI_AnimState, idActor::Event_AnimState )
EVENT( AI_GetAnimState, idActor::Event_GetAnimState )
EVENT( AI_InAnimState, idActor::Event_InAnimState )
EVENT( AI_FinishAction, idActor::Event_FinishAction )
EVENT( AI_AnimDone, idActor::Event_AnimDone )
EVENT( AI_OverrideAnim, idActor::Event_OverrideAnim )
EVENT( AI_EnableAnim, idActor::Event_EnableAnim )
EVENT( AI_HasAnim, idActor::Event_HasAnim )
EVENT( AI_CheckAnim, idActor::Event_CheckAnim )
EVENT( AI_ChooseAnim, idActor::Event_ChooseAnim )
EVENT( AI_AnimLength, idActor::Event_AnimLength )
EVENT( AI_AnimDistance, idActor::Event_AnimDistance )
EVENT( AI_HasEnemies, idActor::Event_HasEnemies )
EVENT( AI_NextEnemy, idActor::Event_NextEnemy )
EVENT( AI_ClosestEnemyToPoint, idActor::Event_ClosestEnemyToPoint )
EVENT( EV_StopSound, idActor::Event_StopSound )
EVENT( AI_SetNextState, idActor::Event_SetNextState )
EVENT( AI_SetState, idActor::Event_SetState )
EVENT( AI_GetState, idActor::Event_GetState )
EVENT( AI_GetHead, idActor::Event_GetHead )
END_CLASS
/*
=====================
idActor::idActor
=====================
*/
idActor::idActor( void ) {
viewAxis.Identity();
scriptThread = NULL; // initialized by ConstructScriptObject, which is called by idEntity::Spawn
use_combat_bbox = false;
head = NULL;
team = 0;
rank = 0;
fovDot = 0.0f;
eyeOffset.Zero();
pain_debounce_time = 0;
pain_delay = 0;
pain_threshold = 0;
state = NULL;
idealState = NULL;
leftEyeJoint = INVALID_JOINT;
rightEyeJoint = INVALID_JOINT;
soundJoint = INVALID_JOINT;
modelOffset.Zero();
deltaViewAngles.Zero();
painTime = 0;
allowPain = false;
allowEyeFocus = false;
waitState = "";
blink_anim = 0;
blink_time = 0;
blink_min = 0;
blink_max = 0;
finalBoss = false;
attachments.SetGranularity( 1 );
enemyNode.SetOwner( this );
enemyList.SetOwner( this );
}
/*
=====================
idActor::~idActor
=====================
*/
idActor::~idActor( void ) {
int i;
idEntity *ent;
DeconstructScriptObject();
scriptObject.Free();
StopSound( SND_CHANNEL_ANY, false );
delete combatModel;
combatModel = NULL;
if ( head.GetEntity() ) {
head.GetEntity()->ClearBody();
head.GetEntity()->PostEventMS( &EV_Remove, 0 );
}
// remove any attached entities
for( i = 0; i < attachments.Num(); i++ ) {
ent = attachments[ i ].ent.GetEntity();
if ( ent ) {
ent->PostEventMS( &EV_Remove, 0 );
}
}
ShutdownThreads();
}
/*
=====================
idActor::Spawn
=====================
*/
void idActor::Spawn( void ) {
idEntity *ent;
idStr jointName;
float fovDegrees;
copyJoints_t copyJoint;
animPrefix = "";
state = NULL;
idealState = NULL;
spawnArgs.GetInt( "rank", "0", rank );
spawnArgs.GetInt( "team", "0", team );
spawnArgs.GetVector( "offsetModel", "0 0 0", modelOffset );
spawnArgs.GetBool( "use_combat_bbox", "0", use_combat_bbox );
viewAxis = GetPhysics()->GetAxis();
spawnArgs.GetFloat( "fov", "90", fovDegrees );
SetFOV( fovDegrees );
pain_debounce_time = 0;
pain_delay = SEC2MS( spawnArgs.GetFloat( "pain_delay" ) );
pain_threshold = spawnArgs.GetInt( "pain_threshold" );
LoadAF();
walkIK.Init( this, IK_ANIM, modelOffset );
// the animation used to be set to the IK_ANIM at this point, but that was fixed, resulting in
// attachments not binding correctly, so we're stuck setting the IK_ANIM before attaching things.
animator.ClearAllAnims( gameLocal.time, 0 );
animator.SetFrame( ANIMCHANNEL_ALL, animator.GetAnim( IK_ANIM ), 0, 0, 0 );
// spawn any attachments we might have
const idKeyValue *kv = spawnArgs.MatchPrefix( "def_attach", NULL );
while ( kv ) {
idDict args;
args.Set( "classname", kv->GetValue().c_str() );
// make items non-touchable so the player can't take them out of the character's hands
args.Set( "no_touch", "1" );
// don't let them drop to the floor
args.Set( "dropToFloor", "0" );
gameLocal.SpawnEntityDef( args, &ent );
if ( !ent ) {
gameLocal.Error( "Couldn't spawn '%s' to attach to entity '%s'", kv->GetValue().c_str(), name.c_str() );
} else {
Attach( ent );
}
kv = spawnArgs.MatchPrefix( "def_attach", kv );
}
SetupDamageGroups();
SetupHead();
// clear the bind anim
animator.ClearAllAnims( gameLocal.time, 0 );
idEntity *headEnt = head.GetEntity();
idAnimator *headAnimator;
if ( headEnt ) {
headAnimator = headEnt->GetAnimator();
} else {
headAnimator = &animator;
}
if ( headEnt ) {
// set up the list of joints to copy to the head
for( kv = spawnArgs.MatchPrefix( "copy_joint", NULL ); kv != NULL; kv = spawnArgs.MatchPrefix( "copy_joint", kv ) ) {
if ( kv->GetValue() == "" ) {
// probably clearing out inherited key, so skip it
continue;
}
jointName = kv->GetKey();
if ( jointName.StripLeadingOnce( "copy_joint_world " ) ) {
copyJoint.mod = JOINTMOD_WORLD_OVERRIDE;
} else {
jointName.StripLeadingOnce( "copy_joint " );
copyJoint.mod = JOINTMOD_LOCAL_OVERRIDE;
}
copyJoint.from = animator.GetJointHandle( jointName );
if ( copyJoint.from == INVALID_JOINT ) {
gameLocal.Warning( "Unknown copy_joint '%s' on entity %s", jointName.c_str(), name.c_str() );
continue;
}
jointName = kv->GetValue();
copyJoint.to = headAnimator->GetJointHandle( jointName );
if ( copyJoint.to == INVALID_JOINT ) {
gameLocal.Warning( "Unknown copy_joint '%s' on head of entity %s", jointName.c_str(), name.c_str() );
continue;
}
copyJoints.Append( copyJoint );
}
}
// set up blinking
blink_anim = headAnimator->GetAnim( "blink" );
blink_time = 0; // it's ok to blink right away
blink_min = SEC2MS( spawnArgs.GetFloat( "blink_min", "0.5" ) );
blink_max = SEC2MS( spawnArgs.GetFloat( "blink_max", "8" ) );
// set up the head anim if necessary
int headAnim = headAnimator->GetAnim( "def_head" );
if ( headAnim ) {
if ( headEnt ) {
headAnimator->CycleAnim( ANIMCHANNEL_ALL, headAnim, gameLocal.time, 0 );
} else {
headAnimator->CycleAnim( ANIMCHANNEL_HEAD, headAnim, gameLocal.time, 0 );
}
}
if ( spawnArgs.GetString( "sound_bone", "", jointName ) ) {
soundJoint = animator.GetJointHandle( jointName );
if ( soundJoint == INVALID_JOINT ) {
gameLocal.Warning( "idAnimated '%s' at (%s): cannot find joint '%s' for sound playback", name.c_str(), GetPhysics()->GetOrigin().ToString(0), jointName.c_str() );
}
}
finalBoss = spawnArgs.GetBool( "finalBoss" );
FinishSetup();
}
/*
================
idActor::FinishSetup
================
*/
void idActor::FinishSetup( void ) {
const char *scriptObjectName;
// setup script object
if ( spawnArgs.GetString( "scriptobject", NULL, &scriptObjectName ) ) {
if ( !scriptObject.SetType( scriptObjectName ) ) {
gameLocal.Error( "Script object '%s' not found on entity '%s'.", scriptObjectName, name.c_str() );
}
ConstructScriptObject();
}
SetupBody();
}
/*
================
idActor::SetupHead
================
*/
void idActor::SetupHead( void ) {
idAFAttachment *headEnt;
idStr jointName;
const char *headModel;
jointHandle_t joint;
jointHandle_t damageJoint;
int i;
const idKeyValue *sndKV;
if ( gameLocal.isClient ) {
return;
}
headModel = spawnArgs.GetString( "def_head", "" );
if ( headModel[ 0 ] ) {
jointName = spawnArgs.GetString( "head_joint" );
joint = animator.GetJointHandle( jointName );
if ( joint == INVALID_JOINT ) {
gameLocal.Error( "Joint '%s' not found for 'head_joint' on '%s'", jointName.c_str(), name.c_str() );
}
// set the damage joint to be part of the head damage group
damageJoint = joint;
for( i = 0; i < damageGroups.Num(); i++ ) {
if ( damageGroups[ i ] == "head" ) {
damageJoint = static_cast<jointHandle_t>( i );
break;
}
}
// copy any sounds in case we have frame commands on the head
idDict args;
sndKV = spawnArgs.MatchPrefix( "snd_", NULL );
while( sndKV ) {
args.Set( sndKV->GetKey(), sndKV->GetValue() );
sndKV = spawnArgs.MatchPrefix( "snd_", sndKV );
}
headEnt = static_cast<idAFAttachment *>( gameLocal.SpawnEntityType( idAFAttachment::Type, &args ) );
headEnt->SetName( va( "%s_head", name.c_str() ) );
headEnt->SetBody( this, headModel, damageJoint );
head = headEnt;
idVec3 origin;
idMat3 axis;
idAttachInfo &attach = attachments.Alloc();
attach.channel = animator.GetChannelForJoint( joint );
animator.GetJointTransform( joint, gameLocal.time, origin, axis );
origin = renderEntity.origin + ( origin + modelOffset ) * renderEntity.axis;
attach.ent = headEnt;
headEnt->SetOrigin( origin );
headEnt->SetAxis( renderEntity.axis );
headEnt->BindToJoint( this, joint, true );
}
}
/*
================
idActor::CopyJointsFromBodyToHead
================
*/
void idActor::CopyJointsFromBodyToHead( void ) {
idEntity *headEnt = head.GetEntity();
idAnimator *headAnimator;
int i;
idMat3 mat;
idMat3 axis;
idVec3 pos;
if ( !headEnt ) {
return;
}
headAnimator = headEnt->GetAnimator();
// copy the animation from the body to the head
for( i = 0; i < copyJoints.Num(); i++ ) {
if ( copyJoints[ i ].mod == JOINTMOD_WORLD_OVERRIDE ) {
mat = headEnt->GetPhysics()->GetAxis().Transpose();
GetJointWorldTransform( copyJoints[ i ].from, gameLocal.time, pos, axis );
pos -= headEnt->GetPhysics()->GetOrigin();
headAnimator->SetJointPos( copyJoints[ i ].to, copyJoints[ i ].mod, pos * mat );
headAnimator->SetJointAxis( copyJoints[ i ].to, copyJoints[ i ].mod, axis * mat );
} else {
animator.GetJointLocalTransform( copyJoints[ i ].from, gameLocal.time, pos, axis );
headAnimator->SetJointPos( copyJoints[ i ].to, copyJoints[ i ].mod, pos );
headAnimator->SetJointAxis( copyJoints[ i ].to, copyJoints[ i ].mod, axis );
}
}
}
/*
================
idActor::Restart
================
*/
void idActor::Restart( void ) {
assert( !head.GetEntity() );
SetupHead();
FinishSetup();
}
/*
================
idActor::Save
archive object for savegame file
================
*/
void idActor::Save( idSaveGame *savefile ) const {
idActor *ent;
int i;
savefile->WriteInt( team );
savefile->WriteInt( rank );
savefile->WriteMat3( viewAxis );
savefile->WriteInt( enemyList.Num() );
for ( ent = enemyList.Next(); ent != NULL; ent = ent->enemyNode.Next() ) {
savefile->WriteObject( ent );
}
savefile->WriteFloat( fovDot );
savefile->WriteVec3( eyeOffset );
savefile->WriteVec3( modelOffset );
savefile->WriteAngles( deltaViewAngles );
savefile->WriteInt( pain_debounce_time );
savefile->WriteInt( pain_delay );
savefile->WriteInt( pain_threshold );
savefile->WriteInt( damageGroups.Num() );
for( i = 0; i < damageGroups.Num(); i++ ) {
savefile->WriteString( damageGroups[ i ] );
}
savefile->WriteInt( damageScale.Num() );
for( i = 0; i < damageScale.Num(); i++ ) {
savefile->WriteFloat( damageScale[ i ] );
}
savefile->WriteBool( use_combat_bbox );
head.Save( savefile );
savefile->WriteInt( copyJoints.Num() );
for( i = 0; i < copyJoints.Num(); i++ ) {
savefile->WriteInt( copyJoints[i].mod );
savefile->WriteJoint( copyJoints[i].from );
savefile->WriteJoint( copyJoints[i].to );
}
savefile->WriteJoint( leftEyeJoint );
savefile->WriteJoint( rightEyeJoint );
savefile->WriteJoint( soundJoint );
walkIK.Save( savefile );
savefile->WriteString( animPrefix );
savefile->WriteString( painAnim );
savefile->WriteInt( blink_anim );
savefile->WriteInt( blink_time );
savefile->WriteInt( blink_min );
savefile->WriteInt( blink_max );
// script variables
savefile->WriteObject( scriptThread );
savefile->WriteString( waitState );
headAnim.Save( savefile );
torsoAnim.Save( savefile );
legsAnim.Save( savefile );
savefile->WriteBool( allowPain );
savefile->WriteBool( allowEyeFocus );
savefile->WriteInt( painTime );
savefile->WriteInt( attachments.Num() );
for ( i = 0; i < attachments.Num(); i++ ) {
attachments[i].ent.Save( savefile );
savefile->WriteInt( attachments[i].channel );
}
savefile->WriteBool( finalBoss );
idToken token;
//FIXME: this is unneccesary
if ( state ) {
idLexer src( state->Name(), idStr::Length( state->Name() ), "idAI::Save" );
src.ReadTokenOnLine( &token );
src.ExpectTokenString( "::" );
src.ReadTokenOnLine( &token );
savefile->WriteString( token );
} else {
savefile->WriteString( "" );
}
if ( idealState ) {
idLexer src( idealState->Name(), idStr::Length( idealState->Name() ), "idAI::Save" );
src.ReadTokenOnLine( &token );
src.ExpectTokenString( "::" );
src.ReadTokenOnLine( &token );
savefile->WriteString( token );
} else {
savefile->WriteString( "" );
}
}
/*
================
idActor::Restore
unarchives object from save game file
================
*/
void idActor::Restore( idRestoreGame *savefile ) {
int i, num;
idActor *ent;
savefile->ReadInt( team );
savefile->ReadInt( rank );
savefile->ReadMat3( viewAxis );
savefile->ReadInt( num );
for ( i = 0; i < num; i++ ) {
savefile->ReadObject( reinterpret_cast<idClass *&>( ent ) );
assert( ent );
if ( ent ) {
ent->enemyNode.AddToEnd( enemyList );
}
}
savefile->ReadFloat( fovDot );
savefile->ReadVec3( eyeOffset );
savefile->ReadVec3( modelOffset );
savefile->ReadAngles( deltaViewAngles );
savefile->ReadInt( pain_debounce_time );
savefile->ReadInt( pain_delay );
savefile->ReadInt( pain_threshold );
savefile->ReadInt( num );
damageGroups.SetGranularity( 1 );
damageGroups.SetNum( num );
for( i = 0; i < num; i++ ) {
savefile->ReadString( damageGroups[ i ] );
}
savefile->ReadInt( num );
damageScale.SetNum( num );
for( i = 0; i < num; i++ ) {
savefile->ReadFloat( damageScale[ i ] );
}
savefile->ReadBool( use_combat_bbox );
head.Restore( savefile );
savefile->ReadInt( num );
copyJoints.SetNum( num );
for( i = 0; i < num; i++ ) {
int val;
savefile->ReadInt( val );
copyJoints[i].mod = static_cast<jointModTransform_t>( val );
savefile->ReadJoint( copyJoints[i].from );
savefile->ReadJoint( copyJoints[i].to );
}
savefile->ReadJoint( leftEyeJoint );
savefile->ReadJoint( rightEyeJoint );
savefile->ReadJoint( soundJoint );
walkIK.Restore( savefile );
savefile->ReadString( animPrefix );
savefile->ReadString( painAnim );
savefile->ReadInt( blink_anim );
savefile->ReadInt( blink_time );
savefile->ReadInt( blink_min );
savefile->ReadInt( blink_max );
savefile->ReadObject( reinterpret_cast<idClass *&>( scriptThread ) );
savefile->ReadString( waitState );
headAnim.Restore( savefile );
torsoAnim.Restore( savefile );
legsAnim.Restore( savefile );
savefile->ReadBool( allowPain );
savefile->ReadBool( allowEyeFocus );
savefile->ReadInt( painTime );
savefile->ReadInt( num );
for ( i = 0; i < num; i++ ) {
idAttachInfo &attach = attachments.Alloc();
attach.ent.Restore( savefile );
savefile->ReadInt( attach.channel );
}
savefile->ReadBool( finalBoss );
idStr statename;
savefile->ReadString( statename );
if ( statename.Length() > 0 ) {
state = GetScriptFunction( statename );
}
savefile->ReadString( statename );
if ( statename.Length() > 0 ) {
idealState = GetScriptFunction( statename );
}
}
/*
================
idActor::Hide
================
*/
void idActor::Hide( void ) {
idEntity *ent;
idEntity *next;
idAFEntity_Base::Hide();
if ( head.GetEntity() ) {
head.GetEntity()->Hide();
}
for( ent = GetNextTeamEntity(); ent != NULL; ent = next ) {
next = ent->GetNextTeamEntity();
if ( ent->GetBindMaster() == this ) {
ent->Hide();
if ( ent->IsType( idLight::Type ) ) {
static_cast<idLight *>( ent )->Off();
}
}
}
UnlinkCombat();
}
/*
================
idActor::Show
================
*/
void idActor::Show( void ) {
idEntity *ent;
idEntity *next;
idAFEntity_Base::Show();
if ( head.GetEntity() ) {
head.GetEntity()->Show();
}
for( ent = GetNextTeamEntity(); ent != NULL; ent = next ) {
next = ent->GetNextTeamEntity();
if ( ent->GetBindMaster() == this ) {
ent->Show();
if ( ent->IsType( idLight::Type ) ) {
static_cast<idLight *>( ent )->On();
}
}
}
LinkCombat();
}
/*
==============
idActor::GetDefaultSurfaceType
==============
*/
int idActor::GetDefaultSurfaceType( void ) const {
return SURFTYPE_FLESH;
}
/*
================
idActor::ProjectOverlay
================
*/
void idActor::ProjectOverlay( const idVec3 &origin, const idVec3 &dir, float size, const char *material ) {
idEntity *ent;
idEntity *next;
idEntity::ProjectOverlay( origin, dir, size, material );
for( ent = GetNextTeamEntity(); ent != NULL; ent = next ) {
next = ent->GetNextTeamEntity();
if ( ent->GetBindMaster() == this ) {
if ( ent->fl.takedamage && ent->spawnArgs.GetBool( "bleed" ) ) {
ent->ProjectOverlay( origin, dir, size, material );
}
}
}
}
/*
================
idActor::LoadAF
================
*/
bool idActor::LoadAF( void ) {
idStr fileName;
if ( !spawnArgs.GetString( "ragdoll", "*unknown*", fileName ) || !fileName.Length() ) {
return false;
}
af.SetAnimator( GetAnimator() );
return af.Load( this, fileName );
}
/*
=====================
idActor::SetupBody
=====================
*/
void idActor::SetupBody( void ) {
const char *jointname;
animator.ClearAllAnims( gameLocal.time, 0 );
animator.ClearAllJoints();
idEntity *headEnt = head.GetEntity();
if ( headEnt ) {
jointname = spawnArgs.GetString( "bone_leftEye" );
leftEyeJoint = headEnt->GetAnimator()->GetJointHandle( jointname );
jointname = spawnArgs.GetString( "bone_rightEye" );
rightEyeJoint = headEnt->GetAnimator()->GetJointHandle( jointname );
// set up the eye height. check if it's specified in the def.
if ( !spawnArgs.GetFloat( "eye_height", "0", eyeOffset.z ) ) {
// if not in the def, then try to base it off the idle animation
int anim = headEnt->GetAnimator()->GetAnim( "idle" );
if ( anim && ( leftEyeJoint != INVALID_JOINT ) ) {
idVec3 pos;
idMat3 axis;
headEnt->GetAnimator()->PlayAnim( ANIMCHANNEL_ALL, anim, gameLocal.time, 0 );
headEnt->GetAnimator()->GetJointTransform( leftEyeJoint, gameLocal.time, pos, axis );
headEnt->GetAnimator()->ClearAllAnims( gameLocal.time, 0 );
headEnt->GetAnimator()->ForceUpdate();
pos += headEnt->GetPhysics()->GetOrigin() - GetPhysics()->GetOrigin();
eyeOffset = pos + modelOffset;
} else {
// just base it off the bounding box size
eyeOffset.z = GetPhysics()->GetBounds()[ 1 ].z - 6;
}
}
headAnim.Init( this, headEnt->GetAnimator(), ANIMCHANNEL_ALL );
} else {
jointname = spawnArgs.GetString( "bone_leftEye" );
leftEyeJoint = animator.GetJointHandle( jointname );
jointname = spawnArgs.GetString( "bone_rightEye" );
rightEyeJoint = animator.GetJointHandle( jointname );
// set up the eye height. check if it's specified in the def.
if ( !spawnArgs.GetFloat( "eye_height", "0", eyeOffset.z ) ) {
// if not in the def, then try to base it off the idle animation
int anim = animator.GetAnim( "idle" );
if ( anim && ( leftEyeJoint != INVALID_JOINT ) ) {
idVec3 pos;
idMat3 axis;
animator.PlayAnim( ANIMCHANNEL_ALL, anim, gameLocal.time, 0 );
animator.GetJointTransform( leftEyeJoint, gameLocal.time, pos, axis );
animator.ClearAllAnims( gameLocal.time, 0 );
animator.ForceUpdate();
eyeOffset = pos + modelOffset;
} else {
// just base it off the bounding box size
eyeOffset.z = GetPhysics()->GetBounds()[ 1 ].z - 6;
}
}
headAnim.Init( this, &animator, ANIMCHANNEL_HEAD );
}
waitState = "";
torsoAnim.Init( this, &animator, ANIMCHANNEL_TORSO );
legsAnim.Init( this, &animator, ANIMCHANNEL_LEGS );
}
/*
=====================
idActor::CheckBlink
=====================
*/
void idActor::CheckBlink( void ) {
// check if it's time to blink
if ( !blink_anim || ( health <= 0 ) || !allowEyeFocus || ( blink_time > gameLocal.time ) ) {
return;
}
idEntity *headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->GetAnimator()->PlayAnim( ANIMCHANNEL_EYELIDS, blink_anim, gameLocal.time, 1 );
} else {
animator.PlayAnim( ANIMCHANNEL_EYELIDS, blink_anim, gameLocal.time, 1 );
}
// set the next blink time
blink_time = gameLocal.time + blink_min + gameLocal.random.RandomFloat() * ( blink_max - blink_min );
}
/*
================
idActor::GetPhysicsToVisualTransform
================
*/
bool idActor::GetPhysicsToVisualTransform( idVec3 &origin, idMat3 &axis ) {
if ( af.IsActive() ) {
af.GetPhysicsToVisualTransform( origin, axis );
return true;
}
origin = modelOffset;
axis = viewAxis;
return true;
}
/*
================
idActor::GetPhysicsToSoundTransform
================
*/
bool idActor::GetPhysicsToSoundTransform( idVec3 &origin, idMat3 &axis ) {
if ( soundJoint != INVALID_JOINT ) {
animator.GetJointTransform( soundJoint, gameLocal.time, origin, axis );
origin += modelOffset;
axis = viewAxis;
} else {
origin = GetPhysics()->GetGravityNormal() * -eyeOffset.z;
axis.Identity();
}
return true;
}
/***********************************************************************
script state management
***********************************************************************/
/*
================
idActor::ShutdownThreads
================
*/
void idActor::ShutdownThreads( void ) {
headAnim.Shutdown();
torsoAnim.Shutdown();
legsAnim.Shutdown();
if ( scriptThread ) {
scriptThread->EndThread();
scriptThread->PostEventMS( &EV_Remove, 0 );
delete scriptThread;
scriptThread = NULL;
}
}
/*
================
idActor::ShouldConstructScriptObjectAtSpawn
Called during idEntity::Spawn to see if it should construct the script object or not.
Overridden by subclasses that need to spawn the script object themselves.
================
*/
bool idActor::ShouldConstructScriptObjectAtSpawn( void ) const {
return false;
}
/*
================
idActor::ConstructScriptObject
Called during idEntity::Spawn. Calls the constructor on the script object.
Can be overridden by subclasses when a thread doesn't need to be allocated.
================
*/
idThread *idActor::ConstructScriptObject( void ) {
const function_t *constructor;
// make sure we have a scriptObject
if ( !scriptObject.HasObject() ) {
gameLocal.Error( "No scriptobject set on '%s'. Check the '%s' entityDef.", name.c_str(), GetEntityDefName() );
}
if ( !scriptThread ) {
// create script thread
scriptThread = new idThread();
scriptThread->ManualDelete();
scriptThread->ManualControl();
scriptThread->SetThreadName( name.c_str() );
} else {
scriptThread->EndThread();
}
// call script object's constructor
constructor = scriptObject.GetConstructor();
if ( !constructor ) {
gameLocal.Error( "Missing constructor on '%s' for entity '%s'", scriptObject.GetTypeName(), name.c_str() );
}
// init the script object's data
scriptObject.ClearObject();
// just set the current function on the script. we'll execute in the subclasses.
scriptThread->CallFunction( this, constructor, true );
return scriptThread;
}
/*
=====================
idActor::GetScriptFunction
=====================
*/
const function_t *idActor::GetScriptFunction( const char *funcname ) {
const function_t *func;
func = scriptObject.GetFunction( funcname );
if ( !func ) {
scriptThread->Error( "Unknown function '%s' in '%s'", funcname, scriptObject.GetTypeName() );
}
return func;
}
/*
=====================
idActor::SetState
=====================
*/
void idActor::SetState( const function_t *newState ) {
if ( !newState ) {
gameLocal.Error( "idActor::SetState: Null state" );
}
if ( ai_debugScript.GetInteger() == entityNumber ) {
gameLocal.Printf( "%d: %s: State: %s\n", gameLocal.time, name.c_str(), newState->Name() );
}
state = newState;
idealState = state;
scriptThread->CallFunction( this, state, true );
}
/*
=====================
idActor::SetState
=====================
*/
void idActor::SetState( const char *statename ) {
const function_t *newState;
newState = GetScriptFunction( statename );
SetState( newState );
}
/*
=====================
idActor::UpdateScript
=====================
*/
void idActor::UpdateScript( void ) {
int i;
if ( ai_debugScript.GetInteger() == entityNumber ) {
scriptThread->EnableDebugInfo();
} else {
scriptThread->DisableDebugInfo();
}
// a series of state changes can happen in a single frame.
// this loop limits them in case we've entered an infinite loop.
for( i = 0; i < 20; i++ ) {
if ( idealState != state ) {
SetState( idealState );
}
// don't call script until it's done waiting
if ( scriptThread->IsWaiting() ) {
break;
}
scriptThread->Execute();
if ( idealState == state ) {
break;
}
}
if ( i == 20 ) {
scriptThread->Warning( "idActor::UpdateScript: exited loop to prevent lockup" );
}
}
/***********************************************************************
vision
***********************************************************************/
/*
=====================
idActor::setFov
=====================
*/
void idActor::SetFOV( float fov ) {
fovDot = (float)cos( DEG2RAD( fov * 0.5f ) );
}
/*
=====================
idActor::SetEyeHeight
=====================
*/
void idActor::SetEyeHeight( float height ) {
eyeOffset.z = height;
}
/*
=====================
idActor::EyeHeight
=====================
*/
float idActor::EyeHeight( void ) const {
return eyeOffset.z;
}
/*
=====================
idActor::EyeOffset
=====================
*/
idVec3 idActor::EyeOffset( void ) const {
return GetPhysics()->GetGravityNormal() * -eyeOffset.z;
}
/*
=====================
idActor::GetEyePosition
=====================
*/
idVec3 idActor::GetEyePosition( void ) const {
return GetPhysics()->GetOrigin() + ( GetPhysics()->GetGravityNormal() * -eyeOffset.z );
}
/*
=====================
idActor::GetViewPos
=====================
*/
void idActor::GetViewPos( idVec3 &origin, idMat3 &axis ) const {
origin = GetEyePosition();
axis = viewAxis;
}
/*
=====================
idActor::CheckFOV
=====================
*/
bool idActor::CheckFOV( const idVec3 &pos ) const {
if ( fovDot == 1.0f ) {
return true;
}
float dot;
idVec3 delta;
delta = pos - GetEyePosition();
// get our gravity normal
const idVec3 &gravityDir = GetPhysics()->GetGravityNormal();
// infinite vertical vision, so project it onto our orientation plane
delta -= gravityDir * ( gravityDir * delta );
delta.Normalize();
dot = viewAxis[ 0 ] * delta;
return ( dot >= fovDot );
}
/*
=====================
idActor::CanSee
=====================
*/
bool idActor::CanSee( idEntity *ent, bool useFov ) const {
trace_t tr;
idVec3 eye;
idVec3 toPos;
if ( ent->IsHidden() ) {
return false;
}
if ( ent->IsType( idActor::Type ) ) {
toPos = ( ( idActor * )ent )->GetEyePosition();
} else {
toPos = ent->GetPhysics()->GetOrigin();
}
if ( useFov && !CheckFOV( toPos ) ) {
return false;
}
eye = GetEyePosition();
gameLocal.clip.TracePoint( tr, eye, toPos, MASK_OPAQUE, this );
if ( tr.fraction >= 1.0f || ( gameLocal.GetTraceEntity( tr ) == ent ) ) {
return true;
}
return false;
}
/*
=====================
idActor::PointVisible
=====================
*/
bool idActor::PointVisible( const idVec3 &point ) const {
trace_t results;
idVec3 start, end;
start = GetEyePosition();
end = point;
end[2] += 1.0f;
gameLocal.clip.TracePoint( results, start, end, MASK_OPAQUE, this );
return ( results.fraction >= 1.0f );
}
/*
=====================
idActor::GetAIAimTargets
Returns positions for the AI to aim at.
=====================
*/
void idActor::GetAIAimTargets( const idVec3 &lastSightPos, idVec3 &headPos, idVec3 &chestPos ) {
headPos = lastSightPos + EyeOffset();
chestPos = ( headPos + lastSightPos + GetPhysics()->GetBounds().GetCenter() ) * 0.5f;
}
/*
=====================
idActor::GetRenderView
=====================
*/
renderView_t *idActor::GetRenderView() {
renderView_t *rv = idEntity::GetRenderView();
rv->viewaxis = viewAxis;
rv->vieworg = GetEyePosition();
return rv;
}
/***********************************************************************
Model/Ragdoll
***********************************************************************/
/*
================
idActor::SetCombatModel
================
*/
void idActor::SetCombatModel( void ) {
idAFAttachment *headEnt;
if ( !use_combat_bbox ) {
if ( combatModel ) {
combatModel->Unlink();
combatModel->LoadModel( modelDefHandle );
} else {
combatModel = new idClipModel( modelDefHandle );
}
headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->SetCombatModel();
}
}
}
/*
================
idActor::GetCombatModel
================
*/
idClipModel *idActor::GetCombatModel( void ) const {
return combatModel;
}
/*
================
idActor::LinkCombat
================
*/
void idActor::LinkCombat( void ) {
idAFAttachment *headEnt;
if ( fl.hidden || use_combat_bbox ) {
return;
}
if ( combatModel ) {
combatModel->Link( gameLocal.clip, this, 0, renderEntity.origin, renderEntity.axis, modelDefHandle );
}
headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->LinkCombat();
}
}
/*
================
idActor::UnlinkCombat
================
*/
void idActor::UnlinkCombat( void ) {
idAFAttachment *headEnt;
if ( combatModel ) {
combatModel->Unlink();
}
headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->UnlinkCombat();
}
}
/*
================
idActor::StartRagdoll
================
*/
bool idActor::StartRagdoll( void ) {
float slomoStart, slomoEnd;
float jointFrictionDent, jointFrictionDentStart, jointFrictionDentEnd;
float contactFrictionDent, contactFrictionDentStart, contactFrictionDentEnd;
// if no AF loaded
if ( !af.IsLoaded() ) {
return false;
}
// if the AF is already active
if ( af.IsActive() ) {
return true;
}
// disable the monster bounding box
GetPhysics()->DisableClip();
// start using the AF
af.StartFromCurrentPose( spawnArgs.GetInt( "velocityTime", "0" ) );
slomoStart = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_slomoStart", "-1.6" );
slomoEnd = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_slomoEnd", "0.8" );
// do the first part of the death in slow motion
af.GetPhysics()->SetTimeScaleRamp( slomoStart, slomoEnd );
jointFrictionDent = spawnArgs.GetFloat( "ragdoll_jointFrictionDent", "0.1" );
jointFrictionDentStart = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_jointFrictionStart", "0.2" );
jointFrictionDentEnd = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_jointFrictionEnd", "1.2" );
// set joint friction dent
af.GetPhysics()->SetJointFrictionDent( jointFrictionDent, jointFrictionDentStart, jointFrictionDentEnd );
contactFrictionDent = spawnArgs.GetFloat( "ragdoll_contactFrictionDent", "0.1" );
contactFrictionDentStart = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_contactFrictionStart", "1.0" );
contactFrictionDentEnd = MS2SEC( gameLocal.time ) + spawnArgs.GetFloat( "ragdoll_contactFrictionEnd", "2.0" );
// set contact friction dent
af.GetPhysics()->SetContactFrictionDent( contactFrictionDent, contactFrictionDentStart, contactFrictionDentEnd );
// drop any items the actor is holding
idMoveableItem::DropItems( this, "death", NULL );
// drop any articulated figures the actor is holding
idAFEntity_Base::DropAFs( this, "death", NULL );
RemoveAttachments();
return true;
}
/*
================
idActor::StopRagdoll
================
*/
void idActor::StopRagdoll( void ) {
if ( af.IsActive() ) {
af.Stop();
}
}
/*
================
idActor::UpdateAnimationControllers
================
*/
bool idActor::UpdateAnimationControllers( void ) {
if ( af.IsActive() ) {
return idAFEntity_Base::UpdateAnimationControllers();
} else {
animator.ClearAFPose();
}
if ( walkIK.IsInitialized() ) {
walkIK.Evaluate();
return true;
}
return false;
}
/*
================
idActor::RemoveAttachments
================
*/
void idActor::RemoveAttachments( void ) {
int i;
idEntity *ent;
// remove any attached entities
for( i = 0; i < attachments.Num(); i++ ) {
ent = attachments[ i ].ent.GetEntity();
if ( ent && ent->spawnArgs.GetBool( "remove" ) ) {
ent->PostEventMS( &EV_Remove, 0 );
}
}
}
/*
================
idActor::Attach
================
*/
void idActor::Attach( idEntity *ent ) {
idVec3 origin;
idMat3 axis;
jointHandle_t joint;
idStr jointName;
idAttachInfo &attach = attachments.Alloc();
idAngles angleOffset;
idVec3 originOffset;
jointName = ent->spawnArgs.GetString( "joint" );
joint = animator.GetJointHandle( jointName );
if ( joint == INVALID_JOINT ) {
gameLocal.Error( "Joint '%s' not found for attaching '%s' on '%s'", jointName.c_str(), ent->GetClassname(), name.c_str() );
}
angleOffset = ent->spawnArgs.GetAngles( "angles" );
originOffset = ent->spawnArgs.GetVector( "origin" );
attach.channel = animator.GetChannelForJoint( joint );
GetJointWorldTransform( joint, gameLocal.time, origin, axis );
attach.ent = ent;
ent->SetOrigin( origin + originOffset * renderEntity.axis );
idMat3 rotate = angleOffset.ToMat3();
idMat3 newAxis = rotate * axis;
ent->SetAxis( newAxis );
ent->BindToJoint( this, joint, true );
ent->cinematic = cinematic;
}
/*
================
idActor::Teleport
================
*/
void idActor::Teleport( const idVec3 &origin, const idAngles &angles, idEntity *destination ) {
GetPhysics()->SetOrigin( origin + idVec3( 0, 0, CM_CLIP_EPSILON ) );
GetPhysics()->SetLinearVelocity( vec3_origin );
viewAxis = angles.ToMat3();
UpdateVisuals();
if ( !IsHidden() ) {
// kill anything at the new position
gameLocal.KillBox( this );
}
}
/*
================
idActor::GetDeltaViewAngles
================
*/
const idAngles &idActor::GetDeltaViewAngles( void ) const {
return deltaViewAngles;
}
/*
================
idActor::SetDeltaViewAngles
================
*/
void idActor::SetDeltaViewAngles( const idAngles &delta ) {
deltaViewAngles = delta;
}
/*
================
idActor::HasEnemies
================
*/
bool idActor::HasEnemies( void ) const {
idActor *ent;
for( ent = enemyList.Next(); ent != NULL; ent = ent->enemyNode.Next() ) {
if ( !ent->fl.hidden ) {
return true;
}
}
return false;
}
/*
================
idActor::ClosestEnemyToPoint
================
*/
idActor *idActor::ClosestEnemyToPoint( const idVec3 &pos ) {
idActor *ent;
idActor *bestEnt;
float bestDistSquared;
float distSquared;
idVec3 delta;
bestDistSquared = idMath::INFINITY;
bestEnt = NULL;
for( ent = enemyList.Next(); ent != NULL; ent = ent->enemyNode.Next() ) {
if ( ent->fl.hidden ) {
continue;
}
delta = ent->GetPhysics()->GetOrigin() - pos;
distSquared = delta.LengthSqr();
if ( distSquared < bestDistSquared ) {
bestEnt = ent;
bestDistSquared = distSquared;
}
}
return bestEnt;
}
/*
================
idActor::EnemyWithMostHealth
================
*/
idActor *idActor::EnemyWithMostHealth() {
idActor *ent;
idActor *bestEnt;
int most = -9999;
bestEnt = NULL;
for( ent = enemyList.Next(); ent != NULL; ent = ent->enemyNode.Next() ) {
if ( !ent->fl.hidden && ( ent->health > most ) ) {
bestEnt = ent;
most = ent->health;
}
}
return bestEnt;
}
/*
================
idActor::OnLadder
================
*/
bool idActor::OnLadder( void ) const {
return false;
}
/*
==============
idActor::GetAASLocation
==============
*/
void idActor::GetAASLocation( idAAS *aas, idVec3 &pos, int &areaNum ) const {
idVec3 size;
idBounds bounds;
GetFloorPos( 64.0f, pos );
if ( !aas ) {
areaNum = 0;
return;
}
size = aas->GetSettings()->boundingBoxes[0][1];
bounds[0] = -size;
size.z = 32.0f;
bounds[1] = size;
areaNum = aas->PointReachableAreaNum( pos, bounds, AREA_REACHABLE_WALK );
if ( areaNum ) {
aas->PushPointIntoAreaNum( areaNum, pos );
}
}
/***********************************************************************
animation state
***********************************************************************/
/*
=====================
idActor::SetAnimState
=====================
*/
void idActor::SetAnimState( int channel, const char *statename, int blendFrames ) {
const function_t *func;
func = scriptObject.GetFunction( statename );
if ( !func ) {
assert( 0 );
gameLocal.Error( "Can't find function '%s' in object '%s'", statename, scriptObject.GetTypeName() );
}
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.SetState( statename, blendFrames );
allowEyeFocus = true;
break;
case ANIMCHANNEL_TORSO :
torsoAnim.SetState( statename, blendFrames );
legsAnim.Enable( blendFrames );
allowPain = true;
allowEyeFocus = true;
break;
case ANIMCHANNEL_LEGS :
legsAnim.SetState( statename, blendFrames );
torsoAnim.Enable( blendFrames );
allowPain = true;
allowEyeFocus = true;
break;
default:
gameLocal.Error( "idActor::SetAnimState: Unknown anim group" );
break;
}
}
/*
=====================
idActor::GetAnimState
=====================
*/
const char *idActor::GetAnimState( int channel ) const {
switch( channel ) {
case ANIMCHANNEL_HEAD :
return headAnim.state;
break;
case ANIMCHANNEL_TORSO :
return torsoAnim.state;
break;
case ANIMCHANNEL_LEGS :
return legsAnim.state;
break;
default:
gameLocal.Error( "idActor::GetAnimState: Unknown anim group" );
return NULL;
break;
}
}
/*
=====================
idActor::InAnimState
=====================
*/
bool idActor::InAnimState( int channel, const char *statename ) const {
switch( channel ) {
case ANIMCHANNEL_HEAD :
if ( headAnim.state == statename ) {
return true;
}
break;
case ANIMCHANNEL_TORSO :
if ( torsoAnim.state == statename ) {
return true;
}
break;
case ANIMCHANNEL_LEGS :
if ( legsAnim.state == statename ) {
return true;
}
break;
default:
gameLocal.Error( "idActor::InAnimState: Unknown anim group" );
break;
}
return false;
}
/*
=====================
idActor::WaitState
=====================
*/
const char *idActor::WaitState( void ) const {
if ( waitState.Length() ) {
return waitState;
} else {
return NULL;
}
}
/*
=====================
idActor::SetWaitState
=====================
*/
void idActor::SetWaitState( const char *_waitstate ) {
waitState = _waitstate;
}
/*
=====================
idActor::UpdateAnimState
=====================
*/
void idActor::UpdateAnimState( void ) {
headAnim.UpdateState();
torsoAnim.UpdateState();
legsAnim.UpdateState();
}
/*
=====================
idActor::GetAnim
=====================
*/
int idActor::GetAnim( int channel, const char *animname ) {
int anim;
const char *temp;
idAnimator *animatorPtr;
if ( channel == ANIMCHANNEL_HEAD ) {
if ( !head.GetEntity() ) {
return 0;
}
animatorPtr = head.GetEntity()->GetAnimator();
} else {
animatorPtr = &animator;
}
if ( animPrefix.Length() ) {
temp = va( "%s_%s", animPrefix.c_str(), animname );
anim = animatorPtr->GetAnim( temp );
if ( anim ) {
return anim;
}
}
anim = animatorPtr->GetAnim( animname );
return anim;
}
/*
===============
idActor::SyncAnimChannels
===============
*/
void idActor::SyncAnimChannels( int channel, int syncToChannel, int blendFrames ) {
idAnimator *headAnimator;
idAFAttachment *headEnt;
int anim;
idAnimBlend *syncAnim;
int starttime;
int blendTime;
int cycle;
blendTime = FRAME2MS( blendFrames );
if ( channel == ANIMCHANNEL_HEAD ) {
headEnt = head.GetEntity();
if ( headEnt ) {
headAnimator = headEnt->GetAnimator();
syncAnim = animator.CurrentAnim( syncToChannel );
if ( syncAnim ) {
anim = headAnimator->GetAnim( syncAnim->AnimFullName() );
if ( !anim ) {
anim = headAnimator->GetAnim( syncAnim->AnimName() );
}
if ( anim ) {
cycle = animator.CurrentAnim( syncToChannel )->GetCycleCount();
starttime = animator.CurrentAnim( syncToChannel )->GetStartTime();
headAnimator->PlayAnim( ANIMCHANNEL_ALL, anim, gameLocal.time, blendTime );
headAnimator->CurrentAnim( ANIMCHANNEL_ALL )->SetCycleCount( cycle );
headAnimator->CurrentAnim( ANIMCHANNEL_ALL )->SetStartTime( starttime );
} else {
headEnt->PlayIdleAnim( blendTime );
}
}
}
} else if ( syncToChannel == ANIMCHANNEL_HEAD ) {
headEnt = head.GetEntity();
if ( headEnt ) {
headAnimator = headEnt->GetAnimator();
syncAnim = headAnimator->CurrentAnim( ANIMCHANNEL_ALL );
if ( syncAnim ) {
anim = GetAnim( channel, syncAnim->AnimFullName() );
if ( !anim ) {
anim = GetAnim( channel, syncAnim->AnimName() );
}
if ( anim ) {
cycle = headAnimator->CurrentAnim( ANIMCHANNEL_ALL )->GetCycleCount();
starttime = headAnimator->CurrentAnim( ANIMCHANNEL_ALL )->GetStartTime();
animator.PlayAnim( channel, anim, gameLocal.time, blendTime );
animator.CurrentAnim( channel )->SetCycleCount( cycle );
animator.CurrentAnim( channel )->SetStartTime( starttime );
}
}
}
} else {
animator.SyncAnimChannels( channel, syncToChannel, gameLocal.time, blendTime );
}
}
/***********************************************************************
Damage
***********************************************************************/
/*
============
idActor::Gib
============
*/
void idActor::Gib( const idVec3 &dir, const char *damageDefName ) {
// no gibbing in multiplayer - by self damage or by moving objects
if ( gameLocal.isMultiplayer ) {
return;
}
// only gib once
if ( gibbed ) {
return;
}
idAFEntity_Gibbable::Gib( dir, damageDefName );
if ( head.GetEntity() ) {
head.GetEntity()->Hide();
}
StopSound( SND_CHANNEL_VOICE, false );
}
/*
============
idActor::Damage
this entity that is being damaged
inflictor entity that is causing the damage
attacker entity that caused the inflictor to damage targ
example: this=monster, inflictor=rocket, attacker=player
dir direction of the attack for knockback in global space
point point at which the damage is being inflicted, used for headshots
damage amount of damage being inflicted
inflictor, attacker, dir, and point can be NULL for environmental effects
Bleeding wounds and surface overlays are applied in the collision code that
calls Damage()
============
*/
void idActor::Damage( idEntity *inflictor, idEntity *attacker, const idVec3 &dir,
const char *damageDefName, const float damageScale, const int location ) {
if ( !fl.takedamage ) {
return;
}
if ( !inflictor ) {
inflictor = gameLocal.world;
}
if ( !attacker ) {
attacker = gameLocal.world;
}
if ( finalBoss && !inflictor->IsType( idSoulCubeMissile::Type ) ) {
return;
}
const idDict *damageDef = gameLocal.FindEntityDefDict( damageDefName );
if ( !damageDef ) {
gameLocal.Error( "Unknown damageDef '%s'", damageDefName );
}
int damage = damageDef->GetInt( "damage" ) * damageScale;
damage = GetDamageForLocation( damage, location );
// inform the attacker that they hit someone
attacker->DamageFeedback( this, inflictor, damage );
if ( damage > 0 ) {
health -= damage;
if ( health <= 0 ) {
if ( health < -999 ) {
health = -999;
}
Killed( inflictor, attacker, damage, dir, location );
if ( ( health < -20 ) && spawnArgs.GetBool( "gib" ) && damageDef->GetBool( "gib" ) ) {
Gib( dir, damageDefName );
}
} else {
Pain( inflictor, attacker, damage, dir, location );
}
} else {
// don't accumulate knockback
if ( af.IsLoaded() ) {
// clear impacts
af.Rest();
// physics is turned off by calling af.Rest()
BecomeActive( TH_PHYSICS );
}
}
}
/*
=====================
idActor::ClearPain
=====================
*/
void idActor::ClearPain( void ) {
pain_debounce_time = 0;
}
/*
=====================
idActor::Pain
=====================
*/
bool idActor::Pain( idEntity *inflictor, idEntity *attacker, int damage, const idVec3 &dir, int location ) {
if ( af.IsLoaded() ) {
// clear impacts
af.Rest();
// physics is turned off by calling af.Rest()
BecomeActive( TH_PHYSICS );
}
if ( gameLocal.time < pain_debounce_time ) {
return false;
}
// don't play pain sounds more than necessary
pain_debounce_time = gameLocal.time + pain_delay;
if ( health > 75 ) {
StartSound( "snd_pain_small", SND_CHANNEL_VOICE, 0, false, NULL );
} else if ( health > 50 ) {
StartSound( "snd_pain_medium", SND_CHANNEL_VOICE, 0, false, NULL );
} else if ( health > 25 ) {
StartSound( "snd_pain_large", SND_CHANNEL_VOICE, 0, false, NULL );
} else {
StartSound( "snd_pain_huge", SND_CHANNEL_VOICE, 0, false, NULL );
}
if ( !allowPain || ( gameLocal.time < painTime ) ) {
// don't play a pain anim
return false;
}
if ( pain_threshold && ( damage < pain_threshold ) ) {
return false;
}
// set the pain anim
idStr damageGroup = GetDamageGroup( location );
painAnim = "";
if ( animPrefix.Length() ) {
if ( damageGroup.Length() && ( damageGroup != "legs" ) ) {
sprintf( painAnim, "%s_pain_%s", animPrefix.c_str(), damageGroup.c_str() );
if ( !animator.HasAnim( painAnim ) ) {
sprintf( painAnim, "pain_%s", damageGroup.c_str() );
if ( !animator.HasAnim( painAnim ) ) {
painAnim = "";
}
}
}
if ( !painAnim.Length() ) {
sprintf( painAnim, "%s_pain", animPrefix.c_str() );
if ( !animator.HasAnim( painAnim ) ) {
painAnim = "";
}
}
} else if ( damageGroup.Length() && ( damageGroup != "legs" ) ) {
sprintf( painAnim, "pain_%s", damageGroup.c_str() );
if ( !animator.HasAnim( painAnim ) ) {
sprintf( painAnim, "pain_%s", damageGroup.c_str() );
if ( !animator.HasAnim( painAnim ) ) {
painAnim = "";
}
}
}
if ( !painAnim.Length() ) {
painAnim = "pain";
}
if ( g_debugDamage.GetBool() ) {
gameLocal.Printf( "Damage: joint: '%s', zone '%s', anim '%s'\n", animator.GetJointName( ( jointHandle_t )location ),
damageGroup.c_str(), painAnim.c_str() );
}
return true;
}
/*
=====================
idActor::SpawnGibs
=====================
*/
void idActor::SpawnGibs( const idVec3 &dir, const char *damageDefName ) {
idAFEntity_Gibbable::SpawnGibs( dir, damageDefName );
RemoveAttachments();
}
/*
=====================
idActor::SetupDamageGroups
FIXME: only store group names once and store an index for each joint
=====================
*/
void idActor::SetupDamageGroups( void ) {
int i;
const idKeyValue *arg;
idStr groupname;
idList<jointHandle_t> jointList;
int jointnum;
float scale;
// create damage zones
damageGroups.SetNum( animator.NumJoints() );
arg = spawnArgs.MatchPrefix( "damage_zone ", NULL );
while ( arg ) {
groupname = arg->GetKey();
groupname.Strip( "damage_zone " );
animator.GetJointList( arg->GetValue(), jointList );
for( i = 0; i < jointList.Num(); i++ ) {
jointnum = jointList[ i ];
damageGroups[ jointnum ] = groupname;
}
jointList.Clear();
arg = spawnArgs.MatchPrefix( "damage_zone ", arg );
}
// initilize the damage zones to normal damage
damageScale.SetNum( animator.NumJoints() );
for( i = 0; i < damageScale.Num(); i++ ) {
damageScale[ i ] = 1.0f;
}
// set the percentage on damage zones
arg = spawnArgs.MatchPrefix( "damage_scale ", NULL );
while ( arg ) {
scale = atof( arg->GetValue() );
groupname = arg->GetKey();
groupname.Strip( "damage_scale " );
for( i = 0; i < damageScale.Num(); i++ ) {
if ( damageGroups[ i ] == groupname ) {
damageScale[ i ] = scale;
}
}
arg = spawnArgs.MatchPrefix( "damage_scale ", arg );
}
}
/*
=====================
idActor::GetDamageForLocation
=====================
*/
int idActor::GetDamageForLocation( int damage, int location ) {
if ( ( location < 0 ) || ( location >= damageScale.Num() ) ) {
return damage;
}
return (int)ceil( damage * damageScale[ location ] );
}
/*
=====================
idActor::GetDamageGroup
=====================
*/
const char *idActor::GetDamageGroup( int location ) {
if ( ( location < 0 ) || ( location >= damageGroups.Num() ) ) {
return "";
}
return damageGroups[ location ];
}
/***********************************************************************
Events
***********************************************************************/
/*
=====================
idActor::Event_EnableEyeFocus
=====================
*/
void idActor::PlayFootStepSound( void ) {
const char *sound = NULL;
const idMaterial *material;
if ( !GetPhysics()->HasGroundContacts() ) {
return;
}
// start footstep sound based on material type
material = GetPhysics()->GetContact( 0 ).material;
if ( material != NULL ) {
sound = spawnArgs.GetString( va( "snd_footstep_%s", gameLocal.sufaceTypeNames[ material->GetSurfaceType() ] ) );
}
if ( *sound == '\0' ) {
sound = spawnArgs.GetString( "snd_footstep" );
}
if ( *sound != '\0' ) {
StartSoundShader( declManager->FindSound( sound ), SND_CHANNEL_BODY, 0, false, NULL );
}
}
/*
=====================
idActor::Event_EnableEyeFocus
=====================
*/
void idActor::Event_EnableEyeFocus( void ) {
allowEyeFocus = true;
blink_time = gameLocal.time + blink_min + gameLocal.random.RandomFloat() * ( blink_max - blink_min );
}
/*
=====================
idActor::Event_DisableEyeFocus
=====================
*/
void idActor::Event_DisableEyeFocus( void ) {
allowEyeFocus = false;
idEntity *headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->GetAnimator()->Clear( ANIMCHANNEL_EYELIDS, gameLocal.time, FRAME2MS( 2 ) );
} else {
animator.Clear( ANIMCHANNEL_EYELIDS, gameLocal.time, FRAME2MS( 2 ) );
}
}
/*
===============
idActor::Event_Footstep
===============
*/
void idActor::Event_Footstep( void ) {
PlayFootStepSound();
}
/*
=====================
idActor::Event_EnableWalkIK
=====================
*/
void idActor::Event_EnableWalkIK( void ) {
walkIK.EnableAll();
}
/*
=====================
idActor::Event_DisableWalkIK
=====================
*/
void idActor::Event_DisableWalkIK( void ) {
walkIK.DisableAll();
}
/*
=====================
idActor::Event_EnableLegIK
=====================
*/
void idActor::Event_EnableLegIK( int num ) {
walkIK.EnableLeg( num );
}
/*
=====================
idActor::Event_DisableLegIK
=====================
*/
void idActor::Event_DisableLegIK( int num ) {
walkIK.DisableLeg( num );
}
/*
=====================
idActor::Event_PreventPain
=====================
*/
void idActor::Event_PreventPain( float duration ) {
painTime = gameLocal.time + SEC2MS( duration );
}
/*
===============
idActor::Event_DisablePain
===============
*/
void idActor::Event_DisablePain( void ) {
allowPain = false;
}
/*
===============
idActor::Event_EnablePain
===============
*/
void idActor::Event_EnablePain( void ) {
allowPain = true;
}
/*
=====================
idActor::Event_GetPainAnim
=====================
*/
void idActor::Event_GetPainAnim( void ) {
if ( !painAnim.Length() ) {
idThread::ReturnString( "pain" );
} else {
idThread::ReturnString( painAnim );
}
}
/*
=====================
idActor::Event_SetAnimPrefix
=====================
*/
void idActor::Event_SetAnimPrefix( const char *prefix ) {
animPrefix = prefix;
}
/*
===============
idActor::Event_StopAnim
===============
*/
void idActor::Event_StopAnim( int channel, int frames ) {
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.StopAnim( frames );
break;
case ANIMCHANNEL_TORSO :
torsoAnim.StopAnim( frames );
break;
case ANIMCHANNEL_LEGS :
legsAnim.StopAnim( frames );
break;
default:
gameLocal.Error( "Unknown anim group" );
break;
}
}
/*
===============
idActor::Event_PlayAnim
===============
*/
void idActor::Event_PlayAnim( int channel, const char *animname ) {
animFlags_t flags;
idEntity *headEnt;
int anim;
anim = GetAnim( channel, animname );
if ( !anim ) {
if ( ( channel == ANIMCHANNEL_HEAD ) && head.GetEntity() ) {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), spawnArgs.GetString( "def_head", "" ) );
} else {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), GetEntityDefName() );
}
idThread::ReturnInt( 0 );
return;
}
switch( channel ) {
case ANIMCHANNEL_HEAD :
headEnt = head.GetEntity();
if ( headEnt ) {
headAnim.idleAnim = false;
headAnim.PlayAnim( anim );
flags = headAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( torsoAnim.IsIdle() ) {
torsoAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
if ( legsAnim.IsIdle() ) {
legsAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
}
}
}
}
break;
case ANIMCHANNEL_TORSO :
torsoAnim.idleAnim = false;
torsoAnim.PlayAnim( anim );
flags = torsoAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( headAnim.IsIdle() ) {
headAnim.animBlendFrames = torsoAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
if ( legsAnim.IsIdle() ) {
legsAnim.animBlendFrames = torsoAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
}
break;
case ANIMCHANNEL_LEGS :
legsAnim.idleAnim = false;
legsAnim.PlayAnim( anim );
flags = legsAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( torsoAnim.IsIdle() ) {
torsoAnim.animBlendFrames = legsAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
if ( headAnim.IsIdle() ) {
headAnim.animBlendFrames = legsAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
}
}
}
break;
default :
gameLocal.Error( "Unknown anim group" );
break;
}
idThread::ReturnInt( 1 );
}
/*
===============
idActor::Event_PlayCycle
===============
*/
void idActor::Event_PlayCycle( int channel, const char *animname ) {
animFlags_t flags;
int anim;
anim = GetAnim( channel, animname );
if ( !anim ) {
if ( ( channel == ANIMCHANNEL_HEAD ) && head.GetEntity() ) {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), spawnArgs.GetString( "def_head", "" ) );
} else {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), GetEntityDefName() );
}
idThread::ReturnInt( false );
return;
}
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.idleAnim = false;
headAnim.CycleAnim( anim );
flags = headAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( torsoAnim.IsIdle() && legsAnim.IsIdle() ) {
torsoAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
legsAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
}
}
break;
case ANIMCHANNEL_TORSO :
torsoAnim.idleAnim = false;
torsoAnim.CycleAnim( anim );
flags = torsoAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( headAnim.IsIdle() ) {
headAnim.animBlendFrames = torsoAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
if ( legsAnim.IsIdle() ) {
legsAnim.animBlendFrames = torsoAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
}
break;
case ANIMCHANNEL_LEGS :
legsAnim.idleAnim = false;
legsAnim.CycleAnim( anim );
flags = legsAnim.GetAnimFlags();
if ( !flags.prevent_idle_override ) {
if ( torsoAnim.IsIdle() ) {
torsoAnim.animBlendFrames = legsAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
if ( headAnim.IsIdle() ) {
headAnim.animBlendFrames = legsAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
}
}
}
break;
default:
gameLocal.Error( "Unknown anim group" );
}
idThread::ReturnInt( true );
}
/*
===============
idActor::Event_IdleAnim
===============
*/
void idActor::Event_IdleAnim( int channel, const char *animname ) {
int anim;
anim = GetAnim( channel, animname );
if ( !anim ) {
if ( ( channel == ANIMCHANNEL_HEAD ) && head.GetEntity() ) {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), spawnArgs.GetString( "def_head", "" ) );
} else {
gameLocal.DPrintf( "missing '%s' animation on '%s' (%s)\n", animname, name.c_str(), GetEntityDefName() );
}
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.BecomeIdle();
break;
case ANIMCHANNEL_TORSO :
torsoAnim.BecomeIdle();
break;
case ANIMCHANNEL_LEGS :
legsAnim.BecomeIdle();
break;
default:
gameLocal.Error( "Unknown anim group" );
}
idThread::ReturnInt( false );
return;
}
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.BecomeIdle();
if ( torsoAnim.GetAnimFlags().prevent_idle_override ) {
// don't sync to torso body if it doesn't override idle anims
headAnim.CycleAnim( anim );
} else if ( torsoAnim.IsIdle() && legsAnim.IsIdle() ) {
// everything is idle, so play the anim on the head and copy it to the torso and legs
headAnim.CycleAnim( anim );
torsoAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
legsAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_HEAD, headAnim.lastAnimBlendFrames );
} else if ( torsoAnim.IsIdle() ) {
// sync the head and torso to the legs
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_LEGS, headAnim.animBlendFrames );
torsoAnim.animBlendFrames = headAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, torsoAnim.animBlendFrames );
} else {
// sync the head to the torso
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, headAnim.animBlendFrames );
}
break;
case ANIMCHANNEL_TORSO :
torsoAnim.BecomeIdle();
if ( legsAnim.GetAnimFlags().prevent_idle_override ) {
// don't sync to legs if legs anim doesn't override idle anims
torsoAnim.CycleAnim( anim );
} else if ( legsAnim.IsIdle() ) {
// play the anim in both legs and torso
torsoAnim.CycleAnim( anim );
legsAnim.animBlendFrames = torsoAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
} else {
// sync the anim to the legs
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, torsoAnim.animBlendFrames );
}
if ( headAnim.IsIdle() ) {
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
break;
case ANIMCHANNEL_LEGS :
legsAnim.BecomeIdle();
if ( torsoAnim.GetAnimFlags().prevent_idle_override ) {
// don't sync to torso if torso anim doesn't override idle anims
legsAnim.CycleAnim( anim );
} else if ( torsoAnim.IsIdle() ) {
// play the anim in both legs and torso
legsAnim.CycleAnim( anim );
torsoAnim.animBlendFrames = legsAnim.lastAnimBlendFrames;
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
if ( headAnim.IsIdle() ) {
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
}
} else {
// sync the anim to the torso
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_TORSO, legsAnim.animBlendFrames );
}
break;
default:
gameLocal.Error( "Unknown anim group" );
}
idThread::ReturnInt( true );
}
/*
================
idActor::Event_SetSyncedAnimWeight
================
*/
void idActor::Event_SetSyncedAnimWeight( int channel, int anim, float weight ) {
idEntity *headEnt;
headEnt = head.GetEntity();
switch( channel ) {
case ANIMCHANNEL_HEAD :
if ( headEnt ) {
animator.CurrentAnim( ANIMCHANNEL_ALL )->SetSyncedAnimWeight( anim, weight );
} else {
animator.CurrentAnim( ANIMCHANNEL_HEAD )->SetSyncedAnimWeight( anim, weight );
}
if ( torsoAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_TORSO )->SetSyncedAnimWeight( anim, weight );
if ( legsAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_LEGS )->SetSyncedAnimWeight( anim, weight );
}
}
break;
case ANIMCHANNEL_TORSO :
animator.CurrentAnim( ANIMCHANNEL_TORSO )->SetSyncedAnimWeight( anim, weight );
if ( legsAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_LEGS )->SetSyncedAnimWeight( anim, weight );
}
if ( headEnt && headAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_ALL )->SetSyncedAnimWeight( anim, weight );
}
break;
case ANIMCHANNEL_LEGS :
animator.CurrentAnim( ANIMCHANNEL_LEGS )->SetSyncedAnimWeight( anim, weight );
if ( torsoAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_TORSO )->SetSyncedAnimWeight( anim, weight );
if ( headEnt && headAnim.IsIdle() ) {
animator.CurrentAnim( ANIMCHANNEL_ALL )->SetSyncedAnimWeight( anim, weight );
}
}
break;
default:
gameLocal.Error( "Unknown anim group" );
}
}
/*
===============
idActor::Event_OverrideAnim
===============
*/
void idActor::Event_OverrideAnim( int channel ) {
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.Disable();
if ( !torsoAnim.IsIdle() ) {
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
} else {
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
}
break;
case ANIMCHANNEL_TORSO :
torsoAnim.Disable();
SyncAnimChannels( ANIMCHANNEL_TORSO, ANIMCHANNEL_LEGS, legsAnim.lastAnimBlendFrames );
if ( headAnim.IsIdle() ) {
SyncAnimChannels( ANIMCHANNEL_HEAD, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
}
break;
case ANIMCHANNEL_LEGS :
legsAnim.Disable();
SyncAnimChannels( ANIMCHANNEL_LEGS, ANIMCHANNEL_TORSO, torsoAnim.lastAnimBlendFrames );
break;
default:
gameLocal.Error( "Unknown anim group" );
break;
}
}
/*
===============
idActor::Event_EnableAnim
===============
*/
void idActor::Event_EnableAnim( int channel, int blendFrames ) {
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.Enable( blendFrames );
break;
case ANIMCHANNEL_TORSO :
torsoAnim.Enable( blendFrames );
break;
case ANIMCHANNEL_LEGS :
legsAnim.Enable( blendFrames );
break;
default:
gameLocal.Error( "Unknown anim group" );
break;
}
}
/*
===============
idActor::Event_SetBlendFrames
===============
*/
void idActor::Event_SetBlendFrames( int channel, int blendFrames ) {
switch( channel ) {
case ANIMCHANNEL_HEAD :
headAnim.animBlendFrames = blendFrames;
headAnim.lastAnimBlendFrames = blendFrames;
break;
case ANIMCHANNEL_TORSO :
torsoAnim.animBlendFrames = blendFrames;
torsoAnim.lastAnimBlendFrames = blendFrames;
break;
case ANIMCHANNEL_LEGS :
legsAnim.animBlendFrames = blendFrames;
legsAnim.lastAnimBlendFrames = blendFrames;
break;
default:
gameLocal.Error( "Unknown anim group" );
break;
}
}
/*
===============
idActor::Event_GetBlendFrames
===============
*/
void idActor::Event_GetBlendFrames( int channel ) {
switch( channel ) {
case ANIMCHANNEL_HEAD :
idThread::ReturnInt( headAnim.animBlendFrames );
break;
case ANIMCHANNEL_TORSO :
idThread::ReturnInt( torsoAnim.animBlendFrames );
break;
case ANIMCHANNEL_LEGS :
idThread::ReturnInt( legsAnim.animBlendFrames );
break;
default:
gameLocal.Error( "Unknown anim group" );
break;
}
}
/*
===============
idActor::Event_AnimState
===============
*/
void idActor::Event_AnimState( int channel, const char *statename, int blendFrames ) {
SetAnimState( channel, statename, blendFrames );
}
/*
===============
idActor::Event_GetAnimState
===============
*/
void idActor::Event_GetAnimState( int channel ) {
const char *state;
state = GetAnimState( channel );
idThread::ReturnString( state );
}
/*
===============
idActor::Event_InAnimState
===============
*/
void idActor::Event_InAnimState( int channel, const char *statename ) {
bool instate;
instate = InAnimState( channel, statename );
idThread::ReturnInt( instate );
}
/*
===============
idActor::Event_FinishAction
===============
*/
void idActor::Event_FinishAction( const char *actionname ) {
if ( waitState == actionname ) {
SetWaitState( "" );
}
}
/*
===============
idActor::Event_AnimDone
===============
*/
void idActor::Event_AnimDone( int channel, int blendFrames ) {
bool result;
switch( channel ) {
case ANIMCHANNEL_HEAD :
result = headAnim.AnimDone( blendFrames );
idThread::ReturnInt( result );
break;
case ANIMCHANNEL_TORSO :
result = torsoAnim.AnimDone( blendFrames );
idThread::ReturnInt( result );
break;
case ANIMCHANNEL_LEGS :
result = legsAnim.AnimDone( blendFrames );
idThread::ReturnInt( result );
break;
default:
gameLocal.Error( "Unknown anim group" );
}
}
/*
================
idActor::Event_HasAnim
================
*/
void idActor::Event_HasAnim( int channel, const char *animname ) {
if ( GetAnim( channel, animname ) != 0 ) {
idThread::ReturnFloat( 1.0f );
} else {
idThread::ReturnFloat( 0.0f );
}
}
/*
================
idActor::Event_CheckAnim
================
*/
void idActor::Event_CheckAnim( int channel, const char *animname ) {
if ( !GetAnim( channel, animname ) ) {
if ( animPrefix.Length() ) {
gameLocal.Error( "Can't find anim '%s_%s' for '%s'", animPrefix.c_str(), animname, name.c_str() );
} else {
gameLocal.Error( "Can't find anim '%s' for '%s'", animname, name.c_str() );
}
}
}
/*
================
idActor::Event_ChooseAnim
================
*/
void idActor::Event_ChooseAnim( int channel, const char *animname ) {
int anim;
anim = GetAnim( channel, animname );
if ( anim ) {
if ( channel == ANIMCHANNEL_HEAD ) {
if ( head.GetEntity() ) {
idThread::ReturnString( head.GetEntity()->GetAnimator()->AnimFullName( anim ) );
return;
}
} else {
idThread::ReturnString( animator.AnimFullName( anim ) );
return;
}
}
idThread::ReturnString( "" );
}
/*
================
idActor::Event_AnimLength
================
*/
void idActor::Event_AnimLength( int channel, const char *animname ) {
int anim;
anim = GetAnim( channel, animname );
if ( anim ) {
if ( channel == ANIMCHANNEL_HEAD ) {
if ( head.GetEntity() ) {
idThread::ReturnFloat( MS2SEC( head.GetEntity()->GetAnimator()->AnimLength( anim ) ) );
return;
}
} else {
idThread::ReturnFloat( MS2SEC( animator.AnimLength( anim ) ) );
return;
}
}
idThread::ReturnFloat( 0.0f );
}
/*
================
idActor::Event_AnimDistance
================
*/
void idActor::Event_AnimDistance( int channel, const char *animname ) {
int anim;
anim = GetAnim( channel, animname );
if ( anim ) {
if ( channel == ANIMCHANNEL_HEAD ) {
if ( head.GetEntity() ) {
idThread::ReturnFloat( head.GetEntity()->GetAnimator()->TotalMovementDelta( anim ).Length() );
return;
}
} else {
idThread::ReturnFloat( animator.TotalMovementDelta( anim ).Length() );
return;
}
}
idThread::ReturnFloat( 0.0f );
}
/*
================
idActor::Event_HasEnemies
================
*/
void idActor::Event_HasEnemies( void ) {
bool hasEnemy;
hasEnemy = HasEnemies();
idThread::ReturnInt( hasEnemy );
}
/*
================
idActor::Event_NextEnemy
================
*/
void idActor::Event_NextEnemy( idEntity *ent ) {
idActor *actor;
if ( !ent || ( ent == this ) ) {
actor = enemyList.Next();
} else {
if ( !ent->IsType( idActor::Type ) ) {
gameLocal.Error( "'%s' cannot be an enemy", ent->name.c_str() );
}
actor = static_cast<idActor *>( ent );
if ( actor->enemyNode.ListHead() != &enemyList ) {
gameLocal.Error( "'%s' is not in '%s' enemy list", actor->name.c_str(), name.c_str() );
}
}
for( ; actor != NULL; actor = actor->enemyNode.Next() ) {
if ( !actor->fl.hidden ) {
idThread::ReturnEntity( actor );
return;
}
}
idThread::ReturnEntity( NULL );
}
/*
================
idActor::Event_ClosestEnemyToPoint
================
*/
void idActor::Event_ClosestEnemyToPoint( const idVec3 &pos ) {
idActor *bestEnt = ClosestEnemyToPoint( pos );
idThread::ReturnEntity( bestEnt );
}
/*
================
idActor::Event_StopSound
================
*/
void idActor::Event_StopSound( int channel, int netSync ) {
if ( channel == SND_CHANNEL_VOICE ) {
idEntity *headEnt = head.GetEntity();
if ( headEnt ) {
headEnt->StopSound( channel, ( netSync != 0 ) );
}
}
StopSound( channel, ( netSync != 0 ) );
}
/*
=====================
idActor::Event_SetNextState
=====================
*/
void idActor::Event_SetNextState( const char *name ) {
idealState = GetScriptFunction( name );
if ( idealState == state ) {
state = NULL;
}
}
/*
=====================
idActor::Event_SetState
=====================
*/
void idActor::Event_SetState( const char *name ) {
idealState = GetScriptFunction( name );
if ( idealState == state ) {
state = NULL;
}
scriptThread->DoneProcessing();
}
/*
=====================
idActor::Event_GetState
=====================
*/
void idActor::Event_GetState( void ) {
if ( state ) {
idThread::ReturnString( state->Name() );
} else {
idThread::ReturnString( "" );
}
}
/*
=====================
idActor::Event_GetHead
=====================
*/
void idActor::Event_GetHead( void ) {
idThread::ReturnEntity( head.GetEntity() );
}