/*** * * 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. * * 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 "soundent.h" #include "gamerules.h" #include "decals.h" #include "sporegrenade.h" #include "gearbox_weapons.h" LINK_ENTITY_TO_CLASS(spore, CSporeGrenade) TYPEDESCRIPTION CSporeGrenade::m_SaveData[] = { DEFINE_FIELD(CSporeGrenade, m_pSporeGlow, FIELD_CLASSPTR), }; IMPLEMENT_SAVERESTORE(CSporeGrenade, CBaseMonster) int gSporeExplode, gSporeExplodeC; void CSporeGrenade::Precache(void) { PRECACHE_MODEL("models/spore.mdl"); PRECACHE_MODEL("sprites/glow02.spr"); g_sModelIndexTinySpit = PRECACHE_MODEL("sprites/tinyspit.spr"); gSporeExplode = PRECACHE_MODEL ("sprites/spore_exp_01.spr"); gSporeExplodeC = PRECACHE_MODEL ("sprites/spore_exp_c_01.spr"); PRECACHE_SOUND("weapons/splauncher_bounce.wav"); PRECACHE_SOUND("weapons/splauncher_impact.wav"); } void CSporeGrenade::Explode(TraceResult *pTrace) { pev->solid = SOLID_NOT;// intangible pev->takedamage = DAMAGE_NO; // Pull out of the wall a bit if (pTrace->flFraction != 1.0) { pev->origin = pTrace->vecEndPos + (pTrace->vecPlaneNormal * (pev->dmg - 24) * 0.6); } Vector vecSpraySpot = pTrace->vecEndPos; float flSpraySpeed = RANDOM_LONG(10, 15); // If the trace is pointing up, then place // spawn position a few units higher. if (pTrace->vecPlaneNormal.z > 0) { vecSpraySpot = vecSpraySpot + (pTrace->vecPlaneNormal * 8); flSpraySpeed *= 2; // Double the speed to make them fly higher // in the air. } // Spawn small particles at the explosion origin. SpawnExplosionParticles( vecSpraySpot, // position pTrace->vecPlaneNormal, // direction g_sModelIndexTinySpit, // modelindex RANDOM_LONG(40, 50), // count flSpraySpeed, // speed RANDOM_FLOAT(600, 640)); // noise MESSAGE_BEGIN( MSG_PAS, SVC_TEMPENTITY, pev->origin ); WRITE_BYTE( TE_SPRITE ); WRITE_COORD( pev->origin.x ); WRITE_COORD( pev->origin.y ); WRITE_COORD( pev->origin.z ); WRITE_SHORT( RANDOM_LONG( 0, 1 ) ? gSporeExplode : gSporeExplodeC ); WRITE_BYTE( 25 ); // scale * 10 WRITE_BYTE( 155 ); // framerate MESSAGE_END(); MESSAGE_BEGIN( MSG_PVS, SVC_TEMPENTITY, pev->origin ); WRITE_BYTE(TE_DLIGHT); WRITE_COORD( pev->origin.x ); // X WRITE_COORD( pev->origin.y ); // Y WRITE_COORD( pev->origin.z ); // Z WRITE_BYTE( 12 ); // radius * 0.1 WRITE_BYTE( 0 ); // r WRITE_BYTE( 180 ); // g WRITE_BYTE( 0 ); // b WRITE_BYTE( 20 ); // time * 10 WRITE_BYTE( 20 ); // decay * 0.1 MESSAGE_END( ); // Play explode sound. EMIT_SOUND(ENT(pev), CHAN_VOICE, "weapons/splauncher_impact.wav", 1, ATTN_NORM); CSoundEnt::InsertSound(bits_SOUND_COMBAT, pev->origin, NORMAL_EXPLOSION_VOLUME, 3.0); entvars_t *pevOwner; if (pev->owner) pevOwner = VARS(pev->owner); else pevOwner = NULL; pev->owner = NULL; // can't traceline attack owner if this is set RadiusDamage(pev, pevOwner, pev->dmg, CLASS_NONE, DMG_BLAST); // Place a decal on the surface that was hit. UTIL_DecalTrace(pTrace, DECAL_SPR_SPLT1 + RANDOM_LONG(0, 2)); UTIL_Remove(this); } void CSporeGrenade::Detonate(void) { TraceResult tr; Vector vecSpot = pev->origin + Vector(0, 0, 8); UTIL_TraceLine(vecSpot, vecSpot + Vector(0, 0, -40), ignore_monsters, ENT(pev), &tr); Explode(&tr); } void CSporeGrenade::BounceSound(void) { DangerSound(); EMIT_SOUND(ENT(pev), CHAN_VOICE, "weapons/splauncher_bounce.wav", 0.25, ATTN_NORM); } void CSporeGrenade::DangerSound() { CSoundEnt::InsertSound(bits_SOUND_DANGER, pev->origin + pev->velocity * 0.5, pev->velocity.Length(), 0.2); } void CSporeGrenade::TumbleThink(void) { if (!IsInWorld()) { UTIL_Remove(this); return; } pev->nextthink = gpGlobals->time + 0.1; if (pev->dmgtime - 1 < gpGlobals->time) { CSoundEnt::InsertSound(bits_SOUND_DANGER, pev->origin + pev->velocity * (pev->dmgtime - gpGlobals->time), 400, 0.1); } if (pev->dmgtime <= gpGlobals->time) { SetThink(&CSporeGrenade::Detonate); } // Spawn particles. SpawnTrailParticles( pev->origin, // position -pev->velocity.Normalize(), // dir g_sModelIndexTinySpit, // modelindex RANDOM_LONG( 2, 4 ), // count RANDOM_FLOAT(10, 15), // speed RANDOM_FLOAT(2, 3) * 100); // noise ( client will divide by 100 ) } // // Contact grenade, explode when it touches something // void CSporeGrenade::ExplodeTouch(CBaseEntity *pOther) { TraceResult tr; Vector vecSpot;// trace starts here! pev->enemy = pOther->edict(); vecSpot = pev->origin - pev->velocity.Normalize() * 32; UTIL_TraceLine(vecSpot, vecSpot + pev->velocity.Normalize() * 64, ignore_monsters, ENT(pev), &tr); Explode(&tr); } void CSporeGrenade::DangerSoundThink(void) { if (!IsInWorld()) { UTIL_Remove(this); return; } DangerSound(); pev->nextthink = gpGlobals->time + 0.2; // Spawn particles. SpawnTrailParticles( pev->origin, // position -pev->velocity.Normalize(), // dir g_sModelIndexTinySpit, // modelindex RANDOM_LONG( 5, 10), // count RANDOM_FLOAT(10, 15), // speed RANDOM_FLOAT(2, 3) * 100); // noise ( client will divide by 100 ) } void CSporeGrenade::BounceTouch(CBaseEntity *pOther) { if ( !pOther->pev->takedamage ) { if (!(pev->flags & FL_ONGROUND)) { if (pev->dmg_save < gpGlobals->time) { BounceSound(); pev->dmg_save = gpGlobals->time + 0.1; } } else { pev->velocity = pev->velocity * 0.9; } if (pev->flags & FL_SWIM) { pev->velocity = pev->velocity * 0.5; } } else { TraceResult tr = UTIL_GetGlobalTrace(); Explode(&tr); } } void CSporeGrenade::Spawn(void) { Precache(); pev->classname = MAKE_STRING("spore"); pev->movetype = MOVETYPE_BOUNCE; pev->solid = SOLID_BBOX; SET_MODEL(ENT(pev), "models/spore.mdl"); UTIL_SetSize(pev, Vector(0, 0, 0), Vector(0, 0, 0)); //pev->gravity = 0.5; pev->dmg = gSkillData.plrDmgSpore; m_pSporeGlow = CSprite::SpriteCreate("sprites/glow02.spr", pev->origin, FALSE); if (m_pSporeGlow) { m_pSporeGlow->SetTransparency(kRenderGlow, 150, 158, 19, 155, kRenderFxNoDissipation); m_pSporeGlow->SetAttachment(edict(), 0); m_pSporeGlow->SetScale(.75f); } } CBaseEntity* CSporeGrenade::ShootTimed(entvars_t *pevOwner, Vector vecStart, Vector vecVelocity, bool ai) { CSporeGrenade *pGrenade = GetClassPtr((CSporeGrenade *)NULL); UTIL_SetOrigin(pGrenade->pev, vecStart); pGrenade->Spawn(); pGrenade->pev->velocity = vecVelocity; pGrenade->pev->angles = UTIL_VecToAngles(pGrenade->pev->velocity); pGrenade->pev->owner = ENT(pevOwner); pGrenade->SetTouch(&CSporeGrenade::BounceTouch); // Bounce if touched float lifetime = 2.0; if (ai) { lifetime = 4.0; pGrenade->pev->gravity = 0.5; pGrenade->pev->friction = 0.9; } pGrenade->pev->dmgtime = gpGlobals->time + lifetime; pGrenade->SetThink(&CSporeGrenade::TumbleThink); pGrenade->pev->nextthink = gpGlobals->time + 0.1; if (lifetime < 0.1) { pGrenade->pev->nextthink = gpGlobals->time; pGrenade->pev->velocity = Vector(0, 0, 0); } return pGrenade; } CBaseEntity *CSporeGrenade::ShootContact(entvars_t *pevOwner, Vector vecStart, Vector vecVelocity) { CSporeGrenade *pGrenade = GetClassPtr((CSporeGrenade *)NULL); UTIL_SetOrigin(pGrenade->pev, vecStart); pGrenade->Spawn(); pGrenade->pev->movetype = MOVETYPE_FLY; pGrenade->pev->velocity = vecVelocity; pGrenade->pev->angles = UTIL_VecToAngles(pGrenade->pev->velocity); pGrenade->pev->owner = ENT(pevOwner); // make monsters afraid of it while in the air pGrenade->SetThink(&CSporeGrenade::DangerSoundThink); pGrenade->pev->nextthink = gpGlobals->time; // Explode on contact pGrenade->SetTouch(&CSporeGrenade::ExplodeTouch); pGrenade->pev->gravity = 0.5; pGrenade->pev->friction = 0.7; return pGrenade; } void CSporeGrenade::SpawnTrailParticles(const Vector& origin, const Vector& direction, int modelindex, int count, float speed, float noise) { MESSAGE_BEGIN(MSG_PVS, SVC_TEMPENTITY, origin); WRITE_BYTE(TE_SPRITE_SPRAY); WRITE_COORD(origin.x); // pos WRITE_COORD(origin.y); WRITE_COORD(origin.z); WRITE_COORD(direction.x); // dir WRITE_COORD(direction.y); WRITE_COORD(direction.z); WRITE_SHORT(modelindex); // model WRITE_BYTE(count); // count WRITE_BYTE(speed); // speed WRITE_BYTE(noise); // noise ( client will divide by 100 ) MESSAGE_END(); } void CSporeGrenade::SpawnExplosionParticles(const Vector& origin, const Vector& direction, int modelindex, int count, float speed, float noise) { MESSAGE_BEGIN(MSG_PVS, SVC_TEMPENTITY, origin); WRITE_BYTE(TE_SPRITE_SPRAY); WRITE_COORD(origin.x); // pos WRITE_COORD(origin.y); WRITE_COORD(origin.z); WRITE_COORD(direction.x); // dir WRITE_COORD(direction.y); WRITE_COORD(direction.z); WRITE_SHORT(modelindex); // model WRITE_BYTE(count); // count WRITE_BYTE(speed); // speed WRITE_BYTE(noise); // noise ( client will divide by 100 ) MESSAGE_END(); } void CSporeGrenade::UpdateOnRemove() { CBaseMonster::UpdateOnRemove(); if (m_pSporeGlow) { UTIL_Remove(m_pSporeGlow); m_pSporeGlow = NULL; } }