mirror of
https://github.com/ZDoom/qzdoom-gpl.git
synced 2025-01-23 15:20:40 +00:00
386 lines
11 KiB
C++
386 lines
11 KiB
C++
|
#include "templates.h"
|
||
|
#include "actor.h"
|
||
|
#include "info.h"
|
||
|
#include "m_random.h"
|
||
|
#include "s_sound.h"
|
||
|
#include "p_local.h"
|
||
|
#include "p_enemy.h"
|
||
|
#include "gstrings.h"
|
||
|
|
||
|
static FRandom pr_imp ("ImpExplode");
|
||
|
static FRandom pr_impmeatk ("ImpMeAttack");
|
||
|
static FRandom pr_impmsatk ("ImpMsAttack");
|
||
|
static FRandom pr_impmsatk2 ("ImpMsAttack2");
|
||
|
|
||
|
void A_ImpExplode (AActor *);
|
||
|
void A_ImpMeAttack (AActor *);
|
||
|
void A_ImpMsAttack (AActor *);
|
||
|
void A_ImpMsAttack2 (AActor *);
|
||
|
void A_ImpDeath (AActor *);
|
||
|
void A_ImpXDeath1 (AActor *);
|
||
|
void A_ImpXDeath2 (AActor *);
|
||
|
|
||
|
// Heretic imp (as opposed to the Doom variety) -----------------------------
|
||
|
|
||
|
class AHereticImp : public AActor
|
||
|
{
|
||
|
DECLARE_ACTOR (AHereticImp, AActor)
|
||
|
public:
|
||
|
const char *GetObituary ();
|
||
|
const char *GetHitObituary ();
|
||
|
};
|
||
|
|
||
|
FState AHereticImp::States[] =
|
||
|
{
|
||
|
#define S_IMP_LOOK 0
|
||
|
S_NORMAL (IMPX, 'A', 10, A_Look , &States[S_IMP_LOOK+1]),
|
||
|
S_NORMAL (IMPX, 'B', 10, A_Look , &States[S_IMP_LOOK+2]),
|
||
|
S_NORMAL (IMPX, 'C', 10, A_Look , &States[S_IMP_LOOK+3]),
|
||
|
S_NORMAL (IMPX, 'B', 10, A_Look , &States[S_IMP_LOOK+0]),
|
||
|
|
||
|
#define S_IMP_FLY (S_IMP_LOOK+4)
|
||
|
S_NORMAL (IMPX, 'A', 3, A_Chase , &States[S_IMP_FLY+1]),
|
||
|
S_NORMAL (IMPX, 'A', 3, A_Chase , &States[S_IMP_FLY+2]),
|
||
|
S_NORMAL (IMPX, 'B', 3, A_Chase , &States[S_IMP_FLY+3]),
|
||
|
S_NORMAL (IMPX, 'B', 3, A_Chase , &States[S_IMP_FLY+4]),
|
||
|
S_NORMAL (IMPX, 'C', 3, A_Chase , &States[S_IMP_FLY+5]),
|
||
|
S_NORMAL (IMPX, 'C', 3, A_Chase , &States[S_IMP_FLY+6]),
|
||
|
S_NORMAL (IMPX, 'B', 3, A_Chase , &States[S_IMP_FLY+7]),
|
||
|
S_NORMAL (IMPX, 'B', 3, A_Chase , &States[S_IMP_FLY+0]),
|
||
|
|
||
|
#define S_IMP_MEATK (S_IMP_FLY+8)
|
||
|
S_NORMAL (IMPX, 'D', 6, A_FaceTarget , &States[S_IMP_MEATK+1]),
|
||
|
S_NORMAL (IMPX, 'E', 6, A_FaceTarget , &States[S_IMP_MEATK+2]),
|
||
|
S_NORMAL (IMPX, 'F', 6, A_ImpMeAttack , &States[S_IMP_FLY+0]),
|
||
|
|
||
|
#define S_IMP_MSATK1 (S_IMP_MEATK+3)
|
||
|
S_NORMAL (IMPX, 'A', 10, A_FaceTarget , &States[S_IMP_MSATK1+1]),
|
||
|
S_NORMAL (IMPX, 'B', 6, A_ImpMsAttack , &States[S_IMP_MSATK1+2]),
|
||
|
S_NORMAL (IMPX, 'C', 6, NULL , &States[S_IMP_MSATK1+3]),
|
||
|
S_NORMAL (IMPX, 'B', 6, NULL , &States[S_IMP_MSATK1+4]),
|
||
|
S_NORMAL (IMPX, 'A', 6, NULL , &States[S_IMP_MSATK1+5]),
|
||
|
S_NORMAL (IMPX, 'B', 6, NULL , &States[S_IMP_MSATK1+2]),
|
||
|
|
||
|
#define S_IMP_PAIN (S_IMP_MSATK1+6)
|
||
|
S_NORMAL (IMPX, 'G', 3, NULL , &States[S_IMP_PAIN+1]),
|
||
|
S_NORMAL (IMPX, 'G', 3, A_Pain , &States[S_IMP_FLY+0]),
|
||
|
|
||
|
#define S_IMP_DIE (S_IMP_PAIN+2)
|
||
|
S_NORMAL (IMPX, 'G', 4, A_ImpDeath , &States[S_IMP_DIE+1]),
|
||
|
S_NORMAL (IMPX, 'H', 5, NULL , &States[S_IMP_DIE+1]),
|
||
|
|
||
|
#define S_IMP_XDIE (S_IMP_DIE+2)
|
||
|
S_NORMAL (IMPX, 'S', 5, A_ImpXDeath1 , &States[S_IMP_XDIE+1]),
|
||
|
S_NORMAL (IMPX, 'T', 5, NULL , &States[S_IMP_XDIE+2]),
|
||
|
S_NORMAL (IMPX, 'U', 5, NULL , &States[S_IMP_XDIE+3]),
|
||
|
S_NORMAL (IMPX, 'V', 5, A_ImpXDeath2 , &States[S_IMP_XDIE+4]),
|
||
|
S_NORMAL (IMPX, 'W', 5, NULL , &States[S_IMP_XDIE+4]),
|
||
|
|
||
|
#define S_IMP_CRASH (S_IMP_XDIE+5)
|
||
|
S_NORMAL (IMPX, 'I', 7, A_ImpExplode , &States[S_IMP_CRASH+1]),
|
||
|
S_NORMAL (IMPX, 'J', 7, A_Scream , &States[S_IMP_CRASH+2]),
|
||
|
S_NORMAL (IMPX, 'K', 7, NULL , &States[S_IMP_CRASH+3]),
|
||
|
S_NORMAL (IMPX, 'L', -1, NULL , NULL),
|
||
|
|
||
|
#define S_IMP_XCRASH (S_IMP_CRASH+4)
|
||
|
S_NORMAL (IMPX, 'X', 7, NULL , &States[S_IMP_XCRASH+1]),
|
||
|
S_NORMAL (IMPX, 'Y', 7, NULL , &States[S_IMP_XCRASH+2]),
|
||
|
S_NORMAL (IMPX, 'Z', -1, NULL , NULL)
|
||
|
};
|
||
|
|
||
|
IMPLEMENT_ACTOR (AHereticImp, Heretic, 66, 5)
|
||
|
PROP_SpawnHealth (40)
|
||
|
PROP_RadiusFixed (16)
|
||
|
PROP_HeightFixed (36)
|
||
|
PROP_Mass (50)
|
||
|
PROP_SpeedFixed (10)
|
||
|
PROP_PainChance (200)
|
||
|
PROP_Flags (MF_SOLID|MF_SHOOTABLE|MF_FLOAT|MF_NOGRAVITY|MF_COUNTKILL)
|
||
|
PROP_Flags2 (MF2_SPAWNFLOAT|MF2_PASSMOBJ|MF2_PUSHWALL)
|
||
|
PROP_Flags3 (MF3_DONTOVERLAP)
|
||
|
PROP_Flags4 (MF4_MISSILEMORE)
|
||
|
|
||
|
PROP_SpawnState (S_IMP_LOOK)
|
||
|
PROP_SeeState (S_IMP_FLY)
|
||
|
PROP_PainState (S_IMP_PAIN)
|
||
|
PROP_MeleeState (S_IMP_MEATK)
|
||
|
PROP_MissileState (S_IMP_MSATK1)
|
||
|
PROP_CrashState (S_IMP_CRASH)
|
||
|
PROP_DeathState (S_IMP_DIE)
|
||
|
PROP_XDeathState (S_IMP_XDIE)
|
||
|
|
||
|
PROP_SeeSound ("himp/sight")
|
||
|
PROP_AttackSound ("himp/attack")
|
||
|
PROP_PainSound ("himp/pain")
|
||
|
PROP_DeathSound ("himp/death")
|
||
|
PROP_ActiveSound ("himp/active")
|
||
|
END_DEFAULTS
|
||
|
|
||
|
const char *AHereticImp::GetObituary ()
|
||
|
{
|
||
|
return GStrings("OB_HERETICIMP");
|
||
|
}
|
||
|
|
||
|
const char *AHereticImp::GetHitObituary ()
|
||
|
{
|
||
|
return GStrings("OB_HERETICIMPHIT");
|
||
|
}
|
||
|
|
||
|
// Heretic imp leader -------------------------------------------------------
|
||
|
|
||
|
class AHereticImpLeader : public AHereticImp
|
||
|
{
|
||
|
DECLARE_ACTOR (AHereticImpLeader, AHereticImp)
|
||
|
};
|
||
|
|
||
|
FState AHereticImpLeader::States[] =
|
||
|
{
|
||
|
#define S_IMP_MSATK2 0
|
||
|
S_NORMAL (IMPX, 'D', 6, A_FaceTarget , &States[S_IMP_MSATK2+1]),
|
||
|
S_NORMAL (IMPX, 'E', 6, A_FaceTarget , &States[S_IMP_MSATK2+2]),
|
||
|
S_NORMAL (IMPX, 'F', 6, A_ImpMsAttack2 , &AHereticImp::States[S_IMP_FLY]),
|
||
|
};
|
||
|
|
||
|
IMPLEMENT_ACTOR (AHereticImpLeader, Heretic, 5, 7)
|
||
|
PROP_SpawnHealth (80)
|
||
|
|
||
|
PROP_MeleeState (~0)
|
||
|
PROP_MissileState (S_IMP_MSATK2)
|
||
|
PROP_Flags4Clear(MF4_MISSILEMORE) // The imp leader does have a 'normal' missile range!
|
||
|
|
||
|
PROP_AttackSound ("himp/leaderattack")
|
||
|
END_DEFAULTS
|
||
|
|
||
|
// Heretic imp chunk 1 ------------------------------------------------------
|
||
|
|
||
|
class AHereticImpChunk1 : public AActor
|
||
|
{
|
||
|
DECLARE_ACTOR (AHereticImpChunk1, AActor)
|
||
|
};
|
||
|
|
||
|
FState AHereticImpChunk1::States[] =
|
||
|
{
|
||
|
S_NORMAL (IMPX, 'M', 5, NULL , &States[1]),
|
||
|
S_NORMAL (IMPX, 'N', 700, NULL , &States[2]),
|
||
|
S_NORMAL (IMPX, 'O', 700, NULL , NULL)
|
||
|
};
|
||
|
|
||
|
IMPLEMENT_ACTOR (AHereticImpChunk1, Heretic, -1, 0)
|
||
|
PROP_Mass (5)
|
||
|
PROP_Radius (4)
|
||
|
|
||
|
PROP_SpawnState (0)
|
||
|
END_DEFAULTS
|
||
|
|
||
|
// Heretic imp chunk 2 ------------------------------------------------------
|
||
|
|
||
|
class AHereticImpChunk2 : public AActor
|
||
|
{
|
||
|
DECLARE_ACTOR (AHereticImpChunk2, AActor)
|
||
|
};
|
||
|
|
||
|
FState AHereticImpChunk2::States[] =
|
||
|
{
|
||
|
S_NORMAL (IMPX, 'P', 5, NULL , &States[1]),
|
||
|
S_NORMAL (IMPX, 'Q', 700, NULL , &States[2]),
|
||
|
S_NORMAL (IMPX, 'R', 700, NULL , NULL)
|
||
|
};
|
||
|
|
||
|
IMPLEMENT_ACTOR (AHereticImpChunk2, Heretic, -1, 0)
|
||
|
PROP_Mass (5)
|
||
|
PROP_Radius (4)
|
||
|
|
||
|
PROP_SpawnState (0)
|
||
|
END_DEFAULTS
|
||
|
|
||
|
// Heretic imp ball ---------------------------------------------------------
|
||
|
|
||
|
class AHereticImpBall : public AActor
|
||
|
{
|
||
|
DECLARE_ACTOR (AHereticImpBall, AActor)
|
||
|
};
|
||
|
|
||
|
FState AHereticImpBall::States[] =
|
||
|
{
|
||
|
#define S_IMPFX 0
|
||
|
S_BRIGHT (FX10, 'A', 6, NULL , &States[S_IMPFX+1]),
|
||
|
S_BRIGHT (FX10, 'B', 6, NULL , &States[S_IMPFX+2]),
|
||
|
S_BRIGHT (FX10, 'C', 6, NULL , &States[S_IMPFX+0]),
|
||
|
|
||
|
#define S_IMPFXI (S_IMPFX+3)
|
||
|
S_BRIGHT (FX10, 'D', 5, NULL , &States[S_IMPFXI+1]),
|
||
|
S_BRIGHT (FX10, 'E', 5, NULL , &States[S_IMPFXI+2]),
|
||
|
S_BRIGHT (FX10, 'F', 5, NULL , &States[S_IMPFXI+3]),
|
||
|
S_BRIGHT (FX10, 'G', 5, NULL , NULL)
|
||
|
};
|
||
|
|
||
|
IMPLEMENT_ACTOR (AHereticImpBall, Heretic, -1, 10)
|
||
|
PROP_RadiusFixed (8)
|
||
|
PROP_HeightFixed (8)
|
||
|
PROP_SpeedFixed (10)
|
||
|
PROP_Damage (1)
|
||
|
PROP_Flags (MF_MISSILE|MF_DROPOFF|MF_NOGRAVITY)
|
||
|
PROP_Flags2 (MF2_WINDTHRUST|MF2_NOTELEPORT)
|
||
|
PROP_RenderStyle (STYLE_Add)
|
||
|
|
||
|
PROP_SpawnState (S_IMPFX)
|
||
|
PROP_DeathState (S_IMPFXI)
|
||
|
END_DEFAULTS
|
||
|
|
||
|
AT_SPEED_SET (HereticImpBall, speed)
|
||
|
{
|
||
|
SimpleSpeedSetter (AHereticImpBall, 10*FRACUNIT, 20*FRACUNIT, speed);
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpExplode
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpExplode (AActor *self)
|
||
|
{
|
||
|
AActor *chunk;
|
||
|
|
||
|
self->flags &= ~MF_NOGRAVITY;
|
||
|
|
||
|
chunk = Spawn<AHereticImpChunk1> (self->x, self->y, self->z);
|
||
|
chunk->momx = pr_imp.Random2 () << 10;
|
||
|
chunk->momy = pr_imp.Random2 () << 10;
|
||
|
chunk->momz = 9*FRACUNIT;
|
||
|
|
||
|
chunk = Spawn<AHereticImpChunk2> (self->x, self->y, self->z);
|
||
|
chunk->momx = pr_imp.Random2 () << 10;
|
||
|
chunk->momy = pr_imp.Random2 () << 10;
|
||
|
chunk->momz = 9*FRACUNIT;
|
||
|
if (self->special1 == 666)
|
||
|
{ // Extreme death crash
|
||
|
self->SetState (&AHereticImp::States[S_IMP_XCRASH]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpMeAttack
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpMeAttack (AActor *self)
|
||
|
{
|
||
|
if (!self->target)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
S_SoundID (self, CHAN_WEAPON, self->AttackSound, 1, ATTN_NORM);
|
||
|
if (self->CheckMeleeRange ())
|
||
|
{
|
||
|
int damage = 5+(pr_impmeatk()&7);
|
||
|
P_DamageMobj (self->target, self, self, damage, MOD_HIT);
|
||
|
P_TraceBleed (damage, self->target, self);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpMsAttack
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpMsAttack (AActor *self)
|
||
|
{
|
||
|
AActor *dest;
|
||
|
angle_t an;
|
||
|
int dist;
|
||
|
|
||
|
if (!self->target || pr_impmsatk() > 64)
|
||
|
{
|
||
|
self->SetState (self->SeeState);
|
||
|
return;
|
||
|
}
|
||
|
dest = self->target;
|
||
|
self->flags |= MF_SKULLFLY;
|
||
|
S_SoundID (self, CHAN_WEAPON, self->AttackSound, 1, ATTN_NORM);
|
||
|
A_FaceTarget (self);
|
||
|
an = self->angle >> ANGLETOFINESHIFT;
|
||
|
self->momx = FixedMul (12*FRACUNIT, finecosine[an]);
|
||
|
self->momy = FixedMul (12*FRACUNIT, finesine[an]);
|
||
|
dist = P_AproxDistance (dest->x - self->x, dest->y - self->y);
|
||
|
dist = dist/(12*FRACUNIT);
|
||
|
if (dist < 1)
|
||
|
{
|
||
|
dist = 1;
|
||
|
}
|
||
|
self->momz = (dest->z + (dest->height>>1) - self->z)/dist;
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpMsAttack2
|
||
|
//
|
||
|
// Fireball attack of the imp leader.
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpMsAttack2 (AActor *self)
|
||
|
{
|
||
|
if (!self->target)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
S_SoundID (self, CHAN_WEAPON, self->AttackSound, 1, ATTN_NORM);
|
||
|
if (self->CheckMeleeRange ())
|
||
|
{
|
||
|
int damage = 5+(pr_impmsatk2()&7);
|
||
|
P_DamageMobj (self->target, self, self, damage, MOD_HIT);
|
||
|
P_TraceBleed (damage, self->target, self);
|
||
|
return;
|
||
|
}
|
||
|
P_SpawnMissile (self, self->target, RUNTIME_CLASS(AHereticImpBall));
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpDeath
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpDeath (AActor *self)
|
||
|
{
|
||
|
self->flags &= ~MF_SOLID;
|
||
|
self->flags2 |= MF2_FLOORCLIP;
|
||
|
if (self->z <= self->floorz)
|
||
|
{
|
||
|
//self->SetState (&AHereticImp::States[S_IMP_CRASH]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpXDeath1
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpXDeath1 (AActor *self)
|
||
|
{
|
||
|
self->flags &= ~MF_SOLID;
|
||
|
self->flags |= MF_NOGRAVITY;
|
||
|
self->flags2 |= MF2_FLOORCLIP;
|
||
|
self->special1 = 666; // Flag the crash routine
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
//
|
||
|
// PROC A_ImpXDeath2
|
||
|
//
|
||
|
//----------------------------------------------------------------------------
|
||
|
|
||
|
void A_ImpXDeath2 (AActor *self)
|
||
|
{
|
||
|
self->flags &= ~MF_NOGRAVITY;
|
||
|
if (self->z <= self->floorz)
|
||
|
{
|
||
|
self->SetState (&AHereticImp::States[S_IMP_CRASH]);
|
||
|
}
|
||
|
}
|
||
|
|