/*** * * 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. * * Use, distribution, and modification of this source code and/or resulting * object code is restricted to non-commercial enhancements to products from * Valve LLC. All other use, distribution, or modification is prohibited * without written permission from Valve LLC. * ****/ #include "extdll.h" #include "util.h" #include "cbase.h" #include "monsters.h" #include "weapons.h" #include "nodes.h" #include "player.h" #include "effects.h" #include "gamerules.h" #define TRIPMINE_PRIMARY_VOLUME 450 enum tripmine_e { TRIPMINE_IDLE1 = 0, TRIPMINE_IDLE2, TRIPMINE_ARM1, TRIPMINE_ARM2, TRIPMINE_FIDGET, TRIPMINE_HOLSTER, TRIPMINE_DRAW, TRIPMINE_WORLD, TRIPMINE_GROUND }; #if !CLIENT_DLL class CTripmineGrenade : public CGrenade { void Spawn( void ); void Precache( void ); void UpdateOnRemove(); virtual int Save( CSave &save ); virtual int Restore( CRestore &restore ); static TYPEDESCRIPTION m_SaveData[]; int TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ); void EXPORT WarningThink( void ); void EXPORT PowerupThink( void ); void EXPORT BeamBreakThink( void ); void EXPORT DelayDeathThink( void ); void Killed( entvars_t *pevAttacker, int iGib ); void MakeBeam( void ); void KillBeam( void ); float m_flPowerUp; Vector m_vecDir; Vector m_vecEnd; float m_flBeamLength; EHANDLE m_hOwner; CBeam *m_pBeam; Vector m_posOwner; Vector m_angleOwner; edict_t *m_pRealOwner;// tracelines don't hit PEV->OWNER, which means a player couldn't detonate his own trip mine, so we store the owner here. }; LINK_ENTITY_TO_CLASS( monster_tripmine, CTripmineGrenade ) TYPEDESCRIPTION CTripmineGrenade::m_SaveData[] = { DEFINE_FIELD( CTripmineGrenade, m_flPowerUp, FIELD_TIME ), DEFINE_FIELD( CTripmineGrenade, m_vecDir, FIELD_VECTOR ), DEFINE_FIELD( CTripmineGrenade, m_vecEnd, FIELD_POSITION_VECTOR ), DEFINE_FIELD( CTripmineGrenade, m_flBeamLength, FIELD_FLOAT ), DEFINE_FIELD( CTripmineGrenade, m_hOwner, FIELD_EHANDLE ), #if !TRIPMINE_BEAM_DUPLICATION_FIX DEFINE_FIELD( CTripmineGrenade, m_pBeam, FIELD_CLASSPTR ), #endif DEFINE_FIELD( CTripmineGrenade, m_posOwner, FIELD_POSITION_VECTOR ), DEFINE_FIELD( CTripmineGrenade, m_angleOwner, FIELD_VECTOR ), DEFINE_FIELD( CTripmineGrenade, m_pRealOwner, FIELD_EDICT ), }; IMPLEMENT_SAVERESTORE( CTripmineGrenade, CGrenade ) void CTripmineGrenade::Spawn( void ) { Precache(); // motor pev->movetype = MOVETYPE_FLY; pev->solid = SOLID_NOT; SET_MODEL( ENT( pev ), "models/v_tripmine.mdl" ); pev->frame = 0; pev->body = 3; pev->sequence = TRIPMINE_WORLD; ResetSequenceInfo(); pev->framerate = 0; UTIL_SetSize( pev, Vector( -8.0f, -8.0f, -8.0f ), Vector( 8.0f, 8.0f, 8.0f ) ); UTIL_SetOrigin( pev, pev->origin ); if( pev->spawnflags & 1 ) { // power up quickly m_flPowerUp = gpGlobals->time + 1.0f; } else { // power up in 2.5 seconds m_flPowerUp = gpGlobals->time + 2.5f; } SetThink( &CTripmineGrenade::PowerupThink ); pev->nextthink = gpGlobals->time + 0.2f; pev->takedamage = DAMAGE_YES; pev->dmg = gSkillData.plrDmgTripmine; pev->health = 1; // don't let die normally if( pev->owner != NULL ) { // play deploy sound EMIT_SOUND( ENT( pev ), CHAN_VOICE, "weapons/mine_deploy.wav", 1.0, ATTN_NORM ); EMIT_SOUND( ENT( pev ), CHAN_BODY, "weapons/mine_charge.wav", 0.2, ATTN_NORM ); // chargeup m_pRealOwner = pev->owner;// see CTripmineGrenade for why. } UTIL_MakeAimVectors( pev->angles ); m_vecDir = gpGlobals->v_forward; m_vecEnd = pev->origin + m_vecDir * 2048.0f; } void CTripmineGrenade::Precache( void ) { PRECACHE_MODEL( "models/v_tripmine.mdl" ); PRECACHE_SOUND( "weapons/mine_deploy.wav" ); PRECACHE_SOUND( "weapons/mine_activate.wav" ); PRECACHE_SOUND( "weapons/mine_charge.wav" ); } void CTripmineGrenade::UpdateOnRemove() { CBaseEntity::UpdateOnRemove(); KillBeam(); } void CTripmineGrenade::WarningThink( void ) { // play warning sound // EMIT_SOUND( ENT( pev ), CHAN_VOICE, "buttons/Blip2.wav", 1.0, ATTN_NORM ); // set to power up SetThink( &CTripmineGrenade::PowerupThink ); pev->nextthink = gpGlobals->time + 1.0f; } void CTripmineGrenade::PowerupThink( void ) { TraceResult tr; if( m_hOwner == 0 ) { // find an owner edict_t *oldowner = pev->owner; pev->owner = NULL; UTIL_TraceLine( pev->origin + m_vecDir * 8.0f, pev->origin - m_vecDir * 32.0f, dont_ignore_monsters, ENT( pev ), &tr ); if( tr.fStartSolid || ( oldowner && tr.pHit == oldowner ) ) { pev->owner = oldowner; m_flPowerUp += 0.1f; pev->nextthink = gpGlobals->time + 0.1f; return; } if( tr.flFraction < 1.0f ) { pev->owner = tr.pHit; m_hOwner = CBaseEntity::Instance( pev->owner ); m_posOwner = m_hOwner->pev->origin; m_angleOwner = m_hOwner->pev->angles; } else { STOP_SOUND( ENT( pev ), CHAN_VOICE, "weapons/mine_deploy.wav" ); STOP_SOUND( ENT( pev ), CHAN_BODY, "weapons/mine_charge.wav" ); SetThink( &CBaseEntity::SUB_Remove ); pev->nextthink = gpGlobals->time + 0.1f; ALERT( at_console, "WARNING:Tripmine at %.0f, %.0f, %.0f removed\n", (double)pev->origin.x, (double)pev->origin.y, (double)pev->origin.z ); KillBeam(); return; } } else if( m_posOwner != m_hOwner->pev->origin || m_angleOwner != m_hOwner->pev->angles ) { // disable STOP_SOUND( ENT( pev ), CHAN_VOICE, "weapons/mine_deploy.wav" ); STOP_SOUND( ENT( pev ), CHAN_BODY, "weapons/mine_charge.wav" ); CBaseEntity *pMine = Create( "weapon_tripmine", pev->origin + m_vecDir * 24.0f, pev->angles ); pMine->pev->spawnflags |= SF_NORESPAWN; SetThink( &CBaseEntity::SUB_Remove ); KillBeam(); pev->nextthink = gpGlobals->time + 0.1f; return; } // ALERT( at_console, "%d %.0f %.0f %0.f\n", pev->owner, m_pOwner->pev->origin.x, m_pOwner->pev->origin.y, m_pOwner->pev->origin.z ); if( gpGlobals->time > m_flPowerUp ) { // make solid pev->solid = SOLID_BBOX; UTIL_SetOrigin( pev, pev->origin ); MakeBeam(); // play enabled sound EMIT_SOUND_DYN( ENT( pev ), CHAN_VOICE, "weapons/mine_activate.wav", 0.5, ATTN_NORM, 1, 75 ); } pev->nextthink = gpGlobals->time + 0.1f; } void CTripmineGrenade::KillBeam( void ) { if( m_pBeam ) { UTIL_Remove( m_pBeam ); m_pBeam = NULL; } } void CTripmineGrenade::MakeBeam( void ) { TraceResult tr; // ALERT( at_console, "serverflags %f\n", gpGlobals->serverflags ); UTIL_TraceLine( pev->origin, m_vecEnd, dont_ignore_monsters, ENT( pev ), &tr ); m_flBeamLength = tr.flFraction; // set to follow laser spot SetThink( &CTripmineGrenade::BeamBreakThink ); pev->nextthink = gpGlobals->time + 0.1f; Vector vecTmpEnd = pev->origin + m_vecDir * 2048.0f * m_flBeamLength; m_pBeam = CBeam::BeamCreate( g_pModelNameLaser, 10 ); #if TRIPMINE_BEAM_DUPLICATION_FIX m_pBeam->pev->spawnflags |= SF_BEAM_TEMPORARY; #endif m_pBeam->PointEntInit( vecTmpEnd, entindex() ); m_pBeam->SetColor( 255, 0, 0 ); // I made this red for the hell of it, make it go cycle rainbow -- GOAHEAD m_pBeam->SetScrollRate( 255 ); m_pBeam->SetBrightness( 64 ); } void CTripmineGrenade::BeamBreakThink( void ) { BOOL bBlowup = 0; TraceResult tr; // HACKHACK Set simple box using this really nice global! gpGlobals->trace_flags = FTRACE_SIMPLEBOX; UTIL_TraceLine( pev->origin, m_vecEnd, dont_ignore_monsters, ENT( pev ), &tr ); // ALERT( at_console, "%f : %f\n", tr.flFraction, m_flBeamLength ); // respawn detect. if( !m_pBeam ) { #if TRIPMINE_BEAM_DUPLICATION_FIX // Use the same trace parameters as the original trace above so the right entity is hit. TraceResult tr2; UTIL_TraceLine( pev->origin + m_vecDir * 8.0f, pev->origin - m_vecDir * 32.0f, dont_ignore_monsters, ENT( pev ), &tr2 ); #endif MakeBeam(); #if TRIPMINE_BEAM_DUPLICATION_FIX if( tr2.pHit ) { // reset owner too pev->owner = tr2.pHit; m_hOwner = CBaseEntity::Instance( tr2.pHit ); } #else if( tr.pHit ) m_hOwner = CBaseEntity::Instance( tr.pHit ); // reset owner too #endif } if( fabs( m_flBeamLength - tr.flFraction ) > 0.001f ) { bBlowup = 1; } else { if( m_hOwner == 0 ) bBlowup = 1; else if( m_posOwner != m_hOwner->pev->origin ) bBlowup = 1; else if( m_angleOwner != m_hOwner->pev->angles ) bBlowup = 1; } if( bBlowup ) { // a bit of a hack, but all CGrenade code passes pev->owner along to make sure the proper player gets credit for the kill // so we have to restore pev->owner from pRealOwner, because an entity's tracelines don't strike it's pev->owner which meant // that a player couldn't trigger his own tripmine. Now that the mine is exploding, it's safe the restore the owner so the // CGrenade code knows who the explosive really belongs to. pev->owner = m_pRealOwner; pev->health = 0; Killed( VARS( pev->owner ), GIB_NORMAL ); return; } m_pBeam->SetColor( RANDOM_FLOAT( 0.0f, 255.0f ), RANDOM_FLOAT( 0.0f, 255.0f ), RANDOM_FLOAT( 0.0f, 255.0f ) ); pev->nextthink = gpGlobals->time + 0.1f; } int CTripmineGrenade::TakeDamage( entvars_t *pevInflictor, entvars_t *pevAttacker, float flDamage, int bitsDamageType ) { if( gpGlobals->time < m_flPowerUp && flDamage < pev->health ) { // disable // Create( "weapon_tripmine", pev->origin + m_vecDir * 24.0f, pev->angles ); SetThink( &CBaseEntity::SUB_Remove ); pev->nextthink = gpGlobals->time + 0.1f; KillBeam(); return FALSE; } return CGrenade::TakeDamage( pevInflictor, pevAttacker, flDamage, bitsDamageType ); } void CTripmineGrenade::Killed( entvars_t *pevAttacker, int iGib ) { pev->takedamage = DAMAGE_NO; if( pevAttacker && ( pevAttacker->flags & FL_CLIENT ) ) { // some client has destroyed this mine, he'll get credit for any kills pev->owner = ENT( pevAttacker ); } SetThink( &CTripmineGrenade::DelayDeathThink ); pev->nextthink = gpGlobals->time + RANDOM_FLOAT( 0.1f, 0.3f ); EMIT_SOUND( ENT( pev ), CHAN_BODY, "common/null.wav", 0.5f, ATTN_NORM ); // shut off chargeup } void CTripmineGrenade::DelayDeathThink( void ) { KillBeam(); TraceResult tr; UTIL_TraceLine( pev->origin + m_vecDir * 8, pev->origin - m_vecDir * 64.0f, dont_ignore_monsters, ENT( pev ), &tr ); Explode( &tr, DMG_BLAST ); } #endif LINK_ENTITY_TO_CLASS( weapon_tripmine, CTripmine ) void CTripmine::Spawn() { Precache(); m_iId = WEAPON_TRIPMINE; SET_MODEL( ENT( pev ), "models/v_tripmine.mdl" ); pev->frame = 0; #ifdef CLIENT_DLL pev->body = 0; #else pev->body = 3; #endif pev->sequence = TRIPMINE_GROUND; // ResetSequenceInfo(); pev->framerate = 0; FallInit();// get ready to fall down m_iDefaultAmmo = TRIPMINE_DEFAULT_GIVE; #if CLIENT_DLL if( !bIsMultiplayer() ) #else if( !g_pGameRules->IsDeathmatch() ) #endif { UTIL_SetSize( pev, Vector( -16.0f, -16.0f, 0.0f ), Vector( 16.0f, 16.0f, 28.0f ) ); } } void CTripmine::Precache( void ) { PRECACHE_MODEL( "models/v_tripmine.mdl" ); PRECACHE_MODEL( "models/p_tripmine.mdl" ); UTIL_PrecacheOther( "monster_tripmine" ); m_usTripFire = PRECACHE_EVENT( 1, "events/tripfire.sc" ); } int CTripmine::GetItemInfo( ItemInfo *p ) { p->pszName = STRING( pev->classname ); p->pszAmmo1 = "Trip Mine"; p->iMaxAmmo1 = TRIPMINE_MAX_CARRY; p->pszAmmo2 = NULL; p->iMaxAmmo2 = -1; p->iMaxClip = WEAPON_NOCLIP; p->iSlot = 4; p->iPosition = 2; p->iId = m_iId = WEAPON_TRIPMINE; p->iWeight = TRIPMINE_WEIGHT; p->iFlags = ITEM_FLAG_LIMITINWORLD | ITEM_FLAG_EXHAUSTIBLE; return 1; } int CTripmine::AddToPlayer( CBasePlayer *pPlayer ) { if( CBasePlayerWeapon::AddToPlayer( pPlayer ) ) { MESSAGE_BEGIN( MSG_ONE, gmsgWeapPickup, NULL, pPlayer->pev ); WRITE_BYTE( m_iId ); MESSAGE_END(); return TRUE; } return FALSE; } BOOL CTripmine::Deploy() { pev->body = 0; return DefaultDeploy( "models/v_tripmine.mdl", "models/p_tripmine.mdl", TRIPMINE_DRAW, "trip" ); } void CTripmine::Holster( int skiplocal /* = 0 */ ) { m_pPlayer->m_flNextAttack = UTIL_WeaponTimeBase() + 0.5f; if( !m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] ) { // out of mines m_pPlayer->pev->weapons &= ~( 1 << WEAPON_TRIPMINE ); DestroyItem(); } SendWeaponAnim( TRIPMINE_HOLSTER ); EMIT_SOUND( ENT( m_pPlayer->pev ), CHAN_WEAPON, "common/null.wav", 1.0f, ATTN_NORM ); } void CTripmine::PrimaryAttack( void ) { if( m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] <= 0 ) return; UTIL_MakeVectors( m_pPlayer->pev->v_angle + m_pPlayer->pev->punchangle ); Vector vecSrc = m_pPlayer->GetGunPosition(); Vector vecAiming = gpGlobals->v_forward; TraceResult tr; UTIL_TraceLine( vecSrc, vecSrc + vecAiming * 128.0f, dont_ignore_monsters, ENT( m_pPlayer->pev ), &tr ); int flags; #if CLIENT_WEAPONS flags = FEV_NOTHOST; #else flags = 0; #endif PLAYBACK_EVENT_FULL( flags, m_pPlayer->edict(), m_usTripFire, 0.0f, g_vecZero, g_vecZero, 0.0f, 0.0f, 0, 0, m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] == 1, 0 ); if( tr.flFraction < 1.0f ) { CBaseEntity *pEntity = CBaseEntity::Instance( tr.pHit ); if( pEntity && !( pEntity->pev->flags & FL_CONVEYOR ) ) { Vector angles = UTIL_VecToAngles( tr.vecPlaneNormal ); CBaseEntity::Create( "monster_tripmine", tr.vecEndPos + tr.vecPlaneNormal * 8.0f, angles, m_pPlayer->edict() ); m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType]--; // player "shoot" animation m_pPlayer->SetAnimation( PLAYER_ATTACK1 ); if( m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] <= 0 ) { // no more mines! RetireWeapon(); return; } } /*else { // ALERT( at_console, "no deploy\n" ); }*/ } /*else { }*/ m_flNextPrimaryAttack = GetNextAttackDelay( 0.3 ); m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + UTIL_SharedRandomFloat( m_pPlayer->random_seed, 10, 15 ); } void CTripmine::WeaponIdle( void ) { pev->body = 0; if( m_flTimeWeaponIdle > UTIL_WeaponTimeBase() ) return; if( m_pPlayer->m_rgAmmo[m_iPrimaryAmmoType] > 0 ) { SendWeaponAnim( TRIPMINE_DRAW ); } else { RetireWeapon(); return; } int iAnim; float flRand = UTIL_SharedRandomFloat( m_pPlayer->random_seed, 0, 1 ); if( flRand <= 0.25f ) { iAnim = TRIPMINE_IDLE1; m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + 90.0f / 30.0f; } else if( flRand <= 0.75f ) { iAnim = TRIPMINE_IDLE2; m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + 60.0f / 30.0f; } else { iAnim = TRIPMINE_FIDGET; m_flTimeWeaponIdle = UTIL_WeaponTimeBase() + 100.0f / 30.0f; } SendWeaponAnim( iAnim ); }