1012 lines
24 KiB
C++
1012 lines
24 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: Bullseyes act as targets for other NPC's to attack and to trigger
|
|
// events
|
|
//
|
|
// $Workfile: $
|
|
// $Date: $
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
// $Log: $
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#include "cbase.h"
|
|
#include "ai_default.h"
|
|
#include "ai_task.h"
|
|
#include "ai_schedule.h"
|
|
#include "ai_node.h"
|
|
#include "ai_hull.h"
|
|
#include "ai_hint.h"
|
|
#include "ai_memory.h"
|
|
#include "ai_route.h"
|
|
#include "ai_motor.h"
|
|
#include "ai_senses.h"
|
|
#include "soundent.h"
|
|
#include "game.h"
|
|
#include "npcevent.h"
|
|
#include "entitylist.h"
|
|
#include "activitylist.h"
|
|
#include "animation.h"
|
|
#include "basecombatweapon.h"
|
|
#include "IEffects.h"
|
|
#include "vstdlib/random.h"
|
|
#include "engine/IEngineSound.h"
|
|
#include "ammodef.h"
|
|
#include "hl1_ai_basenpc.h"
|
|
#include "studio.h" //hitbox parsing
|
|
#include "collisionutils.h" //ComputeSeparatingPlane
|
|
|
|
#include "physics_bone_follower.h" //For BoneFollowerManager
|
|
|
|
#define ACT_T_IDLE 1010
|
|
Activity ACT_1010;
|
|
Activity ACT_1011;
|
|
Activity ACT_1012;
|
|
Activity ACT_1013;
|
|
|
|
#define ACT_T_TAP 1020
|
|
Activity ACT_1020;
|
|
Activity ACT_1021;
|
|
Activity ACT_1022;
|
|
Activity ACT_1023;
|
|
|
|
#define ACT_T_STRIKE 1030
|
|
Activity ACT_1030;
|
|
Activity ACT_1031;
|
|
Activity ACT_1032;
|
|
Activity ACT_1033;
|
|
|
|
#define ACT_T_REARIDLE 1040
|
|
Activity ACT_1040;
|
|
Activity ACT_1041;
|
|
Activity ACT_1042;
|
|
Activity ACT_1043;
|
|
Activity ACT_1044;
|
|
|
|
class CNPC_Tentacle : public CHL1BaseNPC
|
|
{
|
|
DECLARE_CLASS( CNPC_Tentacle, CHL1BaseNPC );
|
|
public:
|
|
|
|
CNPC_Tentacle();
|
|
|
|
void Spawn( );
|
|
void Precache( );
|
|
bool KeyValue( const char *szKeyName, const char *szValue );
|
|
|
|
bool QueryHearSound( CSound *pSound ) { return true; } // Tentacle isn't picky.
|
|
|
|
|
|
int Level( float dz );
|
|
int MyLevel( void );
|
|
float MyHeight( void );
|
|
|
|
// Don't allow the tentacle to go across transitions!!!
|
|
virtual int ObjectCaps( void ) { return CAI_BaseNPC::ObjectCaps() & ~FCAP_ACROSS_TRANSITION; }
|
|
|
|
void Start ( void );
|
|
void Cycle ( void );
|
|
void HitTouch( CBaseEntity *pOther );
|
|
|
|
void HandleAnimEvent( animevent_t *pEvent );
|
|
float HearingSensitivity( void ) { return 2.0; };
|
|
|
|
virtual int OnTakeDamage( const CTakeDamageInfo &info );
|
|
|
|
bool CreateVPhysics( void );
|
|
|
|
void UpdateOnRemove( void );
|
|
|
|
float m_flInitialYaw;
|
|
int m_iGoalAnim;
|
|
int m_iLevel;
|
|
int m_iDir;
|
|
float m_flFramerateAdj;
|
|
float m_flSoundYaw;
|
|
int m_iSoundLevel;
|
|
float m_flSoundTime;
|
|
float m_flSoundRadius;
|
|
int m_iHitDmg;
|
|
float m_flHitTime;
|
|
|
|
float m_flTapRadius;
|
|
|
|
float m_flNextSong;
|
|
static int g_fFlySound;
|
|
static int g_fSquirmSound;
|
|
|
|
float m_flMaxYaw;
|
|
int m_iTapSound;
|
|
|
|
Vector m_vecPrevSound;
|
|
float m_flPrevSoundTime;
|
|
|
|
float MaxYawSpeed( void ) { return 18.0f; }
|
|
|
|
bool HeardAnything( void );
|
|
|
|
Class_T Classify ( void );
|
|
|
|
CBoneFollowerManager m_BoneFollowerManager;
|
|
|
|
DECLARE_DATADESC();
|
|
DEFINE_CUSTOM_AI;
|
|
};
|
|
|
|
// Crane bones that have physics followers
|
|
static const char *pTentacleFollowerBoneNames[] =
|
|
{
|
|
"Bone08",
|
|
"Bone09"
|
|
};
|
|
|
|
int CNPC_Tentacle::g_fFlySound;
|
|
int CNPC_Tentacle::g_fSquirmSound;
|
|
|
|
LINK_ENTITY_TO_CLASS( monster_tentacle, CNPC_Tentacle );
|
|
|
|
// stike sounds
|
|
#define TE_NONE -1
|
|
#define TE_SILO 0
|
|
#define TE_DIRT 1
|
|
#define TE_WATER 2
|
|
|
|
// animation sequence aliases
|
|
typedef enum
|
|
{
|
|
TENTACLE_ANIM_Pit_Idle,
|
|
|
|
TENTACLE_ANIM_rise_to_Temp1,
|
|
TENTACLE_ANIM_Temp1_to_Floor,
|
|
TENTACLE_ANIM_Floor_Idle,
|
|
TENTACLE_ANIM_Floor_Fidget_Pissed,
|
|
TENTACLE_ANIM_Floor_Fidget_SmallRise,
|
|
TENTACLE_ANIM_Floor_Fidget_Wave,
|
|
TENTACLE_ANIM_Floor_Strike,
|
|
TENTACLE_ANIM_Floor_Tap,
|
|
TENTACLE_ANIM_Floor_Rotate,
|
|
TENTACLE_ANIM_Floor_Rear,
|
|
TENTACLE_ANIM_Floor_Rear_Idle,
|
|
TENTACLE_ANIM_Floor_to_Lev1,
|
|
|
|
TENTACLE_ANIM_Lev1_Idle,
|
|
TENTACLE_ANIM_Lev1_Fidget_Claw,
|
|
TENTACLE_ANIM_Lev1_Fidget_Shake,
|
|
TENTACLE_ANIM_Lev1_Fidget_Snap,
|
|
TENTACLE_ANIM_Lev1_Strike,
|
|
TENTACLE_ANIM_Lev1_Tap,
|
|
TENTACLE_ANIM_Lev1_Rotate,
|
|
TENTACLE_ANIM_Lev1_Rear,
|
|
TENTACLE_ANIM_Lev1_Rear_Idle,
|
|
TENTACLE_ANIM_Lev1_to_Lev2,
|
|
|
|
TENTACLE_ANIM_Lev2_Idle,
|
|
TENTACLE_ANIM_Lev2_Fidget_Shake,
|
|
TENTACLE_ANIM_Lev2_Fidget_Swing,
|
|
TENTACLE_ANIM_Lev2_Fidget_Tut,
|
|
TENTACLE_ANIM_Lev2_Strike,
|
|
TENTACLE_ANIM_Lev2_Tap,
|
|
TENTACLE_ANIM_Lev2_Rotate,
|
|
TENTACLE_ANIM_Lev2_Rear,
|
|
TENTACLE_ANIM_Lev2_Rear_Idle,
|
|
TENTACLE_ANIM_Lev2_to_Lev3,
|
|
|
|
TENTACLE_ANIM_Lev3_Idle,
|
|
TENTACLE_ANIM_Lev3_Fidget_Shake,
|
|
TENTACLE_ANIM_Lev3_Fidget_Side,
|
|
TENTACLE_ANIM_Lev3_Fidget_Swipe,
|
|
TENTACLE_ANIM_Lev3_Strike,
|
|
TENTACLE_ANIM_Lev3_Tap,
|
|
TENTACLE_ANIM_Lev3_Rotate,
|
|
TENTACLE_ANIM_Lev3_Rear,
|
|
TENTACLE_ANIM_Lev3_Rear_Idle,
|
|
|
|
TENTACLE_ANIM_Lev1_Door_reach,
|
|
|
|
TENTACLE_ANIM_Lev3_to_Engine,
|
|
TENTACLE_ANIM_Engine_Idle,
|
|
TENTACLE_ANIM_Engine_Sway,
|
|
TENTACLE_ANIM_Engine_Swat,
|
|
TENTACLE_ANIM_Engine_Bob,
|
|
TENTACLE_ANIM_Engine_Death1,
|
|
TENTACLE_ANIM_Engine_Death2,
|
|
TENTACLE_ANIM_Engine_Death3,
|
|
|
|
TENTACLE_ANIM_none
|
|
} TENTACLE_ANIM;
|
|
|
|
BEGIN_DATADESC( CNPC_Tentacle )
|
|
DEFINE_FIELD( m_flInitialYaw, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_iGoalAnim, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_iLevel, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_iDir, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_flFramerateAdj, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_flSoundYaw, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_iSoundLevel, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_flSoundTime, FIELD_TIME ),
|
|
DEFINE_FIELD( m_flSoundRadius, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_iHitDmg, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_flHitTime, FIELD_TIME ),
|
|
DEFINE_FIELD( m_flTapRadius, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_flNextSong, FIELD_TIME ),
|
|
DEFINE_FIELD( m_iTapSound, FIELD_INTEGER ),
|
|
DEFINE_FIELD( m_flMaxYaw, FIELD_FLOAT ),
|
|
DEFINE_FIELD( m_vecPrevSound, FIELD_POSITION_VECTOR ),
|
|
DEFINE_FIELD( m_flPrevSoundTime, FIELD_TIME ),
|
|
|
|
DEFINE_EMBEDDED( m_BoneFollowerManager ),
|
|
|
|
DEFINE_THINKFUNC( Start ),
|
|
DEFINE_THINKFUNC( Cycle ),
|
|
DEFINE_ENTITYFUNC( HitTouch ),
|
|
END_DATADESC()
|
|
|
|
Class_T CNPC_Tentacle::Classify ( void )
|
|
{
|
|
return CLASS_ALIEN_MONSTER;
|
|
}
|
|
|
|
|
|
CNPC_Tentacle::CNPC_Tentacle()
|
|
{
|
|
m_flMaxYaw = 65;
|
|
m_iTapSound = 0;
|
|
}
|
|
|
|
bool CNPC_Tentacle::KeyValue( const char *szKeyName, const char *szValue )
|
|
{
|
|
if ( FStrEq( szKeyName, "sweeparc") )
|
|
{
|
|
m_flMaxYaw = atof( szValue ) / 2.0;
|
|
return true;
|
|
}
|
|
else if (FStrEq( szKeyName, "sound"))
|
|
{
|
|
m_iTapSound = atoi( szValue );
|
|
return true;
|
|
}
|
|
else
|
|
return BaseClass::KeyValue( szKeyName, szValue );
|
|
|
|
return false;
|
|
}
|
|
|
|
//
|
|
// Tentacle Spawn
|
|
//
|
|
void CNPC_Tentacle::Spawn( )
|
|
{
|
|
Precache( );
|
|
|
|
SetSolid( SOLID_BBOX );
|
|
|
|
//Necessary for TestCollision to be called for hitbox ray hits
|
|
AddSolidFlags( FSOLID_CUSTOMRAYTEST );
|
|
|
|
SetMoveType( MOVETYPE_NONE );
|
|
ClearEffects();
|
|
m_iHealth = 75;
|
|
SetSequence( 0 );
|
|
|
|
SetModel( "models/tentacle2.mdl" );
|
|
UTIL_SetSize( this, Vector( -32, -32, 0 ), Vector( 32, 32, 64 ) );
|
|
|
|
// Use our hitboxes to determine our render bounds
|
|
CollisionProp()->SetSurroundingBoundsType( USE_HITBOXES );
|
|
|
|
m_takedamage = DAMAGE_AIM;
|
|
AddFlag( FL_NPC );
|
|
|
|
m_bloodColor = BLOOD_COLOR_GREEN;
|
|
|
|
ResetSequenceInfo( );
|
|
m_iDir = 1;
|
|
|
|
SetThink( &CNPC_Tentacle::Start );
|
|
SetNextThink( gpGlobals->curtime + 0.2 );
|
|
|
|
SetTouch( &CNPC_Tentacle::HitTouch );
|
|
|
|
m_flInitialYaw = GetAbsAngles().y;
|
|
GetMotor()->SetIdealYawAndUpdate( m_flInitialYaw );
|
|
|
|
g_fFlySound = FALSE;
|
|
g_fSquirmSound = FALSE;
|
|
|
|
m_iHitDmg = 200;
|
|
|
|
if (m_flMaxYaw <= 0)
|
|
m_flMaxYaw = 65;
|
|
|
|
m_NPCState = NPC_STATE_IDLE;
|
|
|
|
UTIL_SetOrigin( this, GetAbsOrigin() );
|
|
|
|
CreateVPhysics();
|
|
|
|
AddEffects( EF_NOSHADOW );
|
|
}
|
|
|
|
void CNPC_Tentacle::UpdateOnRemove( void )
|
|
{
|
|
m_BoneFollowerManager.DestroyBoneFollowers();
|
|
BaseClass::UpdateOnRemove();
|
|
}
|
|
|
|
void CNPC_Tentacle::Precache( )
|
|
{
|
|
PrecacheModel("models/tentacle2.mdl");
|
|
|
|
PrecacheScriptSound( "Tentacle.Flies" );
|
|
PrecacheScriptSound( "Tentacle.Squirm" );
|
|
PrecacheScriptSound( "Tentacle.Sing" );
|
|
PrecacheScriptSound( "Tentacle.HitDirt" );
|
|
PrecacheScriptSound( "Tentacle.Swing" );
|
|
PrecacheScriptSound( "Tentacle.Search" );
|
|
PrecacheScriptSound( "Tentacle.Roar" );
|
|
PrecacheScriptSound( "Tentacle.Alert" );
|
|
|
|
BaseClass::Precache();
|
|
}
|
|
|
|
|
|
int CNPC_Tentacle::Level( float dz )
|
|
{
|
|
if (dz < 216)
|
|
return 0;
|
|
if (dz < 408)
|
|
return 1;
|
|
if (dz < 600)
|
|
return 2;
|
|
return 3;
|
|
}
|
|
|
|
|
|
float CNPC_Tentacle::MyHeight( )
|
|
{
|
|
switch ( MyLevel( ) )
|
|
{
|
|
case 1:
|
|
return 256;
|
|
case 2:
|
|
return 448;
|
|
case 3:
|
|
return 640;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int CNPC_Tentacle::MyLevel( )
|
|
{
|
|
switch( GetSequence() )
|
|
{
|
|
case TENTACLE_ANIM_Pit_Idle:
|
|
return -1;
|
|
|
|
case TENTACLE_ANIM_rise_to_Temp1:
|
|
case TENTACLE_ANIM_Temp1_to_Floor:
|
|
case TENTACLE_ANIM_Floor_to_Lev1:
|
|
return 0;
|
|
|
|
case TENTACLE_ANIM_Floor_Idle:
|
|
case TENTACLE_ANIM_Floor_Fidget_Pissed:
|
|
case TENTACLE_ANIM_Floor_Fidget_SmallRise:
|
|
case TENTACLE_ANIM_Floor_Fidget_Wave:
|
|
case TENTACLE_ANIM_Floor_Strike:
|
|
case TENTACLE_ANIM_Floor_Tap:
|
|
case TENTACLE_ANIM_Floor_Rotate:
|
|
case TENTACLE_ANIM_Floor_Rear:
|
|
case TENTACLE_ANIM_Floor_Rear_Idle:
|
|
return 0;
|
|
|
|
case TENTACLE_ANIM_Lev1_Idle:
|
|
case TENTACLE_ANIM_Lev1_Fidget_Claw:
|
|
case TENTACLE_ANIM_Lev1_Fidget_Shake:
|
|
case TENTACLE_ANIM_Lev1_Fidget_Snap:
|
|
case TENTACLE_ANIM_Lev1_Strike:
|
|
case TENTACLE_ANIM_Lev1_Tap:
|
|
case TENTACLE_ANIM_Lev1_Rotate:
|
|
case TENTACLE_ANIM_Lev1_Rear:
|
|
case TENTACLE_ANIM_Lev1_Rear_Idle:
|
|
return 1;
|
|
|
|
case TENTACLE_ANIM_Lev1_to_Lev2:
|
|
return 1;
|
|
|
|
case TENTACLE_ANIM_Lev2_Idle:
|
|
case TENTACLE_ANIM_Lev2_Fidget_Shake:
|
|
case TENTACLE_ANIM_Lev2_Fidget_Swing:
|
|
case TENTACLE_ANIM_Lev2_Fidget_Tut:
|
|
case TENTACLE_ANIM_Lev2_Strike:
|
|
case TENTACLE_ANIM_Lev2_Tap:
|
|
case TENTACLE_ANIM_Lev2_Rotate:
|
|
case TENTACLE_ANIM_Lev2_Rear:
|
|
case TENTACLE_ANIM_Lev2_Rear_Idle:
|
|
return 2;
|
|
|
|
case TENTACLE_ANIM_Lev2_to_Lev3:
|
|
return 2;
|
|
|
|
case TENTACLE_ANIM_Lev3_Idle:
|
|
case TENTACLE_ANIM_Lev3_Fidget_Shake:
|
|
case TENTACLE_ANIM_Lev3_Fidget_Side:
|
|
case TENTACLE_ANIM_Lev3_Fidget_Swipe:
|
|
case TENTACLE_ANIM_Lev3_Strike:
|
|
case TENTACLE_ANIM_Lev3_Tap:
|
|
case TENTACLE_ANIM_Lev3_Rotate:
|
|
case TENTACLE_ANIM_Lev3_Rear:
|
|
case TENTACLE_ANIM_Lev3_Rear_Idle:
|
|
return 3;
|
|
|
|
case TENTACLE_ANIM_Lev1_Door_reach:
|
|
return -1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void CNPC_Tentacle::Start( void )
|
|
{
|
|
SetThink( &CNPC_Tentacle::Cycle );
|
|
|
|
CPASAttenuationFilter filter( this );
|
|
|
|
if ( !g_fFlySound )
|
|
{
|
|
EmitSound( filter, entindex(), "Tentacle.Flies" );
|
|
g_fFlySound = TRUE;
|
|
}
|
|
else if ( !g_fSquirmSound )
|
|
{
|
|
EmitSound( filter, entindex(), "Tentacle.Squirm" );
|
|
g_fSquirmSound = TRUE;
|
|
}
|
|
|
|
SetNextThink( gpGlobals->curtime + 0.1 );
|
|
}
|
|
|
|
bool CNPC_Tentacle::HeardAnything( void )
|
|
{
|
|
if ( HasCondition( COND_HEAR_DANGER ) || // I remove a bunch of sounds from here on purpose. Talk to me if you're changing this!(sjb)
|
|
HasCondition( COND_HEAR_COMBAT ) ||
|
|
HasCondition( COND_HEAR_WORLD ) ||
|
|
HasCondition( COND_HEAR_PLAYER ) )
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void CNPC_Tentacle::Cycle( void )
|
|
{
|
|
//NDebugOverlay::Cross3D( EarPosition(), 32, 255, 0, 0, false, 0.1 );
|
|
|
|
// ALERT( at_console, "%s %.2f %d %d\n", STRING( pev->targetname ), pev->origin.z, m_MonsterState, m_IdealMonsterState );
|
|
SetNextThink( gpGlobals->curtime + 0.1 );
|
|
|
|
// ALERT( at_console, "%s %d %d %d %f %f\n", STRING( pev->targetname ), pev->sequence, m_iGoalAnim, m_iDir, pev->framerate, pev->health );
|
|
|
|
if ( m_NPCState == NPC_STATE_SCRIPT || GetIdealState() == NPC_STATE_SCRIPT)
|
|
{
|
|
SetAbsAngles( QAngle( GetAbsAngles().x, m_flInitialYaw, GetAbsAngles().z ) );
|
|
GetMotor()->SetIdealYaw( m_flInitialYaw );
|
|
RemoveIgnoredConditions();
|
|
NPCThink( );
|
|
m_iGoalAnim = TENTACLE_ANIM_Pit_Idle;
|
|
return;
|
|
}
|
|
|
|
StudioFrameAdvance();
|
|
DispatchAnimEvents( this );
|
|
|
|
GetMotor()->UpdateYaw( MaxYawSpeed() );
|
|
|
|
CSound *pSound = NULL;
|
|
|
|
GetSenses()->Listen();
|
|
m_BoneFollowerManager.UpdateBoneFollowers(this);
|
|
|
|
// Listen will set this if there's something in my sound list
|
|
if ( HeardAnything() )
|
|
pSound = GetSenses()->GetClosestSound( false, (SOUND_DANGER|SOUND_COMBAT|SOUND_WORLD|SOUND_PLAYER) );
|
|
else
|
|
pSound = NULL;
|
|
|
|
if ( pSound )
|
|
{
|
|
//NDebugOverlay::Line( EarPosition(), pSound->GetSoundOrigin(), 0, 255, 0, false, 0.2 );
|
|
|
|
Vector vecDir;
|
|
if ( gpGlobals->curtime - m_flPrevSoundTime < 0.5 )
|
|
{
|
|
float dt = gpGlobals->curtime - m_flPrevSoundTime;
|
|
vecDir = pSound->GetSoundOrigin() + (pSound->GetSoundOrigin() - m_vecPrevSound) / dt - GetAbsOrigin();
|
|
}
|
|
else
|
|
{
|
|
vecDir = pSound->GetSoundOrigin() - GetAbsOrigin();
|
|
}
|
|
|
|
m_flPrevSoundTime = gpGlobals->curtime;
|
|
m_vecPrevSound = pSound->GetSoundOrigin();
|
|
|
|
m_flSoundYaw = VecToYaw ( vecDir ) - m_flInitialYaw;
|
|
|
|
m_iSoundLevel = Level( vecDir.z );
|
|
|
|
if (m_flSoundYaw < -180)
|
|
m_flSoundYaw += 360;
|
|
if (m_flSoundYaw > 180)
|
|
m_flSoundYaw -= 360;
|
|
|
|
// ALERT( at_console, "sound %d %.0f\n", m_iSoundLevel, m_flSoundYaw );
|
|
if (m_flSoundTime < gpGlobals->curtime)
|
|
{
|
|
// play "I hear new something" sound
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), GetAbsOrigin() + Vector( 0, 0, MyHeight()), "Tentacle.Alert", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
}
|
|
m_flSoundTime = gpGlobals->curtime + random->RandomFloat( 5.0, 10.0 );
|
|
}
|
|
|
|
// clip ideal_yaw
|
|
float dy = m_flSoundYaw;
|
|
switch( GetSequence() )
|
|
{
|
|
case TENTACLE_ANIM_Floor_Rear:
|
|
case TENTACLE_ANIM_Floor_Rear_Idle:
|
|
case TENTACLE_ANIM_Lev1_Rear:
|
|
case TENTACLE_ANIM_Lev1_Rear_Idle:
|
|
case TENTACLE_ANIM_Lev2_Rear:
|
|
case TENTACLE_ANIM_Lev2_Rear_Idle:
|
|
case TENTACLE_ANIM_Lev3_Rear:
|
|
case TENTACLE_ANIM_Lev3_Rear_Idle:
|
|
if (dy < 0 && dy > -m_flMaxYaw)
|
|
dy = -m_flMaxYaw;
|
|
if (dy > 0 && dy < m_flMaxYaw)
|
|
dy = m_flMaxYaw;
|
|
break;
|
|
default:
|
|
if (dy < -m_flMaxYaw)
|
|
dy = -m_flMaxYaw;
|
|
if (dy > m_flMaxYaw)
|
|
dy = m_flMaxYaw;
|
|
}
|
|
GetMotor()->SetIdealYaw( m_flInitialYaw + dy );
|
|
|
|
if ( IsSequenceFinished() )
|
|
{
|
|
// ALERT( at_console, "%s done %d %d\n", STRING( pev->targetname ), pev->sequence, m_iGoalAnim );
|
|
if ( m_iHealth <= 1)
|
|
{
|
|
m_iGoalAnim = TENTACLE_ANIM_Pit_Idle;
|
|
|
|
if ( GetSequence() == TENTACLE_ANIM_Pit_Idle)
|
|
{
|
|
m_iHealth = 75;
|
|
}
|
|
}
|
|
else if ( m_flSoundTime > gpGlobals->curtime )
|
|
{
|
|
if (m_flSoundYaw >= -(m_flMaxYaw + 30) && m_flSoundYaw <= (m_flMaxYaw + 30))
|
|
{
|
|
// strike
|
|
switch ( m_iSoundLevel )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1030 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1031 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1032 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1033 );
|
|
break;
|
|
}
|
|
}
|
|
else if (m_flSoundYaw >= -m_flMaxYaw * 2 && m_flSoundYaw <= m_flMaxYaw * 2)
|
|
{
|
|
// tap
|
|
switch ( m_iSoundLevel )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1020 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1021 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1022 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1023 );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// go into rear idle
|
|
switch ( m_iSoundLevel )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1040 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1041 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1042 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1043 );
|
|
break;
|
|
case 4:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1044 );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if ( GetSequence() == TENTACLE_ANIM_Pit_Idle)
|
|
{
|
|
// stay in pit until hear noise
|
|
m_iGoalAnim = TENTACLE_ANIM_Pit_Idle;
|
|
}
|
|
else if ( GetSequence() == m_iGoalAnim)
|
|
{
|
|
if ( MyLevel() >= 0 && gpGlobals->curtime < m_flSoundTime)
|
|
{
|
|
if ( random->RandomInt(0,9) < m_flSoundTime - gpGlobals->curtime )
|
|
{
|
|
// continue stike
|
|
switch ( m_iSoundLevel )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1030 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1031 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1032 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1033 );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// tap
|
|
switch ( m_iSoundLevel )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1020 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1021 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1022 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1023 );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if ( MyLevel( ) < 0 )
|
|
{
|
|
m_iGoalAnim = SelectWeightedSequence( ACT_1010 );
|
|
}
|
|
else
|
|
{
|
|
if (m_flNextSong < gpGlobals->curtime)
|
|
{
|
|
// play "I hear new something" sound
|
|
CPASAttenuationFilter filter( this );
|
|
EmitSound( filter, entindex(), "Tentacle.Sing" );
|
|
|
|
m_flNextSong = gpGlobals->curtime + random->RandomFloat( 10, 20 );
|
|
}
|
|
|
|
if (random->RandomInt(0,15) == 0)
|
|
{
|
|
// idle on new level
|
|
switch ( random->RandomInt( 0, 3 ) )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1010 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1011 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1012 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1013 );
|
|
break;
|
|
}
|
|
}
|
|
else if ( random->RandomInt( 0, 3 ) == 0 )
|
|
{
|
|
// tap
|
|
switch ( MyLevel() )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1020 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1021 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1022 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1023 );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// idle
|
|
switch ( MyLevel() )
|
|
{
|
|
case 0:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1010 );
|
|
break;
|
|
case 1:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1011 );
|
|
break;
|
|
case 2:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1012 );
|
|
break;
|
|
case 3:
|
|
m_iGoalAnim = SelectWeightedSequence ( ACT_1013 );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (m_flSoundYaw < 0)
|
|
m_flSoundYaw += random->RandomFloat( 2, 8 );
|
|
else
|
|
m_flSoundYaw -= random->RandomFloat( 2, 8 );
|
|
}
|
|
|
|
SetSequence( FindTransitionSequence( GetSequence(), m_iGoalAnim, &m_iDir ) );
|
|
|
|
|
|
if (m_iDir > 0)
|
|
{
|
|
SetCycle( 0 );
|
|
}
|
|
else
|
|
{
|
|
m_iDir = -1; // just to safe
|
|
SetCycle( 1.0f );
|
|
}
|
|
|
|
ResetSequenceInfo( );
|
|
|
|
m_flFramerateAdj = random->RandomFloat( -0.2, 0.2 );
|
|
m_flPlaybackRate = m_iDir * 1.0 + m_flFramerateAdj;
|
|
|
|
switch( GetSequence() )
|
|
{
|
|
case TENTACLE_ANIM_Floor_Tap:
|
|
case TENTACLE_ANIM_Lev1_Tap:
|
|
case TENTACLE_ANIM_Lev2_Tap:
|
|
case TENTACLE_ANIM_Lev3_Tap:
|
|
{
|
|
Vector vecSrc, v_forward;
|
|
AngleVectors( GetAbsAngles(), &v_forward );
|
|
|
|
trace_t tr1, tr2;
|
|
|
|
vecSrc = GetAbsOrigin() + Vector( 0, 0, MyHeight() - 4);
|
|
UTIL_TraceLine( vecSrc, vecSrc + v_forward * 512, MASK_NPCSOLID_BRUSHONLY, this, COLLISION_GROUP_NONE, &tr1 );
|
|
|
|
vecSrc = GetAbsOrigin() + Vector( 0, 0, MyHeight() + 8);
|
|
UTIL_TraceLine( vecSrc, vecSrc + v_forward * 512, MASK_NPCSOLID_BRUSHONLY, this, COLLISION_GROUP_NONE, &tr2 );
|
|
|
|
// ALERT( at_console, "%f %f\n", tr1.flFraction * 512, tr2.flFraction * 512 );
|
|
|
|
m_flTapRadius = SetPoseParameter( 0, random->RandomFloat( tr1.fraction * 512, tr2.fraction * 512 ) );
|
|
}
|
|
break;
|
|
default:
|
|
m_flTapRadius = 336; // 400 - 64
|
|
break;
|
|
}
|
|
SetViewOffset( Vector( 0, 0, MyHeight() ) );
|
|
// ALERT( at_console, "seq %d\n", pev->sequence );
|
|
}
|
|
|
|
if (m_flPrevSoundTime + 2.0 > gpGlobals->curtime)
|
|
{
|
|
// 1.5 normal speed if hears sounds
|
|
m_flPlaybackRate = m_iDir * 1.5 + m_flFramerateAdj;
|
|
}
|
|
else if (m_flPrevSoundTime + 5.0 > gpGlobals->curtime)
|
|
{
|
|
// slowdown to normal
|
|
m_flPlaybackRate = m_iDir + m_iDir * (5 - (gpGlobals->curtime - m_flPrevSoundTime)) / 2 + m_flFramerateAdj;
|
|
}
|
|
}
|
|
|
|
void CNPC_Tentacle::HandleAnimEvent( animevent_t *pEvent )
|
|
{
|
|
switch( pEvent->event )
|
|
{
|
|
case 1: // bang
|
|
{
|
|
Vector vecSrc;
|
|
QAngle angAngles;
|
|
GetAttachment( "0", vecSrc, angAngles );
|
|
|
|
// Vector vecSrc = GetAbsOrigin() + m_flTapRadius * Vector( cos( GetAbsAngles().y * (3.14192653 / 180.0) ), sin( GetAbsAngles().y * (M_PI / 180.0) ), 0.0 );
|
|
|
|
// vecSrc.z += MyHeight( );
|
|
|
|
switch( m_iTapSound )
|
|
{
|
|
case TE_SILO:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitSilo", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
case TE_NONE:
|
|
break;
|
|
case TE_DIRT:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitDirt", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
case TE_WATER:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitWater", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 3: // start killing swing
|
|
m_iHitDmg = 200;
|
|
break;
|
|
|
|
case 4: // end killing swing
|
|
m_iHitDmg = 25;
|
|
break;
|
|
|
|
case 5: // just "whoosh" sound
|
|
break;
|
|
|
|
case 2: // tap scrape
|
|
case 6: // light tap
|
|
{
|
|
Vector vecSrc = GetAbsOrigin() + m_flTapRadius * Vector( cos( GetAbsAngles().y * (M_PI / 180.0) ), sin( GetAbsAngles().y * (M_PI / 180.0) ), 0.0 );
|
|
|
|
vecSrc.z += MyHeight( );
|
|
|
|
float flVol = random->RandomFloat( 0.3, 0.5 );
|
|
|
|
switch( m_iTapSound )
|
|
{
|
|
case TE_SILO:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitSilo", flVol, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
case TE_NONE:
|
|
break;
|
|
case TE_DIRT:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitDirt", flVol, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
case TE_WATER:
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), vecSrc, "Tentacle.HitWater", flVol, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
|
|
case 7: // roar
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), GetAbsOrigin() + Vector( 0, 0, MyHeight()), "Tentacle.Roar", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
|
|
case 8: // search
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), GetAbsOrigin() + Vector( 0, 0, MyHeight()), "Tentacle.Search", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
|
|
case 9: // swing
|
|
UTIL_EmitAmbientSound( GetSoundSourceIndex(), GetAbsOrigin() + Vector( 0, 0, MyHeight()), "Tentacle.Swing", 1.0, SNDLVL_GUNFIRE, 0, 100);
|
|
break;
|
|
default:
|
|
BaseClass::HandleAnimEvent( pEvent );
|
|
}
|
|
}
|
|
|
|
void CNPC_Tentacle::HitTouch( CBaseEntity *pOther )
|
|
{
|
|
if (m_flHitTime > gpGlobals->curtime)
|
|
return;
|
|
|
|
// only look at the ones where the player hit me
|
|
if( pOther == NULL || pOther->GetModelIndex() == GetModelIndex() || ( pOther->GetSolidFlags() & FSOLID_TRIGGER ) )
|
|
return;
|
|
|
|
//Right now the BoneFollower will always be hit in box 0, and
|
|
//will pass that to us. Make *any* touch by the physics objects a kill
|
|
//as the ragdoll only covers the top portion of the tentacle.
|
|
if ( pOther->m_takedamage )
|
|
{
|
|
CTakeDamageInfo info( this, this, m_iHitDmg, DMG_CLUB );
|
|
|
|
Vector vDamageForce = pOther->GetAbsOrigin() - GetAbsOrigin();
|
|
VectorNormalize( vDamageForce );
|
|
|
|
CalculateMeleeDamageForce( &info, vDamageForce, pOther->GetAbsOrigin() );
|
|
pOther->TakeDamage( info );
|
|
|
|
m_flHitTime = gpGlobals->curtime + 0.5;
|
|
}
|
|
}
|
|
|
|
int CNPC_Tentacle::OnTakeDamage( const CTakeDamageInfo &info )
|
|
{
|
|
CTakeDamageInfo i = info;
|
|
|
|
//Don't allow the tentacle to die. Instead set health to 1, so we can do our own death and rebirth
|
|
if( (int)i.GetDamage() >= m_iHealth )
|
|
{
|
|
i.SetDamage( 0.0f );
|
|
m_iHealth = 1;
|
|
}
|
|
|
|
return BaseClass::OnTakeDamage( i );
|
|
}
|
|
|
|
bool CNPC_Tentacle::CreateVPhysics( void )
|
|
{
|
|
BaseClass::CreateVPhysics();
|
|
|
|
IPhysicsObject *pPhysics = VPhysicsGetObject();
|
|
if( pPhysics )
|
|
{
|
|
unsigned short flags = pPhysics->GetCallbackFlags();
|
|
|
|
flags |= CALLBACK_GLOBAL_TOUCH;
|
|
|
|
pPhysics->SetCallbackFlags( flags );
|
|
}
|
|
m_BoneFollowerManager.InitBoneFollowers( this, ARRAYSIZE(pTentacleFollowerBoneNames), pTentacleFollowerBoneNames );
|
|
|
|
return true;
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
//
|
|
// Schedules
|
|
//
|
|
//------------------------------------------------------------------------------
|
|
|
|
AI_BEGIN_CUSTOM_NPC( monster_tentacle, CNPC_Tentacle )
|
|
|
|
DECLARE_ACTIVITY( ACT_1010 )
|
|
DECLARE_ACTIVITY( ACT_1011 )
|
|
DECLARE_ACTIVITY( ACT_1012 )
|
|
DECLARE_ACTIVITY( ACT_1013 )
|
|
|
|
DECLARE_ACTIVITY( ACT_1020 )
|
|
DECLARE_ACTIVITY( ACT_1021 )
|
|
DECLARE_ACTIVITY( ACT_1022 )
|
|
DECLARE_ACTIVITY( ACT_1023 )
|
|
|
|
DECLARE_ACTIVITY( ACT_1030 )
|
|
DECLARE_ACTIVITY( ACT_1031 )
|
|
DECLARE_ACTIVITY( ACT_1032 )
|
|
DECLARE_ACTIVITY( ACT_1033 )
|
|
|
|
DECLARE_ACTIVITY( ACT_1040 )
|
|
DECLARE_ACTIVITY( ACT_1041 )
|
|
DECLARE_ACTIVITY( ACT_1042 )
|
|
DECLARE_ACTIVITY( ACT_1043 )
|
|
DECLARE_ACTIVITY( ACT_1044 )
|
|
|
|
AI_END_CUSTOM_NPC()
|