/*
===========================================================================
Copyright (C) 2000 - 2013, Raven Software, Inc.
Copyright (C) 2001 - 2013, Activision, Inc.
Copyright (C) 2013 - 2015, OpenJK contributors
This file is part of the OpenJK source code.
OpenJK is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, see .
===========================================================================
*/
#include "g_local.h"
#include "b_local.h"
#include "g_functions.h"
#include "wp_saber.h"
#include "w_local.h"
#include "bg_local.h"
#include
//---------------------
// Thermal Detonator
//---------------------
//---------------------------------------------------------
void thermalDetonatorExplode( gentity_t *ent )
//---------------------------------------------------------
{
if ( (ent->s.eFlags&EF_HELD_BY_SAND_CREATURE) )
{
ent->takedamage = qfalse; // don't allow double deaths!
G_Damage( ent->activator, ent, ent->owner, vec3_origin, ent->currentOrigin, weaponData[WP_THERMAL].altDamage, 0, MOD_EXPLOSIVE );
G_PlayEffect( "thermal/explosion", ent->currentOrigin );
G_PlayEffect( "thermal/shockwave", ent->currentOrigin );
G_FreeEntity( ent );
}
else if ( !ent->count )
{
G_Sound( ent, G_SoundIndex( "sound/weapons/thermal/warning.wav" ) );
ent->count = 1;
ent->nextthink = level.time + 800;
ent->svFlags |= SVF_BROADCAST;//so everyone hears/sees the explosion?
}
else
{
vec3_t pos;
VectorSet( pos, ent->currentOrigin[0], ent->currentOrigin[1], ent->currentOrigin[2] + 8 );
ent->takedamage = qfalse; // don't allow double deaths!
G_RadiusDamage( ent->currentOrigin, ent->owner, weaponData[WP_THERMAL].splashDamage, weaponData[WP_THERMAL].splashRadius, NULL, MOD_EXPLOSIVE_SPLASH );
G_PlayEffect( "thermal/explosion", ent->currentOrigin );
G_PlayEffect( "thermal/shockwave", ent->currentOrigin );
G_FreeEntity( ent );
}
}
//-------------------------------------------------------------------------------------------------------------
void thermal_die( gentity_t *self, gentity_t *inflictor, gentity_t *attacker, int damage, int mod, int dFlags, int hitLoc )
//-------------------------------------------------------------------------------------------------------------
{
thermalDetonatorExplode( self );
}
//---------------------------------------------------------
qboolean WP_LobFire( gentity_t *self, vec3_t start, vec3_t target, vec3_t mins, vec3_t maxs, int clipmask,
vec3_t velocity, qboolean tracePath, int ignoreEntNum, int enemyNum,
float minSpeed, float maxSpeed, float idealSpeed, qboolean mustHit )
//---------------------------------------------------------
{
float targetDist, shotSpeed, speedInc = 100, travelTime, impactDist, bestImpactDist = Q3_INFINITE;//fireSpeed,
vec3_t targetDir, shotVel, failCase = { 0.0f };
trace_t trace;
trajectory_t tr;
qboolean blocked;
int elapsedTime, skipNum, timeStep = 500, hitCount = 0, maxHits = 7;
vec3_t lastPos, testPos;
gentity_t *traceEnt;
if ( !idealSpeed )
{
idealSpeed = 300;
}
else if ( idealSpeed < speedInc )
{
idealSpeed = speedInc;
}
shotSpeed = idealSpeed;
skipNum = (idealSpeed-speedInc)/speedInc;
if ( !minSpeed )
{
minSpeed = 100;
}
if ( !maxSpeed )
{
maxSpeed = 900;
}
while ( hitCount < maxHits )
{
VectorSubtract( target, start, targetDir );
targetDist = VectorNormalize( targetDir );
VectorScale( targetDir, shotSpeed, shotVel );
travelTime = targetDist/shotSpeed;
shotVel[2] += travelTime * 0.5 * g_gravity->value;
if ( !hitCount )
{//save the first (ideal) one as the failCase (fallback value)
if ( !mustHit )
{//default is fine as a return value
VectorCopy( shotVel, failCase );
}
}
if ( tracePath )
{//do a rough trace of the path
blocked = qfalse;
VectorCopy( start, tr.trBase );
VectorCopy( shotVel, tr.trDelta );
tr.trType = TR_GRAVITY;
tr.trTime = level.time;
travelTime *= 1000.0f;
VectorCopy( start, lastPos );
//This may be kind of wasteful, especially on long throws... use larger steps? Divide the travelTime into a certain hard number of slices? Trace just to apex and down?
for ( elapsedTime = timeStep; elapsedTime < floor(travelTime)+timeStep; elapsedTime += timeStep )
{
if ( (float)elapsedTime > travelTime )
{//cap it
elapsedTime = floor( travelTime );
}
EvaluateTrajectory( &tr, level.time + elapsedTime, testPos );
gi.trace( &trace, lastPos, mins, maxs, testPos, ignoreEntNum, clipmask, (EG2_Collision)0, 0 );
if ( trace.allsolid || trace.startsolid )
{
blocked = qtrue;
break;
}
if ( trace.fraction < 1.0f )
{//hit something
if ( trace.entityNum == enemyNum )
{//hit the enemy, that's perfect!
break;
}
else if ( trace.plane.normal[2] > 0.7 && DistanceSquared( trace.endpos, target ) < 4096 )//hit within 64 of desired location, should be okay
{//close enough!
break;
}
else
{//FIXME: maybe find the extents of this brush and go above or below it on next try somehow?
impactDist = DistanceSquared( trace.endpos, target );
if ( impactDist < bestImpactDist )
{
bestImpactDist = impactDist;
VectorCopy( shotVel, failCase );
}
blocked = qtrue;
//see if we should store this as the failCase
if ( trace.entityNum < ENTITYNUM_WORLD )
{//hit an ent
traceEnt = &g_entities[trace.entityNum];
if ( traceEnt && traceEnt->takedamage && !OnSameTeam( self, traceEnt ) )
{//hit something breakable, so that's okay
//we haven't found a clear shot yet so use this as the failcase
VectorCopy( shotVel, failCase );
}
}
break;
}
}
if ( elapsedTime == floor( travelTime ) )
{//reached end, all clear
break;
}
else
{
//all clear, try next slice
VectorCopy( testPos, lastPos );
}
}
if ( blocked )
{//hit something, adjust speed (which will change arc)
hitCount++;
shotSpeed = idealSpeed + ((hitCount-skipNum) * speedInc);//from min to max (skipping ideal)
if ( hitCount >= skipNum )
{//skip ideal since that was the first value we tested
shotSpeed += speedInc;
}
}
else
{//made it!
break;
}
}
else
{//no need to check the path, go with first calc
break;
}
}
if ( hitCount >= maxHits )
{//NOTE: worst case scenario, use the one that impacted closest to the target (or just use the first try...?)
assert( (failCase[0] + failCase[1] + failCase[2]) > 0.0f );
VectorCopy( failCase, velocity );
return qfalse;
}
VectorCopy( shotVel, velocity );
return qtrue;
}
//---------------------------------------------------------
void WP_ThermalThink( gentity_t *ent )
//---------------------------------------------------------
{
int count;
qboolean blow = qfalse;
// Thermal detonators for the player do occasional radius checks and blow up if there are entities in the blast radius
// This is done so that the main fire is actually useful as an attack. We explode anyway after delay expires.
if ( (ent->s.eFlags&EF_HELD_BY_SAND_CREATURE) )
{//blow once creature is underground (done with anim)
//FIXME: chance of being spit out? Especially if lots of delay left...
ent->e_TouchFunc = touchF_NULL;//don't impact on anything
if ( !ent->activator
|| !ent->activator->client
|| !ent->activator->client->ps.legsAnimTimer )
{//either something happened to the sand creature or it's done with it's attack anim
//blow!
ent->e_ThinkFunc = thinkF_thermalDetonatorExplode;
ent->nextthink = level.time + Q_irand( 50, 2000 );
}
else
{//keep checking
ent->nextthink = level.time + TD_THINK_TIME;
}
return;
}
else if ( ent->delay > level.time )
{
// Finally, we force it to bounce at least once before doing the special checks, otherwise it's just too easy for the player?
if ( ent->has_bounced )
{
count = G_RadiusList( ent->currentOrigin, TD_TEST_RAD, ent, qtrue, ent_list );
for ( int i = 0; i < count; i++ )
{
if ( ent_list[i]->s.number == 0 )
{
// avoid deliberately blowing up next to the player, no matter how close any enemy is..
// ...if the delay time expires though, there is no saving the player...muwhaaa haa ha
blow = qfalse;
break;
}
else if ( ent_list[i]->client
&& ent_list[i]->client->NPC_class != CLASS_SAND_CREATURE//ignore sand creatures
&& ent_list[i]->health > 0 )
{
//FIXME! sometimes the ent_list order changes, so we should make sure that the player isn't anywhere in this list
blow = qtrue;
}
}
}
}
else
{
// our death time has arrived, even if nothing is near us
blow = qtrue;
}
if ( blow )
{
ent->e_ThinkFunc = thinkF_thermalDetonatorExplode;
ent->nextthink = level.time + 50;
}
else
{
// we probably don't need to do this thinking logic very often...maybe this is fast enough?
ent->nextthink = level.time + TD_THINK_TIME;
}
}
#define OLDEST_READING 5
#define NEWEST_READING 2
#define TD_REAL_THROW_VEL_MULT 4.4f
#define VectorDistance(a, b) (sqrt( VectorDistance2( a, b )))
#define VectorDistance2(a, b) (((a)[0] - (b)[0]) * ((a)[0] - (b)[0]) + ((a)[1] - (b)[1]) * ((a)[1] - (b)[1]) + ((a)[2] - (b)[2]) * ((a)[2] - (b)[2]))
//---------------------------------------------------------
gentity_t *WP_FireThermalDetonator( gentity_t *ent, qboolean alt_fire )
//---------------------------------------------------------
{
gentity_t *bolt;
vec3_t dir, start;
float damageScale = 1.0f;
bolt = G_Spawn();
bool realThrow = false;
if ( BG_UseVRPosition(ent) )
{
vec3_t angs;
BG_CalculateVRWeaponPosition(start, angs);
//Caclulate speed between two controller position readings
float distance = VectorDistance(vr->weaponoffset_history[NEWEST_READING], vr->weaponoffset_history[OLDEST_READING]);
float t = vr->weaponoffset_history_timestamp[NEWEST_READING] - vr->weaponoffset_history_timestamp[OLDEST_READING];
float velocity = distance / (t/(float)1000.0);
//Calculate trajectory
VectorSubtract(vr->weaponoffset_history[NEWEST_READING], vr->weaponoffset_history[OLDEST_READING], dir);
VectorNormalize( dir );
BG_ConvertFromVR(dir, NULL, dir);
VectorScale( dir, velocity * TD_REAL_THROW_VEL_MULT, bolt->s.pos.trDelta );
realThrow = true;
}
else {
VectorCopy( forwardVec, dir );
VectorCopy( muzzle, start );
}
bolt->classname = "thermal_detonator";
if ( ent->s.number != 0 )
{
// If not the player, cut the damage a bit so we don't get pounded on so much
damageScale = TD_NPC_DAMAGE_CUT;
}
if ( !alt_fire && ent->s.number == 0 )
{
// Main fires for the players do a little bit of extra thinking
bolt->e_ThinkFunc = thinkF_WP_ThermalThink;
bolt->nextthink = level.time + TD_THINK_TIME;
bolt->delay = level.time + TD_TIME; // How long 'til she blows
}
else
{
bolt->e_ThinkFunc = thinkF_thermalDetonatorExplode;
bolt->nextthink = level.time + TD_TIME; // How long 'til she blows
}
bolt->mass = 10;
// How 'bout we give this thing a size...
VectorSet( bolt->mins, -4.0f, -4.0f, -4.0f );
VectorSet( bolt->maxs, 4.0f, 4.0f, 4.0f );
bolt->clipmask = MASK_SHOT;
bolt->clipmask &= ~CONTENTS_CORPSE;
bolt->contents = CONTENTS_SHOTCLIP;
bolt->takedamage = qtrue;
bolt->health = 15;
bolt->e_DieFunc = dieF_thermal_die;
WP_TraceSetStart( ent, start, bolt->mins, bolt->maxs );//make sure our start point isn't on the other side of a wall
float chargeAmount = 1.0f; // default of full charge
if ( ent->client )
{
chargeAmount = level.time - ent->client->ps.weaponChargeTime;
}
// get charge amount
chargeAmount = chargeAmount / (float)TD_VELOCITY;
if ( chargeAmount > 1.0f )
{
chargeAmount = 1.0f;
}
else if ( chargeAmount < TD_MIN_CHARGE )
{
chargeAmount = TD_MIN_CHARGE;
}
float thrownSpeed = TD_VELOCITY;
const qboolean thisIsAShooter = (qboolean)!Q_stricmp( "misc_weapon_shooter", ent->classname);
if (thisIsAShooter)
{
if (ent->delay != 0)
{
thrownSpeed = ent->delay;
}
}
// normal ones bounce, alt ones explode on impact
bolt->s.pos.trType = TR_GRAVITY;
bolt->owner = ent;
if (!realThrow) {
VectorScale( dir, thrownSpeed * chargeAmount, bolt->s.pos.trDelta );
}
if ( ent->health > 0 )
{
if (!realThrow) {
bolt->s.pos.trDelta[2] += 120;
}
if ( (ent->NPC || (ent->s.number && thisIsAShooter) )
&& ent->enemy )
{//NPC or misc_weapon_shooter
//FIXME: we're assuming he's actually facing this direction...
vec3_t target;
VectorCopy( ent->enemy->currentOrigin, target );
if ( target[2] <= start[2] )
{
vec3_t vec;
VectorSubtract( target, start, vec );
VectorNormalize( vec );
VectorMA( target, Q_flrand( 0, -32 ), vec, target );//throw a little short
}
target[0] += Q_flrand( -5, 5 )+(Q_flrand(-1.0f, 1.0f)*(6-ent->NPC->currentAim)*2);
target[1] += Q_flrand( -5, 5 )+(Q_flrand(-1.0f, 1.0f)*(6-ent->NPC->currentAim)*2);
target[2] += Q_flrand( -5, 5 )+(Q_flrand(-1.0f, 1.0f)*(6-ent->NPC->currentAim)*2);
WP_LobFire( ent, start, target, bolt->mins, bolt->maxs, bolt->clipmask, bolt->s.pos.trDelta, qtrue, ent->s.number, ent->enemy->s.number );
}
else if ( thisIsAShooter && ent->target && !VectorCompare( ent->pos1, vec3_origin ) )
{//misc_weapon_shooter firing at a position
WP_LobFire( ent, start, ent->pos1, bolt->mins, bolt->maxs, bolt->clipmask, bolt->s.pos.trDelta, qtrue, ent->s.number, ent->enemy->s.number );
}
}
if ( alt_fire )
{
bolt->alt_fire = qtrue;
}
else
{
bolt->s.eFlags |= EF_BOUNCE_HALF;
}
bolt->s.loopSound = G_SoundIndex( "sound/weapons/thermal/thermloop.wav" );
bolt->damage = weaponData[WP_THERMAL].damage * damageScale;
bolt->dflags = 0;
bolt->splashDamage = weaponData[WP_THERMAL].splashDamage * damageScale;
bolt->splashRadius = weaponData[WP_THERMAL].splashRadius;
bolt->s.eType = ET_MISSILE;
bolt->svFlags = SVF_USE_CURRENT_ORIGIN;
bolt->s.weapon = WP_THERMAL;
if ( alt_fire )
{
bolt->methodOfDeath = MOD_THERMAL_ALT;
bolt->splashMethodOfDeath = MOD_THERMAL_ALT;//? SPLASH;
}
else
{
bolt->methodOfDeath = MOD_THERMAL;
bolt->splashMethodOfDeath = MOD_THERMAL;//? SPLASH;
}
bolt->s.pos.trTime = level.time; // move a bit on the very first frame
VectorCopy( start, bolt->s.pos.trBase );
SnapVector( bolt->s.pos.trDelta ); // save net bandwidth
VectorCopy (start, bolt->currentOrigin);
VectorCopy( start, bolt->pos2 );
return bolt;
}
//---------------------------------------------------------
gentity_t *WP_DropThermal( gentity_t *ent )
//---------------------------------------------------------
{
AngleVectors( ent->client->ps.viewangles, forwardVec, vrightVec, up );
CalcEntitySpot( ent, SPOT_WEAPON, muzzle );
return (WP_FireThermalDetonator( ent, qfalse ));
}