Portable Half-Life SDK. GoldSource and Xash3D. Crossplatform.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

492 lines
14 KiB

/***
*
* Copyright (c) 1996-2001, 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.
*
****/
//=========================================================
// bullsquid - big, spotty tentacle-mouthed meanie.
//=========================================================
#include "extdll.h"
#include "util.h"
#include "cbase.h"
#include "monsters.h"
#include "schedule.h"
#include "nodes.h"
#include "effects.h"
#include "decals.h"
#include "soundent.h"
#include "game.h"
#include "bullsquid.h"
//=========================================================
// Monster's Anim Events Go Here
//=========================================================
#define ZBULL_AE_SPIT ( 1 )
#define ZBULL_AE_BITE ( 2 )
#define ZBULL_AE_BLINK ( 3 )
#define ZBULL_AE_TAILWHIP ( 4 )
#define ZBULL_AE_HOP ( 5 )
#define ZBULL_AE_THROW ( 6 )
class CEinarZombieBull : public CBullsquid
{
public:
void Spawn();
void Precache();
int Classify();
void SetYawSpeed();
void HandleAnimEvent( MonsterEvent_t *pEvent );
void IdleSound();
void PainSound();
void DeathSound();
void AlertSound();
void AttackSound();
void BiteSound();
void StartTask( Task_t *pTask );
int IgnoreConditions();
void RunAI();
int IRelationship( CBaseEntity *pTarget );
int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType );
BOOL CheckMeleeAttack1( float flDot, float flDist );
BOOL CheckMeleeAttack2( float flDot, float flDist );
BOOL CheckRangeAttack1( float flDot, float flDist ) { return FALSE; }
Schedule_t *GetSchedule();
Schedule_t *GetScheduleOfType( int Type );
static const char *pAttackSounds[];
static const char *pBiteSounds[];
static const char *pIdleSounds[];
static const char *pAlertSounds[];
static const char *pPainSounds[];
static const char *pDeathSounds[];
};
LINK_ENTITY_TO_CLASS( monster_th_zombiebull, CEinarZombieBull )
const char *CEinarZombieBull::pAttackSounds[] =
{
"bull/bu_whip1.wav",
"bull/bu_whip2.wav",
"bull/bu_whip3.wav",
};
const char *CEinarZombieBull::pBiteSounds[] =
{
"bull/bu_gore1.wav",
"bull/bu_gore2.wav",
"bull/bu_gore3.wav",
};
const char *CEinarZombieBull::pIdleSounds[] =
{
"bull/bu_idle1.wav",
"bull/bu_idle2.wav",
"bull/bu_idle3.wav",
};
const char *CEinarZombieBull::pAlertSounds[] =
{
"bull/bu_alert1.wav",
"bull/bu_alert2.wav",
"bull/bu_alert3.wav",
"bull/bu_alert4.wav",
};
const char *CEinarZombieBull::pPainSounds[] =
{
"bull/bu_pain1.wav",
"bull/bu_pain2.wav",
"bull/bu_pain3.wav",
};
const char *CEinarZombieBull::pDeathSounds[] =
{
"bull/bu_die1.wav",
"bull/bu_die2.wav",
"bull/bu_die3.wav",
};
//=========================================================
// Classify - indicates this monster's place in the
// relationship table.
//=========================================================
int CEinarZombieBull::Classify()
{
return CLASS_ALIEN_MONSTER;
}
//=========================================================
// IgnoreConditions
//=========================================================
int CEinarZombieBull::IgnoreConditions()
{
int iIgnore = CBaseMonster::IgnoreConditions();
if( gpGlobals->time - m_flLastHurtTime <= 20 )
{
// haven't been hurt in 20 seconds, so let the squid care about stink.
iIgnore = bits_COND_SMELL | bits_COND_SMELL_FOOD;
}
return iIgnore;
}
//=========================================================
// IRelationship - overridden for bullsquid so that it can
// be made to ignore its love of headcrabs for a while.
//=========================================================
int CEinarZombieBull::IRelationship( CBaseEntity *pTarget )
{
return CBaseMonster::IRelationship( pTarget );
}
//=========================================================
// TakeDamage - overridden for bullsquid so we can keep track
// of how much time has passed since it was last injured
//=========================================================
int CEinarZombieBull::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType )
{
float flDist;
Vector vecApex;
// if the squid is running, has an enemy, was hurt by the enemy, hasn't been hurt in the last 3 seconds, and isn't too close to the enemy,
// it will swerve. (whew).
if( m_hEnemy != 0 && IsMoving() && pevAttacker == m_hEnemy->pev && gpGlobals->time - m_flLastHurtTime > 3 )
{
flDist = ( pev->origin - m_hEnemy->pev->origin ).Length2D();
if( flDist > SQUID_SPRINT_DIST )
{
flDist = ( pev->origin - m_Route[m_iRouteIndex].vecLocation ).Length2D();// reusing flDist.
if( FTriangulate( pev->origin, m_Route[m_iRouteIndex].vecLocation, flDist * 0.5f, m_hEnemy, &vecApex ) )
{
InsertWaypoint( vecApex, bits_MF_TO_DETOUR | bits_MF_DONT_SIMPLIFY );
}
}
}
m_flLastHurtTime = gpGlobals->time;
return CBaseMonster::TakeDamage( pevInflictor, pevAttacker, flDamage, bitsDamageType );
}
//=========================================================
// CheckMeleeAttack1 - bullsquid is a big guy, so has a longer
// melee range than most monsters. This is the tailwhip attack
//=========================================================
BOOL CEinarZombieBull::CheckMeleeAttack1( float flDot, float flDist )
{
if( m_hEnemy->pev->health <= gSkillData.bullsquidDmgWhip && flDist <= 110 && flDot >= 0.7f )
{
return TRUE;
}
return FALSE;
}
//=========================================================
// CheckMeleeAttack2 - bullsquid is a big guy, so has a longer
// melee range than most monsters. This is the bite attack.
// this attack will not be performed if the tailwhip attack
// is valid.
//=========================================================
BOOL CEinarZombieBull::CheckMeleeAttack2( float flDot, float flDist )
{
if( flDist <= 110 && flDot >= 0.7f && !HasConditions( bits_COND_CAN_MELEE_ATTACK1 ) )
{
return TRUE;
}
return FALSE;
}
//=========================================================
// IdleSound
//=========================================================
void CEinarZombieBull::IdleSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pIdleSounds ), 1, ATTN_NORM );
}
//=========================================================
// PainSound
//=========================================================
void CEinarZombieBull::PainSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pPainSounds ), 1, ATTN_NORM );
}
//=========================================================
// AlertSound
//=========================================================
void CEinarZombieBull::AlertSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pAlertSounds ), 1, ATTN_NORM );
}
//=========================================================
// SetYawSpeed - allows each sequence to have a different
// turn rate associated with it.
//=========================================================
void CEinarZombieBull::SetYawSpeed()
{
int ys;
ys = 95;
pev->yaw_speed = ys;
}
//=========================================================
// HandleAnimEvent - catches the monster-specific messages
// that occur when tagged animation frames are played.
//=========================================================
void CEinarZombieBull::HandleAnimEvent( MonsterEvent_t *pEvent )
{
switch( pEvent->event )
{
case ZBULL_AE_BITE:
{
// SOUND HERE!
CBaseEntity *pHurt = CheckTraceHullAttack( 110, gSkillData.bullsquidDmgBite, DMG_SLASH );
if( pHurt )
{
//pHurt->pev->punchangle.z = -15;
//pHurt->pev->punchangle.x = -45;
pHurt->pev->velocity = pHurt->pev->velocity - gpGlobals->v_forward * 50;
pHurt->pev->velocity = pHurt->pev->velocity + gpGlobals->v_up * 30;
}
}
break;
case ZBULL_AE_TAILWHIP:
{
CBaseEntity *pHurt = CheckTraceHullAttack( 110, gSkillData.bullsquidDmgWhip, DMG_CLUB | DMG_ALWAYSGIB );
if( pHurt )
{
// pHurt->pev->punchangle.z = -20;
// pHurt->pev->punchangle.x = 20;
pHurt->pev->velocity = pHurt->pev->velocity + gpGlobals->v_forward * 500;
pHurt->pev->velocity = pHurt->pev->velocity + gpGlobals->v_up * 30;
}
}
break;
case ZBULL_AE_THROW:
{
int iPitch;
// squid throws its prey IF the prey is a client.
CBaseEntity *pHurt = CheckTraceHullAttack( 110, 0, 0 );
if( pHurt )
{
// croonchy bite sound
iPitch = RANDOM_FLOAT( 90, 110 );
EMIT_SOUND_DYN( ENT( pev ), CHAN_WEAPON, RANDOM_SOUND_ARRAY( pBiteSounds ), 1, ATTN_NORM, 0, iPitch );
//pHurt->pev->punchangle.x = RANDOM_LONG( 0, 34 ) - 5;
//pHurt->pev->punchangle.z = RANDOM_LONG( 0, 49 ) - 25;
//pHurt->pev->punchangle.y = RANDOM_LONG( 0, 89 ) - 45;
// screeshake transforms the viewmodel as well as the viewangle. No problems with seeing the ends of the viewmodels.
UTIL_ScreenShake( pHurt->pev->origin, 25.0, 1.5, 0.7, 2 );
if( pHurt->IsPlayer() )
{
UTIL_MakeVectors( pev->angles );
pHurt->pev->velocity = pHurt->pev->velocity + gpGlobals->v_forward * 500 + gpGlobals->v_up * 30;
}
}
}
break;
default:
CBullsquid::HandleAnimEvent( pEvent );
}
}
//=========================================================
// Spawn
//=========================================================
void CEinarZombieBull::Spawn()
{
Precache();
SET_MODEL( ENT( pev ), "models/zombiebull.mdl" );
UTIL_SetSize( pev, Vector( -32, -32, 0 ), Vector( 32, 32, 64 ) );
pev->solid = SOLID_SLIDEBOX;
pev->movetype = MOVETYPE_STEP;
m_bloodColor = BLOOD_COLOR_RED;
pev->effects = 0;
pev->health = gSkillData.bullsquidHealth * 5;
m_flFieldOfView = 0.2;// indicates the width of this monster's forward view cone ( as a dotproduct result )
m_MonsterState = MONSTERSTATE_NONE;
m_fCanThreatDisplay = FALSE;
MonsterInit();
}
//=========================================================
// Precache - precaches all resources this monster needs
//=========================================================
void CEinarZombieBull::Precache()
{
PRECACHE_MODEL( "models/zombiebull.mdl" );
PRECACHE_SOUND( "zombie/claw_miss2.wav" );// because we use the basemonster SWIPE animation event
PRECACHE_SOUND_ARRAY( pAttackSounds );
PRECACHE_SOUND_ARRAY( pBiteSounds );
PRECACHE_SOUND_ARRAY( pIdleSounds );
PRECACHE_SOUND_ARRAY( pAlertSounds );
PRECACHE_SOUND_ARRAY( pPainSounds );
PRECACHE_SOUND_ARRAY( pDeathSounds );
PRECACHE_SOUND( "bull/bu_paw1.wav" );
PRECACHE_SOUND( "bull/bu_runstep1.wav" );
PRECACHE_SOUND( "bull/bu_runstep2.wav" );
PRECACHE_SOUND( "bull/bu_walkstep1.wav" );
PRECACHE_SOUND( "bull/bu_walkstep2.wav" );
}
//=========================================================
// DeathSound
//=========================================================
void CEinarZombieBull::DeathSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pDeathSounds ), 1, ATTN_NORM );
}
//=========================================================
// AttackSound
//=========================================================
void CEinarZombieBull::AttackSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pAttackSounds ), 1, ATTN_NORM );
}
//=========================================================
// BiteSound
//=========================================================
void CEinarZombieBull::BiteSound()
{
EMIT_SOUND( ENT( pev ), CHAN_VOICE, RANDOM_SOUND_ARRAY( pBiteSounds ), 1, ATTN_NORM );
}
//========================================================
// RunAI - overridden for bullsquid because there are things
// that need to be checked every think.
//========================================================
void CEinarZombieBull::RunAI()
{
// first, do base class stuff
CBaseMonster::RunAI();
if( m_hEnemy != 0 && m_Activity == ACT_RUN )
{
// chasing enemy. Sprint for last bit
if( ( pev->origin - m_hEnemy->pev->origin).Length2D() < SQUID_SPRINT_DIST )
{
pev->framerate = 1.25;
}
}
}
//=========================================================
// GetSchedule
//=========================================================
Schedule_t *CEinarZombieBull::GetSchedule()
{
switch( m_MonsterState )
{
case MONSTERSTATE_ALERT:
return CBaseMonster::GetSchedule();
case MONSTERSTATE_COMBAT:
{
// dead enemy
if( HasConditions( bits_COND_ENEMY_DEAD ) )
{
// call base class, all code to handle dead enemies is centralized there.
return CBaseMonster::GetSchedule();
}
if( HasConditions( bits_COND_NEW_ENEMY ) )
{
return GetScheduleOfType( SCHED_WAKE_ANGRY );
}
if( HasConditions( bits_COND_CAN_MELEE_ATTACK1 ) )
{
return GetScheduleOfType( SCHED_MELEE_ATTACK1 );
}
if( HasConditions( bits_COND_CAN_MELEE_ATTACK2 ) )
{
return GetScheduleOfType( SCHED_MELEE_ATTACK2 );
}
return GetScheduleOfType( SCHED_CHASE_ENEMY );
}
default:
break;
}
return CBaseMonster::GetSchedule();
}
//=========================================================
// GetScheduleOfType
//=========================================================
Schedule_t *CEinarZombieBull::GetScheduleOfType( int Type )
{
switch( Type )
{
case SCHED_RANGE_ATTACK1:
case SCHED_SQUID_SNIFF_AND_EAT:
return CBaseMonster::GetScheduleOfType( Type );
break;
}
return CBullsquid::GetScheduleOfType( Type );
}
//=========================================================
// Start task - selects the correct activity and performs
// any necessary calculations to start the next task on the
// schedule. OVERRIDDEN for bullsquid because it needs to
// know explicitly when the last attempt to chase the enemy
// failed, since that impacts its attack choices.
//=========================================================
void CEinarZombieBull::StartTask( Task_t *pTask )
{
m_iTaskStatus = TASKSTATUS_RUNNING;
switch( pTask->iTask )
{
case TASK_MELEE_ATTACK2:
{
AttackSound();
CBaseMonster::StartTask( pTask );
break;
}
default:
{
CBullsquid::StartTask( pTask );
break;
}
}
}