hlsdk-portable/dlls/barney.cpp

918 lines
24 KiB
C++
Raw Normal View History

2017-12-18 02:39:44 +03:00
/***
*
* Copyright (c) 1996-2002, Valve LLC. All rights reserved.
*
* This product contains software technology licensed from Id
* Software, Inc. ("Id Technology"). Id Technology (c) 1996 Id Software, Inc.
* All Rights Reserved.
*
* This source code contains proprietary and confidential information of
* Valve LLC and its suppliers. Access to this code is restricted to
* persons who have executed a written SDK license with Valve. Any access,
* use or distribution of this code by or to any unlicensed person is illegal.
*
****/
//=========================================================
// monster template
//=========================================================
// UNDONE: Holster weapon?
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "talkmonster.h"
#include "schedule.h"
#include "defaultai.h"
#include "scripted.h"
#include "weapons.h"
#include "soundent.h"
//=========================================================
// Monster's Anim Events Go Here
//=========================================================
// first flag is barney dying for scripted sequences?
#define BARNEY_AE_DRAW ( 2 )
#define BARNEY_AE_SHOOT ( 3 )
#define BARNEY_AE_HOLSTER ( 4 )
#define BARNEY_BODY_GUNHOLSTERED 0
#define BARNEY_BODY_GUNDRAWN 1
2016-07-31 18:48:50 +05:00
#define BARNEY_BODY_GUNGONE 2
2017-12-18 02:39:44 +03:00
class CBarney : public CTalkMonster
{
public:
void Spawn( void );
void Precache( void );
void SetYawSpeed( void );
2016-07-31 18:48:50 +05:00
int ISoundMask( void );
2017-12-18 02:39:44 +03:00
void BarneyFirePistol( void );
void AlertSound( void );
2016-07-31 18:48:50 +05:00
int Classify( void );
2017-12-18 02:39:44 +03:00
void HandleAnimEvent( MonsterEvent_t *pEvent );
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
void RunTask( Task_t *pTask );
void StartTask( Task_t *pTask );
2016-07-31 18:48:50 +05:00
virtual int ObjectCaps( void ) { return CTalkMonster :: ObjectCaps() | FCAP_IMPULSE_USE; }
2017-12-18 02:39:44 +03:00
int TakeDamage( entvars_t* pevInflictor, entvars_t* pevAttacker, float flDamage, int bitsDamageType);
2016-07-31 18:48:50 +05:00
BOOL CheckRangeAttack1( float flDot, float flDist );
2017-12-18 02:39:44 +03:00
void DeclineFollowing( void );
// Override these to set behavior
2016-07-31 18:48:50 +05:00
Schedule_t *GetScheduleOfType( int Type );
Schedule_t *GetSchedule( void );
MONSTERSTATE GetIdealState( void );
2017-12-18 02:39:44 +03:00
void DeathSound( void );
void PainSound( void );
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
void TalkInit( void );
void TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType);
void Killed( entvars_t *pevAttacker, int iGib );
2016-06-04 18:24:23 +05:00
2016-07-31 18:48:50 +05:00
virtual int Save( CSave &save );
virtual int Restore( CRestore &restore );
static TYPEDESCRIPTION m_SaveData[];
2017-12-18 02:39:44 +03:00
int m_iBaseBody; //LRC - for barneys with different bodies
2016-07-31 18:48:50 +05:00
BOOL m_fGunDrawn;
float m_painTime;
float m_checkAttackTime;
BOOL m_lastAttackCheck;
2017-12-18 02:39:44 +03:00
// UNDONE: What is this for? It isn't used?
2016-07-31 18:48:50 +05:00
float m_flPlayerDamage;// how much pain has the player inflicted on me?
2017-12-18 02:39:44 +03:00
CUSTOM_SCHEDULES
2017-12-18 02:39:44 +03:00
};
LINK_ENTITY_TO_CLASS( monster_barney, CBarney )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
TYPEDESCRIPTION CBarney::m_SaveData[] =
2017-12-18 02:39:44 +03:00
{
DEFINE_FIELD( CBarney, m_iBaseBody, FIELD_INTEGER ), //LRC
DEFINE_FIELD( CBarney, m_fGunDrawn, FIELD_BOOLEAN ),
DEFINE_FIELD( CBarney, m_painTime, FIELD_TIME ),
DEFINE_FIELD( CBarney, m_checkAttackTime, FIELD_TIME ),
DEFINE_FIELD( CBarney, m_lastAttackCheck, FIELD_BOOLEAN ),
DEFINE_FIELD( CBarney, m_flPlayerDamage, FIELD_FLOAT ),
};
IMPLEMENT_SAVERESTORE( CBarney, CTalkMonster )
2017-12-18 02:39:44 +03:00
//=========================================================
// AI Schedules Specific to this monster
//=========================================================
2016-07-31 18:48:50 +05:00
Task_t tlBaFollow[] =
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
{ TASK_MOVE_TO_TARGET_RANGE, (float)128 }, // Move within 128 of target ent (client)
{ TASK_SET_SCHEDULE, (float)SCHED_TARGET_FACE },
2017-12-18 02:39:44 +03:00
};
2016-07-31 18:48:50 +05:00
Schedule_t slBaFollow[] =
2017-12-18 02:39:44 +03:00
{
{
tlBaFollow,
2016-07-31 18:48:50 +05:00
ARRAYSIZE( tlBaFollow ),
bits_COND_NEW_ENEMY |
bits_COND_LIGHT_DAMAGE |
bits_COND_HEAVY_DAMAGE |
2017-12-18 02:39:44 +03:00
bits_COND_HEAR_SOUND |
bits_COND_PROVOKED,
bits_SOUND_DANGER,
"Follow"
},
};
//=========================================================
2016-07-31 18:48:50 +05:00
// BarneyDraw - much better looking draw schedule for when
2017-12-18 02:39:44 +03:00
// barney knows who he's gonna attack.
//=========================================================
2016-07-31 18:48:50 +05:00
Task_t tlBarneyEnemyDraw[] =
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
{ TASK_STOP_MOVING, 0 },
{ TASK_FACE_ENEMY, 0 },
{ TASK_PLAY_SEQUENCE_FACE_ENEMY, (float) ACT_ARM },
2017-12-18 02:39:44 +03:00
};
2016-07-31 18:48:50 +05:00
Schedule_t slBarneyEnemyDraw[] =
2017-12-18 02:39:44 +03:00
{
{
tlBarneyEnemyDraw,
2016-07-31 18:48:50 +05:00
ARRAYSIZE( tlBarneyEnemyDraw ),
2017-12-18 02:39:44 +03:00
0,
0,
"Barney Enemy Draw"
}
};
2016-07-31 18:48:50 +05:00
Task_t tlBaFaceTarget[] =
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
{ TASK_SET_ACTIVITY, (float)ACT_IDLE },
{ TASK_FACE_TARGET, (float)0 },
{ TASK_SET_ACTIVITY, (float)ACT_IDLE },
{ TASK_SET_SCHEDULE, (float)SCHED_TARGET_CHASE },
2017-12-18 02:39:44 +03:00
};
2016-07-31 18:48:50 +05:00
Schedule_t slBaFaceTarget[] =
2017-12-18 02:39:44 +03:00
{
{
tlBaFaceTarget,
2016-07-31 18:48:50 +05:00
ARRAYSIZE( tlBaFaceTarget ),
bits_COND_CLIENT_PUSH |
bits_COND_NEW_ENEMY |
bits_COND_LIGHT_DAMAGE |
bits_COND_HEAVY_DAMAGE |
2017-12-18 02:39:44 +03:00
bits_COND_HEAR_SOUND |
bits_COND_PROVOKED,
bits_SOUND_DANGER,
"FaceTarget"
},
};
2016-07-31 18:48:50 +05:00
Task_t tlIdleBaStand[] =
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
{ TASK_STOP_MOVING, 0 },
{ TASK_SET_ACTIVITY, (float)ACT_IDLE },
{ TASK_WAIT, (float)2 }, // repick IDLESTAND every two seconds.
{ TASK_TLK_HEADRESET, (float)0 }, // reset head position
2017-12-18 02:39:44 +03:00
};
2016-07-31 18:48:50 +05:00
Schedule_t slIdleBaStand[] =
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
{
2017-12-18 02:39:44 +03:00
tlIdleBaStand,
2016-07-31 18:48:50 +05:00
ARRAYSIZE( tlIdleBaStand ),
bits_COND_NEW_ENEMY |
bits_COND_LIGHT_DAMAGE |
bits_COND_HEAVY_DAMAGE |
bits_COND_HEAR_SOUND |
bits_COND_SMELL |
2017-12-18 02:39:44 +03:00
bits_COND_PROVOKED,
2016-07-31 18:48:50 +05:00
bits_SOUND_COMBAT |// sound flags - change these, and you'll break the talking code.
//bits_SOUND_PLAYER |
//bits_SOUND_WORLD |
bits_SOUND_DANGER |
bits_SOUND_MEAT |// scents
bits_SOUND_CARCASS |
2017-12-18 02:39:44 +03:00
bits_SOUND_GARBAGE,
"IdleStand"
},
};
DEFINE_CUSTOM_SCHEDULES( CBarney )
{
slBaFollow,
slBarneyEnemyDraw,
slBaFaceTarget,
slIdleBaStand,
};
IMPLEMENT_CUSTOM_SCHEDULES( CBarney, CTalkMonster )
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
void CBarney::StartTask( Task_t *pTask )
2017-12-18 02:39:44 +03:00
{
CTalkMonster::StartTask( pTask );
}
2016-07-31 18:48:50 +05:00
void CBarney::RunTask( Task_t *pTask )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
switch( pTask->iTask )
2017-12-18 02:39:44 +03:00
{
case TASK_RANGE_ATTACK1:
2017-06-29 18:56:03 +05:00
if( m_hEnemy != 0 && ( m_hEnemy->IsPlayer() ) )
2017-12-18 02:39:44 +03:00
{
pev->framerate = 1.5;
}
CTalkMonster::RunTask( pTask );
break;
default:
CTalkMonster::RunTask( pTask );
break;
}
}
//=========================================================
// ISoundMask - returns a bit mask indicating which types
// of sounds this monster regards.
//=========================================================
2016-07-31 18:48:50 +05:00
int CBarney::ISoundMask( void)
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
return bits_SOUND_WORLD |
bits_SOUND_COMBAT |
bits_SOUND_CARCASS |
bits_SOUND_MEAT |
bits_SOUND_GARBAGE |
bits_SOUND_DANGER |
2017-12-18 02:39:44 +03:00
bits_SOUND_PLAYER;
}
//=========================================================
// Classify - indicates this monster's place in the
// relationship table.
//=========================================================
2016-07-31 18:48:50 +05:00
int CBarney::Classify( void )
2017-12-18 02:39:44 +03:00
{
return m_iClass?m_iClass:CLASS_PLAYER_ALLY;
}
//=========================================================
// ALertSound - barney says "Freeze!"
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::AlertSound( void )
2017-12-18 02:39:44 +03:00
{
2017-06-29 18:56:03 +05:00
if( m_hEnemy != 0 )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( FOkToSpeak() )
2017-12-18 02:39:44 +03:00
{
2019-11-11 02:59:56 +05:00
if( m_iszSpeakAs )
2017-12-18 02:39:44 +03:00
{
char szBuf[32];
2019-11-11 02:59:56 +05:00
strcpy( szBuf, STRING( m_iszSpeakAs ) );
strcat( szBuf, "_ATTACK" );
PlaySentence( szBuf, RANDOM_FLOAT( 2.8f, 3.2f ), VOL_NORM, ATTN_IDLE );
2017-12-18 02:39:44 +03:00
}
else
{
2019-11-11 02:59:56 +05:00
PlaySentence( "BA_ATTACK", RANDOM_FLOAT( 2.8f, 3.2f ), VOL_NORM, ATTN_IDLE );
2017-12-18 02:39:44 +03:00
}
}
}
}
//=========================================================
// SetYawSpeed - allows each sequence to have a different
// turn rate associated with it.
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::SetYawSpeed( void )
2017-12-18 02:39:44 +03:00
{
int ys;
ys = 0;
switch ( m_Activity )
{
case ACT_IDLE:
ys = 70;
break;
case ACT_WALK:
ys = 70;
break;
case ACT_RUN:
ys = 90;
break;
default:
ys = 70;
break;
}
pev->yaw_speed = ys;
}
//=========================================================
// CheckRangeAttack1
//=========================================================
2016-07-31 18:48:50 +05:00
BOOL CBarney::CheckRangeAttack1( float flDot, float flDist )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
if( flDist <= 1024.0f && flDot >= 0.5f )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( gpGlobals->time > m_checkAttackTime )
2017-12-18 02:39:44 +03:00
{
TraceResult tr;
2016-07-31 18:48:50 +05:00
2019-10-13 16:49:25 +05:00
Vector shootOrigin = pev->origin + Vector( 0.0f, 0.0f, 55.0f );
2017-12-18 02:39:44 +03:00
CBaseEntity *pEnemy = m_hEnemy;
2016-07-31 18:48:50 +05:00
Vector shootTarget = ( ( pEnemy->BodyTarget( shootOrigin ) - pEnemy->pev->origin ) + m_vecEnemyLKP );
UTIL_TraceLine( shootOrigin, shootTarget, dont_ignore_monsters, ENT( pev ), &tr );
2019-10-13 16:49:25 +05:00
m_checkAttackTime = gpGlobals->time + 1.0f;
if( tr.flFraction == 1.0f || ( tr.pHit != NULL && CBaseEntity::Instance( tr.pHit ) == pEnemy ) )
2017-12-18 02:39:44 +03:00
m_lastAttackCheck = TRUE;
else
m_lastAttackCheck = FALSE;
2019-10-13 16:49:25 +05:00
m_checkAttackTime = gpGlobals->time + 1.5f;
2017-12-18 02:39:44 +03:00
}
return m_lastAttackCheck;
}
return FALSE;
}
//=========================================================
// BarneyFirePistol - shoots one round from the pistol at
// the enemy barney is facing.
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::BarneyFirePistol( void )
2017-12-18 02:39:44 +03:00
{
Vector vecShootOrigin;
2016-07-31 18:48:50 +05:00
UTIL_MakeVectors( pev->angles );
2017-12-18 02:39:44 +03:00
vecShootOrigin = pev->origin + Vector( 0, 0, 55 );
Vector vecShootDir = ShootAtEnemy( vecShootOrigin );
Vector angDir = UTIL_VecToAngles( vecShootDir );
SetBlending( 0, angDir.x );
pev->effects = EF_MUZZLEFLASH;
if (pev->frags)
{
FireBullets(1, vecShootOrigin, vecShootDir, VECTOR_CONE_2DEGREES, 1024, BULLET_PLAYER_357);
if (RANDOM_LONG(0, 1))
EMIT_SOUND_DYN( ENT(pev), CHAN_WEAPON, "weapons/357_shot1.wav", 1, ATTN_NORM, 0, 100 );
else
EMIT_SOUND_DYN( ENT(pev), CHAN_WEAPON, "weapons/357_shot2.wav", 1, ATTN_NORM, 0, 100 );
}
else
{
2017-12-18 23:47:12 +03:00
FireBullets( 1, vecShootOrigin, vecShootDir, VECTOR_CONE_2DEGREES, 1024, BULLET_MONSTER_9MM );
2017-12-18 02:39:44 +03:00
int pitchShift = RANDOM_LONG( 0, 20 );
2017-12-18 23:47:12 +03:00
2017-12-18 02:39:44 +03:00
// Only shift about half the time
2017-12-18 23:47:12 +03:00
if( pitchShift > 10 )
2017-12-18 02:39:44 +03:00
pitchShift = 0;
else
pitchShift -= 5;
2017-12-18 23:47:12 +03:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_WEAPON, "barney/ba_attack2.wav", 1, ATTN_NORM, 0, 100 + pitchShift );
2017-12-18 02:39:44 +03:00
}
2019-10-13 16:49:25 +05:00
CSoundEnt::InsertSound( bits_SOUND_COMBAT, pev->origin, 384, 0.3f );
2017-12-18 02:39:44 +03:00
// UNDONE: Reload?
m_cAmmoLoaded--;// take away a bullet!
}
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
//=========================================================
// HandleAnimEvent - catches the monster-specific messages
// that occur when tagged animation frames are played.
//
// Returns number of events handled, 0 if none.
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::HandleAnimEvent( MonsterEvent_t *pEvent )
2017-12-18 02:39:44 +03:00
{
switch( pEvent->event )
{
case BARNEY_AE_SHOOT:
BarneyFirePistol();
break;
case BARNEY_AE_DRAW:
// barney's bodygroup switches here so he can pull gun from holster
pev->body = m_iBaseBody + BARNEY_BODY_GUNDRAWN;
m_fGunDrawn = TRUE;
break;
case BARNEY_AE_HOLSTER:
// change bodygroup to replace gun in holster
pev->body = m_iBaseBody + BARNEY_BODY_GUNHOLSTERED;
m_fGunDrawn = FALSE;
break;
default:
CTalkMonster::HandleAnimEvent( pEvent );
}
}
//=========================================================
// Spawn
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::Spawn()
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
Precache();
2017-12-18 02:39:44 +03:00
if (pev->model)
SET_MODEL(ENT(pev), STRING(pev->model)); //LRC
else
2017-12-18 23:47:12 +03:00
SET_MODEL( ENT( pev ), "models/barney.mdl" );
2016-07-31 18:48:50 +05:00
UTIL_SetSize( pev, VEC_HUMAN_HULL_MIN, VEC_HUMAN_HULL_MAX );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
pev->solid = SOLID_SLIDEBOX;
pev->movetype = MOVETYPE_STEP;
m_bloodColor = BLOOD_COLOR_RED;
2019-11-11 02:59:56 +05:00
if( pev->health == 0 ) //LRC
2017-12-18 23:47:12 +03:00
pev->health = gSkillData.barneyHealth;
2019-10-13 16:49:25 +05:00
pev->view_ofs = Vector( 0.0f, 0.0f, 50.0f );// position of the eyes relative to monster's origin.
2016-07-31 18:48:50 +05:00
m_flFieldOfView = VIEW_FIELD_WIDE; // NOTE: we need a wide field of view so npc will notice player and say hello
m_MonsterState = MONSTERSTATE_NONE;
2017-12-18 02:39:44 +03:00
m_iBaseBody = pev->body; //LRC
pev->body = m_iBaseBody + BARNEY_BODY_GUNHOLSTERED; // gun in holster
2016-07-31 18:48:50 +05:00
m_fGunDrawn = FALSE;
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
m_afCapability = bits_CAP_HEAR | bits_CAP_TURN_HEAD | bits_CAP_DOORS_GROUP;
2017-12-18 02:39:44 +03:00
MonsterInit();
SetUse(&CBarney :: FollowerUse );
}
//=========================================================
// Precache - precaches all resources this monster needs
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::Precache()
2017-12-18 02:39:44 +03:00
{
if (pev->model)
2017-12-18 23:47:12 +03:00
PRECACHE_MODEL(STRING(pev->model)); //LRC
2017-12-18 02:39:44 +03:00
else
2017-12-18 23:47:12 +03:00
PRECACHE_MODEL( "models/barney.mdl" );
2016-06-04 18:24:23 +05:00
2016-07-31 18:48:50 +05:00
PRECACHE_SOUND( "barney/ba_attack1.wav" );
PRECACHE_SOUND( "barney/ba_attack2.wav" );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
PRECACHE_SOUND( "barney/ba_pain1.wav" );
PRECACHE_SOUND( "barney/ba_pain2.wav" );
PRECACHE_SOUND( "barney/ba_pain3.wav" );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
PRECACHE_SOUND( "barney/ba_die1.wav" );
PRECACHE_SOUND( "barney/ba_die2.wav" );
PRECACHE_SOUND( "barney/ba_die3.wav" );
2017-12-18 02:39:44 +03:00
// every new barney must call this, otherwise
// when a level is loaded, nobody will talk (time is reset to 0)
TalkInit();
CTalkMonster::Precache();
}
// Init talk data
2016-07-31 18:48:50 +05:00
void CBarney::TalkInit()
2017-12-18 02:39:44 +03:00
{
CTalkMonster::TalkInit();
// barney speech group names (group names are in sentences.txt)
if (!m_iszSpeakAs)
{
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_ANSWER] = "BA_ANSWER";
m_szGrp[TLK_QUESTION] = "BA_QUESTION";
m_szGrp[TLK_IDLE] = "BA_IDLE";
m_szGrp[TLK_STARE] = "BA_STARE";
2017-12-18 02:39:44 +03:00
if (pev->spawnflags & SF_MONSTER_PREDISASTER) //LRC
m_szGrp[TLK_USE] = "BA_PFOLLOW";
else
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_USE] = "BA_OK";
2017-12-18 02:39:44 +03:00
if (pev->spawnflags & SF_MONSTER_PREDISASTER)
m_szGrp[TLK_UNUSE] = "BA_PWAIT";
else
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_UNUSE] = "BA_WAIT";
2017-12-18 02:39:44 +03:00
if (pev->spawnflags & SF_MONSTER_PREDISASTER)
m_szGrp[TLK_DECLINE] = "BA_POK";
else
m_szGrp[TLK_DECLINE] = "BA_NOTOK";
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_STOP] = "BA_STOP";
2017-12-18 02:39:44 +03:00
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_NOSHOOT] = "BA_SCARED";
m_szGrp[TLK_HELLO] = "BA_HELLO";
2017-12-18 02:39:44 +03:00
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_PLHURT1] = "!BA_CUREA";
m_szGrp[TLK_PLHURT2] = "!BA_CUREB";
m_szGrp[TLK_PLHURT3] = "!BA_CUREC";
2017-12-18 02:39:44 +03:00
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_PHELLO] = NULL; //"BA_PHELLO"; // UNDONE
m_szGrp[TLK_PIDLE] = NULL; //"BA_PIDLE"; // UNDONE
2017-12-18 02:39:44 +03:00
m_szGrp[TLK_PQUESTION] = "BA_PQUEST"; // UNDONE
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_SMELL] = "BA_SMELL";
2016-06-04 18:24:23 +05:00
2017-12-18 23:47:12 +03:00
m_szGrp[TLK_WOUND] = "BA_WOUND";
m_szGrp[TLK_MORTAL] = "BA_MORTAL";
2017-12-18 02:39:44 +03:00
}
// get voice for head - just one barney voice for now
m_voicePitch = 100;
}
static BOOL IsFacing( entvars_t *pevTest, const Vector &reference )
{
2016-07-31 18:48:50 +05:00
Vector vecDir = reference - pevTest->origin;
2019-10-13 16:49:25 +05:00
vecDir.z = 0.0f;
2017-12-18 02:39:44 +03:00
vecDir = vecDir.Normalize();
Vector forward, angle;
angle = pevTest->v_angle;
2019-10-13 16:49:25 +05:00
angle.x = 0.0f;
2017-12-18 02:39:44 +03:00
UTIL_MakeVectorsPrivate( angle, forward, NULL, NULL );
2017-12-18 02:39:44 +03:00
// He's facing me, he meant it
2019-10-13 16:49:25 +05:00
if( DotProduct( forward, vecDir ) > 0.96f ) // +/- 15 degrees or so
2017-12-18 02:39:44 +03:00
{
return TRUE;
}
return FALSE;
}
2016-07-31 18:48:50 +05:00
int CBarney::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType )
2017-12-18 02:39:44 +03:00
{
// make sure friends talk about it if player hurts talkmonsters...
2016-07-31 18:48:50 +05:00
int ret = CTalkMonster::TakeDamage( pevInflictor, pevAttacker, flDamage, bitsDamageType );
if( !IsAlive() || pev->deadflag == DEAD_DYING )
2017-12-18 02:39:44 +03:00
return ret;
// LRC - if my reaction to the player has been overridden, don't do this stuff
if (m_iPlayerReact) return ret;
2016-07-31 18:48:50 +05:00
if( m_MonsterState != MONSTERSTATE_PRONE && ( pevAttacker->flags & FL_CLIENT ) )
2017-12-18 02:39:44 +03:00
{
m_flPlayerDamage += flDamage;
// This is a heurstic to determine if the player intended to harm me
// If I have an enemy, we can't establish intent (may just be crossfire)
2017-06-29 18:56:03 +05:00
if( m_hEnemy == 0 )
2017-12-18 02:39:44 +03:00
{
// If the player was facing directly at me, or I'm already suspicious, get mad
2016-07-31 18:48:50 +05:00
if( ( m_afMemory & bits_MEMORY_SUSPICIOUS ) || IsFacing( pevAttacker, pev->origin ) )
2017-12-18 02:39:44 +03:00
{
// Alright, now I'm pissed!
if (m_iszSpeakAs)
{
char szBuf[32];
strcpy(szBuf,STRING(m_iszSpeakAs));
strcat(szBuf,"_MAD");
PlaySentence( szBuf, 4, VOL_NORM, ATTN_NORM );
}
else
{
PlaySentence( "BA_MAD", 4, VOL_NORM, ATTN_NORM );
}
Remember( bits_MEMORY_PROVOKED );
StopFollowing( TRUE );
}
else
{
// Hey, be careful with that
if (m_iszSpeakAs)
{
char szBuf[32];
strcpy(szBuf,STRING(m_iszSpeakAs));
strcat(szBuf,"_SHOT");
PlaySentence( szBuf, 4, VOL_NORM, ATTN_NORM );
}
else
{
PlaySentence( "BA_SHOT", 4, VOL_NORM, ATTN_NORM );
}
Remember( bits_MEMORY_SUSPICIOUS );
}
}
2016-07-31 18:48:50 +05:00
else if( !( m_hEnemy->IsPlayer()) && pev->deadflag == DEAD_NO )
2017-12-18 02:39:44 +03:00
{
if (m_iszSpeakAs)
{
char szBuf[32];
strcpy(szBuf,STRING(m_iszSpeakAs));
strcat(szBuf,"_SHOT");
PlaySentence( szBuf, 4, VOL_NORM, ATTN_NORM );
}
else
{
PlaySentence( "BA_SHOT", 4, VOL_NORM, ATTN_NORM );
}
}
}
return ret;
}
//=========================================================
// PainSound
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::PainSound( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( gpGlobals->time < m_painTime )
2017-12-18 02:39:44 +03:00
return;
2019-10-13 16:49:25 +05:00
m_painTime = gpGlobals->time + RANDOM_FLOAT( 0.5f, 0.75f );
2016-07-31 18:48:50 +05:00
switch( RANDOM_LONG( 0, 2 ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
case 0:
2019-10-13 16:49:25 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_pain1.wav", 1.0f, ATTN_NORM, 0, GetVoicePitch() );
2016-07-31 18:48:50 +05:00
break;
case 1:
2019-10-13 16:49:25 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_pain2.wav", 1.0f, ATTN_NORM, 0, GetVoicePitch() );
2016-07-31 18:48:50 +05:00
break;
case 2:
2019-10-13 16:49:25 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_pain3.wav", 1.0f, ATTN_NORM, 0, GetVoicePitch() );
2016-07-31 18:48:50 +05:00
break;
2017-12-18 02:39:44 +03:00
}
}
//=========================================================
// DeathSound
//=========================================================
2016-07-31 18:48:50 +05:00
void CBarney::DeathSound( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
switch( RANDOM_LONG( 0, 2 ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
case 0:
2019-10-13 16:49:25 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_die1.wav", 1.0f, ATTN_NORM, 0, GetVoicePitch() );
2016-07-31 18:48:50 +05:00
break;
case 1:
2019-10-13 16:49:25 +05:00
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_die2.wav", 1.0f, ATTN_NORM, 0, GetVoicePitch() );
2016-07-31 18:48:50 +05:00
break;
case 2:
EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "barney/ba_die3.wav", 1, ATTN_NORM, 0, GetVoicePitch() );
break;
2017-12-18 02:39:44 +03:00
}
}
2016-07-31 18:48:50 +05:00
void CBarney::TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
switch( ptr->iHitgroup )
2017-12-18 02:39:44 +03:00
{
case HITGROUP_CHEST:
case HITGROUP_STOMACH:
2016-07-31 18:48:50 +05:00
if (bitsDamageType & ( DMG_BULLET | DMG_SLASH | DMG_BLAST ) )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
flDamage = flDamage * 0.5f;
2017-12-18 02:39:44 +03:00
}
break;
case 10:
2016-07-31 18:48:50 +05:00
if( bitsDamageType & ( DMG_BULLET | DMG_SLASH | DMG_CLUB ) )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
flDamage -= 20.0f;
if( flDamage <= 0.0f )
2017-12-18 02:39:44 +03:00
{
2019-10-13 16:49:25 +05:00
UTIL_Ricochet( ptr->vecEndPos, 1.0f );
flDamage = 0.01f;
2017-12-18 02:39:44 +03:00
}
}
2017-12-18 02:39:44 +03:00
// always a head shot
ptr->iHitgroup = HITGROUP_HEAD;
break;
}
CTalkMonster::TraceAttack( pevAttacker, flDamage, vecDir, ptr, bitsDamageType );
}
void CBarney::Killed( entvars_t *pevAttacker, int iGib )
{
if ( pev->body < m_iBaseBody + BARNEY_BODY_GUNGONE && !(pev->spawnflags & SF_MONSTER_NO_WPN_DROP))
{// drop the gun!
Vector vecGunPos;
Vector vecGunAngles;
pev->body = m_iBaseBody + BARNEY_BODY_GUNGONE;
GetAttachment( 0, vecGunPos, vecGunAngles );
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
CBaseEntity *pGun;
if (pev->frags)
pGun = DropItem( "weapon_357", vecGunPos, vecGunAngles );
else
pGun = DropItem( "weapon_9mmhandgun", vecGunPos, vecGunAngles );
}
SetUse( NULL );
CTalkMonster::Killed( pevAttacker, iGib );
}
//=========================================================
// AI Schedules Specific to this monster
//=========================================================
2016-07-31 18:48:50 +05:00
Schedule_t *CBarney::GetScheduleOfType( int Type )
2017-12-18 02:39:44 +03:00
{
Schedule_t *psched;
switch( Type )
{
case SCHED_ARM_WEAPON:
2017-06-29 18:56:03 +05:00
if( m_hEnemy != 0 )
2017-12-18 02:39:44 +03:00
{
// face enemy, then draw.
return slBarneyEnemyDraw;
}
break;
// Hook these to make a looping schedule
case SCHED_TARGET_FACE:
// call base class default so that barney will talk
// when 'used'
2016-07-31 18:48:50 +05:00
psched = CTalkMonster::GetScheduleOfType( Type );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( psched == slIdleStand )
2017-12-18 02:39:44 +03:00
return slBaFaceTarget; // override this for different target face behavior
else
return psched;
case SCHED_TARGET_CHASE:
return slBaFollow;
case SCHED_IDLE_STAND:
// call base class default so that scientist will talk
// when standing during idle
2016-07-31 18:48:50 +05:00
psched = CTalkMonster::GetScheduleOfType( Type );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
if( psched == slIdleStand )
2017-12-18 02:39:44 +03:00
{
// just look straight ahead.
return slIdleBaStand;
}
else
return psched;
}
return CTalkMonster::GetScheduleOfType( Type );
}
//=========================================================
// GetSchedule - Decides which type of schedule best suits
// the monster's current state and conditions. Then calls
// monster's member function to get a pointer to a schedule
// of the proper type.
//=========================================================
2016-07-31 18:48:50 +05:00
Schedule_t *CBarney::GetSchedule( void )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_HEAR_SOUND ) )
2017-12-18 02:39:44 +03:00
{
CSound *pSound;
pSound = PBestSound();
ASSERT( pSound != NULL );
2016-07-31 18:48:50 +05:00
if( pSound && (pSound->m_iType & bits_SOUND_DANGER) )
2017-12-18 02:39:44 +03:00
return GetScheduleOfType( SCHED_TAKE_COVER_FROM_BEST_SOUND );
}
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_ENEMY_DEAD ) && FOkToSpeak() )
2017-12-18 02:39:44 +03:00
{
// Hey, be careful with that
if (m_iszSpeakAs)
{
char szBuf[32];
strcpy(szBuf,STRING(m_iszSpeakAs));
strcat(szBuf,"_KILL");
PlaySentence( szBuf, 4, VOL_NORM, ATTN_NORM );
}
else
{
PlaySentence( "BA_KILL", 4, VOL_NORM, ATTN_NORM );
}
}
switch( m_MonsterState )
{
case MONSTERSTATE_COMBAT:
{
2016-06-13 19:29:45 +05:00
// dead enemy
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_ENEMY_DEAD ) )
2017-12-18 02:39:44 +03:00
{
// call base class, all code to handle dead enemies is centralized there.
2016-07-31 18:48:50 +05:00
return CBaseMonster::GetSchedule();
2017-12-18 02:39:44 +03:00
}
// always act surprized with a new enemy
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_NEW_ENEMY ) && HasConditions( bits_COND_LIGHT_DAMAGE ) )
2017-12-18 02:39:44 +03:00
return GetScheduleOfType( SCHED_SMALL_FLINCH );
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
// wait for one schedule to draw gun
2016-07-31 18:48:50 +05:00
if( !m_fGunDrawn )
2017-12-18 02:39:44 +03:00
return GetScheduleOfType( SCHED_ARM_WEAPON );
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_HEAVY_DAMAGE ) )
2017-12-18 02:39:44 +03:00
return GetScheduleOfType( SCHED_TAKE_COVER_FROM_ENEMY );
}
break;
case MONSTERSTATE_ALERT:
case MONSTERSTATE_IDLE:
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_LIGHT_DAMAGE | bits_COND_HEAVY_DAMAGE ) )
2017-12-18 02:39:44 +03:00
{
// flinch if hurt
return GetScheduleOfType( SCHED_SMALL_FLINCH );
}
2017-06-29 18:56:03 +05:00
if( m_hEnemy == 0 && IsFollowing() )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
if( !m_hTargetEnt->IsAlive() )
2017-12-18 02:39:44 +03:00
{
// UNDONE: Comment about the recently dead player here?
StopFollowing( FALSE );
break;
}
else
{
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_CLIENT_PUSH ) )
2017-12-18 02:39:44 +03:00
{
return GetScheduleOfType( SCHED_MOVE_AWAY_FOLLOW );
}
return GetScheduleOfType( SCHED_TARGET_FACE );
}
}
2016-07-31 18:48:50 +05:00
if( HasConditions( bits_COND_CLIENT_PUSH ) )
2017-12-18 02:39:44 +03:00
{
return GetScheduleOfType( SCHED_MOVE_AWAY );
}
// try to say something about smells
TrySmellTalk();
break;
default:
break;
}
2016-07-31 18:48:50 +05:00
2017-12-18 02:39:44 +03:00
return CTalkMonster::GetSchedule();
}
2016-07-31 18:48:50 +05:00
MONSTERSTATE CBarney::GetIdealState( void )
2017-12-18 02:39:44 +03:00
{
return CTalkMonster::GetIdealState();
}
void CBarney::DeclineFollowing( void )
{
PlaySentence( m_szGrp[TLK_DECLINE], 2, VOL_NORM, ATTN_NORM ); //LRC
}
//=========================================================
// DEAD BARNEY PROP
//
// Designer selects a pose in worldcraft, 0 through num_poses-1
// this value is added to what is selected as the 'first dead pose'
// among the monster's normal animations. All dead poses must
// appear sequentially in the model file. Be sure and set
// the m_iFirstPose properly!
//
//=========================================================
class CDeadBarney : public CBaseMonster
{
public:
void Spawn( void );
2016-07-31 18:48:50 +05:00
int Classify( void ) { return CLASS_PLAYER_ALLY; }
2017-12-18 02:39:44 +03:00
void KeyValue( KeyValueData *pkvd );
2016-07-31 18:48:50 +05:00
int m_iPose;// which sequence to display -- temporary, don't need to save
2017-06-29 18:56:03 +05:00
static const char *m_szPoses[3];
2017-12-18 02:39:44 +03:00
};
2017-06-29 18:56:03 +05:00
const char *CDeadBarney::m_szPoses[] = { "lying_on_back", "lying_on_side", "lying_on_stomach" };
2017-12-18 02:39:44 +03:00
void CDeadBarney::KeyValue( KeyValueData *pkvd )
{
2016-07-31 18:48:50 +05:00
if( FStrEq( pkvd->szKeyName, "pose" ) )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
m_iPose = atoi( pkvd->szValue );
2017-12-18 02:39:44 +03:00
pkvd->fHandled = TRUE;
}
2016-07-31 18:48:50 +05:00
else
2017-12-18 02:39:44 +03:00
CBaseMonster::KeyValue( pkvd );
}
LINK_ENTITY_TO_CLASS( monster_barney_dead, CDeadBarney )
2017-12-18 02:39:44 +03:00
//=========================================================
// ********** DeadBarney SPAWN **********
//=========================================================
2016-07-31 18:48:50 +05:00
void CDeadBarney::Spawn()
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
PRECACHE_MODEL( "models/barney.mdl" );
SET_MODEL( ENT( pev ), "models/barney.mdl" );
2017-12-18 02:39:44 +03:00
2016-07-31 18:48:50 +05:00
pev->effects = 0;
pev->yaw_speed = 8;
pev->sequence = 0;
m_bloodColor = BLOOD_COLOR_RED;
2017-12-18 02:39:44 +03:00
pev->sequence = LookupSequence( m_szPoses[m_iPose] );
2016-07-31 18:48:50 +05:00
if( pev->sequence == -1 )
2017-12-18 02:39:44 +03:00
{
2016-07-31 18:48:50 +05:00
ALERT( at_console, "Dead barney with bad pose\n" );
2017-12-18 02:39:44 +03:00
}
// Corpses have less health
2016-07-31 18:48:50 +05:00
pev->health = 8;//gSkillData.barneyHealth;
2017-12-18 02:39:44 +03:00
MonsterInitDead();
}