mirror of
https://github.com/ZDoom/qzdoom-gpl.git
synced 2025-01-10 02:00:44 +00:00
063c85b157
with an FString now. - Fixed: The music strings in the default level info were never freed and caused memory leaks when used repeatedly. - Fixed: The intermusic string in the level info was never freed. - Fixed: The default fire obituary should only be printed if the damage came from the environment. If it comes from a monster the monster specific obituary should be used instead. - Added custom damage types from the floating point test release. - Changed Pain Elemental's massacre check. Now A_PainDie checks for the damage type and doesn't spawn anything if it is NAME_Massacre. A_PainDie can also be used by other actors so a more generalized approach is needed than hard coding it into the Pain Elemental. - Converted a few of Doom's monsters to DECORATE because I couldn't test the first version of the custom state code with the corpses inheriting from them. - Added custom states from last year's floating point test release and fixed some bugs I found in that code. Unfortunately it wasn't all salvageable and it was easier to recreate some parts from scratch. SVN r368 (trunk)
319 lines
8.6 KiB
C++
319 lines
8.6 KiB
C++
#include "actor.h"
|
|
#include "info.h"
|
|
#include "m_random.h"
|
|
#include "p_enemy.h"
|
|
#include "p_local.h"
|
|
#include "a_sharedglobal.h"
|
|
#include "s_sound.h"
|
|
|
|
static FRandom pr_thrustraise ("ThrustRaise");
|
|
|
|
// Dirt clump (spawned by spike) --------------------------------------------
|
|
|
|
class ADirtClump : public AActor
|
|
{
|
|
DECLARE_ACTOR (ADirtClump, AActor)
|
|
};
|
|
|
|
FState ADirtClump::States[] =
|
|
{
|
|
S_NORMAL (TSPK, 'C', 20, NULL, &States[0])
|
|
};
|
|
|
|
IMPLEMENT_ACTOR (ADirtClump, Hexen, -1, 0)
|
|
PROP_Flags (MF_NOBLOCKMAP)
|
|
PROP_Flags2 (MF2_NOTELEPORT)
|
|
PROP_SpawnState (0)
|
|
END_DEFAULTS
|
|
|
|
// Spike (thrust floor) -----------------------------------------------------
|
|
|
|
void A_ThrustInitUp (AActor *);
|
|
void A_ThrustInitDn (AActor *);
|
|
void A_ThrustRaise (AActor *);
|
|
void A_ThrustLower (AActor *);
|
|
void A_ThrustBlock (AActor *);
|
|
void A_ThrustImpale (AActor *);
|
|
|
|
AActor *tsthing;
|
|
|
|
bool PIT_ThrustStompThing (AActor *thing)
|
|
{
|
|
fixed_t blockdist;
|
|
|
|
if (!(thing->flags & MF_SHOOTABLE) )
|
|
return true;
|
|
|
|
blockdist = thing->radius + tsthing->radius;
|
|
if ( abs(thing->x - tsthing->x) >= blockdist ||
|
|
abs(thing->y - tsthing->y) >= blockdist ||
|
|
(thing->z > tsthing->z+tsthing->height) )
|
|
return true; // didn't hit it
|
|
|
|
if (thing == tsthing)
|
|
return true; // don't clip against self
|
|
|
|
P_DamageMobj (thing, tsthing, tsthing, 10001, NAME_Crush);
|
|
P_TraceBleed (10001, thing);
|
|
tsthing->args[1] = 1; // Mark thrust thing as bloody
|
|
|
|
return true;
|
|
}
|
|
|
|
void P_ThrustSpike (AActor *actor)
|
|
{
|
|
static TArray<AActor *> spikebt;
|
|
|
|
int xl,xh,yl,yh,bx,by;
|
|
int x0,x2,y0,y2;
|
|
|
|
tsthing = actor;
|
|
|
|
x0 = actor->x - actor->radius;
|
|
x2 = actor->x + actor->radius;
|
|
y0 = actor->y - actor->radius;
|
|
y2 = actor->y + actor->radius;
|
|
|
|
xl = (x0 - bmaporgx - MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
xh = (x2 - bmaporgx + MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
yl = (y0 - bmaporgy - MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
yh = (y2 - bmaporgy + MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
|
|
spikebt.Clear();
|
|
|
|
// stomp on any things contacted
|
|
for (bx = xl; bx <= xh; bx++)
|
|
for (by = yl; by <= yh; by++)
|
|
P_BlockThingsIterator (bx, by, PIT_ThrustStompThing, spikebt);
|
|
}
|
|
|
|
// AThrustFloor is just a container for all the spike states.
|
|
// All the real spikes subclass it.
|
|
|
|
class AThrustFloor : public AActor
|
|
{
|
|
DECLARE_ACTOR (AThrustFloor, AActor)
|
|
HAS_OBJECT_POINTERS
|
|
public:
|
|
void Serialize (FArchive &arc);
|
|
|
|
fixed_t GetSinkSpeed () { return 6*FRACUNIT; }
|
|
fixed_t GetRaiseSpeed () { return special2*FRACUNIT; }
|
|
|
|
void Activate (AActor *activator);
|
|
void Deactivate (AActor *activator);
|
|
|
|
ADirtClump *DirtClump;
|
|
};
|
|
|
|
IMPLEMENT_POINTY_CLASS (AThrustFloor)
|
|
DECLARE_POINTER (DirtClump)
|
|
END_POINTERS
|
|
|
|
void AThrustFloor::Serialize (FArchive &arc)
|
|
{
|
|
Super::Serialize (arc);
|
|
arc << DirtClump;
|
|
}
|
|
|
|
FState AThrustFloor::States[] =
|
|
{
|
|
#define S_THRUSTRAISING 0
|
|
S_NORMAL (TSPK, 'A', 2, A_ThrustRaise , &States[S_THRUSTRAISING]),
|
|
|
|
#define S_BTHRUSTRAISING (S_THRUSTRAISING+1)
|
|
S_NORMAL (TSPK, 'B', 2, A_ThrustRaise , &States[S_BTHRUSTRAISING]),
|
|
|
|
#define S_THRUSTIMPALE (S_BTHRUSTRAISING+1)
|
|
S_NORMAL (TSPK, 'A', 2, A_ThrustImpale , &States[S_THRUSTRAISING]),
|
|
|
|
#define S_BTHRUSTIMPALE (S_THRUSTIMPALE+1)
|
|
S_NORMAL (TSPK, 'B', 2, A_ThrustImpale , &States[S_BTHRUSTRAISING]),
|
|
|
|
#define S_THRUSTBLOCK (S_BTHRUSTIMPALE+1)
|
|
S_NORMAL (TSPK, 'A', 10, NULL , &States[S_THRUSTBLOCK]),
|
|
|
|
#define S_BTHRUSTBLOCK (S_THRUSTBLOCK+1)
|
|
S_NORMAL (TSPK, 'B', 10, NULL , &States[S_BTHRUSTBLOCK]),
|
|
|
|
#define S_THRUSTLOWER (S_BTHRUSTBLOCK+1)
|
|
S_NORMAL (TSPK, 'A', 2, A_ThrustLower , &States[S_THRUSTLOWER]),
|
|
|
|
#define S_BTHRUSTLOWER (S_THRUSTLOWER+1)
|
|
S_NORMAL (TSPK, 'B', 2, A_ThrustLower , &States[S_BTHRUSTLOWER]),
|
|
|
|
#define S_THRUSTSTAY (S_BTHRUSTLOWER+1)
|
|
S_NORMAL (TSPK, 'A', -1, NULL , &States[S_THRUSTSTAY]),
|
|
|
|
#define S_BTHRUSTSTAY (S_THRUSTSTAY+1)
|
|
S_NORMAL (TSPK, 'B', -1, NULL , &States[S_BTHRUSTSTAY]),
|
|
|
|
#define S_THRUSTINIT2 (S_BTHRUSTSTAY+1)
|
|
S_NORMAL (TSPK, 'A', 3, NULL , &States[S_THRUSTINIT2+1]),
|
|
S_NORMAL (TSPK, 'A', 4, A_ThrustInitUp , &States[S_THRUSTBLOCK]),
|
|
|
|
#define S_BTHRUSTINIT2 (S_THRUSTINIT2+2)
|
|
S_NORMAL (TSPK, 'B', 3, NULL , &States[S_BTHRUSTINIT2+1]),
|
|
S_NORMAL (TSPK, 'B', 4, A_ThrustInitUp , &States[S_BTHRUSTBLOCK]),
|
|
|
|
#define S_THRUSTINIT1 (S_BTHRUSTINIT2+2)
|
|
S_NORMAL (TSPK, 'A', 3, NULL , &States[S_THRUSTINIT1+1]),
|
|
S_NORMAL (TSPK, 'A', 4, A_ThrustInitDn , &States[S_THRUSTSTAY]),
|
|
|
|
#define S_BTHRUSTINIT1 (S_THRUSTINIT1+2)
|
|
S_NORMAL (TSPK, 'B', 3, NULL , &States[S_BTHRUSTINIT1+1]),
|
|
S_NORMAL (TSPK, 'B', 4, A_ThrustInitDn , &States[S_BTHRUSTSTAY]),
|
|
|
|
#define S_THRUSTRAISE (S_BTHRUSTINIT1+2)
|
|
S_NORMAL (TSPK, 'A', 8, A_ThrustRaise , &States[S_THRUSTRAISE+1]),
|
|
S_NORMAL (TSPK, 'A', 6, A_ThrustRaise , &States[S_THRUSTRAISE+2]),
|
|
S_NORMAL (TSPK, 'A', 4, A_ThrustRaise , &States[S_THRUSTRAISE+3]),
|
|
S_NORMAL (TSPK, 'A', 3, A_ThrustBlock , &States[S_THRUSTIMPALE]),
|
|
|
|
#define S_BTHRUSTRAISE (S_THRUSTRAISE+4)
|
|
S_NORMAL (TSPK, 'B', 8, A_ThrustRaise , &States[S_BTHRUSTRAISE+1]),
|
|
S_NORMAL (TSPK, 'B', 6, A_ThrustRaise , &States[S_BTHRUSTRAISE+2]),
|
|
S_NORMAL (TSPK, 'B', 4, A_ThrustRaise , &States[S_BTHRUSTRAISE+3]),
|
|
S_NORMAL (TSPK, 'B', 3, A_ThrustBlock , &States[S_BTHRUSTIMPALE]),
|
|
|
|
};
|
|
|
|
BEGIN_DEFAULTS (AThrustFloor, Hexen, -1, 0)
|
|
PROP_RadiusFixed (20)
|
|
PROP_HeightFixed (128)
|
|
END_DEFAULTS
|
|
|
|
void AThrustFloor::Activate (AActor *activator)
|
|
{
|
|
if (args[0] == 0)
|
|
{
|
|
S_Sound (this, CHAN_BODY, "ThrustSpikeLower", 1, ATTN_NORM);
|
|
renderflags &= ~RF_INVISIBLE;
|
|
if (args[1])
|
|
SetState (&States[S_BTHRUSTRAISE]);
|
|
else
|
|
SetState (&States[S_THRUSTRAISE]);
|
|
}
|
|
}
|
|
|
|
void AThrustFloor::Deactivate (AActor *activator)
|
|
{
|
|
if (args[0] == 1)
|
|
{
|
|
S_Sound (this, CHAN_BODY, "ThrustSpikeRaise", 1, ATTN_NORM);
|
|
if (args[1])
|
|
SetState (&States[S_BTHRUSTLOWER]);
|
|
else
|
|
SetState (&States[S_THRUSTLOWER]);
|
|
}
|
|
}
|
|
|
|
// Spike up -----------------------------------------------------------------
|
|
|
|
class AThrustFloorUp : public AThrustFloor
|
|
{
|
|
DECLARE_STATELESS_ACTOR (AThrustFloorUp, AThrustFloor)
|
|
};
|
|
|
|
IMPLEMENT_STATELESS_ACTOR (AThrustFloorUp, Hexen, 10091, 104)
|
|
PROP_Flags (MF_SOLID)
|
|
PROP_Flags2 (MF2_NOTELEPORT|MF2_FLOORCLIP)
|
|
PROP_SpawnState (S_THRUSTINIT2)
|
|
END_DEFAULTS
|
|
|
|
// Spike down ---------------------------------------------------------------
|
|
|
|
class AThrustFloorDown : public AThrustFloor
|
|
{
|
|
DECLARE_STATELESS_ACTOR (AThrustFloorDown, AThrustFloor)
|
|
};
|
|
|
|
IMPLEMENT_STATELESS_ACTOR (AThrustFloorDown, Hexen, 10090, 105)
|
|
PROP_Flags2 (MF2_NOTELEPORT|MF2_FLOORCLIP)
|
|
PROP_RenderFlags (RF_INVISIBLE)
|
|
PROP_SpawnState (S_THRUSTINIT1)
|
|
END_DEFAULTS
|
|
|
|
//===========================================================================
|
|
//
|
|
// Thrust floor stuff
|
|
//
|
|
// Thrust Spike Variables
|
|
// DirtClump pointer to dirt clump actor
|
|
// special2 speed of raise
|
|
// args[0] 0 = lowered, 1 = raised
|
|
// args[1] 0 = normal, 1 = bloody
|
|
//===========================================================================
|
|
|
|
void A_ThrustInitUp (AActor *actor)
|
|
{
|
|
actor->special2 = 5; // Raise speed
|
|
actor->args[0] = 1; // Mark as up
|
|
actor->floorclip = 0;
|
|
actor->flags = MF_SOLID;
|
|
actor->flags2 = MF2_NOTELEPORT|MF2_FLOORCLIP;
|
|
actor->special1 = 0L;
|
|
}
|
|
|
|
void A_ThrustInitDn (AActor *actor)
|
|
{
|
|
actor->special2 = 5; // Raise speed
|
|
actor->args[0] = 0; // Mark as down
|
|
actor->floorclip = actor->GetDefault()->height;
|
|
actor->flags = 0;
|
|
actor->flags2 = MF2_NOTELEPORT|MF2_FLOORCLIP;
|
|
actor->renderflags = RF_INVISIBLE;
|
|
static_cast<AThrustFloor *>(actor)->DirtClump =
|
|
Spawn<ADirtClump> (actor->x, actor->y, actor->z, ALLOW_REPLACE);
|
|
}
|
|
|
|
|
|
void A_ThrustRaise (AActor *self)
|
|
{
|
|
AThrustFloor *actor = static_cast<AThrustFloor *>(self);
|
|
|
|
if (A_RaiseMobj (actor))
|
|
{ // Reached it's target height
|
|
actor->args[0] = 1;
|
|
if (actor->args[1])
|
|
actor->SetStateNF (&AThrustFloor::States[S_BTHRUSTINIT2]);
|
|
else
|
|
actor->SetStateNF (&AThrustFloor::States[S_THRUSTINIT2]);
|
|
}
|
|
|
|
// Lose the dirt clump
|
|
if ((actor->floorclip < actor->height) && actor->DirtClump)
|
|
{
|
|
actor->DirtClump->Destroy ();
|
|
actor->DirtClump = NULL;
|
|
}
|
|
|
|
// Spawn some dirt
|
|
if (pr_thrustraise()<40)
|
|
P_SpawnDirt (actor, actor->radius);
|
|
actor->special2++; // Increase raise speed
|
|
}
|
|
|
|
void A_ThrustLower (AActor *actor)
|
|
{
|
|
if (A_SinkMobj (actor))
|
|
{
|
|
actor->args[0] = 0;
|
|
if (actor->args[1])
|
|
actor->SetStateNF (&AThrustFloor::States[S_BTHRUSTINIT1]);
|
|
else
|
|
actor->SetStateNF (&AThrustFloor::States[S_THRUSTINIT1]);
|
|
}
|
|
}
|
|
|
|
void A_ThrustBlock (AActor *actor)
|
|
{
|
|
actor->flags |= MF_SOLID;
|
|
}
|
|
|
|
void A_ThrustImpale (AActor *actor)
|
|
{
|
|
// Impale all shootables in radius
|
|
P_ThrustSpike (actor);
|
|
}
|
|
|