/*** * * 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. * ****/ /* ===== turret.cpp ======================================================== */ // // TODO: // Take advantage of new monster fields like m_hEnemy and get rid of that OFFSET() stuff // Revisit enemy validation stuff, maybe it's not necessary with the newest monster code // #include "extdll.h" #include "util.h" #include "cbase.h" #include "monsters.h" #include "weapons.h" #include "effects.h" extern Vector VecBModelOrigin( entvars_t* pevBModel ); #define TURRET_SHOTS 2 #define TURRET_RANGE (100 * 12) #define TURRET_SPREAD Vector( 0, 0, 0 ) #define TURRET_TURNRATE 30 //angles per 0.1 second #define TURRET_MAXWAIT 15 // seconds turret will stay active w/o a target #define TURRET_MAXSPIN 5 // seconds turret barrel will spin w/o a target #define TURRET_MACHINE_VOLUME 0.5 typedef enum { TURRET_ANIM_NONE = 0, TURRET_ANIM_FIRE, TURRET_ANIM_SPIN, TURRET_ANIM_DEPLOY, TURRET_ANIM_RETIRE, TURRET_ANIM_DIE } TURRET_ANIM; class CBaseTurret : public CBaseMonster { public: void Spawn( void ); virtual void Precache( void ); void UpdateOnRemove(); void KeyValue( KeyValueData *pkvd ); void EXPORT TurretUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value ); virtual void TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType ); virtual int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ); virtual int Classify( void ); int BloodColor( void ) { return DONT_BLEED; } void GibMonster( void ) {} // UNDONE: Throw turret gibs? // Think functions void EXPORT ActiveThink(void); void EXPORT SearchThink(void); void EXPORT AutoSearchThink(void); void EXPORT TurretDeath(void); virtual void EXPORT SpinDownCall(void) { m_iSpin = 0; } virtual void EXPORT SpinUpCall(void) { m_iSpin = 1; } // void SpinDown( void ); // float EXPORT SpinDownCall( void ) { return SpinDown(); } // virtual float SpinDown( void ) { return 0;} // virtual float Retire( void ) { return 0;} void EXPORT Deploy( void ); void EXPORT Retire( void ); void EXPORT Initialize( void ); virtual void Ping( void ); virtual void EyeOn( void ); virtual void EyeOff( void ); virtual int Save( CSave &save ); virtual int Restore( CRestore &restore ); static TYPEDESCRIPTION m_SaveData[]; // other functions void SetTurretAnim( TURRET_ANIM anim ); int MoveTurret( void ); virtual void Shoot( Vector &vecSrc, Vector &vecDirToEnemy ) { }; float m_flMaxSpin; // Max time to spin the barrel w/o a target int m_iSpin; CSprite *m_pEyeGlow; int m_eyeBrightness; int m_iDeployHeight; int m_iRetractHeight; int m_iMinPitch; int m_iBaseTurnRate; // angles per second float m_fTurnRate; // actual turn rate int m_iOrientation; // 0 = floor, 1 = Ceiling int m_iOn; int m_fBeserk; // Sometimes this bitch will just freak out int m_iAutoStart; // true if the turret auto deploys when a target // enters its range Vector m_vecLastSight; float m_flLastSight; // Last time we saw a target float m_flMaxWait; // Max time to seach w/o a target int m_iSearchSpeed; // Not Used! // movement float m_flStartYaw; Vector m_vecCurAngles; Vector m_vecGoalAngles; float m_flPingTime; // Time until the next ping, used when searching float m_flSpinUpTime; // Amount of time until the barrel should spin down when searching }; TYPEDESCRIPTION CBaseTurret::m_SaveData[] = { DEFINE_FIELD( CBaseTurret, m_flMaxSpin, FIELD_FLOAT ), DEFINE_FIELD( CBaseTurret, m_iSpin, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_pEyeGlow, FIELD_CLASSPTR ), DEFINE_FIELD( CBaseTurret, m_eyeBrightness, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iDeployHeight, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iRetractHeight, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iMinPitch, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iBaseTurnRate, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_fTurnRate, FIELD_FLOAT ), DEFINE_FIELD( CBaseTurret, m_iOrientation, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iOn, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_fBeserk, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_iAutoStart, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_vecLastSight, FIELD_POSITION_VECTOR ), DEFINE_FIELD( CBaseTurret, m_flLastSight, FIELD_TIME ), DEFINE_FIELD( CBaseTurret, m_flMaxWait, FIELD_FLOAT ), DEFINE_FIELD( CBaseTurret, m_iSearchSpeed, FIELD_INTEGER ), DEFINE_FIELD( CBaseTurret, m_flStartYaw, FIELD_FLOAT ), DEFINE_FIELD( CBaseTurret, m_vecCurAngles, FIELD_VECTOR ), DEFINE_FIELD( CBaseTurret, m_vecGoalAngles, FIELD_VECTOR ), DEFINE_FIELD( CBaseTurret, m_flPingTime, FIELD_TIME ), DEFINE_FIELD( CBaseTurret, m_flSpinUpTime, FIELD_TIME ), }; IMPLEMENT_SAVERESTORE( CBaseTurret, CBaseMonster ) class CTurret : public CBaseTurret { public: void Spawn( void ); void Precache( void ); // Think functions void SpinUpCall( void ); void SpinDownCall( void ); virtual int Save( CSave &save ); virtual int Restore( CRestore &restore ); static TYPEDESCRIPTION m_SaveData[]; // other functions void Shoot( Vector &vecSrc, Vector &vecDirToEnemy ); private: int m_iStartSpin; }; TYPEDESCRIPTION CTurret::m_SaveData[] = { DEFINE_FIELD( CTurret, m_iStartSpin, FIELD_INTEGER ), }; IMPLEMENT_SAVERESTORE( CTurret, CBaseTurret ) class CMiniTurret : public CBaseTurret { public: void Spawn(); void Precache( void ); // other functions void Shoot( Vector &vecSrc, Vector &vecDirToEnemy ); }; LINK_ENTITY_TO_CLASS( monster_turret, CTurret ) LINK_ENTITY_TO_CLASS( monster_miniturret, CMiniTurret ) void CBaseTurret::KeyValue( KeyValueData *pkvd ) { if( FStrEq( pkvd->szKeyName, "maxsleep" ) ) { m_flMaxWait = atof( pkvd->szValue ); pkvd->fHandled = TRUE; } else if( FStrEq( pkvd->szKeyName, "orientation" ) ) { m_iOrientation = atoi( pkvd->szValue ); pkvd->fHandled = TRUE; } else if( FStrEq( pkvd->szKeyName, "searchspeed" ) ) { m_iSearchSpeed = atoi( pkvd->szValue ); pkvd->fHandled = TRUE; } else if( FStrEq( pkvd->szKeyName, "turnrate" ) ) { m_iBaseTurnRate = atoi( pkvd->szValue ); pkvd->fHandled = TRUE; } else if( FStrEq( pkvd->szKeyName, "style" ) || FStrEq( pkvd->szKeyName, "height" ) || FStrEq( pkvd->szKeyName, "value1" ) || FStrEq( pkvd->szKeyName, "value2" ) || FStrEq( pkvd->szKeyName, "value3" ) ) pkvd->fHandled = TRUE; else CBaseMonster::KeyValue( pkvd ); } void CBaseTurret::Spawn() { Precache(); pev->nextthink = gpGlobals->time + 1; pev->movetype = MOVETYPE_FLY; pev->sequence = 0; pev->frame = 0; pev->solid = SOLID_SLIDEBOX; pev->takedamage = DAMAGE_AIM; SetBits( pev->flags, FL_MONSTER ); SetUse( &CBaseTurret::TurretUse ); if( ( pev->spawnflags & SF_MONSTER_TURRET_AUTOACTIVATE ) && !( pev->spawnflags & SF_MONSTER_TURRET_STARTINACTIVE ) ) { m_iAutoStart = TRUE; } ResetSequenceInfo(); SetBoneController( 0, 0 ); SetBoneController( 1, 0 ); m_flFieldOfView = VIEW_FIELD_FULL; // m_flSightRange = TURRET_RANGE; } void CBaseTurret::Precache() { PRECACHE_SOUND( "turret/tu_fire1.wav" ); PRECACHE_SOUND( "turret/tu_ping.wav" ); PRECACHE_SOUND( "turret/tu_active2.wav" ); PRECACHE_SOUND( "turret/tu_die.wav" ); PRECACHE_SOUND( "turret/tu_die2.wav" ); PRECACHE_SOUND( "turret/tu_die3.wav" ); // PRECACHE_SOUND( "turret/tu_retract.wav" ); // just use deploy sound to save memory PRECACHE_SOUND( "turret/tu_deploy.wav" ); PRECACHE_SOUND( "turret/tu_spinup.wav" ); PRECACHE_SOUND( "turret/tu_spindown.wav" ); PRECACHE_SOUND( "turret/tu_search.wav" ); PRECACHE_SOUND( "turret/tu_alert.wav" ); } void CBaseTurret::UpdateOnRemove() { CBaseEntity::UpdateOnRemove(); if( m_pEyeGlow ) { UTIL_Remove( m_pEyeGlow ); m_pEyeGlow = 0; } } #define TURRET_GLOW_SPRITE "sprites/flare3.spr" void CTurret::Spawn() { Precache(); SET_MODEL( ENT( pev ), "models/turret.mdl" ); pev->health = gSkillData.turretHealth; m_HackedGunPos = Vector( 0, 0, 12.75 ); m_flMaxSpin = TURRET_MAXSPIN; pev->view_ofs.z = 12.75; CBaseTurret::Spawn(); m_iRetractHeight = 16; m_iDeployHeight = 32; m_iMinPitch = -15; UTIL_SetSize( pev, Vector( -32, -32, -m_iRetractHeight ), Vector( 32, 32, m_iRetractHeight ) ); SetThink( &CBaseTurret::Initialize ); m_pEyeGlow = CSprite::SpriteCreate( TURRET_GLOW_SPRITE, pev->origin, FALSE ); m_pEyeGlow->SetTransparency( kRenderGlow, 255, 0, 0, 0, kRenderFxNoDissipation ); m_pEyeGlow->SetAttachment( edict(), 2 ); m_eyeBrightness = 0; pev->nextthink = gpGlobals->time + 0.3; } void CTurret::Precache() { CBaseTurret::Precache(); PRECACHE_MODEL( "models/turret.mdl" ); PRECACHE_MODEL( TURRET_GLOW_SPRITE ); } void CMiniTurret::Spawn() { Precache(); SET_MODEL( ENT( pev ), "models/miniturret.mdl" ); pev->health = gSkillData.miniturretHealth; m_HackedGunPos = Vector( 0, 0, 12.75 ); m_flMaxSpin = 0; pev->view_ofs.z = 12.75; CBaseTurret::Spawn(); m_iRetractHeight = 16; m_iDeployHeight = 32; m_iMinPitch = -15; UTIL_SetSize( pev, Vector( -16, -16, -m_iRetractHeight ), Vector( 16, 16, m_iRetractHeight ) ); SetThink( &CBaseTurret::Initialize ); pev->nextthink = gpGlobals->time + 0.3; } void CMiniTurret::Precache() { CBaseTurret::Precache(); PRECACHE_MODEL( "models/miniturret.mdl" ); PRECACHE_SOUND( "weapons/hks1.wav" ); PRECACHE_SOUND( "weapons/hks2.wav" ); PRECACHE_SOUND( "weapons/hks3.wav" ); } void CBaseTurret::Initialize( void ) { m_iOn = 0; m_fBeserk = 0; m_iSpin = 0; SetBoneController( 0, 0 ); SetBoneController( 1, 0 ); if( m_iBaseTurnRate == 0 ) m_iBaseTurnRate = TURRET_TURNRATE; if( m_flMaxWait == 0 ) m_flMaxWait = TURRET_MAXWAIT; m_flStartYaw = pev->angles.y; if( m_iOrientation == 1 ) { pev->idealpitch = 180; pev->angles.x = 180; pev->view_ofs.z = -pev->view_ofs.z; pev->effects |= EF_INVLIGHT; pev->angles.y = pev->angles.y + 180; if( pev->angles.y > 360 ) pev->angles.y = pev->angles.y - 360; } m_vecGoalAngles.x = 0; if( m_iAutoStart ) { m_flLastSight = gpGlobals->time + m_flMaxWait; SetThink( &CBaseTurret::AutoSearchThink ); pev->nextthink = gpGlobals->time + .1; } else SetThink( &CBaseEntity::SUB_DoNothing ); } void CBaseTurret::TurretUse( CBaseEntity *pActivator, CBaseEntity *pCaller, USE_TYPE useType, float value ) { if( !ShouldToggle( useType, m_iOn ) ) return; if( m_iOn ) { m_hEnemy = NULL; pev->nextthink = gpGlobals->time + 0.1; m_iAutoStart = FALSE;// switching off a turret disables autostart //!!!! this should spin down first!!BUGBUG SetThink( &CBaseTurret::Retire ); } else { pev->nextthink = gpGlobals->time + 0.1; // turn on delay // if the turret is flagged as an autoactivate turret, re-enable it's ability open self. if( pev->spawnflags & SF_MONSTER_TURRET_AUTOACTIVATE ) { m_iAutoStart = TRUE; } SetThink( &CBaseTurret::Deploy ); } } void CBaseTurret::Ping( void ) { // make the pinging noise every second while searching if( m_flPingTime == 0 ) m_flPingTime = gpGlobals->time + 1; else if( m_flPingTime <= gpGlobals->time ) { m_flPingTime = gpGlobals->time + 1; EMIT_SOUND( ENT( pev ), CHAN_ITEM, "turret/tu_ping.wav", 1, ATTN_NORM ); EyeOn(); } else if( m_eyeBrightness > 0 ) { EyeOff(); } } void CBaseTurret::EyeOn() { if( m_pEyeGlow ) { if( m_eyeBrightness != 255 ) { m_eyeBrightness = 255; } m_pEyeGlow->SetBrightness( m_eyeBrightness ); } } void CBaseTurret::EyeOff() { if( m_pEyeGlow ) { if( m_eyeBrightness > 0 ) { m_eyeBrightness = Q_max( 0, m_eyeBrightness - 30 ); m_pEyeGlow->SetBrightness( m_eyeBrightness ); } } } void CBaseTurret::ActiveThink( void ) { int fAttack = 0; Vector vecDirToEnemy; pev->nextthink = gpGlobals->time + 0.1; StudioFrameAdvance(); GlowShellUpdate(); if( ( !m_iOn ) || ( m_hEnemy == 0 ) ) { m_hEnemy = NULL; m_flLastSight = gpGlobals->time + m_flMaxWait; SetThink( &CBaseTurret::SearchThink ); return; } // if it's dead, look for something new if( !m_hEnemy->IsAlive() ) { if( !m_flLastSight ) { m_flLastSight = gpGlobals->time + 0.5; // continue-shooting timeout } else { if( gpGlobals->time > m_flLastSight ) { m_hEnemy = NULL; m_flLastSight = gpGlobals->time + m_flMaxWait; SetThink( &CBaseTurret::SearchThink ); return; } } } Vector vecMid = pev->origin + pev->view_ofs; Vector vecMidEnemy = m_hEnemy->BodyTarget( vecMid ); // Look for our current enemy int fEnemyVisible = FBoxVisible( pev, m_hEnemy->pev, vecMidEnemy ); vecDirToEnemy = vecMidEnemy - vecMid; // calculate dir and dist to enemy float flDistToEnemy = vecDirToEnemy.Length(); Vector vec = UTIL_VecToAngles( vecMidEnemy - vecMid ); // Current enmey is not visible. if( !fEnemyVisible || ( flDistToEnemy > TURRET_RANGE ) ) { if( !m_flLastSight ) m_flLastSight = gpGlobals->time + 0.5; else { // Should we look for a new target? if( gpGlobals->time > m_flLastSight ) { m_hEnemy = NULL; m_flLastSight = gpGlobals->time + m_flMaxWait; SetThink( &CBaseTurret::SearchThink ); return; } } fEnemyVisible = 0; } else { m_vecLastSight = vecMidEnemy; } UTIL_MakeAimVectors(m_vecCurAngles); /* ALERT( at_console, "%.0f %.0f : %.2f %.2f %.2f\n", m_vecCurAngles.x, m_vecCurAngles.y, gpGlobals->v_forward.x, gpGlobals->v_forward.y, gpGlobals->v_forward.z ); */ Vector vecLOS = vecDirToEnemy; //vecMid - m_vecLastSight; vecLOS = vecLOS.Normalize(); // Is the Gun looking at the target if( DotProduct( vecLOS, gpGlobals->v_forward ) <= 0.866 ) // 30 degree slop fAttack = FALSE; else fAttack = TRUE; // fire the gun if( m_iSpin && ( ( fAttack ) || ( m_fBeserk ) ) ) { Vector vecSrc, vecAng; GetAttachment( 0, vecSrc, vecAng ); SetTurretAnim( TURRET_ANIM_FIRE ); Shoot( vecSrc, gpGlobals->v_forward ); } else { SetTurretAnim( TURRET_ANIM_SPIN ); } //move the gun if( m_fBeserk ) { if( RANDOM_LONG( 0, 9 ) == 0 ) { m_vecGoalAngles.y = RANDOM_FLOAT( 0, 360 ); m_vecGoalAngles.x = RANDOM_FLOAT( 0, 90 ) - 90 * m_iOrientation; TakeDamage( pev, pev, 1, DMG_GENERIC ); // don't beserk forever return; } } else if( fEnemyVisible ) { if( vec.y > 360 ) vec.y -= 360; if( vec.y < 0 ) vec.y += 360; //ALERT( at_console, "[%.2f]", vec.x ); if( vec.x < -180 ) vec.x += 360; if( vec.x > 180 ) vec.x -= 360; // now all numbers should be in [1...360] // pin to turret limitations to [-90...15] if( m_iOrientation == 0 ) { if( vec.x > 90 ) vec.x = 90; else if( vec.x < m_iMinPitch ) vec.x = m_iMinPitch; } else { if( vec.x < -90 ) vec.x = -90; else if( vec.x > -m_iMinPitch ) vec.x = -m_iMinPitch; } // ALERT( at_console, "->[%.2f]\n", vec.x ); m_vecGoalAngles.y = vec.y; m_vecGoalAngles.x = vec.x; } SpinUpCall(); MoveTurret(); } void CTurret::Shoot( Vector &vecSrc, Vector &vecDirToEnemy ) { FireBullets( 1, vecSrc, vecDirToEnemy, TURRET_SPREAD, TURRET_RANGE, BULLET_MONSTER_12MM, 1 ); EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "turret/tu_fire1.wav", 1, 0.6 ); pev->effects = pev->effects | EF_MUZZLEFLASH; } void CMiniTurret::Shoot( Vector &vecSrc, Vector &vecDirToEnemy ) { FireBullets( 1, vecSrc, vecDirToEnemy, TURRET_SPREAD, TURRET_RANGE, BULLET_MONSTER_9MM, 1 ); switch( RANDOM_LONG( 0, 2 ) ) { case 0: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks1.wav", 1, ATTN_NORM ); break; case 1: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks2.wav", 1, ATTN_NORM ); break; case 2: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks3.wav", 1, ATTN_NORM ); break; } pev->effects = pev->effects | EF_MUZZLEFLASH; } void CBaseTurret::Deploy( void ) { pev->nextthink = gpGlobals->time + 0.1; StudioFrameAdvance(); GlowShellUpdate(); if( pev->sequence != TURRET_ANIM_DEPLOY ) { m_iOn = 1; SetTurretAnim( TURRET_ANIM_DEPLOY ); EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_deploy.wav", TURRET_MACHINE_VOLUME, ATTN_NORM ); SUB_UseTargets( this, USE_ON, 0 ); } if( m_fSequenceFinished ) { pev->maxs.z = m_iDeployHeight; pev->mins.z = -m_iDeployHeight; UTIL_SetSize( pev, pev->mins, pev->maxs ); m_vecCurAngles.x = 0; if( m_iOrientation == 1 ) { m_vecCurAngles.y = UTIL_AngleMod( pev->angles.y + 180 ); } else { m_vecCurAngles.y = UTIL_AngleMod( pev->angles.y ); } SetTurretAnim( TURRET_ANIM_SPIN ); pev->framerate = 0; SetThink( &CBaseTurret::SearchThink ); } m_flLastSight = gpGlobals->time + m_flMaxWait; } void CBaseTurret::Retire( void ) { // make the turret level m_vecGoalAngles.x = 0; m_vecGoalAngles.y = m_flStartYaw; pev->nextthink = gpGlobals->time + 0.1; StudioFrameAdvance(); GlowShellUpdate(); EyeOff(); if( !MoveTurret() ) { if( m_iSpin ) { SpinDownCall(); } else if( pev->sequence != TURRET_ANIM_RETIRE ) { SetTurretAnim( TURRET_ANIM_RETIRE ); EMIT_SOUND_DYN( ENT( pev ), CHAN_BODY, "turret/tu_deploy.wav", TURRET_MACHINE_VOLUME, ATTN_NORM, 0, 120 ); SUB_UseTargets( this, USE_OFF, 0 ); } else if( m_fSequenceFinished ) { m_iOn = 0; m_flLastSight = 0; SetTurretAnim( TURRET_ANIM_NONE ); pev->maxs.z = m_iRetractHeight; pev->mins.z = -m_iRetractHeight; UTIL_SetSize( pev, pev->mins, pev->maxs ); if( m_iAutoStart ) { SetThink( &CBaseTurret::AutoSearchThink ); pev->nextthink = gpGlobals->time + .1; } else SetThink( &CBaseEntity::SUB_DoNothing ); } } else { SetTurretAnim( TURRET_ANIM_SPIN ); } } void CTurret::SpinUpCall( void ) { StudioFrameAdvance(); pev->nextthink = gpGlobals->time + 0.1; // Are we already spun up? If not start the two stage process. if( !m_iSpin ) { SetTurretAnim( TURRET_ANIM_SPIN ); // for the first pass, spin up the the barrel if( !m_iStartSpin ) { pev->nextthink = gpGlobals->time + 1.0; // spinup delay EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_spinup.wav", TURRET_MACHINE_VOLUME, ATTN_NORM ); m_iStartSpin = 1; pev->framerate = 0.1; } // after the barrel is spun up, turn on the hum else if( pev->framerate >= 1.0 ) { pev->nextthink = gpGlobals->time + 0.1; // retarget delay EMIT_SOUND( ENT( pev ), CHAN_STATIC, "turret/tu_active2.wav", TURRET_MACHINE_VOLUME, ATTN_NORM ); SetThink( &CBaseTurret::ActiveThink ); m_iStartSpin = 0; m_iSpin = 1; } else { pev->framerate += 0.075; } } if( m_iSpin ) { SetThink( &CBaseTurret::ActiveThink ); } } void CTurret::SpinDownCall( void ) { if( m_iSpin ) { SetTurretAnim( TURRET_ANIM_SPIN ); if( pev->framerate == 1.0 ) { EMIT_SOUND_DYN( ENT( pev ), CHAN_STATIC, "turret/tu_active2.wav", 0, 0, SND_STOP, 100 ); EMIT_SOUND( ENT( pev ), CHAN_ITEM, "turret/tu_spindown.wav", TURRET_MACHINE_VOLUME, ATTN_NORM ); } pev->framerate -= 0.02; if( pev->framerate <= 0 ) { pev->framerate = 0; m_iSpin = 0; } } } void CBaseTurret::SetTurretAnim( TURRET_ANIM anim ) { if( pev->sequence != anim ) { switch( anim ) { case TURRET_ANIM_FIRE: case TURRET_ANIM_SPIN: if( pev->sequence != TURRET_ANIM_FIRE && pev->sequence != TURRET_ANIM_SPIN ) { pev->frame = 0; } break; default: pev->frame = 0; break; } pev->sequence = anim; ResetSequenceInfo(); switch( anim ) { case TURRET_ANIM_RETIRE: pev->frame = 255; pev->framerate = -1.0; break; case TURRET_ANIM_DIE: pev->framerate = 1.0; break; default: break; } //ALERT( at_console, "Turret anim #%d\n", anim ); } } // // This search function will sit with the turret deployed and look for a new target. // After a set amount of time, the barrel will spin down. After m_flMaxWait, the turret will // retact. // void CBaseTurret::SearchThink( void ) { // ensure rethink SetTurretAnim( TURRET_ANIM_SPIN ); StudioFrameAdvance(); pev->nextthink = gpGlobals->time + 0.1; GlowShellUpdate(); if( m_flSpinUpTime == 0 && m_flMaxSpin ) m_flSpinUpTime = gpGlobals->time + m_flMaxSpin; Ping(); // If we have a target and we're still healthy if( m_hEnemy != 0 ) { if( !m_hEnemy->IsAlive() ) m_hEnemy = NULL;// Dead enemy forces a search for new one } // Acquire Target if( m_hEnemy == 0 ) { Look( TURRET_RANGE ); m_hEnemy = BestVisibleEnemy(); } // If we've found a target, spin up the barrel and start to attack if( m_hEnemy != 0 ) { m_flLastSight = 0; m_flSpinUpTime = 0; SetThink( &CBaseTurret::ActiveThink ); } else { // Are we out of time, do we need to retract? if( gpGlobals->time > m_flLastSight ) { //Before we retrace, make sure that we are spun down. m_flLastSight = 0; m_flSpinUpTime = 0; SetThink( &CBaseTurret::Retire ); } // should we stop the spin? else if( ( m_flSpinUpTime ) && ( gpGlobals->time > m_flSpinUpTime ) ) { SpinDownCall(); } // generic hunt for new victims m_vecGoalAngles.y = ( m_vecGoalAngles.y + 0.1 * m_fTurnRate ); if( m_vecGoalAngles.y >= 360 ) m_vecGoalAngles.y -= 360; MoveTurret(); } } // // This think function will deploy the turret when something comes into range. This is for // automatically activated turrets. // void CBaseTurret::AutoSearchThink( void ) { // ensure rethink StudioFrameAdvance(); pev->nextthink = gpGlobals->time + 0.3; GlowShellUpdate(); // If we have a target and we're still healthy if( m_hEnemy != 0 ) { if( !m_hEnemy->IsAlive() ) m_hEnemy = NULL;// Dead enemy forces a search for new one } // Acquire Target if( m_hEnemy == 0 ) { Look( TURRET_RANGE ); m_hEnemy = BestVisibleEnemy(); } if( m_hEnemy != 0 ) { SetThink( &CBaseTurret::Deploy ); EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_alert.wav", TURRET_MACHINE_VOLUME, ATTN_NORM ); } } void CBaseTurret::TurretDeath( void ) { //BOOL iActive = FALSE; StudioFrameAdvance(); pev->nextthink = gpGlobals->time + 0.1; GlowShellUpdate(); if( pev->deadflag != DEAD_DEAD ) { pev->deadflag = DEAD_DEAD; float flRndSound = RANDOM_FLOAT( 0, 1 ); if( flRndSound <= 0.33 ) EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die.wav", 1.0, ATTN_NORM ); else if( flRndSound <= 0.66 ) EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die2.wav", 1.0, ATTN_NORM ); else EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die3.wav", 1.0, ATTN_NORM ); EMIT_SOUND_DYN( ENT( pev ), CHAN_STATIC, "turret/tu_active2.wav", 0, 0, SND_STOP, 100 ); if( m_iOrientation == 0 ) m_vecGoalAngles.x = -15; else m_vecGoalAngles.x = -90; SetTurretAnim( TURRET_ANIM_DIE ); EyeOn(); } EyeOff(); if( pev->dmgtime + RANDOM_FLOAT( 0, 2 ) > gpGlobals->time ) { // lots of smoke MESSAGE_BEGIN( MSG_BROADCAST, SVC_TEMPENTITY ); WRITE_BYTE( TE_SMOKE ); WRITE_COORD( RANDOM_FLOAT( pev->absmin.x, pev->absmax.x ) ); WRITE_COORD( RANDOM_FLOAT( pev->absmin.y, pev->absmax.y ) ); WRITE_COORD( pev->origin.z - m_iOrientation * 64 ); WRITE_SHORT( g_sModelIndexSmoke ); WRITE_BYTE( 25 ); // scale * 10 WRITE_BYTE( 10 - m_iOrientation * 5 ); // framerate MESSAGE_END(); } if( pev->dmgtime + RANDOM_FLOAT( 0, 5 ) > gpGlobals->time ) { Vector vecSrc = Vector( RANDOM_FLOAT( pev->absmin.x, pev->absmax.x ), RANDOM_FLOAT( pev->absmin.y, pev->absmax.y ), 0 ); if( m_iOrientation == 0 ) vecSrc = vecSrc + Vector( 0, 0, RANDOM_FLOAT( pev->origin.z, pev->absmax.z ) ); else vecSrc = vecSrc + Vector( 0, 0, RANDOM_FLOAT( pev->absmin.z, pev->origin.z ) ); UTIL_Sparks( vecSrc ); } if( m_fSequenceFinished && !MoveTurret() && pev->dmgtime + 5 < gpGlobals->time ) { pev->framerate = 0; SetThink( NULL ); } } void CBaseTurret::TraceAttack( entvars_t *pevAttacker, float flDamage, Vector vecDir, TraceResult *ptr, int bitsDamageType ) { if( ptr->iHitgroup == 10 ) { // hit armor if( pev->dmgtime != gpGlobals->time || (RANDOM_LONG( 0, 10 ) < 1 ) ) { UTIL_Ricochet( ptr->vecEndPos, RANDOM_FLOAT( 1, 2 ) ); pev->dmgtime = gpGlobals->time; } flDamage = 0.1;// don't hurt the monster much, but allow bits_COND_LIGHT_DAMAGE to be generated } if( !pev->takedamage ) return; AddMultiDamage( pevAttacker, this, flDamage, bitsDamageType ); } // take damage. bitsDamageType indicates type of damage sustained, ie: DMG_BULLET int CBaseTurret::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ) { if( !pev->takedamage ) return 0; if( !m_iOn ) flDamage /= 10.0; pev->health -= flDamage; if( pev->health <= 0 ) { pev->health = 0; pev->takedamage = DAMAGE_NO; pev->dmgtime = gpGlobals->time; ClearBits( pev->flags, FL_MONSTER ); // why are they set in the first place??? SetUse( NULL ); SetThink( &CBaseTurret::TurretDeath ); SUB_UseTargets( this, USE_ON, 0 ); // wake up others pev->nextthink = gpGlobals->time + 0.1; return 0; } if( pev->health <= 10 ) { if( m_iOn && ( 1 || RANDOM_LONG( 0, 0x7FFF ) > 800 ) ) { m_fBeserk = 1; SetThink( &CBaseTurret::SearchThink ); } } return 1; } int CBaseTurret::MoveTurret( void ) { int state = 0; // any x movement? if( m_vecCurAngles.x != m_vecGoalAngles.x ) { float flDir = m_vecGoalAngles.x > m_vecCurAngles.x ? 1 : -1 ; m_vecCurAngles.x += 0.1 * m_fTurnRate * flDir; // if we started below the goal, and now we're past, peg to goal if( flDir == 1 ) { if( m_vecCurAngles.x > m_vecGoalAngles.x ) m_vecCurAngles.x = m_vecGoalAngles.x; } else { if( m_vecCurAngles.x < m_vecGoalAngles.x ) m_vecCurAngles.x = m_vecGoalAngles.x; } if( m_iOrientation == 0 ) SetBoneController( 1, -m_vecCurAngles.x ); else SetBoneController( 1, m_vecCurAngles.x ); state = 1; } if( m_vecCurAngles.y != m_vecGoalAngles.y ) { float flDir = m_vecGoalAngles.y > m_vecCurAngles.y ? 1 : -1 ; float flDist = fabs( m_vecGoalAngles.y - m_vecCurAngles.y ); if( flDist > 180 ) { flDist = 360 - flDist; flDir = -flDir; } if( flDist > 30 ) { if( m_fTurnRate < m_iBaseTurnRate * 10 ) { m_fTurnRate += m_iBaseTurnRate; } } else if( m_fTurnRate > 45 ) { m_fTurnRate -= m_iBaseTurnRate; } else { m_fTurnRate += m_iBaseTurnRate; } m_vecCurAngles.y += 0.1 * m_fTurnRate * flDir; if( m_vecCurAngles.y < 0 ) m_vecCurAngles.y += 360; else if( m_vecCurAngles.y >= 360 ) m_vecCurAngles.y -= 360; if( flDist < ( 0.05 * m_iBaseTurnRate ) ) m_vecCurAngles.y = m_vecGoalAngles.y; //ALERT( at_console, "%.2f -> %.2f\n", m_vecCurAngles.y, y ); if( m_iOrientation == 0 ) SetBoneController( 0, m_vecCurAngles.y - pev->angles.y ); else SetBoneController( 0, pev->angles.y - 180 - m_vecCurAngles.y ); state = 1; } if( !state ) m_fTurnRate = m_iBaseTurnRate; //ALERT( at_console, "(%.2f, %.2f)->(%.2f, %.2f)\n", m_vecCurAngles.x, // m_vecCurAngles.y, m_vecGoalAngles.x, m_vecGoalAngles.y ); return state; } // // ID as a machine // int CBaseTurret::Classify( void ) { if( m_iOn || m_iAutoStart ) return CLASS_MACHINE; return CLASS_NONE; } //========================================================= // Sentry gun - smallest turret, placed near grunt entrenchments //========================================================= class CSentry : public CBaseTurret { public: void Spawn(); void Precache( void ); // other functions void Shoot( Vector &vecSrc, Vector &vecDirToEnemy ); int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ); void EXPORT SentryTouch( CBaseEntity *pOther ); void EXPORT SentryDeath( void ); }; LINK_ENTITY_TO_CLASS( monster_sentry, CSentry ) void CSentry::Precache() { CBaseTurret::Precache(); PRECACHE_MODEL( "models/sentry.mdl" ); } void CSentry::Spawn() { Precache(); SET_MODEL( ENT( pev ), "models/sentry.mdl" ); pev->health = gSkillData.sentryHealth; m_HackedGunPos = Vector( 0, 0, 48 ); pev->view_ofs.z = 48; m_flMaxWait = 1E6; m_flMaxSpin = 1E6; CBaseTurret::Spawn(); m_iRetractHeight = 64; m_iDeployHeight = 64; m_iMinPitch = -60; UTIL_SetSize( pev, Vector( -16, -16, -m_iRetractHeight ), Vector( 16, 16, m_iRetractHeight ) ); SetTouch( &CSentry::SentryTouch ); SetThink( &CBaseTurret::Initialize ); pev->nextthink = gpGlobals->time + 0.3; } void CSentry::Shoot( Vector &vecSrc, Vector &vecDirToEnemy ) { FireBullets( 1, vecSrc, vecDirToEnemy, TURRET_SPREAD, TURRET_RANGE, BULLET_MONSTER_MP5, 1 ); switch( RANDOM_LONG( 0, 2 ) ) { case 0: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks1.wav", 1, ATTN_NORM ); break; case 1: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks2.wav", 1, ATTN_NORM ); break; case 2: EMIT_SOUND( ENT( pev ), CHAN_WEAPON, "weapons/hks3.wav", 1, ATTN_NORM ); break; } pev->effects = pev->effects | EF_MUZZLEFLASH; } int CSentry::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ) { if( !pev->takedamage ) return 0; if( !m_iOn ) { SetThink( &CBaseTurret::Deploy ); SetUse( NULL ); pev->nextthink = gpGlobals->time + 0.1; } pev->health -= flDamage; if( pev->health <= 0 ) { pev->health = 0; pev->takedamage = DAMAGE_NO; pev->dmgtime = gpGlobals->time; ClearBits( pev->flags, FL_MONSTER ); // why are they set in the first place??? SetUse( NULL ); SetThink( &CSentry::SentryDeath ); SUB_UseTargets( this, USE_ON, 0 ); // wake up others pev->nextthink = gpGlobals->time + 0.1; return 0; } return 1; } void CSentry::SentryTouch( CBaseEntity *pOther ) { if( pOther && ( pOther->IsPlayer() || ( pOther->pev->flags & FL_MONSTER ) ) ) { TakeDamage( pOther->pev, pOther->pev, 0, 0 ); } } void CSentry::SentryDeath( void ) { //BOOL iActive = FALSE; StudioFrameAdvance(); pev->nextthink = gpGlobals->time + 0.1; GlowShellUpdate(); if( pev->deadflag != DEAD_DEAD ) { pev->deadflag = DEAD_DEAD; float flRndSound = RANDOM_FLOAT( 0, 1 ); if( flRndSound <= 0.33 ) EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die.wav", 1.0, ATTN_NORM ); else if( flRndSound <= 0.66 ) EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die2.wav", 1.0, ATTN_NORM ); else EMIT_SOUND( ENT( pev ), CHAN_BODY, "turret/tu_die3.wav", 1.0, ATTN_NORM ); EMIT_SOUND_DYN( ENT( pev ), CHAN_STATIC, "turret/tu_active2.wav", 0, 0, SND_STOP, 100 ); SetBoneController( 0, 0 ); SetBoneController( 1, 0 ); SetTurretAnim( TURRET_ANIM_DIE ); pev->solid = SOLID_NOT; pev->angles.y = UTIL_AngleMod( pev->angles.y + RANDOM_LONG( 0, 2 ) * 120 ); EyeOn(); } EyeOff(); Vector vecSrc, vecAng; GetAttachment( 1, vecSrc, vecAng ); if( pev->dmgtime + RANDOM_FLOAT( 0, 2 ) > gpGlobals->time ) { // lots of smoke MESSAGE_BEGIN( MSG_BROADCAST, SVC_TEMPENTITY ); WRITE_BYTE( TE_SMOKE ); WRITE_COORD( vecSrc.x + RANDOM_FLOAT( -16, 16 ) ); WRITE_COORD( vecSrc.y + RANDOM_FLOAT( -16, 16 ) ); WRITE_COORD( vecSrc.z - 32 ); WRITE_SHORT( g_sModelIndexSmoke ); WRITE_BYTE( 15 ); // scale * 10 WRITE_BYTE( 8 ); // framerate MESSAGE_END(); } if( pev->dmgtime + RANDOM_FLOAT( 0, 8 ) > gpGlobals->time ) { UTIL_Sparks( vecSrc ); } if( m_fSequenceFinished && pev->dmgtime + 5 < gpGlobals->time ) { pev->framerate = 0; SetThink( NULL ); } }