rpg-x2/game/g_breakable.c
2011-06-01 14:20:56 +02:00

743 lines
17 KiB
C

#include "g_local.h"
#include "g_breakable.h"
/**
* \brief A func_breakables health has sunk to or under zero
*
* Removes entity entirely blow chunks.
* If it is repairable it's not removed but made invisible.
*/
void breakable_die( gentity_t *self, gentity_t *inflictor, gentity_t *attacker, int damage, int meansOfDeath )
{
vec3_t size, org, dir;
gentity_t *te = NULL;
entityState_t *eState = &self->s;
entityShared_t *eShared = &self->r;
//RPG-X | GSIO01 | 09/05/2009:
if(!(self->spawnflags & 256)) {
eState->frame = 0;
//eShared->svFlags &= ~SVF_ANIMATING;
self->health = 0;
}
self->pain = 0/*NULL*/;
self->die = 0/*NULL*/;
self->use = 0/*NULL*/;
self->takedamage = qfalse;
if( self->target )
{
G_UseTargets(self, attacker);
}
if ( !(self->spawnflags & 4) )
{//We don't want to stay solid
eState->solid = 0;
eShared->contents = 0;
self->clipmask = 0;
if(self->spawnflags & 256) {
eShared->svFlags |= SVF_NOCLIENT;
eState->eFlags |= EF_NODRAW;
}
trap_LinkEntity(self);
}
/*if( self->target )
{
G_UseTargets(self, attacker);
}*/
if ( eShared->bmodel )
{
VectorSubtract( eShared->absmax, eShared->absmin, size );
VectorMA( eShared->absmin, 0.5, size, org );
}
else
{
VectorSubtract( eShared->maxs, eShared->mins, size );
VectorCopy( eShared->currentOrigin, org );
}
// Create a chunk effect
te = G_TempEntity( org, EV_FX_CHUNKS );
VectorSet( te->s.angles2, 0, 0, 1 ); // FIXME: temp direction
te->s.time2 = VectorNormalize( size );
te->s.powerups = eState->powerups;
// Ok, we are allowed to explode, so do it now!
if ( self->splashDamage > 0 && self->splashRadius > 0 )
{
//fixme: what about chain reactions?
G_RadiusDamage( org, attacker, self->splashDamage, self->splashRadius, self, DAMAGE_RADIUS|DAMAGE_ALL_TEAMS, MOD_EXPLOSION );
//explosion effect
te = G_TempEntity( org, EV_MISSILE_MISS );
VectorSet( dir, 0, 0, 1 );
te->s.eventParm = DirToByte( dir );
te->s.weapon = WP_GRENADE_LAUNCHER;
//G_Sound( self, G_SoundIndex("sound/weapons/explosions/cargoexplode.wav") );
}
if ( eShared->bmodel )
{
trap_AdjustAreaPortalState( self, qtrue );
}
else if ( eState->modelindex2 != -1 && !(self->spawnflags & 8) )
{
eState->modelindex = self->s.modelindex2;
return;
}
if(!(self->spawnflags & 256))
G_FreeEntity( self );
if(self->spawnflags & 256)
self->count = 0;
}
/**
* Called when a breakable takes damage
*/
void breakable_pain ( gentity_t *self, gentity_t *attacker, int damage )
{
if ( self->pain_debounce_time > level.time )
{
return;
}
if ( self->paintarget )
{
G_UseTargets2 ( self, self->activator, self->paintarget );
}
if(self->wait == -1)
{
self->pain = 0/*NULL*/;
return;
}
self->pain_debounce_time = level.time + self->wait;
}
/**
* Called if a brealable has been used
*/
void breakable_use (gentity_t *self, gentity_t *other, gentity_t *activator)
{
breakable_die( self, other, activator, self->health, MOD_UNKNOWN );
}
/**
* Inits a breakable brush entity
*/
void InitBBrush ( gentity_t *ent )
{
float light;
vec3_t color;
qboolean lightSet, colorSet;
entityState_t *eState = &ent->s;
VectorCopy( eState->origin, ent->pos1 );
trap_SetBrushModel( ent, ent->model );
ent->die = breakable_die;
//ent->r.svFlags |= SVF_BBRUSH;
// if the "model2" key is set, use a seperate model
// for drawing, but clip against the brushes
if ( ent->model2 )
{
eState->modelindex2 = G_ModelIndex( ent->model2 );
}
// if the "color" or "light" keys are set, setup constantLight
lightSet = G_SpawnFloat( "light", "100", &light );
colorSet = G_SpawnVector( "color", "1 1 1", color );
if ( lightSet || colorSet )
{
int r, g, b, i;
r = color[0] * 255;
if ( r > 255 ) {
r = 255;
}
g = color[1] * 255;
if ( g > 255 ) {
g = 255;
}
b = color[2] * 255;
if ( b > 255 ) {
b = 255;
}
i = light / 4;
if ( i > 255 ) {
i = 255;
}
eState->constantLight = r | ( g << 8 ) | ( b << 16 ) | ( i << 24 );
}
eState->eType = ET_MOVER;
trap_LinkEntity (ent);
eState->pos.trType = TR_STATIONARY;
//VectorCopy( ent->pos1, eState->pos.trBase );
}
//RPG-X | GSIO01 | 09/05/2009 SOE
/**
* Gets called if someone is inside a breakables trigger.
* Breables only have trigges if they are repairable
* This function sets the touched gentity_t pointer to the tirgger
* so it is possible to check if the player is inside the trigger
* and there fore can repair the breakable.
*
* \author Ubergames - GSIO01
* \date 09/05/2009
*/
void Touch_breakable_trigger(gentity_t *ent, gentity_t *other, trace_t *trace) {
other->touched = ent;
}
/**
* Spawns a trigger for a breakable.
* Only gets called if the breakable is repairable.
*/
void breakable_spawn_trigger(gentity_t *ent) {
vec3_t maxs, mins;
gentity_t *other;
int best, i;
entityShared_t *eShared;
VectorCopy(ent->r.absmin, mins);
VectorCopy(ent->r.absmax, maxs);
for(other = ent->teamchain; other; other=other->teamchain) {
eShared = &other->r;
AddPointToBounds(eShared->absmin, mins, maxs);
AddPointToBounds(eShared->absmax, mins, maxs);
}
best = 0;
for(i = 1; i < 3; i++) {
if(maxs[i] - mins[i] < maxs[best] - mins[best])
best = i;
}
maxs[best] += 48;
mins[best] -= 48;
other = G_Spawn();
if(!other) {
G_Printf(S_COLOR_RED "Unable to spawn trigger for func_breakable at %s!\n", vtos(ent->s.origin));
G_FreeEntity(ent);
return;
}
eShared = &other->r;
VectorCopy(maxs, eShared->maxs);
VectorCopy(mins, eShared->mins);
eShared->contents = CONTENTS_TRIGGER;
other->touch = Touch_breakable_trigger;
other->count = best;
other->touched = ent;
other->target = ent->target;
ent->touched = other;
trap_LinkEntity(other);
}
//RPG-X | GSIO01 | 09/05/2009 EOE
/*QUAKED func_breakable (0 .8 .5) ? x x x x INVINCIBLE x x x REPAIRABLE
INVINCIBLE - can only be broken by being used
REPAIRABLE - con be repaired with hyperspanner
When destroyed, fires it's trigger and explodes
"targetname" entities with matching target will fire it
"paintarget" target to fire when hit (but not destroyed)
"wait" how long minimum to wait between firing paintarget each time hit
"model2" .md3 model to also draw
"target" all entities with a matching targetname will be used when this is destoryed
"health" default is 10
"team" - This cannot take damage from members of this team (2 = blue, 1 = red)
Damage: default is none
"splashDamage" - damage to do (will make it explode on death
"splashRadius" - radius for above damage
"material" - sets the chunk type:
0 - none (default)
1 - metal
2 - glass
3 - glass and metal
4 - wood
5 - stone
Don't know if these work:
"color" constantLight color
"light" constantLight radius
*/
/**
* Spawnfunction for func_breakable entity.
*/
void SP_func_breakable( gentity_t *self )
{
if(!(self->spawnflags & 1))
{
if(!self->health)
{
self->health = 10;
}
}
if (self->health)
{
self->takedamage = qtrue; //RPG-X - TiM: qtrue
}
//RPG-X | GSIO01 | 09/05/2009: store max health for repairing
self->damage = self->health;
G_SoundIndex("sound/weapons/explosions/cargoexplode.wav");//precaching
self->use = breakable_use;
self->count = 1; // GSIO01
if ( self->paintarget )
{
self->pain = breakable_pain;
}
if (!self->model) {
G_Error("func_breakable with NULL model\n");
}
VectorCopy(self->s.origin, self->pos1);
trap_LinkEntity(self);
InitBBrush( self );
breakable_spawn_trigger(self);
level.numBrushEnts++;
}
/*QUAKED misc_model_breakable (1 0 0) (-16 -16 -16) (16 16 16) SOLID AUTOANIMATE DEADSOLID NO_DMODEL INVINCIBLE x x x x
SOLID - Movement is blocked by it, if not set, can still be broken by explosions and shots if it has health
AUTOANIMATE - Will cycle it's anim
DEADSOLID - Stay solid even when destroyed (in case damage model is rather large).
NO_DMODEL - Makes it NOT display a damage model when destroyed, even if one exists
INVINCIBLE - Can only be broken by being used
"model" arbitrary .md3 file to display
"health" how much health to have - default is zero (not breakable) If you don't set the SOLID flag, but give it health, it can be shot but will not block NPCs or players from moving
"targetname" when used, dies and displays damagemodel, if any (if not, removes itself)
"target" What to use when it dies
"paintarget" target to fire when hit (but not destroyed)
"wait" how long minimum to wait between firing paintarget each time hit
Damage: default is none
"splashDamage" - damage to do (will make it explode on death)
"splashRadius" - radius for above damage
"team" - This cannot take damage from members of this team (2 = blue, 1 = red)
"material" - sets the chunk type:
0 - none (default)
1 - metal
2 - glass
3 - glass and metal
4 - wood
5 - stone
FIXME/TODO:
set size better?
multiple damage models?
don't throw chunks on pain, or throw level 1 chunks only on pains?
custom explosion effect/sound?
*/
/**
* Spawnfunction for misc_model_breakable entity.
*/
void SP_misc_model_breakable( gentity_t *ent )
{
char damageModel[MAX_QPATH];
int len;
entityShared_t *eShared = &ent->r;
entityState_t *eState = &ent->s;
//Main model
eState->modelindex = ent->sound2to1 = G_ModelIndex( ent->model );
if ( ent->spawnflags & 1 )
{//Blocks movement
eShared->contents = CONTENTS_BODY;//Was CONTENTS_SOLID, but only architecture should be this
}
else if ( ent->health )
{//Can only be shot
eShared->contents = CONTENTS_SHOTCLIP;
}
ent->use = breakable_use;
if ( ent->health )
{
G_SoundIndex("sound/weapons/explosions/cargoexplode.wav");
//ent->max_health = ent->health;
ent->takedamage = qtrue;
ent->pain = breakable_pain;
ent->die = breakable_die;
}
len = strlen( ent->model ) - 4;
strncpy( damageModel, ent->model, len );
damageModel[len] = 0; //chop extension
if (ent->takedamage) {
//Dead/damaged model
if( !(ent->spawnflags & 8) ) { //no dmodel
strcat( damageModel, "_d1.md3" );
eState->modelindex2 = G_ModelIndex( damageModel );
}
}
if ( !eShared->mins[0] && !eShared->mins[1] && !eShared->mins[2] )
{
VectorSet (eShared->mins, -16, -16, -16);
}
if ( !eShared->maxs[0] && !eShared->maxs[1] && !eShared->maxs[2] )
{
VectorSet (eShared->maxs, 16, 16, 16);
}
if ( ent->spawnflags & 2 )
{
eState->eFlags |= EF_ANIM_ALLFAST;
}
G_SetOrigin( ent, eState->origin );
VectorCopy( eState->angles, eState->apos.trBase );
trap_LinkEntity (ent);
}
//-------------------------------------------------------------------------------
// --------------------------------------------------------------------
//
// AMMO plugin functions
//
// --------------------------------------------------------------------
void ammo_use( gentity_t *self, gentity_t *other, gentity_t *activator);
//!give a player ammo for a weapon
int Add_Ammo2 (gentity_t *ent, int weapon, int count)
{
playerState_t *ps = &ent->client->ps;
ps->ammo[weapon] += count;
if ( ps->ammo[weapon] > Max_Weapon(weapon) )
{
ps->ammo[weapon] = Max_Weapon(weapon);
return qfalse;
}
return qtrue;
}
//!Shuts down a ammo station
void ammo_shutdown( gentity_t *self )
{
if (!(self->s.eFlags & EF_ANIM_ONCE))
{
self->s.eFlags &= ~ EF_ANIM_ALLFAST;
self->s.eFlags |= EF_ANIM_ONCE;
trap_LinkEntity (self);
}
}
//!Fades out a ammo station
void ammo_fade_out( gentity_t *ent )
{
G_Sound( ent, G_SoundIndex( "sound/items/respawn1.wav" ) );
ent->s.eFlags |= EF_ITEMPLACEHOLDER;
ent->s.eventParm = -1;
ent->think = G_FreeEntity;
ent->nextthink = level.time + 1000;
}
//! Think function for the ammo station
void ammo_think( gentity_t *ent )
{
int dif = 0;
int i;
// Still has ammo to give
if ( ent->enemy && ent->enemy->client )
{
//assume that we'll finish here, if we don't, it will be overridden
ent->use = ammo_use;
ent->think = 0;//qvm complains about using NULL
ent->nextthink = -1;
for ( i = 0; i < WP_NUM_WEAPONS && ent->count > 0; i++ )
{//go through all weapons
if ( (ent->enemy->client->ps.stats[STAT_WEAPONS]&( 1 << i )) )
{//has this weapon
dif = Max_Weapon(i) - ent->enemy->client->ps.ammo[i];//needs ammo?
if (dif > 2 )
{
dif= 2;
}
else if (dif < 0)
{
dif= 0;
}
if (ent->count < dif) // Can't give more than count
{
dif = ent->count;
}
// Give player ammo
if (Add_Ammo2(ent->enemy,i,dif) && (dif!=0))
{
ent->count-=dif;
if ( ent->splashDamage )
{
ent->splashDamage = floor((float)ent->count/10);
}
ent->use = 0; /*NULL*/
ent->think = ammo_think;
ent->nextthink = level.time + 10;
}
else // User has taken all ammo he can hold
{
}
}
}
}
if (ent->count < 1)
{
ammo_shutdown(ent);
ent->think = ammo_fade_out;
ent->nextthink = level.time + 3000;
}
}
//------------------------------------------------------------
//! use function for a ammo station
void ammo_use( gentity_t *self, gentity_t *other, gentity_t *activator)
{
int dif = 0;
int i;
if (self->think != NULL)
{
if (self->use != NULL)
{
self->think = 0; /*NULL*/
self->nextthink = -1;
}
}
else
{
if ( other && other->client )
{
for ( i = 0; i < WP_NUM_WEAPONS && dif == 0; i++ )
{//go through all weapons
if ( (other->client->ps.stats[STAT_WEAPONS]&( 1 << i )) )
{//has this weapon
dif = Max_Weapon(i) - other->client->ps.ammo[i];//needs ammo?
}
}
}
else
{ // Being triggered to be used up
dif = 1;
self->count = 0;
}
// Does player already have full ammo?
if ( dif > 0 )
{
G_Sound(self, G_SoundIndex("sound/player/suitenergy.wav") );
if ((dif >= self->count) || (self->count<1)) // use it all up?
{
ammo_shutdown(self);
}
}
else
{
G_Sound(self, G_SoundIndex("sound/weapons/noammo.wav") );
}
// Use target when used
if (self->spawnflags & 8)
{
G_UseTargets( self, activator );
}
self->use = 0; /*NULL*/
self->enemy = other;
self->think = ammo_think;
self->nextthink = level.time + 50;
}
}
/**
* Finishs off the spawning of an ammo station.
*/
void ammo_station_finish_spawning ( gentity_t *self )
{
self->s.eFlags &= ~EF_ITEMPLACEHOLDER;
self->think = 0; /*NULL*/
self->nextthink = -1;
self->use = ammo_use;
}
//------------------------------------------------------------
/*QUAKED misc_ammo_station (1 0 0) (-16 -16 -16) (16 16 16)
"health" - how much health the model has - default 60 (zero makes non-breakable)
"target" - what to use when it dies
"paintarget" - target to fire when hit (but not destroyed)
"count" - the amount of health given when used (default 1000)
"team" - This cannot take damage from members of this team and only members of this team can use it (2 = blue, 1 = red)
*/
//------------------------------------------------------------
/**
* Spawnfunction for misc_ammo_station entity
*/
void SP_misc_ammo_station( gentity_t *ent )
{
if (!ent->health)
{
ent->health = 60;
}
if ( !ent->count )
{
ent->count = 1000;
}
ent->s.powerups = 3;//material
if ( ent->team && atoi(ent->team) == 1 )
{
ent->s.modelindex = G_ModelIndex( "models/mapobjects/dn/powercell2.md3" );
}
else
{
ent->s.modelindex = G_ModelIndex( "models/mapobjects/dn/powercell.md3" );
}
ent->r.contents = CONTENTS_CORPSE;
// Set a generic use function
ent->use = ammo_use;
G_SoundIndex( "sound/player/suitenergy.wav" );
if ( ent->health )
{
ent->takedamage = qtrue;
ent->pain = breakable_pain;
ent->die = breakable_die;
}
//FIXME: if set at an angle (say, on a floor), BBOX will be wrong
VectorSet( ent->r.mins, -16, -16, -16 );
VectorSet( ent->r.maxs, 16, 16, 16 );
G_SetOrigin( ent, ent->s.origin );
VectorCopy( ent->s.angles, ent->s.apos.trBase );
trap_LinkEntity (ent);
}
//RPG-X | GSIO01 | 09/05/2009 SOE
/*QUAKED target_repair (1 0 0) (-8 -8 -8) (8 8 8)
Repairs a func_breakable.
"target" breakable to repair (targetname2)
*/
/**
* Use function of target_repair entity.
* Repairs it's target entity (stored in ent->lastEnemy)
* if it is damaged.
*/
void target_repair_use(gentity_t *ent, gentity_t *other, gentity_t *activator) {
gentity_t *target;
target = ent->lastEnemy;
if(!(target->spawnflags & 256)) return;
target->r.contents = CONTENTS_BODY;
trap_SetBrushModel(target, target->model);
target->r.svFlags &= ~SVF_NOCLIENT;
target->s.eFlags &= ~EF_NODRAW;
InitBBrush(target);
target->health = target->damage;
target->takedamage = qtrue;
target->use = breakable_use;
if(target->paintarget)
target->pain = breakable_pain;
target->clipmask = 0;
target->count = 1;
if(target->touched->target)
G_UseTargets2(target->touched, target, target->touched->target);
}
/**
* Link function finishes off spawning of the entity.
*/
void target_repair_link(gentity_t *ent) {
gentity_t *target;
ent->nextthink = -1;
target = G_Find(NULL, FOFS(targetname), ent->target);
if(!target) {
target = G_Find(NULL, FOFS(targetname2), ent->target);
if(!target) {
G_Printf("target_repair at %s with an unfound target: %s\n", vtos(ent->s.origin), ent->target);
return;
}
}
ent->lastEnemy = target;
if(Q_stricmp(target->classname, "func_breakable")) {
G_Printf("target_repair at %s with an invalid target entity %s\n", vtos(ent->s.origin), target->classname);
return;
}
ent->use = target_repair_use;
}
/**
* Spawn function of target_repair entity
*/
void SP_target_repair(gentity_t *ent) {
if(!ent->target) {
G_Printf("target_repair without target at %s\n", vtos(ent->s.origin));
return;
}
ent->think = target_repair_link;
ent->nextthink = level.time + 1000; // give the breakables some time so init
}
//RPG-X | GSIO01 | 09/05/2009 EOE