2016-03-01 15:47:10 +00:00
|
|
|
/*
|
|
|
|
#include "actor.h"
|
|
|
|
#include "gi.h"
|
|
|
|
#include "m_random.h"
|
|
|
|
#include "s_sound.h"
|
|
|
|
#include "d_player.h"
|
|
|
|
#include "a_action.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "a_action.h"
|
|
|
|
#include "p_pspr.h"
|
|
|
|
#include "gstrings.h"
|
|
|
|
#include "a_hexenglobal.h"
|
|
|
|
#include "thingdef/thingdef.h"
|
|
|
|
#include "g_level.h"
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define ZAGSPEED FRACUNIT
|
|
|
|
|
|
|
|
static FRandom pr_lightningready ("LightningReady");
|
|
|
|
static FRandom pr_lightningclip ("LightningClip");
|
|
|
|
static FRandom pr_zap ("LightningZap");
|
|
|
|
static FRandom pr_zapf ("LightningZapF");
|
|
|
|
static FRandom pr_hit ("LightningHit");
|
|
|
|
|
|
|
|
DECLARE_ACTION(A_LightningClip)
|
|
|
|
DECLARE_ACTION(A_LightningZap)
|
|
|
|
|
|
|
|
// Lightning ----------------------------------------------------------------
|
|
|
|
|
|
|
|
class ALightning : public AActor
|
|
|
|
{
|
|
|
|
DECLARE_CLASS (ALightning, AActor)
|
|
|
|
public:
|
|
|
|
int SpecialMissileHit (AActor *victim);
|
|
|
|
};
|
|
|
|
|
|
|
|
IMPLEMENT_CLASS(ALightning)
|
|
|
|
|
|
|
|
int ALightning::SpecialMissileHit (AActor *thing)
|
|
|
|
{
|
|
|
|
if (thing->flags&MF_SHOOTABLE && thing != target)
|
|
|
|
{
|
|
|
|
if (thing->Mass != INT_MAX)
|
|
|
|
{
|
2016-03-12 13:11:43 +00:00
|
|
|
thing->vel.x += vel.x>>4;
|
|
|
|
thing->vel.y += vel.y>>4;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
if ((!thing->player && !(thing->flags2&MF2_BOSS))
|
|
|
|
|| !(level.time&1))
|
|
|
|
{
|
|
|
|
P_DamageMobj(thing, this, target, 3, NAME_Electric);
|
|
|
|
if (!(S_IsActorPlayingSomething (this, CHAN_WEAPON, -1)))
|
|
|
|
{
|
|
|
|
S_Sound (this, CHAN_WEAPON, this->AttackSound, 1, ATTN_NORM);
|
|
|
|
}
|
|
|
|
if (thing->flags3&MF3_ISMONSTER && pr_hit() < 64)
|
|
|
|
{
|
|
|
|
thing->Howl ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
health--;
|
|
|
|
if (health <= 0 || thing->health <= 0)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (flags3 & MF3_FLOORHUGGER)
|
|
|
|
{
|
|
|
|
if (lastenemy && ! lastenemy->tracer)
|
|
|
|
{
|
|
|
|
lastenemy->tracer = thing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!tracer)
|
|
|
|
{
|
|
|
|
tracer = thing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1; // lightning zaps through all sprites
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lightning Zap ------------------------------------------------------------
|
|
|
|
|
|
|
|
class ALightningZap : public AActor
|
|
|
|
{
|
|
|
|
DECLARE_CLASS (ALightningZap, AActor)
|
|
|
|
public:
|
|
|
|
int SpecialMissileHit (AActor *thing);
|
|
|
|
};
|
|
|
|
|
|
|
|
IMPLEMENT_CLASS (ALightningZap)
|
|
|
|
|
|
|
|
int ALightningZap::SpecialMissileHit (AActor *thing)
|
|
|
|
{
|
|
|
|
AActor *lmo;
|
|
|
|
|
|
|
|
if (thing->flags&MF_SHOOTABLE && thing != target)
|
|
|
|
{
|
|
|
|
lmo = lastenemy;
|
|
|
|
if (lmo)
|
|
|
|
{
|
|
|
|
if (lmo->flags3 & MF3_FLOORHUGGER)
|
|
|
|
{
|
|
|
|
if (lmo->lastenemy && !lmo->lastenemy->tracer)
|
|
|
|
{
|
|
|
|
lmo->lastenemy->tracer = thing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!lmo->tracer)
|
|
|
|
{
|
|
|
|
lmo->tracer = thing;
|
|
|
|
}
|
|
|
|
if (!(level.time&3))
|
|
|
|
{
|
|
|
|
lmo->health--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_LightningReady
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_LightningReady)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
|
|
|
|
DoReadyWeapon(self);
|
|
|
|
if (pr_lightningready() < 160)
|
|
|
|
{
|
|
|
|
S_Sound (self, CHAN_WEAPON, "MageLightningReady", 1, ATTN_NORM);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_LightningClip
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_LightningClip)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
|
|
|
|
AActor *cMo;
|
|
|
|
AActor *target = NULL;
|
|
|
|
int zigZag;
|
|
|
|
|
|
|
|
if (self->flags3 & MF3_FLOORHUGGER)
|
|
|
|
{
|
|
|
|
if (self->lastenemy == NULL)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
self->SetZ(self->floorz);
|
|
|
|
target = self->lastenemy->tracer;
|
|
|
|
}
|
|
|
|
else if (self->flags3 & MF3_CEILINGHUGGER)
|
|
|
|
{
|
|
|
|
self->SetZ(self->ceilingz-self->height);
|
|
|
|
target = self->tracer;
|
|
|
|
}
|
|
|
|
if (self->flags3 & MF3_FLOORHUGGER)
|
|
|
|
{ // floor lightning zig-zags, and forces the ceiling lightning to mimic
|
|
|
|
cMo = self->lastenemy;
|
|
|
|
zigZag = pr_lightningclip();
|
|
|
|
if((zigZag > 128 && self->special1 < 2) || self->special1 < -2)
|
|
|
|
{
|
2016-03-16 11:41:26 +00:00
|
|
|
P_ThrustMobj(self, self->_f_angle()+ANG90, ZAGSPEED);
|
2016-03-01 15:47:10 +00:00
|
|
|
if(cMo)
|
|
|
|
{
|
2016-03-16 11:41:26 +00:00
|
|
|
P_ThrustMobj(cMo, self->_f_angle()+ANG90, ZAGSPEED);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
self->special1++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-03-16 11:41:26 +00:00
|
|
|
P_ThrustMobj(self, self->_f_angle()-ANG90, ZAGSPEED);
|
2016-03-01 15:47:10 +00:00
|
|
|
if(cMo)
|
|
|
|
{
|
2016-03-16 11:41:26 +00:00
|
|
|
P_ThrustMobj(cMo, cMo->_f_angle()-ANG90, ZAGSPEED);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
self->special1--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(target)
|
|
|
|
{
|
|
|
|
if(target->health <= 0)
|
|
|
|
{
|
|
|
|
P_ExplodeMissile(self, NULL, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-03-16 11:41:26 +00:00
|
|
|
self->Angles.Yaw = self->_f_AngleTo(target);
|
2016-03-12 13:11:43 +00:00
|
|
|
self->vel.x = 0;
|
|
|
|
self->vel.y = 0;
|
2016-03-16 11:41:26 +00:00
|
|
|
P_ThrustMobj (self, self->Angles.Yaw, self->Speed>>1);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_LightningZap
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_LightningZap)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
|
|
|
|
PClassActor *lightning = PClass::FindActor(self->GetClass()->MissileName);
|
|
|
|
AActor *mo;
|
|
|
|
fixed_t deltaZ;
|
|
|
|
|
|
|
|
if (lightning == NULL)
|
|
|
|
{
|
|
|
|
lightning = PClass::FindActor(NAME_LightningZap);
|
|
|
|
}
|
|
|
|
|
|
|
|
CALL_ACTION(A_LightningClip, self);
|
|
|
|
|
|
|
|
self->health -= 8;
|
|
|
|
if (self->health <= 0)
|
|
|
|
{
|
|
|
|
self->SetState (self->FindState(NAME_Death));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (self->flags3 & MF3_FLOORHUGGER)
|
|
|
|
{
|
|
|
|
deltaZ = 10*FRACUNIT;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
deltaZ = -10*FRACUNIT;
|
|
|
|
}
|
|
|
|
fixed_t xo = ((pr_zap() - 128)*self->radius / 256);
|
|
|
|
fixed_t yo = ((pr_zap() - 128)*self->radius / 256);
|
|
|
|
|
|
|
|
mo = Spawn(lightning, self->Vec3Offset(xo, yo, deltaZ), ALLOW_REPLACE);
|
|
|
|
if (mo)
|
|
|
|
{
|
|
|
|
mo->lastenemy = self;
|
2016-03-12 13:11:43 +00:00
|
|
|
mo->vel.x = self->vel.x;
|
|
|
|
mo->vel.y = self->vel.y;
|
2016-03-01 15:47:10 +00:00
|
|
|
mo->target = self->target;
|
|
|
|
if (self->flags3 & MF3_FLOORHUGGER)
|
|
|
|
{
|
2016-03-12 13:11:43 +00:00
|
|
|
mo->vel.z = 20*FRACUNIT;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-03-12 13:11:43 +00:00
|
|
|
mo->vel.z = -20*FRACUNIT;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((self->flags3 & MF3_FLOORHUGGER) && pr_zapf() < 160)
|
|
|
|
{
|
|
|
|
S_Sound (self, CHAN_BODY, self->ActiveSound, 1, ATTN_NORM);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_MLightningAttack
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION_PARAMS(AActor, A_MLightningAttack)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
PARAM_CLASS_OPT(floor, AActor) { floor = PClass::FindActor("LightningFloor"); }
|
|
|
|
PARAM_CLASS_OPT(ceiling, AActor) { ceiling = PClass::FindActor("LightningCeiling"); }
|
|
|
|
|
|
|
|
AActor *fmo, *cmo;
|
|
|
|
|
|
|
|
fmo = P_SpawnPlayerMissile (self, floor);
|
|
|
|
cmo = P_SpawnPlayerMissile (self, ceiling);
|
|
|
|
if (fmo)
|
|
|
|
{
|
|
|
|
fmo->special1 = 0;
|
|
|
|
fmo->lastenemy = cmo;
|
|
|
|
CALL_ACTION(A_LightningZap, fmo);
|
|
|
|
}
|
|
|
|
if (cmo)
|
|
|
|
{
|
|
|
|
cmo->tracer = NULL;
|
|
|
|
cmo->lastenemy = fmo;
|
|
|
|
CALL_ACTION(A_LightningZap, cmo);
|
|
|
|
}
|
|
|
|
S_Sound (self, CHAN_BODY, "MageLightningFire", 1, ATTN_NORM);
|
|
|
|
|
|
|
|
if (self->player != NULL)
|
|
|
|
{
|
|
|
|
AWeapon *weapon = self->player->ReadyWeapon;
|
|
|
|
if (weapon != NULL)
|
|
|
|
{
|
|
|
|
weapon->DepleteAmmo (weapon->bAltFire);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_ZapMimic
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_ZapMimic)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
|
|
|
|
AActor *mo;
|
|
|
|
|
|
|
|
mo = self->lastenemy;
|
|
|
|
if (mo)
|
|
|
|
{
|
|
|
|
if (mo->state >= mo->FindState(NAME_Death))
|
|
|
|
{
|
|
|
|
P_ExplodeMissile (self, NULL, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-03-12 13:11:43 +00:00
|
|
|
self->vel.x = mo->vel.x;
|
|
|
|
self->vel.y = mo->vel.y;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_LastZap
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_LastZap)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
PClassActor *lightning = PClass::FindActor(self->GetClass()->MissileName);
|
|
|
|
|
|
|
|
AActor *mo;
|
|
|
|
|
|
|
|
if (lightning == NULL)
|
|
|
|
{
|
|
|
|
lightning = PClass::FindActor(NAME_LightningZap);
|
|
|
|
}
|
|
|
|
mo = Spawn(lightning, self->Pos(), ALLOW_REPLACE);
|
|
|
|
if (mo)
|
|
|
|
{
|
|
|
|
mo->SetState (mo->FindState (NAME_Death));
|
2016-03-12 13:11:43 +00:00
|
|
|
mo->vel.z = 40*FRACUNIT;
|
2016-03-01 15:47:10 +00:00
|
|
|
mo->Damage = NULL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// A_LightningRemove
|
|
|
|
//
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_LightningRemove)
|
|
|
|
{
|
|
|
|
PARAM_ACTION_PROLOGUE;
|
|
|
|
|
|
|
|
AActor *mo;
|
|
|
|
|
|
|
|
mo = self->lastenemy;
|
|
|
|
if (mo)
|
|
|
|
{
|
|
|
|
mo->lastenemy = NULL;
|
|
|
|
P_ExplodeMissile (mo, NULL, NULL);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|