2008-09-15 14:11:05 +00:00
|
|
|
/*
|
2006-02-24 04:48:15 +00:00
|
|
|
#include "actor.h"
|
|
|
|
#include "m_random.h"
|
|
|
|
#include "a_action.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "s_sound.h"
|
|
|
|
#include "m_random.h"
|
|
|
|
#include "a_strifeglobal.h"
|
2008-08-10 20:48:55 +00:00
|
|
|
#include "thingdef/thingdef.h"
|
2008-09-15 14:11:05 +00:00
|
|
|
*/
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
AActor *P_SpawnSubMissile (AActor *source, const PClass *type, AActor *target);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
class ASpectralMonster : public AActor
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2008-08-05 22:51:51 +00:00
|
|
|
DECLARE_CLASS (ASpectralMonster, AActor)
|
|
|
|
public:
|
|
|
|
void Touch (AActor *toucher);
|
2006-02-24 04:48:15 +00:00
|
|
|
};
|
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
IMPLEMENT_CLASS (ASpectralMonster)
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
void ASpectralMonster::Touch (AActor *toucher)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2008-08-05 22:51:51 +00:00
|
|
|
P_DamageMobj (toucher, this, this, 5, NAME_Melee);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-10 20:48:55 +00:00
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_SpectralLightningTail)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2009-06-30 20:57:51 +00:00
|
|
|
AActor *foo = Spawn("SpectralLightningHTail", self->x - self->velx, self->y - self->vely, self->z, ALLOW_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
foo->angle = self->angle;
|
|
|
|
foo->health = self->health;
|
|
|
|
}
|
|
|
|
|
2008-08-10 20:48:55 +00:00
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_SpectralBigBallLightning)
|
2008-07-12 10:59:36 +00:00
|
|
|
{
|
2008-08-05 22:51:51 +00:00
|
|
|
const PClass *cls = PClass::FindClass("SpectralLightningH3");
|
|
|
|
if (cls)
|
|
|
|
{
|
|
|
|
self->angle += ANGLE_90;
|
|
|
|
P_SpawnSubMissile (self, cls, self->target);
|
|
|
|
self->angle += ANGLE_180;
|
|
|
|
P_SpawnSubMissile (self, cls, self->target);
|
|
|
|
self->angle += ANGLE_90;
|
|
|
|
P_SpawnSubMissile (self, cls, self->target);
|
|
|
|
}
|
2008-07-12 10:59:36 +00:00
|
|
|
}
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
static FRandom pr_zap5 ("Zap5");
|
|
|
|
|
2008-08-10 20:48:55 +00:00
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_SpectralLightning)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
AActor *flash;
|
|
|
|
fixed_t x, y;
|
|
|
|
|
|
|
|
if (self->threshold != 0)
|
|
|
|
--self->threshold;
|
|
|
|
|
2009-06-30 20:57:51 +00:00
|
|
|
self->velx += pr_zap5.Random2(3) << FRACBITS;
|
|
|
|
self->vely += pr_zap5.Random2(3) << FRACBITS;
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
x = self->x + pr_zap5.Random2(3) * FRACUNIT * 50;
|
|
|
|
y = self->y + pr_zap5.Random2(3) * FRACUNIT * 50;
|
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
flash = Spawn (self->threshold > 25 ? PClass::FindClass("SpectralLightningV2") :
|
|
|
|
PClass::FindClass("SpectralLightningV1"), x, y, ONCEILINGZ, ALLOW_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
flash->target = self->target;
|
2009-06-30 20:57:51 +00:00
|
|
|
flash->velz = -18*FRACUNIT;
|
2006-02-24 04:48:15 +00:00
|
|
|
flash->health = self->health;
|
|
|
|
|
2008-08-05 22:51:51 +00:00
|
|
|
flash = Spawn("SpectralLightningV2", self->x, self->y, ONCEILINGZ, ALLOW_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
flash->target = self->target;
|
2009-06-30 20:57:51 +00:00
|
|
|
flash->velz = -18*FRACUNIT;
|
2006-02-24 04:48:15 +00:00
|
|
|
flash->health = self->health;
|
|
|
|
}
|
2006-12-06 10:38:47 +00:00
|
|
|
|
|
|
|
// In Strife, this number is stored in the data segment, but it doesn't seem to be
|
|
|
|
// altered anywhere.
|
|
|
|
#define TRACEANGLE (0xe000000)
|
|
|
|
|
2008-08-10 20:48:55 +00:00
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_Tracer2)
|
2006-12-06 10:38:47 +00:00
|
|
|
{
|
|
|
|
AActor *dest;
|
|
|
|
angle_t exact;
|
|
|
|
fixed_t dist;
|
|
|
|
fixed_t slope;
|
|
|
|
|
|
|
|
dest = self->tracer;
|
|
|
|
|
2009-05-30 08:56:40 +00:00
|
|
|
if (!dest || dest->health <= 0 || self->Speed == 0 || !self->CanSeek(dest))
|
2006-12-06 10:38:47 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// change angle
|
|
|
|
exact = R_PointToAngle2 (self->x, self->y, dest->x, dest->y);
|
|
|
|
|
|
|
|
if (exact != self->angle)
|
|
|
|
{
|
|
|
|
if (exact - self->angle > 0x80000000)
|
|
|
|
{
|
|
|
|
self->angle -= TRACEANGLE;
|
|
|
|
if (exact - self->angle < 0x80000000)
|
|
|
|
self->angle = exact;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self->angle += TRACEANGLE;
|
|
|
|
if (exact - self->angle > 0x80000000)
|
|
|
|
self->angle = exact;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
exact = self->angle >> ANGLETOFINESHIFT;
|
2009-06-30 20:57:51 +00:00
|
|
|
self->velx = FixedMul (self->Speed, finecosine[exact]);
|
|
|
|
self->vely = FixedMul (self->Speed, finesine[exact]);
|
2006-12-06 10:38:47 +00:00
|
|
|
|
|
|
|
// change slope
|
|
|
|
dist = P_AproxDistance (dest->x - self->x, dest->y - self->y);
|
|
|
|
dist /= self->Speed;
|
|
|
|
|
|
|
|
if (dist < 1)
|
|
|
|
{
|
|
|
|
dist = 1;
|
|
|
|
}
|
2007-10-19 08:49:02 +00:00
|
|
|
if (dest->height >= 56*FRACUNIT)
|
|
|
|
{
|
|
|
|
slope = (dest->z+40*FRACUNIT - self->z) / dist;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
slope = (dest->z + self->height*2/3 - self->z) / dist;
|
|
|
|
}
|
2009-06-30 20:57:51 +00:00
|
|
|
if (slope < self->velz)
|
2006-12-06 10:38:47 +00:00
|
|
|
{
|
2009-06-30 20:57:51 +00:00
|
|
|
self->velz -= FRACUNIT/8;
|
2006-12-06 10:38:47 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-06-30 20:57:51 +00:00
|
|
|
self->velz += FRACUNIT/8;
|
2006-12-06 10:38:47 +00:00
|
|
|
}
|
|
|
|
}
|