raze/source/games/duke/src/player_d.cpp

3208 lines
74 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 1996, 2003 - 3D Realms Entertainment
Copyright (C) 2000, 2003 - Matt Saettler (EDuke Enhancements)
2020-06-28 07:03:31 +00:00
Copyright (C) 2020 - Christoph Oelckers
This file is part of Enhanced Duke Nukem 3D version 1.5 - Atomic Edition
Duke Nukem 3D is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
Original Source: 1996 - Todd Replogle
Prepared for public release: 03/21/2003 - Charlie Wiederhold, 3D Realms
EDuke enhancements integrated: 04/13/2003 - Matt Saettler
Note: EDuke source was in transition. Changes are in-progress in the
source as it is released.
*/
//-------------------------------------------------------------------------
#include "ns.h"
#include "global.h"
#include "gamevar.h"
#include "names_d.h"
2020-10-21 17:38:53 +00:00
#include "dukeactor.h"
BEGIN_DUKE_NS
2020-05-18 22:26:07 +00:00
void fireweapon_ww(int snum);
2021-11-21 07:48:36 +00:00
void operateweapon_ww(int snum, ESyncBits actions);
2020-05-18 22:26:07 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-02-07 10:04:19 +00:00
void incur_damage_d(player_struct* p)
{
int damage = 0L, shield_damage = 0L;
p->GetActor()->spr.extra -= p->extra_extra8 >> 8;
damage = p->GetActor()->spr.extra - p->last_extra;
if (damage < 0)
{
p->extra_extra8 = 0;
if (p->shield_amount > 0)
{
shield_damage = damage * (20 + (rand() % 30)) / 100;
damage -= shield_damage;
p->shield_amount += shield_damage;
if (p->shield_amount < 0)
{
damage += p->shield_amount;
p->shield_amount = 0;
}
}
p->GetActor()->spr.extra = p->last_extra + damage;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 19:15:30 +00:00
static void shootfireball(DDukeActor *actor, int p, DVector3 pos, DAngle ang)
{
2022-09-13 19:15:30 +00:00
// World Tour's values for angles and velocities are quite arbitrary...
double vel, zvel;
2021-12-21 19:40:35 +00:00
if (actor->spr.extra >= 0)
actor->spr.shade = -96;
2022-09-13 19:15:30 +00:00
pos.Z -= 2;
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum != BOSS5)
2022-09-13 19:15:30 +00:00
vel = 840/16.;
else {
2022-09-13 19:15:30 +00:00
vel = 968/16.;
pos.Z += 24;
}
if (p < 0)
{
2022-09-13 19:15:30 +00:00
ang += DAngle22_5 / 8 - randomAngle(22.5 / 4);
double scratch;
2020-11-02 22:10:19 +00:00
int j = findplayer(actor, &scratch);
2022-09-13 19:15:30 +00:00
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].opos.Z - pos.Z + 3) * vel) / dist;
}
else
{
zvel = ps[p].horizon.sum().Tan() * 49.;
2022-09-13 20:30:46 +00:00
pos += (ang + DAngle1 * 61).ToVector() * (1024 / 448.);
2022-09-13 19:15:30 +00:00
pos.Z += 3;
}
int sizx = 18;
int sizy = 18;
if (p >= 0)
{
sizx = 7;
sizy = 7;
}
auto spawned = CreateActor(actor->sector(), pos, FIREBALL, -127, sizx, sizy, ang, vel, zvel, actor, (short)4);
if (spawned)
{
2021-12-21 19:33:39 +00:00
spawned->spr.extra += (krand() & 7);
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS5 || p >= 0)
{
spawned->spr.SetScale(0.625, 0.625);
}
spawned->spr.yint = p;
2021-12-21 19:33:39 +00:00
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
spawned->clipdist = 1;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-08-30 19:42:44 +00:00
static void shootflamethrowerflame(DDukeActor* actor, int p, DVector3 spos, DAngle sang)
{
2022-08-30 19:42:44 +00:00
double vel, zvel = 0;
2021-12-21 19:40:35 +00:00
if (actor->spr.extra >= 0)
actor->spr.shade = -96;
2022-08-30 19:42:44 +00:00
vel = 25;
DDukeActor* spawned = nullptr;
if (p < 0)
{
double x;
int j = findplayer(actor, &x);
sang = (ps[j].opos.XY() - spos.XY()).Angle();
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS5)
{
2022-08-30 19:42:44 +00:00
vel = 33;
spos.Z += 24;
}
2021-12-21 19:40:35 +00:00
else if (actor->spr.picnum == BOSS3)
2022-08-30 19:42:44 +00:00
spos.Z -= 32;
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
if (dist != 0)
zvel = (((ps[j].opos.Z - spos.Z) * vel) / dist);
2021-12-21 19:40:35 +00:00
if (badguy(actor) && (actor->spr.hitag & face_player_smart) != 0)
sang = actor->spr.angle + mapangle((krand() & 31) - 16);
if (actor->sector()->lotag == 2 && (krand() % 5) == 0)
spawned = spawn(actor, WATERBUBBLE);
}
else
{
zvel = ps[p].horizon.sum().Tan() * 40.5;
2022-08-30 19:42:44 +00:00
// WTF???
DAngle myang = DAngle90 - (DAngle180 - abs(abs((spos.XY() - ps[p].pos.XY()).Angle() - sang) - DAngle180));
if (ps[p].GetActor()->vel.X != 0)
vel = ((myang / DAngle90) * ps[p].GetActor()->vel.X) + 25;
if (actor->sector()->lotag == 2 && (krand() % 5) == 0)
spawned = spawn(actor, WATERBUBBLE);
}
if (spawned == nullptr)
{
spawned = spawn(actor, FLAMETHROWERFLAME);
if (!spawned) return;
spawned->vel.X = vel;
spawned->vel.Z = zvel;
}
DVector3 offset;
offset.X = (sang + DAngle::fromBuild(118)).Cos() * (1024 / 448.); // Yes, these angles are really different!
offset.Y = (sang + DAngle::fromBuild(112)).Sin() * (1024 / 448.);
offset.Z = -1;
spawned->spr.pos = spos + offset;
2022-08-30 19:42:44 +00:00
spawned->spr.pos.Z--;
spawned->setsector(actor->sector());
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
2022-08-30 19:42:44 +00:00
spawned->spr.angle = sang;
2022-10-05 22:29:10 +00:00
spawned->spr.SetScale(0.03125, 0.03125);
spawned->clipdist = 10;
spawned->spr.yint = p;
spawned->SetOwner(actor);
if (p == -1)
{
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS5)
{
2022-08-30 19:42:44 +00:00
spawned->spr.pos += sang.ToVector() * (128. / 7);
2022-10-05 22:29:10 +00:00
spawned->spr.SetScale(0.15625, 0.15625);
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 19:44:12 +00:00
static void shootknee(DDukeActor* actor, int p, DVector3 pos, DAngle ang)
{
auto sectp = actor->sector();
2022-09-13 19:44:12 +00:00
double zvel;
HitInfo hit{};
if (p >= 0)
{
zvel = ps[p].horizon.sum().Tan() * 16.;
2022-09-13 19:44:12 +00:00
pos.Z += 6;
ang += DAngle1 * 2.64;
}
else
{
2022-09-13 19:44:12 +00:00
double x;
2020-11-02 22:10:19 +00:00
auto pactor = ps[findplayer(actor, &x)].GetActor();
2022-09-13 19:44:12 +00:00
zvel = ((pactor->spr.pos.Z - pos.Z) * 16) / (x + 1/16.);
ang = (pactor->spr.pos.XY() - pos.XY()).Angle();
}
2022-09-13 19:44:12 +00:00
hitscan(pos, sectp, DVector3(ang.ToVector() * 1024, zvel * 64), hit, CLIPMASK1);
if (hit.hitSector == nullptr) return;
2022-09-13 19:44:12 +00:00
if ((pos.XY() - hit.hitpos.XY()).Sum() < 64)
{
if (hit.hitWall || hit.actor())
{
auto knee = CreateActor(hit.hitSector, hit.hitpos, KNEE, -15, 0, 0, ang, 2., 0., actor, 4);
if (knee)
{
knee->spr.extra += (krand() & 7);
if (p >= 0)
{
auto k = spawn(knee, SMALLSMOKE);
2022-02-07 07:47:18 +00:00
if (k) k->spr.pos.Z -= 8;
S_PlayActorSound(KICK_HIT, knee);
}
if (p >= 0 && ps[p].steroids_amount > 0 && ps[p].steroids_amount < 400)
knee->spr.extra += (gs.max_player_health >> 2);
}
if (hit.actor() && hit.actor()->spr.picnum != ACCESSSWITCH && hit.actor()->spr.picnum != ACCESSSWITCH2)
{
fi.checkhitsprite(hit.actor(), knee);
if (p >= 0) fi.checkhitswitch(p, nullptr, hit.actor());
}
else if (hit.hitWall)
{
if (hit.hitWall->cstat & CSTAT_WALL_BOTTOM_SWAP)
if (hit.hitWall->twoSided())
if (hit.hitpos.Z >= hit.hitWall->nextSector()->floorz)
hit.hitWall =hit.hitWall->nextWall();
if (hit.hitWall->picnum != ACCESSSWITCH && hit.hitWall->picnum != ACCESSSWITCH2)
{
2022-08-23 20:39:07 +00:00
fi.checkhitwall(knee, hit.hitWall, hit.hitpos, KNEE);
if (p >= 0) fi.checkhitswitch(p, hit.hitWall, nullptr);
}
}
}
else if (p >= 0 && zvel > 0 && hit.hitSector->lotag == 1)
{
2020-10-23 20:06:02 +00:00
auto splash = spawn(ps[p].GetActor(), WATERSPLASH2);
if (splash)
{
2022-08-29 17:30:19 +00:00
splash->spr.pos.XY() = hit.hitpos.XY();
splash->spr.angle = ps[p].angle.ang; // Total tweek
2022-09-03 08:05:20 +00:00
splash->vel.X = 2;
ssp(actor, CLIPMASK0);
2022-09-03 08:02:25 +00:00
splash->vel.X = 0;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 20:09:27 +00:00
static void shootweapon(DDukeActor *actor, int p, DVector3 pos, DAngle ang, int atwith)
{
auto sectp = actor->sector();
2022-09-13 20:09:27 +00:00
double zvel = 0;
HitInfo hit{};
2021-12-21 19:40:35 +00:00
if (actor->spr.extra >= 0) actor->spr.shade = -96;
if (p >= 0)
{
SetGameVarID(g_iAimAngleVarID, AUTO_AIM_ANGLE, actor, p);
OnEvent(EVENT_GETAUTOAIMANGLE, p, ps[p].GetActor(), -1);
int varval = GetGameVarID(g_iAimAngleVarID, actor, p).value();
DDukeActor* aimed = nullptr;
if (varval > 0)
{
aimed = aim(actor, varval);
}
if (aimed)
{
double dal = ((aimed->spr.ScaleX() * tileHeight(aimed->spr.picnum)) * 0.5) + 5;
switch (aimed->spr.picnum)
{
case GREENSLIME:
case GREENSLIME + 1:
case GREENSLIME + 2:
case GREENSLIME + 3:
case GREENSLIME + 4:
case GREENSLIME + 5:
case GREENSLIME + 6:
case GREENSLIME + 7:
case ROTATEGUN:
2022-09-13 20:09:27 +00:00
dal -= 8;
break;
}
2022-09-13 20:09:27 +00:00
double dist = (ps[p].GetActor()->spr.pos.XY() - aimed->spr.pos.XY()).Length();
zvel = ((aimed->spr.pos.Z - pos.Z - dal) * 16) / dist;
ang = (aimed->spr.pos - pos).Angle();
}
if (isWW2GI())
{
int angRange = 32;
2022-09-13 20:09:27 +00:00
double zRange = 1;
SetGameVarID(g_iAngRangeVarID, 32, actor, p);
SetGameVarID(g_iZRangeVarID, 256, actor, p);
OnEvent(EVENT_GETSHOTRANGE, p, ps[p].GetActor(), -1);
angRange = GetGameVarID(g_iAngRangeVarID, actor, p).value();
2022-09-13 20:09:27 +00:00
zRange = GetGameVarID(g_iZRangeVarID, actor, p).value() / 256.;
2022-09-13 20:09:27 +00:00
ang += DAngle::fromBuild((angRange / 2) - (krand() & (angRange - 1)));
if (aimed == nullptr)
{
// no target
zvel = ps[p].horizon.sum().Tan() * 16.;
}
2022-09-13 20:09:27 +00:00
zvel += (zRange / 2) - krandf(zRange);
}
else if (aimed == nullptr)
{
2022-09-13 20:09:27 +00:00
ang += DAngle22_5 / 8 - randomAngle(22.5 / 4);
zvel = ps[p].horizon.sum().Tan() * 16.;
2022-09-13 20:09:27 +00:00
zvel += 0.5 - krandf(1);
}
2022-09-13 20:09:27 +00:00
pos.Z -= 2;
}
else
{
2022-09-13 20:09:27 +00:00
double x;
int j = findplayer(actor, &x);
2022-09-13 20:09:27 +00:00
pos.Z -= 4;
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].pos.Z - pos.Z) * 16) / dist;
zvel += 0.5 - krandf(1);
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum != BOSS1)
{
2022-09-13 20:09:27 +00:00
ang += DAngle22_5 / 8 - randomAngle(22.5 / 4);
}
else
{
ang = (ps[j].pos - pos).Angle() + DAngle22_5 / 2 - randomAngle(22.5);
}
}
2021-12-21 19:40:35 +00:00
actor->spr.cstat &= ~CSTAT_SPRITE_BLOCK_ALL;
2022-09-13 20:09:27 +00:00
hitscan(pos, sectp, DVector3(ang.ToVector() * 1024, zvel * 64), hit, CLIPMASK1);
2021-12-21 19:40:35 +00:00
actor->spr.cstat |= CSTAT_SPRITE_BLOCK_ALL;
2020-10-23 20:06:02 +00:00
if (hit.hitSector == nullptr) return;
if ((krand() & 15) == 0 && hit.hitSector->lotag == 2)
2022-09-13 21:07:08 +00:00
tracers(hit.hitpos, pos, 8 - (ud.multimode >> 1));
2020-10-24 04:33:31 +00:00
DDukeActor* spark;
if (p >= 0)
{
spark = CreateActor(hit.hitSector, hit.hitpos, SHOTSPARK1, -15, 10, 10, ang, 0., 0., actor, 4);
if (!spark) return;
spark->spr.extra = ScriptCode[gs.actorinfo[atwith].scriptaddress];
spark->spr.extra += (krand() % 6);
if (hit.hitWall == nullptr && hit.actor() == nullptr)
{
if (zvel < 0)
{
if (hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY)
{
spark->spr.SetScale(0, 0);
return;
}
else
fi.checkhitceiling(hit.hitSector);
}
2020-10-23 20:06:02 +00:00
spawn(spark, SMALLSMOKE);
}
if (hit.actor())
{
fi.checkhitsprite(hit.actor(), spark);
if (hit.actor()->isPlayer() && (ud.coop != 1 || ud.ffire == 1))
{
spark->spr.SetScale(0, 0);
auto jib = spawn(spark, JIBS6);
if (jib)
{
2022-02-07 07:47:18 +00:00
jib->spr.pos.Z += 4;
2022-09-03 08:05:20 +00:00
jib->vel.X = 1;
2022-10-05 22:29:10 +00:00
jib->spr.SetScale(0.375, 0.375);
2022-09-13 20:09:27 +00:00
jib->spr.angle += DAngle22_5 / 2 - randomAngle(22.5);
}
}
2020-10-23 20:06:02 +00:00
else spawn(spark, SMALLSMOKE);
if (p >= 0 && (
hit.actor()->spr.picnum == DIPSWITCH ||
hit.actor()->spr.picnum == DIPSWITCH + 1 ||
hit.actor()->spr.picnum == DIPSWITCH2 ||
hit.actor()->spr.picnum == DIPSWITCH2 + 1 ||
hit.actor()->spr.picnum == DIPSWITCH3 ||
hit.actor()->spr.picnum == DIPSWITCH3 + 1 ||
hit.actor()->spr.picnum == HANDSWITCH ||
hit.actor()->spr.picnum == HANDSWITCH + 1))
{
fi.checkhitswitch(p, nullptr, hit.actor());
return;
}
}
else if (hit.hitWall)
{
2020-10-23 20:06:02 +00:00
spawn(spark, SMALLSMOKE);
if (fi.isadoorwall(hit.hitWall->picnum) == 1)
goto SKIPBULLETHOLE;
if (p >= 0 && (
hit.hitWall->picnum == DIPSWITCH ||
hit.hitWall->picnum == DIPSWITCH + 1 ||
hit.hitWall->picnum == DIPSWITCH2 ||
hit.hitWall->picnum == DIPSWITCH2 + 1 ||
hit.hitWall->picnum == DIPSWITCH3 ||
hit.hitWall->picnum == DIPSWITCH3 + 1 ||
hit.hitWall->picnum == HANDSWITCH ||
hit.hitWall->picnum == HANDSWITCH + 1))
{
fi.checkhitswitch(p, hit.hitWall, nullptr);
return;
}
if (hit.hitWall->hitag != 0 || (hit.hitWall->twoSided() && hit.hitWall->nextWall()->hitag != 0))
goto SKIPBULLETHOLE;
if (hit.hitSector && hit.hitSector->lotag == 0)
if (hit.hitWall->overpicnum != BIGFORCE)
if ((hit.hitWall->twoSided() && hit.hitWall->nextSector()->lotag == 0) ||
(!hit.hitWall->twoSided() && hit.hitSector->lotag == 0))
if ((hit.hitWall->cstat & CSTAT_WALL_MASKED) == 0)
{
if (hit.hitWall->twoSided())
{
DukeSectIterator it(hit.hitWall->nextSector());
2020-10-24 04:33:31 +00:00
while (auto l = it.Next())
{
if (l->spr.statnum == STAT_EFFECTOR && l->spr.lotag == SE_13_EXPLOSIVE)
goto SKIPBULLETHOLE;
}
}
2020-10-24 04:33:31 +00:00
DukeStatIterator it(STAT_MISC);
while (auto l = it.Next())
{
if (l->spr.picnum == BULLETHOLE)
if ((l->spr.pos - spark->spr.pos).Length() < 0.75 + krandf(0.5))
goto SKIPBULLETHOLE;
}
2020-10-23 20:06:02 +00:00
auto hole = spawn(spark, BULLETHOLE);
if (hole)
{
2022-09-13 17:56:17 +00:00
hole->vel.X = -1 / 16.;
hole->spr.angle = hit.hitWall->delta().Angle() - DAngle90;
ssp(hole, CLIPMASK0);
hole->spr.cstat2 |= CSTAT2_SPRITE_DECAL;
}
}
SKIPBULLETHOLE:
if (hit.hitWall->cstat & CSTAT_WALL_BOTTOM_SWAP)
if (hit.hitWall->twoSided())
if (hit.hitpos.Z >= hit.hitWall->nextSector()->floorz)
hit.hitWall = hit.hitWall->nextWall();
2022-08-23 20:39:07 +00:00
fi.checkhitwall(spark, hit.hitWall, hit.hitpos, SHOTSPARK1);
}
}
else
{
spark = CreateActor(hit.hitSector, hit.hitpos, SHOTSPARK1, -15, 24, 24, ang, 0., 0., actor, 4);
if (spark)
{
spark->spr.extra = ScriptCode[gs.actorinfo[atwith].scriptaddress];
if (hit.actor())
{
fi.checkhitsprite(hit.actor(), spark);
if (!hit.actor()->isPlayer())
spawn(spark, SMALLSMOKE);
else spark->spr.SetScale(0, 0);
}
else if (hit.hitWall)
2022-08-23 20:39:07 +00:00
fi.checkhitwall(spark, hit.hitWall, hit.hitpos, SHOTSPARK1);
}
}
if ((krand() & 255) < 4)
{
S_PlaySound3D(PISTOL_RICOCHET, spark, hit.hitpos);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 20:21:07 +00:00
static void shootstuff(DDukeActor* actor, int p, DVector3 pos, DAngle ang, int atwith)
{
sectortype* sect = actor->sector();
2022-09-13 20:21:07 +00:00
double vel, zvel;
2022-09-13 19:14:29 +00:00
int scount;
2021-12-21 19:40:35 +00:00
if (actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
if (atwith == SPIT) vel = 292;
else
{
if (atwith == COOLEXPLOSION1)
{
if (actor->spr.picnum == BOSS2) vel = 644 / 16.;
2022-09-13 20:21:07 +00:00
else vel = 348 / 16.;
pos.Z -= 2;
}
else
{
2022-09-13 20:21:07 +00:00
vel = 840 / 16.;
pos.Z -= 2;
}
}
if (p >= 0)
{
auto aimed = aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
double dal = ((aimed->spr.ScaleX() * tileHeight(aimed->spr.picnum)) * 0.5) - 12;
2022-09-13 20:21:07 +00:00
double dist = (ps[p].GetActor()->spr.pos.XY() - aimed->spr.pos.XY()).Length();
zvel = ((aimed->spr.pos.Z - pos.Z - dal) * vel) / dist;
ang = (aimed->spr.pos.XY() - pos.XY()).Angle();
}
else
zvel = ps[p].horizon.sum().Tan() * 49.;
}
else
{
2022-09-13 20:21:07 +00:00
double x;
int j = findplayer(actor, &x);
2022-09-13 20:21:07 +00:00
ang += DAngle22_5 / 8 - randomAngle(22.5 / 4);
#if 1
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].opos.Z - pos.Z + 3) * vel) / dist;
#else
// this is for pitch corrected velocity
auto dist = (ps[j].GetActor()->spr.pos - actor->spr.pos).Resized(vel);
vel = dist.XY().Length();
zvel = dist.Z;
#endif
}
2022-09-13 20:21:07 +00:00
double oldzvel = zvel;
int sizx, sizy;
2022-09-13 20:21:07 +00:00
if (atwith == SPIT)
{
sizx = 18;
sizy = 18;
pos.Z -= 10;
}
else
{
if (atwith == FIRELASER)
{
if (p >= 0)
{
sizx = 34;
sizy = 34;
}
else
{
sizx = 18;
sizy = 18;
}
}
else
{
sizx = 18;
sizy = 18;
}
}
if (p >= 0) sizx = 7, sizy = 7;
while (scount > 0)
{
auto spawned = CreateActor(sect, pos, atwith, -127, sizx, sizy, ang, vel, zvel, actor, 4);
if (!spawned) return;
spawned->spr.extra += (krand() & 7);
if (atwith == COOLEXPLOSION1)
{
spawned->spr.shade = 0;
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS2)
{
2022-09-13 17:56:17 +00:00
auto ovel = spawned->vel.X;
spawned->vel.X = 64;
2020-10-24 04:40:41 +00:00
ssp(spawned, CLIPMASK0);
2022-09-13 17:56:17 +00:00
spawned->vel.X = ovel;
spawned->spr.angle += DAngle22_5 - randomAngle(45);
}
}
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
spawned->clipdist = 1;
2022-09-13 20:21:07 +00:00
ang = actor->spr.angle + DAngle22_5 / 4 - randomAngle(22.5 / 2);
zvel = oldzvel + 2 - krandf(4);
scount--;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 20:30:46 +00:00
static void shootrpg(DDukeActor *actor, int p, DVector3 pos, DAngle ang, int atwith)
{
auto sect = actor->sector();
2022-09-13 20:30:46 +00:00
double vel, zvel;
2022-09-13 20:52:30 +00:00
int scount;
2021-12-21 19:40:35 +00:00
if (actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
2022-09-13 20:30:46 +00:00
vel = 644 / 16.;
DDukeActor* aimed = nullptr;
if (p >= 0)
{
2022-09-14 16:08:23 +00:00
aimed = aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
double dal = ((aimed->spr.ScaleX() * tileHeight(aimed->spr.picnum)) * 0.5) + 8;
2022-09-13 20:30:46 +00:00
double dist = (ps[p].GetActor()->spr.pos.XY() - aimed->spr.pos.XY()).Length();
zvel = ((aimed->spr.pos.Z - pos.Z - dal) * vel) / dist;
if (aimed->spr.picnum != RECON)
ang = (aimed->spr.pos.XY() - pos.XY()).Angle();
}
2022-09-13 20:30:46 +00:00
else
zvel = ps[p].horizon.sum().Tan() * 40.5;
2022-09-13 20:30:46 +00:00
if (atwith == RPG)
2020-10-24 04:40:41 +00:00
S_PlayActorSound(RPG_SHOOT, actor);
}
else
{
2022-09-13 20:30:46 +00:00
double x;
int j = findplayer(actor, &x);
ang = (ps[j].opos.XY() - pos.XY()).Angle();
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS3)
{
2022-09-13 20:30:46 +00:00
double zoffs = 32;
if (isWorldTour()) // Twentieth Anniversary World Tour
2022-09-13 20:30:46 +00:00
zoffs *= (actor->spr.yrepeat / 80.0f);
pos.Z += zoffs;
}
2021-12-21 19:40:35 +00:00
else if (actor->spr.picnum == BOSS2)
{
2022-09-13 20:30:46 +00:00
vel += 8;
double zoffs = 24;
if (isWorldTour()) // Twentieth Anniversary World Tour
2022-09-13 20:30:46 +00:00
zoffs *= (actor->spr.yrepeat / 80.0f);
pos.Z -= zoffs;
}
2022-09-13 20:30:46 +00:00
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].opos.Z - pos.Z) * vel) / dist;
2021-12-21 19:40:35 +00:00
if (badguy(actor) && (actor->spr.hitag & face_player_smart))
2022-09-13 20:30:46 +00:00
ang = actor->spr.angle + randomAngle(DAngle22_5 / 4) - DAngle22_5 / 8;
}
if (p < 0) aimed = nullptr;
2022-09-13 20:30:46 +00:00
auto offset = (ang + DAngle1 * 61).ToVector() * (1024 / 448);
auto spawned = CreateActor(sect, pos.plusZ(-1) + offset, atwith, 0, 14, 14, ang, vel, zvel, actor, 4);
if (!spawned) return;
2021-12-21 19:33:39 +00:00
spawned->spr.extra += (krand() & 7);
if (atwith != FREEZEBLAST)
spawned->temp_actor = aimed;
else
{
spawned->spr.yint = gs.numfreezebounces;
2021-12-21 19:33:39 +00:00
spawned->spr.xrepeat >>= 1;
spawned->spr.yrepeat >>= 1;
2022-09-11 17:28:45 +00:00
spawned->vel.Z -= 0.25;
}
if (p == -1)
{
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum == BOSS3)
{
2022-09-13 20:52:30 +00:00
DVector2 spawnofs(ang.Sin() * 4, ang.Cos() * -4);
2022-09-13 20:30:46 +00:00
DAngle aoffs = DAngle22_5 / 32.;
if ((krand() & 1) != 0)
{
2022-09-13 20:52:30 +00:00
spawnofs = -spawnofs;
2022-09-13 20:30:46 +00:00
aoffs = -aoffs;
}
if (isWorldTour()) // Twentieth Anniversary World Tour
{
float siz = actor->spr.yrepeat / 80.0f;
2022-09-13 20:52:30 +00:00
spawnofs *= siz;
2022-09-13 20:30:46 +00:00
aoffs *= siz;
}
2022-09-13 20:52:30 +00:00
spawned->spr.pos += spawnofs;
2022-09-13 20:30:46 +00:00
spawned->spr.angle += aoffs;
2022-10-05 22:29:10 +00:00
spawned->spr.SetScale(0.65625, 0.65625);
2021-12-21 19:33:39 +00:00
spawned->spr.yrepeat = 42;
}
2021-12-21 19:40:35 +00:00
else if (actor->spr.picnum == BOSS2)
{
2022-09-13 20:52:30 +00:00
DVector2 spawnofs(ang.Sin() * (1024. / 56.), ang.Cos() * -(1024. / 56.));
2022-09-13 20:30:46 +00:00
DAngle aoffs = DAngle22_5 / 16. - DAngle45 + randomAngle(90);
if (isWorldTour()) { // Twentieth Anniversary World Tour
2022-09-13 20:30:46 +00:00
double siz = actor->spr.yrepeat / 70.;
2022-09-13 20:52:30 +00:00
spawnofs *= siz;
2022-09-13 20:30:46 +00:00
aoffs *= siz;
}
2022-09-13 20:52:30 +00:00
spawned->spr.pos += spawnofs;
2022-09-13 20:30:46 +00:00
spawned->spr.angle += aoffs;
2022-10-05 22:29:10 +00:00
spawned->spr.SetScale(0.375, 0.375);
}
else if (atwith != FREEZEBLAST)
{
2022-10-05 22:29:10 +00:00
spawned->spr.SetScale(0.46875, 0.46875);
2021-12-21 19:33:39 +00:00
spawned->spr.extra >>= 2;
}
}
else if ((isWW2GI() && aplWeaponWorksLike(ps[p].curr_weapon, p) == DEVISTATOR_WEAPON) || (!isWW2GI() && ps[p].curr_weapon == DEVISTATOR_WEAPON))
{
2021-12-21 19:33:39 +00:00
spawned->spr.extra >>= 2;
2022-09-13 20:30:46 +00:00
spawned->spr.angle += DAngle22_5 / 8 - randomAngle(22.5 / 4);
2022-09-13 18:40:49 +00:00
spawned->vel.Z += 1 - krandf(2);
if (ps[p].hbomb_hold_delay)
{
2022-09-13 20:52:30 +00:00
DVector2 spawnofs(-ang.Sin()* (1024. / 644.), ang.Cos() * (1024. / 644.));
spawned->spr.pos += spawnofs;
}
else
{
2022-09-13 20:52:30 +00:00
DVector2 spawnofs(ang.Sin()* 4, ang.Cos() * -4);
spawned->spr.pos += spawnofs;
}
2021-12-21 19:33:39 +00:00
spawned->spr.xrepeat >>= 1;
spawned->spr.yrepeat >>= 1;
}
2021-12-21 19:33:39 +00:00
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
if (atwith == RPG)
spawned->clipdist = 1;
else
spawned->clipdist = 10;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 20:39:39 +00:00
static void shootlaser(DDukeActor* actor, int p, DVector3 pos, DAngle ang)
{
auto sectp = actor->sector();
2022-09-13 20:39:39 +00:00
double zvel;
2021-11-17 22:57:17 +00:00
int j;
HitInfo hit{};
if (p >= 0)
zvel = ps[p].horizon.sum().Tan() * 16.;
else zvel = 0;
2022-09-13 20:39:39 +00:00
hitscan(pos, sectp, DVector3(ang.ToVector() * 1024, zvel * 64), hit, CLIPMASK1);
j = 0;
if (hit.actor()) return;
if (hit.hitWall && hit.hitSector)
{
2022-09-13 20:39:39 +00:00
if ((hit.hitpos.XY() - pos.XY()).LengthSquared() < 18.125 * 18.125)
{
if (hit.hitWall->twoSided())
{
if (hit.hitWall->nextSector()->lotag <= 2 && hit.hitSector->lotag <= 2)
j = 1;
}
else if (hit.hitSector->lotag <= 2)
j = 1;
}
if (j == 1)
{
auto bomb = CreateActor(hit.hitSector, hit.hitpos, TRIPBOMB, -16, 4, 5, ang, 0., 0., actor, STAT_STANDABLE);
if (!bomb) return;
if (isWW2GI())
{
int lTripBombControl = GetGameVar("TRIPBOMB_CONTROL", TRIPBOMB_TRIPWIRE, nullptr, -1).value();
if (lTripBombControl & TRIPBOMB_TIMER)
{
int lLifetime = GetGameVar("STICKYBOMB_LIFETIME", NAM_GRENADE_LIFETIME, nullptr, p).value();
int lLifetimeVar = GetGameVar("STICKYBOMB_LIFETIME_VAR", NAM_GRENADE_LIFETIME_VAR, nullptr, p).value();
// set timer. blows up when at zero....
bomb->spr.extra = lLifetime
+ MulScale(krand(), lLifetimeVar, 14)
- lLifetimeVar;
}
}
// this originally used the sprite index as tag to link the laser segments.
// This value is never used again to reference an actor by index. Decouple this for robustness.
ud.bomb_tag = (ud.bomb_tag + 1) & 32767;
bomb->spr.hitag = ud.bomb_tag;
S_PlayActorSound(LASERTRIP_ONWALL, bomb);
2022-09-13 17:56:17 +00:00
bomb->vel.X = -1.25;
ssp(bomb, CLIPMASK0);
bomb->spr.cstat = CSTAT_SPRITE_ALIGNMENT_WALL;
auto delta = -hit.hitWall->delta();
bomb->spr.angle = delta.Angle() - DAngle90;
bomb->temp_angle = bomb->spr.angle;
if (p >= 0)
ps[p].ammo_amount[TRIPBOMB_WEAPON]--;
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 21:10:08 +00:00
static void shootgrowspark(DDukeActor* actor, int p, DVector3 pos, DAngle ang)
{
auto sect = actor->sector();
2022-09-13 21:10:08 +00:00
double zvel;
2021-11-17 22:57:17 +00:00
int k;
HitInfo hit{};
if (p >= 0)
{
auto aimed = aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
double dal = ((aimed->spr.ScaleX() * tileHeight(aimed->spr.picnum)) * 0.5) + 5;
switch (aimed->spr.picnum)
{
case GREENSLIME:
case GREENSLIME + 1:
case GREENSLIME + 2:
case GREENSLIME + 3:
case GREENSLIME + 4:
case GREENSLIME + 5:
case GREENSLIME + 6:
case GREENSLIME + 7:
case ROTATEGUN:
2022-09-13 21:10:08 +00:00
dal -= 8;
break;
}
2022-09-13 21:10:08 +00:00
double dist = (ps[p].GetActor()->spr.pos.XY() - aimed->spr.pos.XY()).Length();
zvel = ((aimed->spr.pos.Z - pos.Z - dal) * 16) / dist;
ang = (aimed->spr.pos.XY() - pos.XY()).Angle();
}
else
{
2022-09-13 21:10:08 +00:00
ang += DAngle22_5 / 8 - randomAngle(22.5 / 4);
zvel = ps[p].horizon.sum().Tan() * 16.;
2022-09-13 21:10:08 +00:00
zvel += 0.5 - krandf(1);
}
2022-09-13 21:10:08 +00:00
pos.Z -= 2;
}
else
{
2022-09-13 21:10:08 +00:00
double x;
2020-11-02 22:10:19 +00:00
int j = findplayer(actor, &x);
2022-09-13 21:10:08 +00:00
pos.Z -= 4;
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].pos.Z - pos.Z) * 16) / dist;
zvel += 0.5 - krandf(1);
ang += DAngle22_5 / 4 - randomAngle(22.5 / 2);
}
k = 0;
//RESHOOTGROW:
2021-12-21 19:40:35 +00:00
actor->spr.cstat &= ~CSTAT_SPRITE_BLOCK_ALL;
2022-09-13 21:10:08 +00:00
hitscan(pos, sect, DVector3(ang.ToVector() * 1024, zvel * 64), hit, CLIPMASK1);
2021-12-21 19:40:35 +00:00
actor->spr.cstat |= CSTAT_SPRITE_BLOCK_ALL;
auto spark = CreateActor(sect, hit.hitpos, GROWSPARK, -16, 28, 28, ang, 0., 0., actor, 1);
if (!spark) return;
spark->spr.pal = 2;
spark->spr.cstat |= CSTAT_SPRITE_YCENTER | CSTAT_SPRITE_TRANSLUCENT;
2022-10-05 22:29:10 +00:00
spark->spr.SetScale(REPEAT_SCALE, REPEAT_SCALE);
if (hit.hitWall == nullptr && hit.actor() == nullptr && hit.hitSector != nullptr)
{
if (zvel < 0 && (hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY) == 0)
fi.checkhitceiling(hit.hitSector);
}
else if (hit.actor() != nullptr) fi.checkhitsprite(hit.actor(), spark);
else if (hit.hitWall != nullptr)
{
if (hit.hitWall->picnum != ACCESSSWITCH && hit.hitWall->picnum != ACCESSSWITCH2)
{
2022-08-23 20:39:07 +00:00
fi.checkhitwall(spark, hit.hitWall, hit.hitpos, GROWSPARK);
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 21:14:20 +00:00
static void shootmortar(DDukeActor* actor, int p, const DVector3& pos, DAngle ang, int atwith)
{
auto sect = actor->sector();
if (actor->spr.extra >= 0) actor->spr.shade = -96;
2022-09-13 21:14:20 +00:00
double x;
auto plActor = ps[findplayer(actor, &x)].GetActor();
2022-09-13 21:14:20 +00:00
x = (plActor->spr.pos.XY() - actor->spr.pos.XY()).Length();
2022-09-13 21:14:20 +00:00
double zvel = -x * 0.5;
2022-09-13 21:14:20 +00:00
if (zvel < -8)
zvel = -4;
double vel = x / 16.;
CreateActor(sect, pos.plusZ(-6) + ang.ToVector() * 4, atwith, -64, 32, 32, ang, vel, zvel, actor, 1);
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-13 21:20:00 +00:00
static void shootshrinker(DDukeActor* actor, int p, const DVector3& pos, DAngle ang, int atwith)
{
2022-09-13 21:20:00 +00:00
double zvel;
if (actor->spr.extra >= 0) actor->spr.shade = -96;
if (p >= 0)
{
auto aimed = isNamWW2GI() ? nullptr : aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
double dal = ((aimed->spr.ScaleX() * tileHeight(aimed->spr.picnum)) * 0.5);
2022-09-13 21:20:00 +00:00
double dist = (ps[p].GetActor()->spr.pos.XY() - aimed->spr.pos.XY()).Length();
zvel = ((aimed->spr.pos.Z - pos.Z - dal - 4) * 48) / dist;
ang = (aimed->spr.pos.XY() - pos.XY()).Angle();
}
2022-09-13 21:20:00 +00:00
else
zvel = ps[p].horizon.sum().Tan() * 49.;
}
else if (actor->spr.statnum != 3)
{
double x;
int j = findplayer(actor, &x);
2022-09-13 21:20:00 +00:00
double dist = (ps[j].GetActor()->spr.pos.XY() - actor->spr.pos.XY()).Length();
zvel = ((ps[j].pos.Z - pos.Z) * 32) / dist;
}
else zvel = 0;
2022-09-13 21:20:00 +00:00
auto spawned = CreateActor(actor->sector(),
pos.plusZ(2) + ang.ToVector() * 0.25, SHRINKSPARK, -16, 28, 28, ang, 48., zvel, actor, 4);
if (spawned)
{
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
spawned->clipdist = 8;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void shoot_d(DDukeActor* actor, int atwith)
{
2022-09-13 22:02:49 +00:00
int p;
DVector3 spos;
DAngle sang;
auto const sect = actor->sector();
if (actor->isPlayer())
{
p = actor->PlayerIndex();
}
else
{
p = -1;
}
SetGameVarID(g_iAtWithVarID, atwith, actor, p);
SetGameVarID(g_iReturnVarID, 0, actor, p);
OnEvent(EVENT_SHOOT, p, ps[p].GetActor(), -1);
if (GetGameVarID(g_iReturnVarID, actor, p).safeValue() != 0)
{
return;
}
if (actor->isPlayer())
{
2022-09-13 22:02:49 +00:00
spos = ps[p].pos.plusZ(ps[p].pyoff + 4);
sang = ps[p].angle.ang;
ps[p].crack_time = CRACK_TIME;
}
else
{
2022-09-13 22:02:49 +00:00
sang = actor->spr.angle;
spos = actor->spr.pos.plusZ(-(actor->spr.ScaleY() * tileHeight(actor->spr.picnum) * 0.5) + 4);
2022-09-13 22:02:49 +00:00
2021-12-21 19:40:35 +00:00
if (actor->spr.picnum != ROTATEGUN)
{
2022-09-13 22:02:49 +00:00
spos.Z -= 7;
2021-12-21 19:40:35 +00:00
if (badguy(actor) && actor->spr.picnum != COMMANDER)
{
2022-09-13 22:02:49 +00:00
spos.X -= (sang + DAngle22_5 * 0.75).Sin() * 8;
spos.Y += (sang + DAngle22_5 * 0.75).Cos() * 8;
}
}
}
if (isWorldTour())
{ // Twentieth Anniversary World Tour
switch (atwith)
{
case FIREBALL:
2022-09-13 19:15:30 +00:00
shootfireball(actor, p, spos, sang);
return;
case FLAMETHROWERFLAME:
2022-08-30 19:42:44 +00:00
shootflamethrowerflame(actor, p, spos, sang);
return;
case FIREFLY: // BOSS5 shot
{
auto k = spawn(actor, atwith);
if (k)
{
k->setsector(sect);
2022-08-30 19:43:34 +00:00
k->spr.pos = spos;
k->spr.angle = sang;
2022-09-13 17:56:17 +00:00
k->vel.X = 500 / 16.;
2022-09-03 08:02:25 +00:00
k->vel.Z = 0;
}
return;
}
}
}
switch (atwith)
{
case BLOODSPLAT1:
case BLOODSPLAT2:
case BLOODSPLAT3:
case BLOODSPLAT4:
2022-09-13 18:54:34 +00:00
shootbloodsplat(actor, p, spos, sang, atwith, BIGFORCE, OOZFILTER, NEWBEAST);
break;
case KNEE:
2022-09-13 19:44:12 +00:00
shootknee(actor, p, spos, sang);
break;
case SHOTSPARK1:
case SHOTGUN:
case CHAINGUN:
2022-09-13 20:09:27 +00:00
shootweapon(actor, p, spos, sang, atwith);
return;
case FIRELASER:
case SPIT:
case COOLEXPLOSION1:
2022-09-13 20:21:07 +00:00
shootstuff(actor, p, spos, sang, atwith);
return;
case FREEZEBLAST:
2022-09-13 22:02:49 +00:00
spos.Z += 3;
[[fallthrough]];
2022-09-13 22:02:49 +00:00
case RPG:
2022-09-13 20:30:46 +00:00
shootrpg(actor, p, spos, sang, atwith);
break;
case HANDHOLDINGLASER:
2022-09-13 20:39:39 +00:00
shootlaser(actor, p, spos, sang);
return;
case BOUNCEMINE:
case MORTER:
2022-09-13 21:14:20 +00:00
shootmortar(actor, p, spos, sang, atwith);
return;
case GROWSPARK:
2022-09-13 21:10:08 +00:00
shootgrowspark(actor, p, spos, sang);
break;
case SHRINKER:
2022-09-13 21:20:00 +00:00
shootshrinker(actor, p, spos, sang, atwith);
break;
}
return;
}
2020-05-17 11:25:39 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void selectweapon_d(int snum, int weap) // playernum, weaponnum
2020-05-17 11:25:39 +00:00
{
int i, j, k;
2020-05-17 11:25:39 +00:00
auto p = &ps[snum];
if (p->last_pissed_time <= (26 * 218) && p->show_empty_weapon == 0 && p->kickback_pic == 0 && p->quick_kick == 0 && p->GetActor()->spr.xrepeat > 32 && p->access_incs == 0 && p->knee_incs == 0)
2020-05-17 11:25:39 +00:00
{
if ((p->weapon_pos == 0 || (p->holster_weapon && p->weapon_pos == -9)))
{
if (weap == WeaponSel_Alt)
{
j = p->curr_weapon;
switch (p->curr_weapon)
{
case SHRINKER_WEAPON:
if (p->ammo_amount[GROW_WEAPON] > 0 && p->gotweapon[GROW_WEAPON] && isPlutoPak())
{
j = GROW_WEAPON;
p->subweapon |= (1 << GROW_WEAPON);
}
break;
case GROW_WEAPON:
if (p->ammo_amount[SHRINKER_WEAPON] > 0 && p->gotweapon[SHRINKER_WEAPON])
{
j = SHRINKER_WEAPON;
p->subweapon &= ~(1 << GROW_WEAPON);
}
break;
case FREEZE_WEAPON:
if (p->ammo_amount[FLAMETHROWER_WEAPON] > 0 && p->gotweapon[FLAMETHROWER_WEAPON] && isWorldTour())
{
j = FLAMETHROWER_WEAPON;
p->subweapon |= (1 << FLAMETHROWER_WEAPON);
}
break;
case FLAMETHROWER_WEAPON:
if (p->ammo_amount[FREEZE_WEAPON] > 0 && p->gotweapon[FREEZE_WEAPON])
{
j = FREEZE_WEAPON;
p->subweapon &= ~(1 << FLAMETHROWER_WEAPON);
}
break;
default:
break;
}
}
else if (weap == WeaponSel_Next || weap == WeaponSel_Prev)
2020-05-17 11:25:39 +00:00
{
k = p->curr_weapon;
j = (weap == WeaponSel_Prev ? -1 : 1); // JBF: prev (-1) or next (1) weapon choice
2020-05-17 11:25:39 +00:00
i = 0;
while ((k >= 0 && k < 10) || (isPlutoPak() && k == GROW_WEAPON && (p->subweapon & (1 << GROW_WEAPON)) != 0)
|| (isWorldTour() && k == FLAMETHROWER_WEAPON && (p->subweapon & (1 << FLAMETHROWER_WEAPON)) != 0))
2020-05-17 11:25:39 +00:00
{
if (k == FLAMETHROWER_WEAPON) //Twentieth Anniversary World Tour
{
if (j == -1) k = TRIPBOMB_WEAPON;
else k = PISTOL_WEAPON;
}
else if (k == GROW_WEAPON) // JBF: this is handling next/previous with the grower selected
2020-05-17 11:25:39 +00:00
{
2021-10-08 17:21:29 +00:00
if (j == -1)
2020-05-17 11:25:39 +00:00
k = 5;
else k = 7;
}
else
{
k += j;
// JBF 20040116: so we don't select grower with v1.3d
if (isPlutoPak() && k == SHRINKER_WEAPON && (p->subweapon & (1 << GROW_WEAPON))) // JBF: activates grower
k = GROW_WEAPON; // if enabled
if (isWorldTour() && k == FREEZE_WEAPON && (p->subweapon & (1 << FLAMETHROWER_WEAPON)) != 0)
k = FLAMETHROWER_WEAPON;
2020-05-17 11:25:39 +00:00
}
if (k == -1) k = 9;
else if (k == 10) k = 0;
if (p->gotweapon[k] && p->ammo_amount[k] > 0)
{
if (isPlutoPak()) // JBF 20040116: so we don't select grower with v1.3d
{
2020-05-17 11:25:39 +00:00
if (k == SHRINKER_WEAPON && (p->subweapon & (1 << GROW_WEAPON)))
k = GROW_WEAPON;
if (isWorldTour() && k == FREEZE_WEAPON && (p->subweapon & (1 << FLAMETHROWER_WEAPON)) != 0)
k = FLAMETHROWER_WEAPON;
}
2020-05-17 11:25:39 +00:00
j = k;
break;
}
else if (isPlutoPak() && k == GROW_WEAPON && p->ammo_amount[GROW_WEAPON] == 0 && p->gotweapon[SHRINKER_WEAPON] && p->ammo_amount[SHRINKER_WEAPON] > 0) // JBF 20040116: added isPlutoPak() so we don't select grower with v1.3d
{
j = SHRINKER_WEAPON;
p->subweapon &= ~(1 << GROW_WEAPON);
break;
}
else if (isPlutoPak() && k == SHRINKER_WEAPON && p->ammo_amount[SHRINKER_WEAPON] == 0 && p->gotweapon[SHRINKER_WEAPON] && p->ammo_amount[GROW_WEAPON] > 0) // JBF 20040116: added isPlutoPak() so we don't select grower with v1.3d
{
j = GROW_WEAPON;
p->subweapon |= (1 << GROW_WEAPON);
break;
}
//Twentieth Anniversary World Tour
else if (isWorldTour() && k == FLAMETHROWER_WEAPON && p->ammo_amount[FLAMETHROWER_WEAPON] == 0 && p->gotweapon[FREEZE_WEAPON] && p->ammo_amount[FREEZE_WEAPON] > 0)
{
j = FREEZE_WEAPON;
p->subweapon &= ~(1 << FLAMETHROWER_WEAPON);
break;
}
else if (isWorldTour() && k == FREEZE_WEAPON && p->ammo_amount[FREEZE_WEAPON] == 0 && p->gotweapon[FLAMETHROWER_WEAPON] && p->ammo_amount[FLAMETHROWER_WEAPON] > 0)
{
j = FLAMETHROWER_WEAPON;
p->subweapon |= (1 << FLAMETHROWER_WEAPON);
break;
}
2020-05-17 11:25:39 +00:00
i++; // absolutely no weapons, so use foot
if (i == 10)
{
fi.addweapon(p, KNEE_WEAPON);
break;
}
}
}
else j = weap - 1;
2020-05-17 11:25:39 +00:00
k = -1;
if (j == HANDBOMB_WEAPON && p->ammo_amount[HANDBOMB_WEAPON] == 0)
{
2020-10-25 05:34:25 +00:00
DukeStatIterator it(STAT_ACTOR);
while (auto act = it.Next())
2020-05-17 11:25:39 +00:00
{
if (act->spr.picnum == HEAVYHBOMB && act->GetOwner() == p->GetActor())
2020-05-17 11:25:39 +00:00
{
p->gotweapon[HANDBOMB_WEAPON] = true;
2020-05-17 11:25:39 +00:00
j = HANDREMOTE_WEAPON;
break;
}
}
}
//Twentieth Anniversary World Tour
if (j == FREEZE_WEAPON && isWorldTour())
{
if (p->curr_weapon != FLAMETHROWER_WEAPON && p->curr_weapon != FREEZE_WEAPON)
{
if (p->ammo_amount[FLAMETHROWER_WEAPON] > 0)
{
if ((p->subweapon & (1 << FLAMETHROWER_WEAPON)) == (1 << FLAMETHROWER_WEAPON))
j = FLAMETHROWER_WEAPON;
else if (p->ammo_amount[FREEZE_WEAPON] == 0)
{
j = FLAMETHROWER_WEAPON;
p->subweapon |= (1 << FLAMETHROWER_WEAPON);
}
}
else if (p->ammo_amount[FREEZE_WEAPON] > 0)
p->subweapon &= ~(1 << FLAMETHROWER_WEAPON);
}
else if (p->curr_weapon == FREEZE_WEAPON)
{
p->subweapon |= (1 << FLAMETHROWER_WEAPON);
j = FLAMETHROWER_WEAPON;
}
else
p->subweapon &= ~(1 << FLAMETHROWER_WEAPON);
}
if (j == SHRINKER_WEAPON && isPlutoPak()) // JBF 20040116: so we don't select the grower with v1.3d
2020-05-17 11:25:39 +00:00
{
if (p->curr_weapon != GROW_WEAPON && p->curr_weapon != SHRINKER_WEAPON)
{
if (p->ammo_amount[GROW_WEAPON] > 0)
{
if ((p->subweapon & (1 << GROW_WEAPON)) == (1 << GROW_WEAPON))
j = GROW_WEAPON;
else if (p->ammo_amount[SHRINKER_WEAPON] == 0)
{
j = GROW_WEAPON;
p->subweapon |= (1 << GROW_WEAPON);
}
}
else if (p->ammo_amount[SHRINKER_WEAPON] > 0)
p->subweapon &= ~(1 << GROW_WEAPON);
}
else if (p->curr_weapon == SHRINKER_WEAPON)
{
p->subweapon |= (1 << GROW_WEAPON);
j = GROW_WEAPON;
}
else
p->subweapon &= ~(1 << GROW_WEAPON);
}
if (p->holster_weapon)
{
PlayerSetInput(snum, SB_HOLSTER);
p->oweapon_pos = p->weapon_pos = -9;
2020-05-17 11:25:39 +00:00
}
2021-10-08 17:21:29 +00:00
else if (j >= MIN_WEAPON && p->gotweapon[j] && p->curr_weapon != j) switch (j)
2020-05-17 11:25:39 +00:00
{
case KNEE_WEAPON:
fi.addweapon(p, KNEE_WEAPON);
break;
case PISTOL_WEAPON:
case SHOTGUN_WEAPON:
case CHAINGUN_WEAPON:
case RPG_WEAPON:
case DEVISTATOR_WEAPON:
case FREEZE_WEAPON:
case FLAMETHROWER_WEAPON:
2020-05-17 11:25:39 +00:00
case GROW_WEAPON:
case SHRINKER_WEAPON:
if (p->ammo_amount[j] == 0 && p->show_empty_weapon == 0)
{
p->show_empty_weapon = 32;
p->last_full_weapon = p->curr_weapon;
}
fi.addweapon(p, j);
break;
case HANDREMOTE_WEAPON:
if (k >= 0) // Found in list of [1]'s
{
p->curr_weapon = HANDREMOTE_WEAPON;
p->last_weapon = -1;
p->oweapon_pos = p->weapon_pos = 10;
2020-05-17 11:25:39 +00:00
}
break;
case HANDBOMB_WEAPON:
if (p->ammo_amount[HANDBOMB_WEAPON] > 0 && p->gotweapon[HANDBOMB_WEAPON])
fi.addweapon(p, HANDBOMB_WEAPON);
break;
case TRIPBOMB_WEAPON:
if (p->ammo_amount[TRIPBOMB_WEAPON] > 0 && p->gotweapon[TRIPBOMB_WEAPON])
fi.addweapon(p, TRIPBOMB_WEAPON);
break;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-02-07 10:04:19 +00:00
int doincrements_d(player_struct* p)
{
int snum;
2020-11-02 19:24:07 +00:00
auto pact = p->GetActor();
snum = pact->PlayerIndex();
p->player_par++;
if (p->invdisptime > 0)
p->invdisptime--;
if (p->tipincs > 0)
{
p->otipincs = p->tipincs;
p->tipincs--;
}
if (p->last_pissed_time > 0)
{
p->last_pissed_time--;
if (p->last_pissed_time == (26 * 219))
{
2020-11-02 19:24:07 +00:00
S_PlayActorSound(FLUSH_TOILET, pact);
if (snum == screenpeek || ud.coop == 1)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_PISSRELIEF, pact);
}
if (p->last_pissed_time == (26 * 218))
{
p->holster_weapon = 0;
p->oweapon_pos = p->weapon_pos = 10;
}
}
if (p->crack_time > 0)
{
p->crack_time--;
if (p->crack_time == 0)
{
p->knuckle_incs = 1;
p->crack_time = CRACK_TIME;
}
}
if (p->steroids_amount > 0 && p->steroids_amount < 400)
{
p->steroids_amount--;
if (p->steroids_amount == 0)
checkavailinven(p);
if (!(p->steroids_amount & 7))
if (snum == screenpeek || ud.coop == 1)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_HARTBEAT, pact);
}
if (p->heat_on && p->heat_amount > 0)
{
p->heat_amount--;
if (p->heat_amount == 0)
{
p->heat_on = 0;
checkavailinven(p);
2020-11-02 19:24:07 +00:00
S_PlayActorSound(NITEVISION_ONOFF, pact);
}
}
if (p->holoduke_on != nullptr)
{
p->holoduke_amount--;
if (p->holoduke_amount <= 0)
{
2020-11-02 19:24:07 +00:00
S_PlayActorSound(TELEPORTER, pact);
p->holoduke_on = nullptr;
checkavailinven(p);
}
}
if (p->jetpack_on && p->jetpack_amount > 0)
{
p->jetpack_amount--;
if (p->jetpack_amount <= 0)
{
p->jetpack_on = 0;
checkavailinven(p);
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_JETPACK_OFF, pact);
S_StopSound(DUKE_JETPACK_IDLE, pact);
S_StopSound(DUKE_JETPACK_ON, pact);
}
}
if (p->quick_kick > 0 && p->GetActor()->spr.pal != 1)
{
p->last_quick_kick = p->quick_kick + 1;
p->quick_kick--;
if (p->quick_kick == 8)
fi.shoot(p->GetActor(), KNEE);
}
else if (p->last_quick_kick > 0)
p->last_quick_kick--;
if (p->access_incs && p->GetActor()->spr.pal != 1)
{
p->oaccess_incs = p->access_incs;
p->access_incs++;
if (p->GetActor()->spr.extra <= 0)
p->access_incs = 12;
if (p->access_incs == 12)
{
2020-10-25 05:34:25 +00:00
if (p->access_spritenum != nullptr)
{
fi.checkhitswitch(snum, nullptr, p->access_spritenum);
switch (p->access_spritenum->spr.pal)
{
case 0:p->got_access &= (0xffff - 0x1); break;
case 21:p->got_access &= (0xffff - 0x2); break;
case 23:p->got_access &= (0xffff - 0x4); break;
}
2020-10-25 05:34:25 +00:00
p->access_spritenum = nullptr;
}
else
{
2021-11-17 23:20:39 +00:00
fi.checkhitswitch(snum, p->access_wall, nullptr);
switch (p->access_wall->pal)
{
case 0:p->got_access &= (0xffff - 0x1); break;
case 21:p->got_access &= (0xffff - 0x2); break;
case 23:p->got_access &= (0xffff - 0x4); break;
}
}
}
if (p->access_incs > 20)
{
p->oaccess_incs = p->access_incs = 0;
p->oweapon_pos = p->weapon_pos = 10;
p->okickback_pic = p->kickback_pic = 0;
}
}
if (p->scuba_on == 0 && p->insector() && p->cursector->lotag == 2)
{
if (p->scuba_amount > 0)
{
p->scuba_on = 1;
p->inven_icon = 6;
FTA(76, p);
}
else
{
if (p->airleft > 0)
p->airleft--;
else
{
p->extra_extra8 += 32;
if (p->last_extra < (gs.max_player_health >> 1) && (p->last_extra & 3) == 0)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_LONGTERM_PAIN, pact);
}
}
}
else if (p->scuba_amount > 0 && p->scuba_on)
{
p->scuba_amount--;
if (p->scuba_amount == 0)
{
p->scuba_on = 0;
checkavailinven(p);
}
}
if (p->knuckle_incs)
{
p->knuckle_incs++;
if (p->knuckle_incs == 10 && !isWW2GI())
{
if (PlayClock > 1024)
if (snum == screenpeek || ud.coop == 1)
{
if (rand() & 1)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_CRACK, pact);
else S_PlayActorSound(DUKE_CRACK2, pact);
}
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_CRACK_FIRST, pact);
}
else if (p->knuckle_incs == 22 || PlayerInput(snum, SB_FIRE))
p->knuckle_incs = 0;
return 1;
}
return 0;
}
2020-05-17 21:44:53 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-02-07 10:04:19 +00:00
void checkweapons_d(player_struct* p)
2020-05-17 21:44:53 +00:00
{
static const uint16_t weapon_sprites[MAX_WEAPONS] = { KNEE, FIRSTGUNSPRITE, SHOTGUNSPRITE,
2020-05-17 21:44:53 +00:00
CHAINGUNSPRITE, RPGSPRITE, HEAVYHBOMB, SHRINKERSPRITE, DEVISTATORSPRITE,
TRIPBOMBSPRITE, FREEZESPRITE, HEAVYHBOMB, SHRINKERSPRITE };
int cw;
if (isWW2GI())
{
int snum = p->GetActor()->PlayerIndex();
cw = aplWeaponWorksLike(p->curr_weapon, snum);
2020-05-17 21:44:53 +00:00
}
else
cw = p->curr_weapon;
if (cw < 1 || cw >= MAX_WEAPONS) return;
if (cw)
{
if (krand() & 1)
2020-11-02 19:24:07 +00:00
spawn(p->GetActor(), weapon_sprites[cw]);
2020-05-17 21:44:53 +00:00
else switch (cw)
{
case RPG_WEAPON:
case HANDBOMB_WEAPON:
2020-11-02 19:24:07 +00:00
spawn(p->GetActor(), EXPLOSION2);
2020-05-17 21:44:53 +00:00
break;
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2022-09-14 19:29:37 +00:00
static void operateJetpack(int snum, ESyncBits actions, int psectlotag, double floorz, double ceilingz, int shrunk)
2020-05-17 21:44:53 +00:00
{
auto p = &ps[snum];
auto pact = p->GetActor();
2020-05-17 21:44:53 +00:00
p->on_ground = 0;
p->jumping_counter = 0;
p->hard_landing = 0;
p->falling_counter = 0;
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = BobVal(p->pycount);
2020-05-17 21:44:53 +00:00
if (p->jetpack_on && S_CheckActorSoundPlaying(pact, DUKE_SCREAM))
{
S_StopSound(DUKE_SCREAM, pact);
}
2020-05-17 21:44:53 +00:00
if (p->jetpack_on < 11)
{
p->jetpack_on++;
2022-02-05 12:40:30 +00:00
p->pos.Z -= (p->jetpack_on * 0.5); //Goin up
2020-05-17 21:44:53 +00:00
}
else if (p->jetpack_on == 11 && !S_CheckActorSoundPlaying(pact, DUKE_JETPACK_IDLE))
S_PlayActorSound(DUKE_JETPACK_IDLE, pact);
2020-05-17 21:44:53 +00:00
2022-02-05 12:40:30 +00:00
double dist;
if (shrunk) dist = 2;
else dist = 8;
2020-05-17 21:44:53 +00:00
if (actions & SB_JUMP) //A (soar high)
2020-05-17 21:44:53 +00:00
{
// jump
SetGameVarID(g_iReturnVarID, 0, p->GetActor(), snum);
OnEvent(EVENT_SOARUP, snum, p->GetActor(), -1);
if (GetGameVarID(g_iReturnVarID, p->GetActor(), snum).value() == 0)
2020-05-17 21:44:53 +00:00
{
2022-02-05 12:40:30 +00:00
p->pos.Z -= dist;
p->crack_time = CRACK_TIME;
2020-05-17 21:44:53 +00:00
}
}
if (actions & SB_CROUCH) //Z (soar low)
2020-05-17 21:44:53 +00:00
{
// crouch
SetGameVarID(g_iReturnVarID, 0, p->GetActor(), snum);
OnEvent(EVENT_SOARDOWN, snum, p->GetActor(), -1);
if (GetGameVarID(g_iReturnVarID, p->GetActor(), snum).value() == 0)
2020-05-17 21:44:53 +00:00
{
2022-02-05 12:40:30 +00:00
p->pos.Z += dist;
p->crack_time = CRACK_TIME;
2020-05-17 21:44:53 +00:00
}
}
int k;
if (shrunk == 0 && (psectlotag == 0 || psectlotag == 2)) k = 32;
else k = 16;
if (psectlotag != 2 && p->scuba_on == 1)
p->scuba_on = 0;
2022-09-14 19:29:37 +00:00
if (p->pos.Z > floorz - k)
p->pos.Z += ((floorz - k) - p->pos.Z) * 0.5;
if (p->pos.Z < pact->ceilingz + 18)
2022-02-05 12:40:30 +00:00
p->pos.Z = pact->ceilingz + 18;
2020-05-17 21:44:53 +00:00
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void movement(int snum, ESyncBits actions, sectortype* psect, double floorz, double ceilingz, int shrunk, double truefdist, int psectlotag)
2020-05-17 21:44:53 +00:00
{
int j;
auto p = &ps[snum];
auto pact = p->GetActor();
2020-05-17 21:44:53 +00:00
if (p->airleft != 15 * 26)
p->airleft = 15 * 26; //Aprox twenty seconds.
if (p->scuba_on == 1)
p->scuba_on = 0;
2022-08-30 22:03:40 +00:00
double i = 40;
if (psectlotag == ST_1_ABOVE_WATER && p->spritebridge == 0)
2020-05-17 21:44:53 +00:00
{
if (shrunk == 0)
{
i = 34;
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = BobVal(p->pycount) * 2;
2020-05-17 21:44:53 +00:00
}
else i = 12;
if (shrunk == 0 && truefdist <= gs.playerheight)
2020-05-17 21:44:53 +00:00
{
if (p->on_ground == 1)
{
2020-10-25 06:21:33 +00:00
if (p->dummyplayersprite == nullptr)
p->dummyplayersprite = spawn(pact, PLAYERONWATER);
2020-05-17 21:44:53 +00:00
p->footprintcount = 6;
if (p->cursector->floorpicnum == FLOORSLIME)
2020-05-17 21:44:53 +00:00
p->footprintpal = 8;
else p->footprintpal = 0;
p->footprintshade = 0;
}
}
}
else
{
footprints(snum);
}
2022-08-30 22:03:40 +00:00
if (p->pos.Z < floorz - i) //falling
2020-05-17 21:44:53 +00:00
{
// not jumping or crouching
2022-08-30 22:03:40 +00:00
if ((actions & (SB_JUMP|SB_CROUCH)) == 0 && p->on_ground && (psect->floorstat & CSTAT_SECTOR_SLOPE) && p->pos.Z >= (floorz - i - 16))
p->pos.Z = floorz - i;
2020-05-17 21:44:53 +00:00
else
{
p->on_ground = 0;
p->vel.Z += (gs.gravity + 5/16.); // (TICSPERFRAME<<6);
if (p->vel.Z >= (16 + 8)) p->vel.Z = (16 + 8);
if (p->vel.Z > 2400 / 256 && p->falling_counter < 255)
2020-05-17 21:44:53 +00:00
{
p->falling_counter++;
if (p->falling_counter == 38 && !S_CheckActorSoundPlaying(pact, DUKE_SCREAM))
S_PlayActorSound(DUKE_SCREAM, pact);
2020-05-17 21:44:53 +00:00
}
if (p->pos.Z + p->vel.Z >= floorz - i) // hit the ground
{
S_StopSound(DUKE_SCREAM, pact);
if (!p->insector() || p->cursector->lotag != 1)
2020-05-17 21:44:53 +00:00
{
if (p->falling_counter > 62) quickkill(p);
else if (p->falling_counter > 9)
{
j = p->falling_counter;
pact->spr.extra -= j - (krand() & 3);
if (pact->spr.extra <= 0)
2020-05-17 21:44:53 +00:00
{
S_PlayActorSound(SQUISHED, pact);
2020-05-17 21:44:53 +00:00
SetPlayerPal(p, PalEntry(63, 63, 0, 0));
}
else
{
S_PlayActorSound(DUKE_LAND, pact);
S_PlayActorSound(DUKE_LAND_HURT, pact);
2020-05-17 21:44:53 +00:00
}
SetPlayerPal(p, PalEntry(32, 16, 0, 0));
}
else if (p->vel.Z > 8) S_PlayActorSound(DUKE_LAND, pact);
2020-05-17 21:44:53 +00:00
}
}
2020-05-17 21:44:53 +00:00
}
}
else
{
p->falling_counter = 0;
S_StopSound(-1, pact, CHAN_VOICE);
2020-05-17 21:44:53 +00:00
if (psectlotag != ST_1_ABOVE_WATER && psectlotag != ST_2_UNDERWATER && p->on_ground == 0 && p->vel.Z > 12)
p->hard_landing = uint8_t(p->vel.Z / 4. );
2020-05-17 21:44:53 +00:00
p->on_ground = 1;
if (i == 40)
{
//Smooth on the ground
2022-08-30 22:03:40 +00:00
double k = (floorz - i - p->pos.Z) * 0.5;
if (abs(k) < 1) k = 0;
p->pos.Z += k;
p->vel.Z -= 3;
if (p->vel.Z < 0) p->vel.Z = 0;
2020-05-17 21:44:53 +00:00
}
else if (p->jumping_counter == 0)
{
2022-08-30 22:03:40 +00:00
p->pos.Z += ((floorz - i * 0.5) - p->pos.Z) * 0.5; //Smooth on the water
if (p->on_warping_sector == 0 && p->pos.Z > floorz - 16)
2020-05-17 21:44:53 +00:00
{
2022-08-30 22:03:40 +00:00
p->pos.Z = floorz - 16;
p->vel.Z *= 0.5;
2020-05-17 21:44:53 +00:00
}
}
p->on_warping_sector = 0;
if ((actions & SB_CROUCH) || crouch_toggle) // FIXME: The crouch_toggle check here is not network safe and needs revision when multiplayer is going.
2020-05-17 21:44:53 +00:00
{
playerCrouch(snum);
}
// jumping
if ((actions & SB_JUMP) == 0 && p->jumping_toggle == 1)
2020-05-17 21:44:53 +00:00
p->jumping_toggle = 0;
else if ((actions & SB_JUMP))
2020-05-17 21:44:53 +00:00
{
2022-09-14 20:04:41 +00:00
playerJump(snum, floorz, ceilingz);
2020-05-17 21:44:53 +00:00
}
if (p->jumping_counter && (actions & SB_JUMP) == 0)
2020-05-17 21:44:53 +00:00
p->jumping_toggle = 0;
}
if (p->jumping_counter)
{
if ((actions & SB_JUMP) == 0 && p->jumping_toggle == 1)
2020-05-17 21:44:53 +00:00
p->jumping_toggle = 0;
if (p->jumping_counter < (1024 + 256))
{
if (psectlotag == 1 && p->jumping_counter > 768)
{
p->jumping_counter = 0;
p->vel.Z = -2;
2020-05-17 21:44:53 +00:00
}
else
{
p->vel.Z -= BobVal(2048 - 128 + p->jumping_counter) * (64. / 12);
2020-05-17 21:44:53 +00:00
p->jumping_counter += 180;
p->on_ground = 0;
}
}
else
{
p->jumping_counter = 0;
p->vel.Z = 0;
2020-05-17 21:44:53 +00:00
}
}
p->pos.Z += p->vel.Z ;
2020-05-17 21:44:53 +00:00
2022-08-30 22:03:40 +00:00
if (p->pos.Z < ceilingz + 4)
2020-05-17 21:44:53 +00:00
{
p->jumping_counter = 0;
if (p->vel.Z < 0)
p->vel.X = p->vel.Y = 0;
p->vel.Z = 0.5;
2022-08-30 22:03:40 +00:00
p->pos.Z = ceilingz + 4;
2020-05-17 21:44:53 +00:00
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void underwater(int snum, ESyncBits actions, double floorz, double ceilingz)
2020-05-17 21:44:53 +00:00
{
auto p = &ps[snum];
2020-10-25 06:01:58 +00:00
auto pact = p->GetActor();
2020-05-17 21:44:53 +00:00
// under water
p->jumping_counter = 0;
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = BobVal(p->pycount);
2020-05-17 21:44:53 +00:00
2020-10-25 06:01:58 +00:00
if (!S_CheckActorSoundPlaying(pact, DUKE_UNDERWATER))
S_PlayActorSound(DUKE_UNDERWATER, pact);
2020-05-17 21:44:53 +00:00
if (actions & SB_JUMP)
2020-05-17 21:44:53 +00:00
{
// jump
if (p->vel.Z > 0) p->vel.Z = 0;
p->vel.Z -= (348 / 256.);
if (p->vel.Z < -6) p->vel.Z = -6;
2020-05-17 21:44:53 +00:00
}
else if (actions & SB_CROUCH)
2020-05-17 21:44:53 +00:00
{
// crouch
if (p->vel.Z < 0) p->vel.Z = 0;
p->vel.Z += (348 / 256.);
if (p->vel.Z > 6) p->vel.Z = 6;
2020-05-17 21:44:53 +00:00
}
else
{
// normal view
if (p->vel.Z < 0)
2020-05-17 21:44:53 +00:00
{
p->vel.Z += 1;
if (p->vel.Z > 0)
p->vel.Z = 0;
2020-05-17 21:44:53 +00:00
}
if (p->vel.Z > 0)
2020-05-17 21:44:53 +00:00
{
p->vel.Z -= 1;
if (p->vel.Z < 0)
p->vel.Z = 0;
2020-05-17 21:44:53 +00:00
}
}
if (p->vel.Z > 8)
p->vel.Z *= 0.5;
2020-05-17 21:44:53 +00:00
p->pos.Z += p->vel.Z ;
2020-05-17 21:44:53 +00:00
if (p->pos.Z > floorz - 15)
p->pos.Z += (((floorz - 15) - p->pos.Z) * 0.5);
2020-05-17 21:44:53 +00:00
if (p->pos.Z < ceilingz + 4)
2020-05-17 21:44:53 +00:00
{
p->pos.Z = ceilingz + 4;
p->vel.Z = 0;
2020-05-17 21:44:53 +00:00
}
if (p->scuba_on && (krand() & 255) < 8)
{
2020-10-25 06:01:58 +00:00
auto j = spawn(pact, WATERBUBBLE);
if (j)
{
j->spr.pos += (p->angle.ang.ToVector() + DVector2(4 - (global_random & 8), 4 - (global_random & 8))) * 16;
j->spr.xrepeat = 3;
j->spr.yrepeat = 2;
j->spr.pos.Z = p->pos.Z + 8;
}
2020-05-17 21:44:53 +00:00
}
}
2020-05-18 20:28:12 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int operateTripbomb(int snum)
{
auto p = &ps[snum];
HitInfo hit{};
2022-10-11 06:32:16 +00:00
hitscan(p->pos, p->cursector, DVector3(p->angle.ang.ToVector() * 1024, p->horizon.sum().Tan() * 16.), hit, CLIPMASK1);
if (hit.hitSector == nullptr || hit.actor())
return 0;
if (hit.hitWall != nullptr && hit.hitSector->lotag > 2)
return 0;
if (hit.hitWall != nullptr && hit.hitWall->overpicnum >= 0)
if (hit.hitWall->overpicnum == BIGFORCE)
return 0;
2021-12-21 19:33:39 +00:00
DDukeActor* act;
DukeSectIterator it(hit.hitSector);
2021-12-21 19:33:39 +00:00
while ((act = it.Next()))
{
if (!actorflag(act, SFLAG_BLOCK_TRIPBOMB))
{
auto delta = act->spr.pos - hit.hitpos;
if (abs(delta.Z) < 12 && delta.XY().LengthSquared() < (18.125 * 18.125))
return 0;
}
}
2021-12-21 19:33:39 +00:00
if (act == nullptr && hit.hitWall != nullptr && (hit.hitWall->cstat & CSTAT_WALL_MASKED) == 0)
if ((hit.hitWall->twoSided() && hit.hitWall->nextSector()->lotag <= 2) || (!hit.hitWall->twoSided() && hit.hitSector->lotag <= 2))
{
auto delta = hit.hitpos.XY() - p->pos.XY();
if (delta.LengthSquared() < (18.125 * 18.125))
{
2022-02-05 12:40:30 +00:00
p->pos.Z = p->opos.Z;
p->vel.Z = 0;
return 1;
}
}
return 0;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2020-05-18 22:26:07 +00:00
static void fireweapon(int snum)
{
auto p = &ps[snum];
2020-11-02 19:24:07 +00:00
auto pact = p->GetActor();
p->crack_time = CRACK_TIME;
if (p->holster_weapon == 1)
{
if (p->last_pissed_time <= (26 * 218) && p->weapon_pos == -9)
{
p->holster_weapon = 0;
p->oweapon_pos = p->weapon_pos = 10;
FTA(74, p);
}
}
else switch (p->curr_weapon)
{
case HANDBOMB_WEAPON:
p->hbomb_hold_delay = 0;
if (p->ammo_amount[HANDBOMB_WEAPON] > 0)
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
break;
case HANDREMOTE_WEAPON:
p->hbomb_hold_delay = 0;
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
break;
case PISTOL_WEAPON:
if (p->ammo_amount[PISTOL_WEAPON] > 0)
{
p->ammo_amount[PISTOL_WEAPON]--;
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
}
break;
case CHAINGUN_WEAPON:
if (p->ammo_amount[CHAINGUN_WEAPON] > 0) // && p->random_club_frame == 0)
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
break;
case SHOTGUN_WEAPON:
if (p->ammo_amount[SHOTGUN_WEAPON] > 0 && p->random_club_frame == 0)
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
break;
case TRIPBOMB_WEAPON:
if (p->ammo_amount[TRIPBOMB_WEAPON] > 0)
{
if (operateTripbomb(snum))
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
}
break;
case SHRINKER_WEAPON:
case GROW_WEAPON:
if (p->curr_weapon == GROW_WEAPON)
{
if (p->ammo_amount[GROW_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
2020-11-02 19:24:07 +00:00
S_PlayActorSound(EXPANDERSHOOT, pact);
}
}
else if (p->ammo_amount[SHRINKER_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
2020-11-02 19:24:07 +00:00
S_PlayActorSound(SHRINKER_FIRE, pact);
}
break;
case FREEZE_WEAPON:
if (p->ammo_amount[FREEZE_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
2020-11-02 19:24:07 +00:00
S_PlayActorSound(CAT_FIRE, pact);
}
break;
case DEVISTATOR_WEAPON:
if (p->ammo_amount[DEVISTATOR_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
p->hbomb_hold_delay = !p->hbomb_hold_delay;
2020-11-02 19:24:07 +00:00
S_PlayActorSound(CAT_FIRE, pact);
}
break;
case RPG_WEAPON:
if (p->ammo_amount[RPG_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
}
break;
2020-05-18 20:28:12 +00:00
case FLAMETHROWER_WEAPON: // Twentieth Anniversary World Tour
if (isWorldTour() && p->ammo_amount[FLAMETHROWER_WEAPON] > 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
if (p->cursector->lotag != 2)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(FLAMETHROWER_INTRO, pact);
2020-05-18 20:28:12 +00:00
}
break;
case KNEE_WEAPON:
if (p->quick_kick == 0)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic = 1;
}
break;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2021-11-21 07:48:36 +00:00
static void operateweapon(int snum, ESyncBits actions)
{
auto p = &ps[snum];
auto pact = p->GetActor();
// already firing...
2020-05-18 20:28:12 +00:00
switch (p->curr_weapon)
{
2020-05-18 20:28:12 +00:00
case HANDBOMB_WEAPON: // grenade in NAM
if (p->kickback_pic == 6 && (actions & SB_FIRE))
{
p->rapid_fire_hold = 1;
2020-05-18 20:28:12 +00:00
break;
}
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->kickback_pic == 12)
{
double zvel, vel;
2020-05-18 20:28:12 +00:00
p->ammo_amount[HANDBOMB_WEAPON]--;
if (p->on_ground && (actions & SB_CROUCH))
{
vel = 15/16.;
zvel = p->horizon.sum().Tan() * 10.;
}
else
{
vel = 140/16.;
zvel = -4 + p->horizon.sum().Tan() * 10.;
}
auto spawned = CreateActor(p->cursector, p->pos + p->angle.ang.ToVector() * 16, HEAVYHBOMB, -16, 9, 9,
p->angle.ang, vel + p->hbomb_hold_delay * 2, zvel, pact, 1);
2020-05-18 20:28:12 +00:00
if (isNam())
{
spawned->spr.extra = MulScale(krand(), NAM_GRENADE_LIFETIME_VAR, 14);
}
if (vel == 15 / 16.)
{
spawned->spr.yint = 3;
2022-02-07 07:47:18 +00:00
spawned->spr.pos.Z += 8;
}
2022-09-13 22:05:32 +00:00
double hd = hits(pact);
if (hd < 32)
{
2022-09-03 08:04:16 +00:00
spawned->spr.angle += DAngle180;
spawned->vel *= 1./3.;
}
p->hbomb_on = 1;
}
else if (p->kickback_pic < 12 && (actions & SB_FIRE))
p->hbomb_hold_delay++;
2020-05-18 22:26:07 +00:00
else if (p->kickback_pic > 19)
{
p->okickback_pic = p->kickback_pic = 0;
// don't change to remote when in NAM: grenades are timed
2020-05-18 20:28:12 +00:00
if (isNam()) checkavailweapon(p);
else
{
p->curr_weapon = HANDREMOTE_WEAPON;
p->last_weapon = -1;
p->oweapon_pos = p->weapon_pos = 10;
2020-05-18 20:28:12 +00:00
}
}
2020-05-18 20:28:12 +00:00
break;
case HANDREMOTE_WEAPON: // knife in NAM
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 2)
{
2020-05-18 20:28:12 +00:00
p->hbomb_on = 0;
}
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 10)
{
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
int weapon = isNam() ? TRIPBOMB_WEAPON : HANDBOMB_WEAPON;
if (p->ammo_amount[weapon] > 0)
fi.addweapon(p, weapon);
else
checkavailweapon(p);
}
2020-05-18 20:28:12 +00:00
break;
2020-05-18 20:28:12 +00:00
case PISTOL_WEAPON: // m-16 in NAM
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 1)
{
fi.shoot(pact, SHOTSPARK1);
S_PlayActorSound(PISTOL_FIRE, pact);
lastvisinc = PlayClock + 32;
2020-05-18 20:28:12 +00:00
p->visibility = 0;
}
2020-05-18 22:26:07 +00:00
else if (p->kickback_pic == 2)
spawn(pact, SHELL);
2020-05-18 20:28:12 +00:00
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
2020-05-18 20:28:12 +00:00
2020-05-18 22:26:07 +00:00
if (p->kickback_pic >= 5)
2020-05-18 20:28:12 +00:00
{
if (p->ammo_amount[PISTOL_WEAPON] <= 0 || (p->ammo_amount[PISTOL_WEAPON] % (isNam() ? 20 : 12)))
{
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
checkavailweapon(p);
}
2020-05-18 20:28:12 +00:00
else
{
2020-05-18 22:26:07 +00:00
switch (p->kickback_pic)
{
2020-05-18 20:28:12 +00:00
case 5:
S_PlayActorSound(EJECT_CLIP, pact);
2020-05-18 20:28:12 +00:00
break;
//#ifdef NAM
// case WEAPON2_RELOAD_TIME - 15:
2020-05-18 20:28:12 +00:00
//#else
case 8:
//#endif
S_PlayActorSound(INSERT_CLIP, pact);
2020-05-18 20:28:12 +00:00
break;
}
}
2020-05-18 20:28:12 +00:00
}
2020-05-18 20:28:12 +00:00
// 3 second re-load time
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == (isNam() ? 50 : 27))
2020-05-18 20:28:12 +00:00
{
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
checkavailweapon(p);
}
break;
2020-05-18 20:28:12 +00:00
case SHOTGUN_WEAPON:
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 4)
2020-05-18 20:28:12 +00:00
{
for(int ii = 0; ii < 7; ii++)
fi.shoot(pact, SHOTGUN);
2020-05-18 20:28:12 +00:00
p->ammo_amount[SHOTGUN_WEAPON]--;
S_PlayActorSound(SHOTGUN_FIRE, pact);
lastvisinc = PlayClock + 32;
2020-05-18 20:28:12 +00:00
p->visibility = 0;
}
2020-05-18 22:26:07 +00:00
switch(p->kickback_pic)
{
case 13:
checkavailweapon(p);
break;
case 15:
S_PlayActorSound(SHOTGUN_COCK, pact);
break;
case 17:
case 20:
p->kickback_pic++;
break;
case 24:
{
auto j = spawn(pact, SHOTGUNSHELL);
if (j)
{
2022-09-09 16:37:55 +00:00
j->spr.angle += DAngle180;
ssp(j, CLIPMASK0);
2022-09-09 16:37:55 +00:00
j->spr.angle -= DAngle180;
}
p->kickback_pic++;
break;
}
case 31:
p->okickback_pic = p->kickback_pic = 0;
return;
}
break;
case CHAINGUN_WEAPON: // m-60 in NAM
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
2020-05-18 22:26:07 +00:00
if (p->kickback_pic <= 12)
{
2020-05-18 22:26:07 +00:00
if (((p->kickback_pic) % 3) == 0)
{
p->ammo_amount[CHAINGUN_WEAPON]--;
2020-05-18 22:26:07 +00:00
if ((p->kickback_pic % 3) == 0)
{
auto j = spawn(pact, SHELL);
if (j)
{
2022-09-09 16:37:55 +00:00
j->spr.angle += DAngle180;
2022-09-11 17:28:45 +00:00
j->vel.X += 2.;
2022-02-07 07:47:18 +00:00
j->spr.pos.Z += 3;
ssp(j, CLIPMASK0);
}
}
S_PlayActorSound(CHAINGUN_FIRE, pact);
fi.shoot(pact, CHAINGUN);
lastvisinc = PlayClock + 32;
p->visibility = 0;
checkavailweapon(p);
if ((actions & SB_FIRE) == 0)
{
p->okickback_pic = p->kickback_pic = 0;
break;
}
}
}
2020-05-18 22:26:07 +00:00
else if (p->kickback_pic > 10)
{
if (actions & SB_FIRE) p->okickback_pic = p->kickback_pic = 1;
else p->okickback_pic = p->kickback_pic = 0;
}
break;
case GROW_WEAPON: // m-14 with scope (sniper rifle)
2020-05-18 20:28:12 +00:00
bool check;
if (isNam())
{
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
check = (p->kickback_pic == 3);
}
else
{
2020-05-18 22:26:07 +00:00
check = (p->kickback_pic > 3);
}
2020-05-18 20:28:12 +00:00
if (check)
{
2020-05-18 20:28:12 +00:00
// fire now, but don't reload right away...
if (isNam())
{
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->ammo_amount[p->curr_weapon] <= 1)
p->okickback_pic = p->kickback_pic = 0;
}
2020-05-18 20:28:12 +00:00
else
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
p->ammo_amount[p->curr_weapon]--;
fi.shoot(pact, GROWSPARK);
2020-05-18 20:28:12 +00:00
//#ifdef NAM
//#else
if (!(aplWeaponFlags(p->curr_weapon, snum) & WEAPON_FLAG_NOVISIBLE))
2020-05-18 20:28:12 +00:00
{
// make them visible if not set...
p->visibility = 0;
lastvisinc = PlayClock + 32;
}
2020-05-18 20:28:12 +00:00
checkavailweapon(p);
//#endif
}
2020-05-18 22:26:07 +00:00
else if (!isNam()) p->kickback_pic++;
if (isNam() && p->kickback_pic > 30)
{
2020-05-18 20:28:12 +00:00
// reload now...
p->okickback_pic = p->kickback_pic = 0;
if (!(aplWeaponFlags(p->curr_weapon, snum) & WEAPON_FLAG_NOVISIBLE))
{
2020-05-18 20:28:12 +00:00
// make them visible if not set...
p->visibility = 0;
lastvisinc = PlayClock + 32;
}
2020-05-18 20:28:12 +00:00
checkavailweapon(p);
}
break;
case SHRINKER_WEAPON: // m-79 in NAM (Grenade launcher)
2020-05-18 22:26:07 +00:00
if ((!isNam() && p->kickback_pic > 10) || (isNam() && p->kickback_pic == 10))
2020-05-18 20:28:12 +00:00
{
2020-05-18 22:26:07 +00:00
if (isNam()) p->kickback_pic++; // fire now, but wait for reload...
else p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
p->ammo_amount[SHRINKER_WEAPON]--;
fi.shoot(pact, SHRINKER);
2020-05-18 20:28:12 +00:00
if (!isNam())
{
p->visibility = 0;
2020-05-18 20:28:12 +00:00
//flashColor = 176 + (252 << 8) + (120 << 16);
lastvisinc = PlayClock + 32;
checkavailweapon(p);
}
}
2020-05-18 22:26:07 +00:00
else if (isNam() && p->kickback_pic > 30)
2020-05-18 20:28:12 +00:00
{
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
p->visibility = 0;
lastvisinc = PlayClock + 32;
2020-05-18 20:28:12 +00:00
checkavailweapon(p);
}
2020-05-18 22:26:07 +00:00
else p->kickback_pic++;
break;
case DEVISTATOR_WEAPON:
2020-05-18 22:26:07 +00:00
if (p->kickback_pic)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (isNam())
{
if ((p->kickback_pic >= 2) &&
p->kickback_pic < 5 &&
2020-05-18 22:26:07 +00:00
(p->kickback_pic & 1))
{
p->visibility = 0;
lastvisinc = PlayClock + 32;
fi.shoot(pact, RPG);
p->ammo_amount[DEVISTATOR_WEAPON]--;
checkavailweapon(p);
}
if (p->kickback_pic > 5) p->okickback_pic = p->kickback_pic = 0;
}
else if (p->kickback_pic & 1)
{
p->visibility = 0;
lastvisinc = PlayClock + 32;
fi.shoot(pact, RPG);
p->ammo_amount[DEVISTATOR_WEAPON]--;
checkavailweapon(p);
if (p->ammo_amount[DEVISTATOR_WEAPON] <= 0) p->okickback_pic = p->kickback_pic = 0;
}
if (p->kickback_pic > 5) p->okickback_pic = p->kickback_pic = 0;
}
break;
case FREEZE_WEAPON:
// flame thrower in NAM
2020-05-18 22:26:07 +00:00
if (p->kickback_pic < 4)
{
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->kickback_pic == 3)
{
p->ammo_amount[p->curr_weapon]--;
p->visibility = 0;
lastvisinc = PlayClock + 32;
fi.shoot(pact, FREEZEBLAST);
checkavailweapon(p);
}
if (pact->spr.xrepeat < 32)
{
p->okickback_pic = p->kickback_pic = 0; break;
}
}
else
{
if (actions & SB_FIRE)
{
p->okickback_pic = p->kickback_pic = 1;
S_PlayActorSound(CAT_FIRE, pact);
}
else p->okickback_pic = p->kickback_pic = 0;
}
break;
2020-05-18 20:28:12 +00:00
case FLAMETHROWER_WEAPON:
if (!isWorldTour()) // Twentieth Anniversary World Tour
break;
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->kickback_pic == 2)
2020-05-18 20:28:12 +00:00
{
if (p->cursector->lotag != 2)
2020-05-18 20:28:12 +00:00
{
p->ammo_amount[FLAMETHROWER_WEAPON]--;
fi.shoot(pact, FIREBALL);
2020-05-18 20:28:12 +00:00
}
checkavailweapon(p);
}
2020-05-18 22:26:07 +00:00
else if (p->kickback_pic == 16)
2020-05-18 20:28:12 +00:00
{
if ((actions & SB_FIRE) != 0)
2020-05-18 20:28:12 +00:00
{
p->okickback_pic = p->kickback_pic = 1;
S_PlayActorSound(FLAMETHROWER_INTRO, pact);
2020-05-18 20:28:12 +00:00
}
else
p->okickback_pic = p->kickback_pic = 0;
2020-05-18 20:28:12 +00:00
}
break;
case TRIPBOMB_WEAPON: // Claymore in NAM
2020-05-18 22:26:07 +00:00
if (p->kickback_pic < 4)
{
2022-02-05 12:40:30 +00:00
p->pos.Z = p->opos.Z;
p->vel.Z = 0;
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 3)
fi.shoot(pact, HANDHOLDINGLASER);
}
2020-05-18 22:26:07 +00:00
if (p->kickback_pic == 16)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
p->oweapon_pos = p->weapon_pos = -9;
}
2020-05-18 22:26:07 +00:00
else p->kickback_pic++;
break;
case KNEE_WEAPON:
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->kickback_pic == 7) fi.shoot(pact, KNEE);
2020-05-18 22:26:07 +00:00
else if (p->kickback_pic == 14)
{
if (actions & SB_FIRE)
p->okickback_pic = p->kickback_pic = 1 + (krand() & 3);
else p->okickback_pic = p->kickback_pic = 0;
}
if (p->wantweaponfire >= 0)
checkavailweapon(p);
break;
case RPG_WEAPON: // m-72 in NAM (LAW)
2020-05-18 22:26:07 +00:00
p->kickback_pic++;
if (p->kickback_pic == 4)
{
p->ammo_amount[RPG_WEAPON]--;
lastvisinc = PlayClock + 32;
p->visibility = 0;
fi.shoot(pact, RPG);
checkavailweapon(p);
}
else if (p->kickback_pic == 20)
p->okickback_pic = p->kickback_pic = 0;
break;
}
}
2020-05-18 20:28:12 +00:00
2020-05-19 07:54:52 +00:00
//---------------------------------------------------------------------------
//
// this function exists because gotos suck. :P
//
//---------------------------------------------------------------------------
2021-11-21 07:48:36 +00:00
static void processweapon(int snum, ESyncBits actions)
2020-05-19 07:54:52 +00:00
{
auto p = &ps[snum];
2020-10-25 05:34:25 +00:00
auto pact = p->GetActor();
2021-12-21 19:33:39 +00:00
int shrunk = (pact->spr.yrepeat < 32);
2020-05-19 07:54:52 +00:00
if (isNamWW2GI() && (actions & SB_HOLSTER)) // 'Holster Weapon
2020-05-19 07:54:52 +00:00
{
if (isWW2GI())
{
SetGameVarID(g_iReturnVarID, 0, p->GetActor(), snum);
SetGameVarID(g_iWeaponVarID, p->curr_weapon, p->GetActor(), snum);
SetGameVarID(g_iWorksLikeVarID, aplWeaponWorksLike(p->curr_weapon, snum), p->GetActor(), snum);
OnEvent(EVENT_HOLSTER, snum, p->GetActor(), -1);
if (GetGameVarID(g_iReturnVarID, p->GetActor(), snum).value() == 0)
2020-05-19 07:54:52 +00:00
{
// now it uses the game definitions...
if (aplWeaponFlags(p->curr_weapon, snum) & WEAPON_FLAG_HOLSTER_CLEARS_CLIP)
2020-05-19 07:54:52 +00:00
{
if (p->ammo_amount[p->curr_weapon] > aplWeaponClip(p->curr_weapon, snum)
&& (p->ammo_amount[p->curr_weapon] % aplWeaponClip(p->curr_weapon, snum)) != 0)
2020-05-19 07:54:52 +00:00
{
// throw away the remaining clip
p->ammo_amount[p->curr_weapon] -=
p->ammo_amount[p->curr_weapon] % aplWeaponClip(p->curr_weapon, snum);
// p->kickback_pic = aplWeaponFireDelay(p->curr_weapon, snum)+1; // animate, but don't shoot...
p->kickback_pic = aplWeaponTotalTime(p->curr_weapon, snum) + 1; // animate, but don't shoot...
actions &= ~SB_FIRE; // not firing...
2020-05-19 07:54:52 +00:00
}
return;
}
}
}
else if (p->curr_weapon == PISTOL_WEAPON)
{
if (p->ammo_amount[PISTOL_WEAPON] > 20)
{
// throw away the remaining clip
p->ammo_amount[PISTOL_WEAPON] -= p->ammo_amount[PISTOL_WEAPON] % 20;
p->kickback_pic = 3; // animate, but don't shoot...
actions &= ~SB_FIRE; // not firing...
2020-05-19 07:54:52 +00:00
}
return;
}
}
if (isWW2GI() && (aplWeaponFlags(p->curr_weapon, snum) & WEAPON_FLAG_GLOWS))
2020-05-19 07:54:52 +00:00
p->random_club_frame += 64; // Glowing
if (!isWW2GI() && (p->curr_weapon == SHRINKER_WEAPON || p->curr_weapon == GROW_WEAPON))
p->random_club_frame += 64; // Glowing
if (p->rapid_fire_hold == 1)
{
if (actions & SB_FIRE) return;
2020-05-19 07:54:52 +00:00
p->rapid_fire_hold = 0;
}
if (shrunk || p->tipincs || p->access_incs)
actions &= ~SB_FIRE;
else if (shrunk == 0 && (actions & SB_FIRE) && p->kickback_pic == 0 && p->fist_incs == 0 &&
2020-05-19 07:54:52 +00:00
p->last_weapon == -1 && (p->weapon_pos == 0 || p->holster_weapon == 1))
{
if (!isWW2GI()) fireweapon(snum);
else fireweapon_ww(snum);
}
else if (p->kickback_pic)
{
2021-11-21 07:48:36 +00:00
if (!isWW2GI()) operateweapon(snum, actions);
else operateweapon_ww(snum, actions);
2020-05-19 07:54:52 +00:00
}
}
2020-05-18 22:26:07 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2020-05-19 07:03:07 +00:00
void processinput_d(int snum)
2020-05-18 22:26:07 +00:00
{
int k, doubvel;
double floorz, ceilingz, truefdist;
Collision chz, clz;
bool shrunk;
2021-11-21 07:48:36 +00:00
int psectlotag;
2022-02-07 10:04:19 +00:00
player_struct* p;
2020-05-18 22:26:07 +00:00
p = &ps[snum];
2020-10-25 05:34:25 +00:00
auto pact = p->GetActor();
2020-05-18 22:26:07 +00:00
p->horizon.resetadjustment();
p->angle.resetadjustment();
2020-05-19 17:50:31 +00:00
ESyncBits& actions = p->sync.actions;
2020-05-18 22:26:07 +00:00
2020-07-03 19:44:57 +00:00
auto sb_fvel = PlayerInputForwardVel(snum);
auto sb_svel = PlayerInputSideVel(snum);
auto sb_avel = PlayerInputAngVel(snum);
2020-05-18 22:26:07 +00:00
auto psectp = p->cursector;
2021-11-21 07:48:36 +00:00
if (psectp == nullptr)
2020-05-18 22:26:07 +00:00
{
2021-12-21 19:33:39 +00:00
if (pact->spr.extra > 0 && ud.clipping == 0)
2020-05-18 22:26:07 +00:00
{
quickkill(p);
2020-11-02 19:24:07 +00:00
S_PlayActorSound(SQUISHED, pact);
2020-05-18 22:26:07 +00:00
}
2021-11-21 07:48:36 +00:00
psectp = &sector[0];
2020-05-18 22:26:07 +00:00
}
psectlotag = psectp->lotag;
2020-05-18 22:26:07 +00:00
p->spritebridge = 0;
2021-12-21 19:33:39 +00:00
shrunk = (pact->spr.yrepeat < 32);
getzrange(p->pos, psectp, &ceilingz, chz, &floorz, clz, 10.1875, CLIPMASK0);
2020-05-18 22:26:07 +00:00
p->truefz = getflorzofslopeptr(psectp, p->pos);
p->truecz = getceilzofslopeptr(psectp, p->pos);
2020-05-18 22:26:07 +00:00
truefdist = abs(p->pos.Z - p->truefz);
if (clz.type == kHitSector && psectlotag == 1 && truefdist > gs.playerheight + 16)
2020-05-18 22:26:07 +00:00
psectlotag = 0;
pact->floorz = floorz;
pact->ceilingz = ceilingz;
2020-05-18 22:26:07 +00:00
if (SyncInput())
2020-05-18 22:26:07 +00:00
{
p->horizon.backup();
doslopetilting(p);
2020-05-18 22:26:07 +00:00
}
if (chz.type == kHitSprite)
2020-05-18 22:26:07 +00:00
{
if (chz.actor()->spr.statnum == 1 && chz.actor()->spr.extra >= 0)
2020-05-18 22:26:07 +00:00
{
chz.setNone();
ceilingz = p->truecz;
2020-05-18 22:26:07 +00:00
}
}
if (clz.type == kHitSprite)
2020-05-18 22:26:07 +00:00
{
if ((clz.actor()->spr.cstat & (CSTAT_SPRITE_ALIGNMENT_FLOOR | CSTAT_SPRITE_BLOCK)) == (CSTAT_SPRITE_ALIGNMENT_FLOOR | CSTAT_SPRITE_BLOCK))
2020-05-18 22:26:07 +00:00
{
psectlotag = 0;
p->footprintcount = 0;
p->spritebridge = 1;
}
else if (badguy(clz.actor()) && clz.actor()->spr.xrepeat > 24 && abs(pact->spr.pos.Z - clz.actor()->spr.pos.Z) < 84)
2020-05-18 22:26:07 +00:00
{
auto ang = (clz.actor()->spr.pos - p->pos).Angle();
p->vel.XY() -= ang.ToVector();
2020-05-18 22:26:07 +00:00
}
}
2021-12-21 19:33:39 +00:00
if (pact->spr.extra > 0) fi.incur_damage(p);
2020-05-18 22:26:07 +00:00
else
{
2021-12-21 19:33:39 +00:00
pact->spr.extra = 0;
2020-05-18 22:26:07 +00:00
p->shield_amount = 0;
}
2021-12-21 19:33:39 +00:00
p->last_extra = pact->spr.extra;
2020-05-18 22:26:07 +00:00
if (p->loogcnt > 0)
{
p->oloogcnt = p->loogcnt;
p->loogcnt--;
}
else
{
p->oloogcnt = p->loogcnt = 0;
}
2020-05-18 22:26:07 +00:00
if (p->fist_incs)
{
if (endoflevel(snum)) return;
}
if (p->timebeforeexit > 1 && p->last_extra > 0)
{
if (timedexit(snum))
return;
}
2021-12-21 19:33:39 +00:00
if (pact->spr.extra <= 0 && !ud.god)
2020-05-18 22:26:07 +00:00
{
playerisdead(snum, psectlotag, floorz, ceilingz);
2020-07-16 13:03:09 +00:00
return;
2020-05-18 22:26:07 +00:00
}
if (p->GetActor()->spr.xrepeat < 40 && p->jetpack_on == 0)
{
2022-06-06 00:51:57 +00:00
p->ofistsign = p->fistsign;
2022-09-15 16:41:01 +00:00
p->fistsign += p->GetActor()->vel.X * 16;
}
2020-05-18 22:26:07 +00:00
if (p->transporter_hold > 0)
{
p->transporter_hold--;
if (p->transporter_hold == 0 && p->on_warping_sector)
p->transporter_hold = 2;
}
if (p->transporter_hold < 0)
p->transporter_hold++;
2020-11-02 23:20:51 +00:00
if (p->newOwner != nullptr)
2020-05-18 22:26:07 +00:00
{
p->vel.X = p->vel.Y = 0;
2022-09-03 08:02:25 +00:00
pact->vel.X = 0;
2020-05-18 22:26:07 +00:00
fi.doincrements(p);
if (isWW2GI() && aplWeaponWorksLike(p->curr_weapon, snum) == HANDREMOTE_WEAPON) processweapon(snum, actions);
2021-11-21 07:48:36 +00:00
if (!isWW2GI() && p->curr_weapon == HANDREMOTE_WEAPON) processweapon(snum, actions);
2020-05-18 22:26:07 +00:00
return;
}
doubvel = TICSPERFRAME;
checklook(snum,actions);
double iif = 2.5;
2022-02-05 11:30:00 +00:00
auto oldpos = p->opos;
2020-05-18 22:26:07 +00:00
2020-10-17 08:44:00 +00:00
if (p->on_crane != nullptr)
2020-05-18 22:26:07 +00:00
goto HORIZONLY;
2022-09-15 16:41:01 +00:00
p->playerweaponsway(pact->vel.X);
2020-05-18 22:26:07 +00:00
pact->vel.X = clamp((p->pos.XY() - p->bobpos).Length(), 0., 32.);
if (p->on_ground) p->bobcounter += int(p->GetActor()->vel.X * 8);
2020-05-18 22:26:07 +00:00
p->backuppos(ud.clipping == 0 && ((p->insector() && p->cursector->floorpicnum == MIRROR) || !p->insector()));
2020-05-18 22:26:07 +00:00
// Shrinking code
2020-07-16 13:03:09 +00:00
if (psectlotag == ST_2_UNDERWATER)
2020-05-18 22:26:07 +00:00
{
underwater(snum, actions, floorz, ceilingz);
2020-05-18 22:26:07 +00:00
}
else if (p->jetpack_on)
{
operateJetpack(snum, actions, psectlotag, floorz, ceilingz, shrunk);
2020-05-18 22:26:07 +00:00
}
2020-07-16 13:03:09 +00:00
else if (psectlotag != ST_2_UNDERWATER)
2020-05-18 22:26:07 +00:00
{
movement(snum, actions, psectp, floorz, ceilingz, shrunk, truefdist, psectlotag);
2020-05-18 22:26:07 +00:00
}
p->psectlotag = psectlotag;
2020-05-18 22:26:07 +00:00
//Do the quick lefts and rights
if (movementBlocked(p))
2020-05-18 22:26:07 +00:00
{
doubvel = 0;
p->vel.X = 0;
p->vel.Y = 0;
2020-05-18 22:26:07 +00:00
}
else if (SyncInput())
{
//p->ang += syncangvel * constant
//ENGINE calculates angvel for you
// may still be needed later for demo recording
2020-05-18 22:26:07 +00:00
2020-11-29 08:00:00 +00:00
sb_avel = p->adjustavel(sb_avel);
p->angle.applyinput(sb_avel, &actions);
2020-05-18 22:26:07 +00:00
}
if (p->spritebridge == 0 && pact->insector())
2020-05-18 22:26:07 +00:00
{
2022-09-13 17:51:25 +00:00
int j = pact->sector()->floorpicnum;
2020-05-18 22:26:07 +00:00
if (j == PURPLELAVA || pact->sector()->ceilingpicnum == PURPLELAVA)
2020-05-18 22:26:07 +00:00
{
if (p->boot_amount > 0)
{
p->boot_amount--;
p->inven_icon = 7;
if (p->boot_amount <= 0)
checkavailinven(p);
}
else
{
2020-11-02 19:24:07 +00:00
if (!S_CheckActorSoundPlaying(pact, DUKE_LONGTERM_PAIN))
S_PlayActorSound(DUKE_LONGTERM_PAIN, pact);
2020-05-18 22:26:07 +00:00
SetPlayerPal(p, PalEntry(32, 0, 8, 0));
2021-12-21 19:33:39 +00:00
pact->spr.extra--;
2020-05-18 22:26:07 +00:00
}
}
k = 0;
if (p->on_ground && truefdist <= gs.playerheight + 16)
2020-05-18 22:26:07 +00:00
{
2022-01-21 13:02:59 +00:00
int whichsound = (gs.tileinfo[j].flags & TFLAG_ELECTRIC) ? 0 : j == FLOORSLIME ? 1 : j == FLOORPLASMA ? 2 : -1;
2020-05-18 22:26:07 +00:00
if (j >= 0) k = makepainsounds(snum, whichsound);
}
if (k)
{
FTA(75, p);
p->boot_amount -= 2;
if (p->boot_amount <= 0)
checkavailinven(p);
}
}
if (p->vel.X || p->vel.Y || sb_fvel || sb_svel)
2020-05-18 22:26:07 +00:00
{
p->crack_time = CRACK_TIME;
2020-05-18 22:26:07 +00:00
k = int(BobVal(p->bobcounter) * 4);
2020-05-18 22:26:07 +00:00
if (truefdist < gs.playerheight + 8 && (k == 1 || k == 3))
2020-05-18 22:26:07 +00:00
{
if (p->spritebridge == 0 && p->walking_snd_toggle == 0 && p->on_ground)
{
2022-09-13 17:51:25 +00:00
int j;
2020-05-18 22:26:07 +00:00
switch (psectlotag)
{
case 0:
if (clz.type == kHitSprite)
j = clz.actor()->spr.picnum;
else
j = psectp->floorpicnum;
2020-05-18 22:26:07 +00:00
switch (j)
{
case PANNEL1:
case PANNEL2:
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_WALKINDUCTS, pact);
2020-05-18 22:26:07 +00:00
p->walking_snd_toggle = 1;
break;
}
break;
case 1:
if ((krand() & 1) == 0)
2020-11-02 19:24:07 +00:00
S_PlayActorSound(DUKE_ONWATER, pact);
2020-05-18 22:26:07 +00:00
p->walking_snd_toggle = 1;
break;
}
}
}
else if (p->walking_snd_toggle > 0)
p->walking_snd_toggle--;
if (p->jetpack_on == 0 && p->steroids_amount > 0 && p->steroids_amount < 400)
doubvel <<= 1;
p->vel.X += sb_fvel * doubvel * (5. / 16.);
p->vel.Y += sb_svel * doubvel * (5. / 16.);
2020-05-18 22:26:07 +00:00
bool check;
if (!isWW2GI()) check = ((p->curr_weapon == KNEE_WEAPON && p->kickback_pic > 10 && p->on_ground) || (p->on_ground && (actions & SB_CROUCH)));
else check = ((aplWeaponWorksLike(p->curr_weapon, snum) == KNEE_WEAPON && p->kickback_pic > 10 && p->on_ground) || (p->on_ground && (actions & SB_CROUCH)));
2020-05-18 22:26:07 +00:00
if (check)
{
2022-09-11 22:08:42 +00:00
p->vel.XY() *= gs.playerfriction - 0.125;
2020-05-18 22:26:07 +00:00
}
else
{
if (psectlotag == 2)
{
2022-09-11 22:08:42 +00:00
p->vel.XY() *= gs.playerfriction - FixedToFloat(0x1400);
2020-05-18 22:26:07 +00:00
}
else
{
2022-09-11 22:08:42 +00:00
p->vel.XY() *= gs.playerfriction;
2020-05-18 22:26:07 +00:00
}
}
if (abs(p->vel.X) < 1/128. && abs(p->vel.Y) < 1 / 128.)
p->vel.X = p->vel.Y = 0;
2020-05-18 22:26:07 +00:00
if (shrunk)
{
p->vel.XY() *= gs.playerfriction * 0.75;
2020-05-18 22:26:07 +00:00
}
}
HORIZONLY:
if (psectlotag == 1 || p->spritebridge == 1) iif = 4;
else iif = 20;
2020-05-18 22:26:07 +00:00
if (p->insector() && p->cursector->lotag == 2) k = 0;
2020-05-18 22:26:07 +00:00
else k = 1;
Collision clip{};
2020-05-18 22:26:07 +00:00
if (ud.clipping)
{
p->pos.XY() += p->vel.XY() ;
2022-09-03 08:47:36 +00:00
updatesector(p->pos, &p->cursector);
ChangeActorSect(pact, p->cursector);
2020-05-18 22:26:07 +00:00
}
else
clipmove(p->pos, &p->cursector, p->vel, 10.25, 4., iif, CLIPMASK0, clip);
2020-05-18 22:26:07 +00:00
if (p->jetpack_on == 0 && psectlotag != 2 && psectlotag != 1 && shrunk)
2022-02-05 12:40:30 +00:00
p->pos.Z += 32;
2020-05-18 22:26:07 +00:00
if (clip.type != kHitNone)
checkplayerhurt_d(p, clip);
2020-05-18 22:26:07 +00:00
if (p->jetpack_on == 0)
{
2022-09-13 17:51:25 +00:00
if (pact->vel.X > 1)
2020-05-18 22:26:07 +00:00
{
if (psectlotag != 1 && psectlotag != 2 && p->on_ground)
{
p->pycount += 52;
p->pycount &= 2047;
2022-09-14 17:30:20 +00:00
const double factor = 1024. / 1596; // What is 1596?
p->pyoff = abs(pact->vel.X * BobVal(p->pycount)) * factor;
2020-05-18 22:26:07 +00:00
}
}
else if (psectlotag != 2 && psectlotag != 1)
p->pyoff = 0;
}
// RBG***
2022-08-22 16:34:01 +00:00
SetActor(pact, p->pos.plusZ(gs.playerheight));
2020-05-18 22:26:07 +00:00
if (psectlotag < 3)
{
psectp = pact->sector();
if (ud.clipping == 0 && psectp->lotag == 31)
2020-05-18 22:26:07 +00:00
{
auto secact = barrier_cast<DDukeActor*>(psectp->hitagactor);
if (secact && secact->vel.X != 0 && secact->temp_data[0] == 0)
2020-05-18 22:26:07 +00:00
{
quickkill(p);
return;
}
}
}
if (truefdist < gs.playerheight && p->on_ground && psectlotag != 1 && shrunk == 0 && p->insector() && p->cursector->lotag == 1)
if (!S_CheckActorSoundPlaying(pact, DUKE_ONWATER))
S_PlayActorSound(DUKE_ONWATER, pact);
2020-05-18 22:26:07 +00:00
if (p->cursector != pact->sector())
ChangeActorSect(pact, p->cursector);
2020-05-18 22:26:07 +00:00
int retry = 0;
while (ud.clipping == 0)
2020-05-18 22:26:07 +00:00
{
int blocked;
blocked = (pushmove(p->pos, &p->cursector, 10.25, 4, 4, CLIPMASK0) < 0 && furthestangle(p->GetActor(), 8) < DAngle90);
2022-02-03 23:55:12 +00:00
if (fabs(pact->floorz - pact->ceilingz) < 48 || blocked)
2020-05-18 22:26:07 +00:00
{
if (!(pact->sector()->lotag & 0x8000) && (isanunderoperator(pact->sector()->lotag) ||
isanearoperator(pact->sector()->lotag)))
fi.activatebysector(pact->sector(), pact);
if (blocked)
2020-05-18 22:26:07 +00:00
{
if (!retry++)
{
2022-02-05 12:29:21 +00:00
p->pos = p->opos = oldpos;
continue;
}
2020-05-18 22:26:07 +00:00
quickkill(p);
return;
}
}
else if (abs(floorz - ceilingz) < 32 && isanunderoperator(psectp->lotag))
2021-11-18 17:51:19 +00:00
fi.activatebysector(psectp, pact);
break;
2020-05-18 22:26:07 +00:00
}
// center_view
if (actions & SB_CENTERVIEW || p->hard_landing)
2020-05-18 22:26:07 +00:00
{
playerCenterView(snum);
}
else if (actions & SB_LOOK_UP)
2020-05-18 22:26:07 +00:00
{
2020-08-28 22:57:07 +00:00
playerLookUp(snum, actions);
2020-05-18 22:26:07 +00:00
}
else if (actions & SB_LOOK_DOWN)
2020-05-18 22:26:07 +00:00
{
2020-08-28 22:57:07 +00:00
playerLookDown(snum, actions);
2020-05-18 22:26:07 +00:00
}
else if (actions & SB_AIM_UP)
2020-05-18 22:26:07 +00:00
{
2020-08-28 22:57:07 +00:00
playerAimUp(snum, actions);
2020-05-18 22:26:07 +00:00
}
else if (actions & SB_AIM_DOWN)
2020-05-18 22:26:07 +00:00
{ // aim_down
2020-08-28 22:57:07 +00:00
playerAimDown(snum, actions);
2020-05-18 22:26:07 +00:00
}
if (SyncInput())
{
p->horizon.applyinput(GetPlayerHorizon(snum), &actions);
}
2020-11-29 08:00:00 +00:00
p->checkhardlanding();
2020-05-18 22:26:07 +00:00
//Shooting code/changes
if (p->show_empty_weapon > 0)
{
p->show_empty_weapon--;
if (p->show_empty_weapon == 0)
{
if (p->last_full_weapon == GROW_WEAPON)
p->subweapon |= (1 << GROW_WEAPON);
else if (p->last_full_weapon == SHRINKER_WEAPON)
p->subweapon &= ~(1 << GROW_WEAPON);
fi.addweapon(p, p->last_full_weapon);
return;
}
}
dokneeattack(snum, { FEM1, FEM2, FEM3, FEM4, FEM5, FEM6, FEM7, FEM8, FEM9, FEM10, PODFEM1, NAKED1, STATUE });
2020-05-18 22:26:07 +00:00
if (fi.doincrements(p)) return;
if (p->weapon_pos != 0)
{
if (p->weapon_pos == -9)
{
if (p->last_weapon >= 0)
{
p->oweapon_pos = p->weapon_pos = 10;
// if(p->curr_weapon == KNEE_WEAPON) p->kickback_pic = 1;
2020-05-18 22:26:07 +00:00
p->last_weapon = -1;
}
else if (p->holster_weapon == 0)
p->oweapon_pos = p->weapon_pos = 10;
2020-05-18 22:26:07 +00:00
}
else p->weapon_pos--;
}
// HACKS
2021-11-21 07:48:36 +00:00
processweapon(snum, actions);
2020-05-19 07:54:52 +00:00
}
2020-05-18 22:26:07 +00:00
END_DUKE_NS