2016-03-01 15:47:10 +00:00
|
|
|
|
#include "actor.h"
|
|
|
|
|
#include "info.h"
|
|
|
|
|
#include "gi.h"
|
|
|
|
|
#include "m_random.h"
|
2016-10-12 17:22:33 +00:00
|
|
|
|
#include "vm.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
static FRandom pr_orbit ("Orbit");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Custom bridge --------------------------------------------------------
|
|
|
|
|
/*
|
2016-03-23 17:07:04 +00:00
|
|
|
|
args[0]: Bridge radius, in mapunits
|
2016-03-01 15:47:10 +00:00
|
|
|
|
args[1]: Bridge height, in mapunits
|
|
|
|
|
args[2]: Amount of bridge balls (if 0: Doom bridge)
|
|
|
|
|
args[3]: Rotation speed of bridge balls, in byte angle per seconds, sorta:
|
|
|
|
|
Since an arg is only a byte, it can only go from 0 to 255, while ZDoom's
|
|
|
|
|
BAM go from 0 to 65535. Plus, it needs to be able to go either way. So,
|
|
|
|
|
up to 128, it goes counterclockwise; 129-255 is clockwise, substracting
|
|
|
|
|
256 from it to get the angle. A few example values:
|
|
|
|
|
0: Hexen default
|
|
|
|
|
11: 15<EFBFBD> / seconds
|
|
|
|
|
21: 30<EFBFBD> / seconds
|
|
|
|
|
32: 45<EFBFBD> / seconds
|
|
|
|
|
64: 90<EFBFBD> / seconds
|
|
|
|
|
128: 180<EFBFBD> / seconds
|
|
|
|
|
192: -90<EFBFBD> / seconds
|
|
|
|
|
223: -45<EFBFBD> / seconds
|
|
|
|
|
233: -30<EFBFBD> / seconds
|
|
|
|
|
244: -15<EFBFBD> / seconds
|
|
|
|
|
This value only matters if args[2] is not zero.
|
2016-03-22 21:07:38 +00:00
|
|
|
|
args[4]: Rotation radius of bridge balls, in bridge radius %.
|
|
|
|
|
If 0, use Hexen default: ORBIT_RADIUS, regardless of bridge radius.
|
2016-03-01 15:47:10 +00:00
|
|
|
|
This value only matters if args[2] is not zero.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
class ACustomBridge : public AActor
|
|
|
|
|
{
|
|
|
|
|
DECLARE_CLASS (ACustomBridge, AActor)
|
|
|
|
|
public:
|
|
|
|
|
void BeginPlay ();
|
|
|
|
|
void Destroy();
|
|
|
|
|
};
|
|
|
|
|
|
2016-11-06 18:43:19 +00:00
|
|
|
|
IMPLEMENT_CLASS(ACustomBridge, false, false, false, false)
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
void ACustomBridge::BeginPlay ()
|
|
|
|
|
{
|
|
|
|
|
if (args[2]) // Hexen bridge if there are balls
|
|
|
|
|
{
|
|
|
|
|
SetState(SeeState);
|
2016-03-20 14:04:13 +00:00
|
|
|
|
radius = args[0] ? args[0] : 32;
|
2016-03-20 19:55:06 +00:00
|
|
|
|
Height = args[1] ? args[1] : 2;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
}
|
|
|
|
|
else // No balls? Then a Doom bridge.
|
|
|
|
|
{
|
2016-03-20 14:04:13 +00:00
|
|
|
|
radius = args[0] ? args[0] : 36;
|
2016-03-20 19:55:06 +00:00
|
|
|
|
Height = args[1] ? args[1] : 4;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
RenderStyle = STYLE_Normal;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ACustomBridge::Destroy()
|
|
|
|
|
{
|
|
|
|
|
// Hexen originally just set a flag to make the bridge balls remove themselves in A_BridgeOrbit.
|
|
|
|
|
// But this is not safe with custom bridge balls that do not necessarily call that function.
|
|
|
|
|
// So the best course of action is to look for all bridge balls here and destroy them ourselves.
|
|
|
|
|
|
|
|
|
|
TThinkerIterator<AActor> it;
|
|
|
|
|
AActor *thing;
|
|
|
|
|
|
|
|
|
|
while ((thing = it.Next()))
|
|
|
|
|
{
|
|
|
|
|
if (thing->target == this)
|
|
|
|
|
{
|
|
|
|
|
thing->Destroy();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Super::Destroy();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Action functions for the non-Doom bridge --------------------------------
|
|
|
|
|
|
|
|
|
|
#define ORBIT_RADIUS 15
|
|
|
|
|
|
|
|
|
|
// New bridge stuff
|
|
|
|
|
// Parent
|
|
|
|
|
// special1 true == removing from world
|
|
|
|
|
//
|
|
|
|
|
// Child
|
|
|
|
|
// target pointer to center mobj
|
|
|
|
|
// angle angle of ball
|
|
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION(AActor, A_BridgeOrbit)
|
|
|
|
|
{
|
2016-10-22 15:49:08 +00:00
|
|
|
|
PARAM_SELF_PROLOGUE(AActor);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
if (self->target == NULL)
|
|
|
|
|
{ // Don't crash if somebody spawned this into the world
|
|
|
|
|
// independantly of a Bridge actor.
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
// Set default values
|
|
|
|
|
// Every five tics, Hexen moved the ball 3/256th of a revolution.
|
2016-03-16 11:41:26 +00:00
|
|
|
|
DAngle rotationspeed = 45./32*3/5;
|
2016-03-22 21:07:38 +00:00
|
|
|
|
double rotationradius = ORBIT_RADIUS;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
// If the bridge is custom, set non-default values if any.
|
|
|
|
|
|
|
|
|
|
// Set angular speed; 1--128: counterclockwise rotation ~=1--180<38>; 129--255: clockwise rotation ~= 180--1<>
|
2016-03-16 11:41:26 +00:00
|
|
|
|
if (self->target->args[3] > 128) rotationspeed = 45./32 * (self->target->args[3]-256) / TICRATE;
|
|
|
|
|
else if (self->target->args[3] > 0) rotationspeed = 45./32 * (self->target->args[3]) / TICRATE;
|
2016-03-22 21:07:38 +00:00
|
|
|
|
// Set rotation radius
|
|
|
|
|
if (self->target->args[4]) rotationradius = ((self->target->args[4] * self->target->radius) / 100);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
2016-03-16 11:41:26 +00:00
|
|
|
|
self->Angles.Yaw += rotationspeed;
|
2016-03-22 21:07:38 +00:00
|
|
|
|
self->SetOrigin(self->target->Vec3Angle(rotationradius, self->Angles.Yaw, 0), true);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
self->floorz = self->target->floorz;
|
|
|
|
|
self->ceilingz = self->target->ceilingz;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_ACTION_FUNCTION_PARAMS(AActor, A_BridgeInit)
|
|
|
|
|
{
|
2016-10-22 15:49:08 +00:00
|
|
|
|
PARAM_SELF_PROLOGUE(AActor);
|
2016-10-27 22:32:52 +00:00
|
|
|
|
PARAM_CLASS_DEF(balltype, AActor);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
AActor *ball;
|
|
|
|
|
|
|
|
|
|
if (balltype == NULL)
|
|
|
|
|
{
|
|
|
|
|
balltype = PClass::FindActor("BridgeBall");
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-16 11:41:26 +00:00
|
|
|
|
DAngle startangle = pr_orbit() * (360./256.);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
// Spawn triad into world -- may be more than a triad now.
|
|
|
|
|
int ballcount = self->args[2]==0 ? 3 : self->args[2];
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < ballcount; i++)
|
|
|
|
|
{
|
|
|
|
|
ball = Spawn(balltype, self->Pos(), ALLOW_REPLACE);
|
2016-03-16 11:41:26 +00:00
|
|
|
|
ball->Angles.Yaw = startangle + (45./32) * (256/ballcount) * i;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
ball->target = self;
|
|
|
|
|
CALL_ACTION(A_BridgeOrbit, ball);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Invisible bridge --------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
class AInvisibleBridge : public AActor
|
|
|
|
|
{
|
|
|
|
|
DECLARE_CLASS (AInvisibleBridge, AActor)
|
|
|
|
|
public:
|
|
|
|
|
void BeginPlay ();
|
|
|
|
|
};
|
|
|
|
|
|
2016-11-06 18:43:19 +00:00
|
|
|
|
IMPLEMENT_CLASS(AInvisibleBridge, false, false, false, false)
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
|
|
void AInvisibleBridge::BeginPlay ()
|
|
|
|
|
{
|
|
|
|
|
Super::BeginPlay ();
|
|
|
|
|
if (args[0])
|
2016-03-20 14:04:13 +00:00
|
|
|
|
radius = args[0];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
if (args[1])
|
2016-03-20 19:55:06 +00:00
|
|
|
|
Height = args[1];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
}
|
|
|
|
|
|