2008-09-15 14:11:05 +00:00
|
|
|
/*
|
2006-02-24 04:48:15 +00:00
|
|
|
#include "actor.h"
|
|
|
|
#include "info.h"
|
|
|
|
#include "p_enemy.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "a_action.h"
|
2008-04-06 17:33:43 +00:00
|
|
|
#include "templates.h"
|
|
|
|
#include "m_bbox.h"
|
2007-05-28 14:46:49 +00:00
|
|
|
#include "thingdef/thingdef.h"
|
2008-09-14 23:54:38 +00:00
|
|
|
#include "doomstat.h"
|
2008-09-15 14:11:05 +00:00
|
|
|
*/
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2008-08-10 22:48:37 +00:00
|
|
|
DECLARE_ACTION(A_SkullAttack)
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2008-08-12 14:30:07 +00:00
|
|
|
static const PClass *GetSpawnType(DECLARE_PARAMINFO)
|
2007-12-06 08:55:17 +00:00
|
|
|
{
|
2008-08-13 22:54:24 +00:00
|
|
|
ACTION_PARAM_START(1);
|
|
|
|
ACTION_PARAM_CLASS(spawntype, 0);
|
|
|
|
|
2007-12-06 08:55:17 +00:00
|
|
|
if (spawntype == NULL) spawntype = PClass::FindClass("LostSoul");
|
|
|
|
return spawntype;
|
|
|
|
}
|
|
|
|
|
2008-04-06 17:33:43 +00:00
|
|
|
|
2011-06-06 13:23:28 +00:00
|
|
|
enum PA_Flags
|
|
|
|
{
|
|
|
|
PAF_NOSKULLATTACK = 1,
|
|
|
|
PAF_AIMFACING = 2,
|
|
|
|
PAF_NOTARGET = 4,
|
|
|
|
};
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
//
|
|
|
|
// A_PainShootSkull
|
|
|
|
// Spawn a lost soul and launch it at the target
|
|
|
|
//
|
2011-06-06 13:23:28 +00:00
|
|
|
void A_PainShootSkull (AActor *self, angle_t angle, const PClass *spawntype, int flags = 0, int limit = -1)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
AActor *other;
|
|
|
|
int prestep;
|
|
|
|
|
2008-03-14 09:27:02 +00:00
|
|
|
if (spawntype == NULL) return;
|
2006-10-31 14:53:21 +00:00
|
|
|
if (self->DamageType==NAME_Massacre) return;
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
// [RH] check to make sure it's not too close to the ceiling
|
2016-01-18 15:49:24 +00:00
|
|
|
if (self->Top() + 8*FRACUNIT > self->ceilingz)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
if (self->flags & MF_FLOAT)
|
|
|
|
{
|
2009-06-30 20:57:51 +00:00
|
|
|
self->velz -= 2*FRACUNIT;
|
2006-02-24 04:48:15 +00:00
|
|
|
self->flags |= MF_INFLOAT;
|
|
|
|
self->flags4 |= MF4_VFRICTION;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// [RH] make this optional
|
2011-06-06 13:23:28 +00:00
|
|
|
if (limit == -1 && (i_compatflags & COMPATF_LIMITPAIN))
|
|
|
|
limit = 21;
|
|
|
|
|
|
|
|
if (limit)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
// count total number of skulls currently on the level
|
2008-11-29 16:37:54 +00:00
|
|
|
// if there are already 21 skulls on the level, don't spit another one
|
2011-06-06 13:23:28 +00:00
|
|
|
int count = limit;
|
2006-02-24 04:48:15 +00:00
|
|
|
FThinkerIterator iterator (spawntype);
|
|
|
|
DThinker *othink;
|
|
|
|
|
|
|
|
while ( (othink = iterator.Next ()) )
|
|
|
|
{
|
|
|
|
if (--count == 0)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// okay, there's room for another one
|
|
|
|
prestep = 4*FRACUNIT +
|
|
|
|
3*(self->radius + GetDefaultByType(spawntype)->radius)/2;
|
2016-01-18 15:49:24 +00:00
|
|
|
|
|
|
|
// NOTE: The following code contains some advance work for line-to-line portals which is currenty inactive.
|
|
|
|
|
|
|
|
fixedvec2 dist = Vec2Angle(prestep, angle);
|
|
|
|
fixedvec3 pos = self->Vec3Offset(dist.x, dist.y, 8 * FRACUNIT, true);
|
|
|
|
fixedvec3 src = { self->X(), self->Y(), self->Z() };
|
|
|
|
|
|
|
|
for (int i = 0; i < 2; i++)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2016-01-18 15:49:24 +00:00
|
|
|
// Check whether the Lost Soul is being fired through a 1-sided // phares
|
|
|
|
// wall or an impassible line, or a "monsters can't cross" line.// |
|
|
|
|
// If it is, then we don't allow the spawn. // V
|
|
|
|
|
|
|
|
FBoundingBox box(MIN(src.x, pos.x), MIN(src.y, pos.y), MAX(src.x, pos.x), MAX(src.y, pos.y));
|
|
|
|
FBlockLinesIterator it(box);
|
|
|
|
line_t *ld;
|
|
|
|
bool inportal = false;
|
|
|
|
|
|
|
|
while ((ld = it.Next()))
|
2008-04-06 17:33:43 +00:00
|
|
|
{
|
2016-01-18 15:49:24 +00:00
|
|
|
if (ld->isLinePortal() && i == 0)
|
2008-04-06 17:33:43 +00:00
|
|
|
{
|
2016-01-18 15:49:24 +00:00
|
|
|
if (P_PointOnLineSidePrecise(src.x, src.y, ld) == 0 &&
|
|
|
|
P_PointOnLineSidePrecise(pos.x, pos.y, ld) == 1)
|
|
|
|
{
|
|
|
|
// crossed a portal line from front to back, we need to repeat the check on the other side as well.
|
|
|
|
inportal = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!(ld->flags & ML_TWOSIDED) ||
|
|
|
|
(ld->flags & (ML_BLOCKING | ML_BLOCKMONSTERS | ML_BLOCKEVERYTHING)))
|
|
|
|
{
|
|
|
|
if (!(box.Left() > ld->bbox[BOXRIGHT] ||
|
|
|
|
box.Right() < ld->bbox[BOXLEFT] ||
|
|
|
|
box.Top() < ld->bbox[BOXBOTTOM] ||
|
|
|
|
box.Bottom() > ld->bbox[BOXTOP]))
|
|
|
|
{
|
|
|
|
if (P_PointOnLineSidePrecise(src.x, src.y, ld) != P_PointOnLineSidePrecise(pos.x, pos.y, ld))
|
|
|
|
return; // line blocks trajectory // ^
|
|
|
|
}
|
2008-04-06 17:33:43 +00:00
|
|
|
}
|
|
|
|
}
|
2016-01-18 15:49:24 +00:00
|
|
|
if (!inportal) break;
|
|
|
|
|
|
|
|
// recalculate position and redo the check on the other side of the portal
|
|
|
|
pos = self->Vec3Offset(dist.x, dist.y, 8 * FRACUNIT, false);
|
|
|
|
src.x = pos.x - dist.x;
|
|
|
|
src.y = pos.y - dist.y;
|
|
|
|
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
2016-01-18 15:49:24 +00:00
|
|
|
other = Spawn (spawntype, pos.x, pos.y, pos.z, ALLOW_REPLACE);
|
2006-02-24 04:48:15 +00:00
|
|
|
|
|
|
|
// Check to see if the new Lost Soul's z value is above the
|
|
|
|
// ceiling of its new sector, or below the floor. If so, kill it.
|
|
|
|
|
2016-01-18 15:49:24 +00:00
|
|
|
if ((other->Z() >
|
|
|
|
(other->Sector->HighestCeiling(other) - other->height)) ||
|
|
|
|
(other->Z() < other->Sector->LowestFloor(other)))
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
// kill it immediately
|
2009-08-07 03:57:03 +00:00
|
|
|
P_DamageMobj (other, self, self, TELEFRAG_DAMAGE, NAME_None);// ^
|
2006-02-24 04:48:15 +00:00
|
|
|
return; // |
|
|
|
|
} // phares
|
|
|
|
|
|
|
|
// Check for movements.
|
|
|
|
|
2016-01-18 15:49:24 +00:00
|
|
|
if (!P_CheckPosition (other, other->X(), other->Y()))
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
// kill it immediately
|
2009-08-07 03:57:03 +00:00
|
|
|
P_DamageMobj (other, self, self, TELEFRAG_DAMAGE, NAME_None);
|
2006-02-24 04:48:15 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// [RH] Lost souls hate the same things as their pain elementals
|
2011-06-06 13:23:28 +00:00
|
|
|
other->CopyFriendliness (self, !(flags & PAF_NOTARGET));
|
2006-02-24 04:48:15 +00:00
|
|
|
|
2011-06-06 13:23:28 +00:00
|
|
|
if (!(flags & PAF_NOSKULLATTACK))
|
|
|
|
A_SkullAttack(other, SKULLSPEED);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// A_PainAttack
|
|
|
|
// Spawn a lost soul and launch it at the target
|
|
|
|
//
|
2008-08-12 14:30:07 +00:00
|
|
|
DEFINE_ACTION_FUNCTION_PARAMS(AActor, A_PainAttack)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
|
|
|
if (!self->target)
|
|
|
|
return;
|
|
|
|
|
2011-06-06 13:23:28 +00:00
|
|
|
ACTION_PARAM_START(4);
|
|
|
|
ACTION_PARAM_CLASS(spawntype, 0);
|
|
|
|
ACTION_PARAM_ANGLE(angle, 1);
|
|
|
|
ACTION_PARAM_INT(flags, 2);
|
|
|
|
ACTION_PARAM_INT(limit, 3);
|
|
|
|
|
|
|
|
if (spawntype == NULL) spawntype = PClass::FindClass("LostSoul");
|
|
|
|
|
|
|
|
if (!(flags & PAF_AIMFACING))
|
|
|
|
A_FaceTarget (self);
|
|
|
|
A_PainShootSkull (self, self->angle+angle, spawntype, flags, limit);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|
|
|
|
|
2008-08-12 14:30:07 +00:00
|
|
|
DEFINE_ACTION_FUNCTION_PARAMS(AActor, A_DualPainAttack)
|
2006-04-11 08:36:23 +00:00
|
|
|
{
|
|
|
|
if (!self->target)
|
|
|
|
return;
|
|
|
|
|
2008-08-12 14:30:07 +00:00
|
|
|
const PClass *spawntype = GetSpawnType(PUSH_PARAMINFO);
|
2006-04-11 08:36:23 +00:00
|
|
|
A_FaceTarget (self);
|
2007-12-06 08:55:17 +00:00
|
|
|
A_PainShootSkull (self, self->angle + ANG45, spawntype);
|
|
|
|
A_PainShootSkull (self, self->angle - ANG45, spawntype);
|
2006-04-11 08:36:23 +00:00
|
|
|
}
|
|
|
|
|
2008-08-12 14:30:07 +00:00
|
|
|
DEFINE_ACTION_FUNCTION_PARAMS(AActor, A_PainDie)
|
2006-02-24 04:48:15 +00:00
|
|
|
{
|
2006-05-03 14:54:48 +00:00
|
|
|
if (self->target != NULL && self->IsFriend (self->target))
|
2006-02-24 04:48:15 +00:00
|
|
|
{ // And I thought you were my friend!
|
|
|
|
self->flags &= ~MF_FRIENDLY;
|
|
|
|
}
|
2008-08-12 14:30:07 +00:00
|
|
|
const PClass *spawntype = GetSpawnType(PUSH_PARAMINFO);
|
2010-04-23 08:12:47 +00:00
|
|
|
A_Unblock(self, true);
|
2007-12-06 08:55:17 +00:00
|
|
|
A_PainShootSkull (self, self->angle + ANG90, spawntype);
|
|
|
|
A_PainShootSkull (self, self->angle + ANG180, spawntype);
|
|
|
|
A_PainShootSkull (self, self->angle + ANG270, spawntype);
|
2006-02-24 04:48:15 +00:00
|
|
|
}
|