raze/source/games/duke/src/player_r.cpp
Christoph Oelckers 8946a66350 - handle zvel multiplications with a wrapper.
This completes wrapping of zvel.
2022-10-14 20:31:56 +02:00

4181 lines
94 KiB
C++

//-------------------------------------------------------------------------
/*
Copyright (C) 1996, 2003 - 3D Realms Entertainment
Copyright (C) 2017-2019 Nuke.YKT
Copyright (C) 2020 - Christoph Oelckers
This file is part of 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
*/
//-------------------------------------------------------------------------
#include "ns.h"
#include "global.h"
#include "names_r.h"
#include "mapinfo.h"
#include "dukeactor.h"
BEGIN_DUKE_NS
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void incur_damage_r(player_struct* p)
{
int damage = 0, shield_damage = 0;
int gut = 0;
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->steroids_amount > 0 && p->steroids_amount < 400)
{
shield_damage = damage * (20 + (krand() % 30)) / 100;
damage -= shield_damage;
}
if (p->drink_amt > 31 && p->drink_amt < 65)
gut++;
if (p->eat > 31 && p->eat < 65)
gut++;
switch (gut)
{
case 1:
damage = damage * 3 / 4;
break;
case 2:
damage /= 4;
break;
}
p->GetActor()->spr.extra = p->last_extra + damage;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void shootmelee(DDukeActor *actor, int p, int sx, int sy, int sz, int sa, int atwith)
{
auto sectp = actor->sector();
int zvel;
HitInfo hit{};
if (p >= 0)
{
zvel = -ps[p].horizon.sum().asq16() >> 11;
sz += (6 << 8);
sa += 15;
}
else
{
int x;
auto pspr = ps[findplayer(actor, &x)].GetActor();
zvel = ((pspr->int_pos().Z - sz) << 8) / (x + 1);
sa = getangle(pspr->int_pos().X - sx, pspr->int_pos().Y - sy);
}
hitscan(vec3_t( sx, sy, sz ), sectp, { bcos(sa), bsin(sa), zvel << 6 }, hit, CLIPMASK1);
if (isRRRA() && hit.hitSector != nullptr && ((hit.hitSector->lotag == 160 && zvel > 0) || (hit.hitSector->lotag == 161 && zvel < 0))
&& hit.actor() == nullptr && hit.hitWall == nullptr)
{
DukeStatIterator its(STAT_EFFECTOR);
while (auto effector = its.Next())
{
if (effector->sector() == hit.hitSector && effector->spr.picnum == SECTOREFFECTOR && effector->GetOwner()
&& effector->spr.lotag == SE_7_TELEPORT)
{
int nx, ny, nz;
nx = hit.int_hitpos().X + (effector->GetOwner()->int_pos().X - effector->int_pos().X);
ny = hit.int_hitpos().Y + (effector->GetOwner()->int_pos().Y - effector->int_pos().Y);
if (hit.hitSector->lotag == 161)
{
nz = effector->GetOwner()->sector()->int_floorz();
}
else
{
nz = effector->GetOwner()->sector()->int_ceilingz();
}
hitscan(vec3_t( nx, ny, nz ), effector->GetOwner()->sector(), { bcos(sa), bsin(sa), zvel << 6 }, hit, CLIPMASK1);
break;
}
}
}
if (hit.hitSector == nullptr) return;
if ((abs(sx - hit.int_hitpos().X) + abs(sy - hit.int_hitpos().Y)) < 1024)
{
if (hit.hitWall != nullptr || hit.actor())
{
DDukeActor* wpn;
if (isRRRA() && atwith == SLINGBLADE)
{
wpn = CreateActor(hit.hitSector, hit.hitpos, SLINGBLADE, -15, 0, 0, sa, 32, 0, actor, 4);
if (!wpn) return;
wpn->spr.extra += 50;
}
else
{
wpn = CreateActor(hit.hitSector, hit.hitpos, KNEE, -15, 0, 0, sa, 32, 0, actor, 4);
if (!wpn) return;
wpn->spr.extra += (krand() & 7);
}
if (p >= 0)
{
auto k = spawn(wpn, SMALLSMOKE);
if (k) k->spr.pos.Z -= 8;
if (atwith == KNEE) S_PlayActorSound(KICK_HIT, wpn);
else if (isRRRA() && atwith == SLINGBLADE) S_PlayActorSound(260, wpn);
}
if (p >= 0 && ps[p].steroids_amount > 0 && ps[p].steroids_amount < 400)
wpn->spr.extra += (gs.max_player_health >> 2);
if (hit.actor() && hit.actor()->spr.picnum != ACCESSSWITCH && hit.actor()->spr.picnum != ACCESSSWITCH2)
{
fi.checkhitsprite(hit.actor(), wpn);
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)
{
fi.checkhitwall(wpn, hit.hitWall, hit.hitpos, atwith);
if (p >= 0) fi.checkhitswitch(p, hit.hitWall, nullptr);
}
}
}
else if (p >= 0 && zvel > 0 && hit.hitSector->lotag == 1)
{
auto splash = spawn(ps[p].GetActor(), WATERSPLASH2);
if (splash)
{
splash->spr.pos.XY() = hit.hitpos.XY();
splash->spr.angle = ps[p].angle.ang; // Total tweek
splash->spr.xvel = 32;
ssp(actor, 0);
splash->spr.xvel = 0;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void shootweapon(DDukeActor* actor, int p, int sx, int sy, int sz, int sa, int atwith)
{
auto sectp = actor->sector();
int zvel = 0;
HitInfo hit{};
if (actor->spr.extra >= 0) actor->spr.shade = -96;
if (p >= 0)
{
auto aimed = aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
int dal = ((aimed->spr.xrepeat * tileHeight(aimed->spr.picnum)) << 1) + (5 << 8);
zvel = ((aimed->int_pos().Z - sz - dal) << 8) / ldist(ps[p].GetActor(), aimed);
sa = getangle(aimed->int_pos().X - sx, aimed->int_pos().Y - sy);
}
if (atwith == SHOTSPARK1)
{
if (aimed == nullptr)
{
sa += 16 - (krand() & 31);
zvel = -ps[p].horizon.sum().asq16() >> 11;
zvel += 128 - (krand() & 255);
}
}
else
{
if (atwith == SHOTGUN)
sa += 64 - (krand() & 127);
else
sa += 16 - (krand() & 31);
if (aimed == nullptr) zvel = -ps[p].horizon.sum().asq16() >> 11;
zvel += 128 - (krand() & 255);
}
sz -= (2 << 8);
}
else
{
int x;
int j = findplayer(actor, &x);
sz -= (4 << 8);
zvel = ((ps[j].player_int_pos().Z - sz) << 8) / (ldist(ps[j].GetActor(), actor));
if (actor->spr.picnum != BOSS1)
{
zvel += 128 - (krand() & 255);
sa += 32 - (krand() & 63);
}
else
{
zvel += 128 - (krand() & 255);
sa = getangle(ps[j].player_int_pos().X - sx, ps[j].player_int_pos().Y - sy) + 64 - (krand() & 127);
}
}
actor->spr.cstat &= ~CSTAT_SPRITE_BLOCK_ALL;
hitscan(vec3_t( sx, sy, sz ), sectp, { bcos(sa), bsin(sa),zvel << 6 }, hit, CLIPMASK1);
if (isRRRA() && hit.hitSector != nullptr && (((hit.hitSector->lotag == 160 && zvel > 0) || (hit.hitSector->lotag == 161 && zvel < 0))
&& hit.actor() == nullptr && hit.hitWall == nullptr))
{
DukeStatIterator its(STAT_EFFECTOR);
while (auto effector = its.Next())
{
if (effector->sector() == hit.hitSector && effector->spr.picnum == SECTOREFFECTOR && effector->GetOwner()
&& effector->spr.lotag == SE_7_TELEPORT)
{
int nx, ny, nz;
nx = hit.int_hitpos().X + (effector->GetOwner()->int_pos().X - effector->int_pos().X);
ny = hit.int_hitpos().Y + (effector->GetOwner()->int_pos().Y - effector->int_pos().Y);
if (hit.hitSector->lotag == 161)
{
nz = effector->GetOwner()->sector()->int_floorz();
}
else
{
nz = effector->GetOwner()->sector()->int_ceilingz();
}
hitscan(vec3_t( nx, ny, nz ), effector->GetOwner()->sector(), { bcos(sa), bsin(sa), zvel << 6 }, hit, CLIPMASK1);
break;
}
}
}
actor->spr.cstat |= CSTAT_SPRITE_BLOCK_ALL;
if (hit.hitSector == nullptr) return;
if (atwith == SHOTGUN)
if (hit.hitSector->lotag == 1)
if (krand() & 1)
return;
if ((krand() & 15) == 0 && hit.hitSector->lotag == 2)
tracers(hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, sx, sy, sz, 8 - (ud.multimode >> 1));
DDukeActor* spark;
if (p >= 0)
{
spark = CreateActor(hit.hitSector, hit.hitpos, SHOTSPARK1, -15, 10, 10, sa, 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.xrepeat = 0;
spark->spr.yrepeat = 0;
return;
}
else
fi.checkhitceiling(hit.hitSector);
}
if (hit.hitSector->lotag != 1)
spawn(spark, SMALLSMOKE);
}
if (hit.actor())
{
if (hit.actor()->spr.picnum == 1930)
return;
fi.checkhitsprite(hit.actor(), spark);
if (hit.actor()->isPlayer() && (ud.coop != 1 || ud.ffire == 1))
{
auto l = spawn(spark, JIBS6);
spark->spr.xrepeat = spark->spr.yrepeat = 0;
if (l)
{
l->spr.pos.Z += 4;
l->spr.xvel = 16;
l->spr.xrepeat = l->spr.yrepeat = 24;
l->add_int_ang(64 - (krand() & 127));
}
}
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 ||
(isRRRA() && hit.actor()->spr.picnum == RRTILE8660) ||
hit.actor()->spr.picnum == HANDSWITCH ||
hit.actor()->spr.picnum == HANDSWITCH + 1))
{
fi.checkhitswitch(p, nullptr, hit.actor());
return;
}
}
else if (hit.hitWall != nullptr)
{
spawn(spark, SMALLSMOKE);
if (fi.isadoorwall(hit.hitWall->picnum) == 1)
goto SKIPBULLETHOLE;
if (isablockdoor(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 ||
(isRRRA() && hit.hitWall->picnum == RRTILE8660) ||
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 != nullptr && 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());
while (auto l = it.Next())
{
if (l->spr.statnum == STAT_EFFECTOR && l->spr.lotag == SE_13_EXPLOSIVE)
goto SKIPBULLETHOLE;
}
}
DukeStatIterator it(STAT_MISC);
while (auto l = it.Next())
{
if (l->spr.picnum == BULLETHOLE)
if (dist(l, spark) < (12 + (krand() & 7)))
goto SKIPBULLETHOLE;
}
auto hole = spawn(spark, BULLETHOLE);
if (hole)
{
hole->spr.xvel = -1;
hole->set_int_ang(getangle(-hit.hitWall->delta()) + 512);
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();
fi.checkhitwall(spark, hit.hitWall, hit.hitpos, SHOTSPARK1);
}
}
else
{
spark = CreateActor(hit.hitSector, hit.hitpos, SHOTSPARK1, -15, 24, 24, sa, 0, 0, actor, 4);
if (!spark) return;
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.xrepeat = spark->spr.yrepeat = 0;
}
else if (hit.hitWall != nullptr)
fi.checkhitwall(spark, hit.hitWall, hit.hitpos, SHOTSPARK1);
}
if ((krand() & 255) < 10)
{
S_PlaySound3D(PISTOL_RICOCHET, spark, hit.int_hitpos());
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void shootstuff(DDukeActor* actor, int p, int sx, int sy, int sz, int sa, int atwith)
{
auto sect = actor->sector();
int vel = 0, zvel;
int scount;
if (isRRRA())
{
if (atwith != SPIT && actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
if (atwith == SPIT)
{
if (actor->spr.picnum == 8705)
vel = 600;
else
vel = 400;
}
}
else
{
if (actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
if (atwith == SPIT) vel = 400;
}
if (atwith != SPIT)
{
vel = 840;
sz -= (4 << 7);
if (actor->spr.picnum == 4649)
{
sx += bcos(actor->int_ang() + 256, -6);
sy += bsin(actor->int_ang() + 256, -6);
sz += (12 << 8);
}
if (actor->spr.picnum == VIXEN)
{
sz -= (12 << 8);
}
}
if (p >= 0)
{
auto aimed = aim(actor, AUTO_AIM_ANGLE);
sx += bcos(actor->int_ang() + 160, -7);
sy += bsin(actor->int_ang() + 160, -7);
if (aimed)
{
int dal = ((aimed->spr.xrepeat * tileHeight(aimed->spr.picnum)) << 1) - (12 << 8);
zvel = ((aimed->int_pos().Z - sz - dal) * vel) / ldist(ps[p].GetActor(), aimed);
sa = getangle(aimed->int_pos().X - sx, aimed->int_pos().Y - sy);
}
else
{
zvel = -MulScale(ps[p].horizon.sum().asq16(), 98, 16);
}
}
else
{
int x;
int j = findplayer(actor, &x);
// sa = getangle(ps[j].oposx-sx,ps[j].oposy-sy);
if (actor->spr.picnum == HULK)
sa -= (krand() & 31);
else if (actor->spr.picnum == VIXEN)
sa -= (krand() & 16);
else if (actor->spr.picnum != UFOBEAM)
sa += 16 - (krand() & 31);
zvel = (((ps[j].player_int_opos().Z - sz + (3 << 8))) * vel) / ldist(ps[j].GetActor(), actor);
}
int oldzvel = zvel;
int sizx, sizy;
if (atwith == SPIT)
{
sizx = 18; sizy = 18;
if (!isRRRA() || actor->spr.picnum != MAMA) sz -= (10 << 8); else sz -= (20 << 8);
}
else
{
if (atwith == COOLEXPLOSION1)
{
sizx = 8;
sizy = 8;
}
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 j = EGS(sect, sx, sy, sz, atwith, -127, sizx, sizy, sa, vel, zvel, actor, 4);
if (!j) return;
j->spr.extra += (krand() & 7);
j->spr.cstat = CSTAT_SPRITE_YCENTER;
j->spr.clipdist = 4;
sa = actor->int_ang() + 32 - (krand() & 63);
zvel = oldzvel + 512 - (krand() & 1023);
if (atwith == FIRELASER)
{
j->spr.xrepeat = 8;
j->spr.yrepeat = 8;
}
scount--;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void shootrpg(DDukeActor* actor, int p, int sx, int sy, int sz, int sa, int atwith)
{
auto sect = actor->sector();
int vel, zvel;
int l, scount;
DDukeActor* act90 = nullptr;
if (actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
vel = 644;
DDukeActor* aimed = nullptr;
if (p >= 0)
{
aimed = aim(actor, 48);
if (aimed)
{
if (isRRRA() && atwith == RPG2)
{
if (aimed->spr.picnum == HEN || aimed->spr.picnum == HENSTAYPUT)
act90 = ps[screenpeek].GetActor();
else
act90 = aimed;
}
int dal = ((aimed->spr.xrepeat * tileHeight(aimed->spr.picnum)) << 1) + (8 << 8);
zvel = ((aimed->int_pos().Z - sz - dal) * vel) / ldist(ps[p].GetActor(), aimed);
if (aimed->spr.picnum != RECON)
sa = getangle(aimed->int_pos().X - sx, aimed->int_pos().Y - sy);
}
else zvel = -MulScale(ps[p].horizon.sum().asq16(), 81, 16);
if (atwith == RPG)
S_PlayActorSound(RPG_SHOOT, actor);
else if (isRRRA())
{
if (atwith == RPG2)
S_PlayActorSound(244, actor);
else if (atwith == RRTILE1790)
S_PlayActorSound(94, actor);
}
}
else
{
int x;
int j = findplayer(actor, &x);
sa = getangle(ps[j].player_int_opos().X - sx, ps[j].player_int_opos().Y - sy);
if (actor->spr.picnum == BOSS3)
sz -= (32 << 8);
else if (actor->spr.picnum == BOSS2)
{
vel += 128;
sz += 24 << 8;
}
l = ldist(ps[j].GetActor(), actor);
zvel = ((ps[j].player_int_opos().Z - sz) * vel) / l;
if (badguy(actor) && (actor->spr.hitag & face_player_smart))
sa = actor->int_ang() + (krand() & 31) - 16;
}
if (p < 0) aimed = nullptr;
if (isRRRA() && atwith == RRTILE1790)
{
zvel = -(10 << 8);
vel <<= 1;
}
auto spawned = EGS(sect,
sx + (bcos(sa + 348) / 448),
sy + (bsin(sa + 348) / 448),
sz - (1 << 8), atwith, 0, 14, 14, sa, vel, zvel, actor, 4);
if (!spawned) return;
if (isRRRA())
{
if (atwith == RRTILE1790)
{
spawned->spr.extra = 10;
spawned->set_int_zvel(-(10 << 8));
}
else if (atwith == RPG2)
{
spawned->seek_actor = act90;
spawned->spr.hitag = 0;
fi.lotsofmoney(spawned, (krand() & 3) + 1);
}
}
spawned->spr.extra += (krand() & 7);
if (atwith != FREEZEBLAST)
spawned->temp_actor = aimed;
else
{
spawned->spr.yvel = gs.numfreezebounces;
spawned->spr.xrepeat >>= 1;
spawned->spr.yrepeat >>= 1;
spawned->add_int_zvel(- (2 << 4));
}
if (p == -1)
{
if (actor->spr.picnum == HULK)
{
spawned->spr.xrepeat = 8;
spawned->spr.yrepeat = 8;
}
else if (atwith != FREEZEBLAST)
{
spawned->spr.xrepeat = 30;
spawned->spr.yrepeat = 30;
spawned->spr.extra >>= 2;
}
}
else if (ps[p].curr_weapon == TIT_WEAPON)
{
spawned->spr.extra >>= 2;
spawned->add_int_ang(16 - (krand() & 31));
spawned->add_int_zvel( 256 - (krand() & 511));
if (ps[p].hbomb_hold_delay)
{
spawned->add_int_pos({ -bsin(sa) / 644, bcos(sa) / 644, 0 });
}
else
{
spawned->add_int_pos({ bsin(sa, -8), -bcos(sa, -8), 0 });
}
spawned->spr.xrepeat >>= 1;
spawned->spr.yrepeat >>= 1;
}
spawned->spr.cstat = CSTAT_SPRITE_YCENTER;
if (atwith == RPG || (atwith == RPG2 && isRRRA()))
spawned->spr.clipdist = 4;
else
spawned->spr.clipdist = 40;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void shootwhip(DDukeActor* actor, int p, int sx, int sy, int sz, int sa, int atwith)
{
auto sect = actor->sector();
int vel = 0, zvel;
int scount;
if (actor->spr.extra >= 0) actor->spr.shade = -96;
scount = 1;
if (atwith == 3471)
{
vel = 300;
sz -= (15 << 8);
scount = 1;
}
else if (atwith == 3475)
{
vel = 300;
sz += (4 << 8);
scount = 1;
}
if (p >= 0)
{
auto aimed = aim(actor, AUTO_AIM_ANGLE);
if (aimed)
{
int dal = ((aimed->spr.xrepeat * tileHeight(aimed->spr.picnum)) << 1) - (12 << 8);
zvel = ((aimed->int_pos().Z - sz - dal) * vel) / ldist(ps[p].GetActor(), aimed);
sa = getangle(aimed->int_pos().X - sx, aimed->int_pos().Y - sy);
}
else
zvel = -MulScale(ps[p].horizon.sum().asq16(), 98, 16);
}
else
{
int x;
int j = findplayer(actor, &x);
// sa = getangle(ps[j].oposx-sx,ps[j].oposy-sy);
if (actor->spr.picnum == VIXEN)
sa -= (krand() & 16);
else
sa += 16 - (krand() & 31);
zvel = (((ps[j].player_int_opos().Z - sz + (3 << 8))) * vel) / ldist(ps[j].GetActor(), actor);
}
int oldzvel = zvel;
int sizx = 18;
int sizy = 18;
if (p >= 0) sizx = 7, sizy = 7;
else sizx = 8, sizy = 8;
while (scount > 0)
{
auto j = EGS(sect, sx, sy, sz, atwith, -127, sizx, sizy, sa, vel, zvel, actor, 4);
if (!j) return;
j->spr.extra += (krand() & 7);
j->spr.cstat = CSTAT_SPRITE_YCENTER;
j->spr.clipdist = 4;
sa = actor->int_ang() + 32 - (krand() & 63);
zvel = oldzvel + 512 - (krand() & 1023);
scount--;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void shoot_r(DDukeActor* actor, int atwith)
{
int sa, p;
int sx, sy, sz, vel, zvel, x;
auto const sect = actor->sector();
zvel = 0;
if (actor->isPlayer())
{
p = actor->spr.yvel;
sx = ps[p].player_int_pos().X;
sy = ps[p].player_int_pos().Y;
sz = ps[p].player_int_pos().Z + ps[p].pyoff * zworldtoint + (4 << 8);
sa = ps[p].angle.ang.Buildang();
if (isRRRA()) ps[p].crack_time = CRACK_TIME;
}
else
{
p = -1;
sa = actor->int_ang();
sx = actor->int_pos().X;
sy = actor->int_pos().Y;
sz = actor->int_pos().Z - ((actor->spr.yrepeat * tileHeight(actor->spr.picnum)) << 1) + (4 << 8);
sz -= (7 << 8);
if (badguy(actor))
{
sx -= bsin(sa, -7);
sy += bcos(sa, -7);
}
}
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;
}
switch (atwith)
{
case BLOODSPLAT1:
case BLOODSPLAT2:
case BLOODSPLAT3:
case BLOODSPLAT4:
shootbloodsplat(actor, p, sx, sy, sz, sa, atwith, BIGFORCE, OOZFILTER, -1);
return;
case SLINGBLADE:
if (!isRRRA()) break;
[[fallthrough]];
case KNEE:
case GROWSPARK:
shootmelee(actor, p, sx, sy, sz, sa, atwith);
return;
case SHOTSPARK1:
case SHOTGUN:
case CHAINGUN:
shootweapon(actor, p, sx, sy, sz, sa, atwith);
return;
case POWDERKEG:
{
auto j = spawn(actor, atwith);
if (j)
{
j->spr.xvel = 32;
j->spr.angle = actor->spr.angle;
j->spr.pos.Z -= 5;
}
break;
}
case BOWLINGBALL:
{
auto j = spawn(actor, atwith);
if (j)
{
j->spr.xvel = 250;
j->spr.angle = actor->spr.angle;
j->spr.pos.Z -= 15;
}
break;
}
case OWHIP:
case UWHIP:
shootwhip(actor, p, sx, sy, sz, sa, atwith);
return;
case FIRELASER:
case SPIT:
case COOLEXPLOSION1:
shootstuff(actor, p, sx, sy, sz, sa, atwith);
return;
case RPG2:
case RRTILE1790:
if (isRRRA()) goto rrra_rpg2;
else break;
case FREEZEBLAST:
sz += (3 << 8);
case RPG:
case SHRINKSPARK:
rrra_rpg2:
shootrpg(actor, p, sx, sy, sz, sa, atwith);
break;
case CHEERBOMB:
if (!isRRRA()) break;
case MORTER:
{
if (actor->spr.extra >= 0) actor->spr.shade = -96;
auto j = ps[findplayer(actor, &x)].GetActor();
x = ldist(j, actor);
zvel = -x >> 1;
if (zvel < -4096)
zvel = -2048;
vel = x >> 4;
if (atwith == CHEERBOMB)
EGS(sect,
sx - bsin(sa + 512, -8),
sy + bcos(sa + 512, -8),
sz + (6 << 8), atwith, -64, 16, 16, sa, vel, zvel, actor, 1);
else
EGS(sect,
sx - bsin(sa + 512, -8),
sy + bcos(sa + 512, -8),
sz + (6 << 8), atwith, -64, 32, 32, sa, vel, zvel, actor, 1);
break;
}
}
return;
}
//---------------------------------------------------------------------------
//
// this is one lousy hack job...
//
//---------------------------------------------------------------------------
void selectweapon_r(int snum, int weap)
{
int i, j, k;
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 > 8 && p->access_incs == 0 && p->knee_incs == 0)
{
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 THROWSAW_WEAPON:
if (p->ammo_amount[BUZZSAW_WEAPON] > 0)
{
j = BUZZSAW_WEAPON;
p->subweapon = 1 << BUZZSAW_WEAPON;
}
break;
case BUZZSAW_WEAPON:
if (p->ammo_amount[THROWSAW_WEAPON] > 0)
{
j = THROWSAW_WEAPON;
p->subweapon = 0;
}
break;
case POWDERKEG_WEAPON:
if (p->ammo_amount[BOWLING_WEAPON] > 0)
{
j = BOWLING_WEAPON;
p->subweapon = 1 << BOWLING_WEAPON;
}
break;
case BOWLING_WEAPON:
if (p->ammo_amount[POWDERKEG_WEAPON] > 0)
{
j = POWDERKEG_WEAPON;
p->subweapon = 0;
}
break;
case KNEE_WEAPON:
if (isRRRA())
{
j = SLINGBLADE_WEAPON;
p->subweapon = 2;
}
break;
case SLINGBLADE_WEAPON:
j = KNEE_WEAPON;
p->subweapon = 0;
break;
case CROSSBOW_WEAPON:
if (p->ammo_amount[CHICKEN_WEAPON] > 0 && isRRRA())
{
j = CHICKEN_WEAPON;
p->subweapon = 4;
}
break;
case CHICKEN_WEAPON:
if (p->ammo_amount[CROSSBOW_WEAPON] > 0)
{
j = CROSSBOW_WEAPON;
p->subweapon = 0;
}
break;
default:
break;
}
}
else if (weap == WeaponSel_Next || weap == WeaponSel_Prev)
{
k = p->curr_weapon;
if (isRRRA())
{
if (k == CHICKEN_WEAPON) k = CROSSBOW_WEAPON;
else if (k == BUZZSAW_WEAPON) k = THROWSAW_WEAPON;
else if (k == SLINGBLADE_WEAPON) k = KNEE_WEAPON;
}
j = (weap == WeaponSel_Prev ? -1 : 1); // JBF: prev (-1) or next (1) weapon choice
i = 0;
while (k >= 0 && k < 10)
{
k += j;
if (k == -1) k = 9;
else if (k == 10) k = 0;
if (p->gotweapon[k] && p->ammo_amount[k] > 0)
{
j = k;
break;
}
i++;
if (i == 10)
{
fi.addweapon(p, KNEE_WEAPON);
break;
}
}
}
else j = weap - 1;
k = -1;
if (j == DYNAMITE_WEAPON && p->ammo_amount[DYNAMITE_WEAPON] == 0)
{
DukeStatIterator it(STAT_ACTOR);
while (auto act = it.Next())
{
if (act->spr.picnum == HEAVYHBOMB && act->GetOwner() == p->GetActor())
{
p->gotweapon[DYNAMITE_WEAPON] = true;
j = THROWINGDYNAMITE_WEAPON;
break;
}
}
}
else if (j == KNEE_WEAPON && isRRRA())
{
if (p->curr_weapon == KNEE_WEAPON)
{
p->subweapon = 2;
j = SLINGBLADE_WEAPON;
}
else if (p->subweapon & 2)
{
p->subweapon = 0;
j = KNEE_WEAPON;
}
}
else if (j == CROSSBOW_WEAPON && isRRRA())
{
if (p->curr_weapon == CROSSBOW_WEAPON || p->ammo_amount[CROSSBOW_WEAPON] == 0)
{
if (p->ammo_amount[CHICKEN_WEAPON] == 0)
return;
p->subweapon = 4;
j = CHICKEN_WEAPON;
}
else if ((p->subweapon & 4) || p->ammo_amount[CHICKEN_WEAPON] == 0)
{
p->subweapon = 0;
j = CROSSBOW_WEAPON;
}
}
else if (j == THROWSAW_WEAPON)
{
if (p->curr_weapon == THROWSAW_WEAPON || p->ammo_amount[THROWSAW_WEAPON] == 0)
{
p->subweapon = (1 << BUZZSAW_WEAPON);
j = BUZZSAW_WEAPON;
}
else if ((p->subweapon & (1 << BUZZSAW_WEAPON)) || p->ammo_amount[BUZZSAW_WEAPON] == 0)
{
p->subweapon = 0;
j = THROWSAW_WEAPON;
}
}
else if (j == POWDERKEG_WEAPON)
{
if (p->curr_weapon == POWDERKEG_WEAPON || p->ammo_amount[POWDERKEG_WEAPON] == 0)
{
p->subweapon = (1 << BOWLING_WEAPON);
j = BOWLING_WEAPON;
}
else if ((p->subweapon & (1 << BOWLING_WEAPON)) || p->ammo_amount[BOWLING_WEAPON] == 0)
{
p->subweapon = 0;
j = POWDERKEG_WEAPON;
}
}
if (p->holster_weapon)
{
PlayerSetInput(snum, SB_HOLSTER);
p->oweapon_pos = p->weapon_pos = -9;
}
else if (j >= MIN_WEAPON && p->gotweapon[j] && p->curr_weapon != j) switch (j)
{
case KNEE_WEAPON:
fi.addweapon(p, j);
break;
case SLINGBLADE_WEAPON:
if (isRRRA())
{
S_PlayActorSound(496, ps[screenpeek].GetActor());
fi.addweapon(p, j);
}
break;
case PISTOL_WEAPON:
if (p->ammo_amount[PISTOL_WEAPON] == 0)
if (p->show_empty_weapon == 0)
{
p->last_full_weapon = p->curr_weapon;
p->show_empty_weapon = 32;
}
fi.addweapon(p, PISTOL_WEAPON);
break;
case CHICKEN_WEAPON:
if (!isRRRA()) break;
[[fallthrough]];
case SHOTGUN_WEAPON:
case RIFLEGUN_WEAPON:
case CROSSBOW_WEAPON:
case TIT_WEAPON:
case ALIENBLASTER_WEAPON:
case THROWSAW_WEAPON:
case BUZZSAW_WEAPON:
case POWDERKEG_WEAPON:
case BOWLING_WEAPON:
if (p->ammo_amount[j] == 0 && p->show_empty_weapon == 0)
{
p->last_full_weapon = p->curr_weapon;
p->show_empty_weapon = 32;
}
fi.addweapon(p, j);
break;
case MOTORCYCLE_WEAPON:
case BOAT_WEAPON:
if (isRRRA())
{
if (p->ammo_amount[j] == 0 && p->show_empty_weapon == 0)
{
p->show_empty_weapon = 32;
}
fi.addweapon(p, j);
}
break;
case THROWINGDYNAMITE_WEAPON:
if (k >= 0) // Found in list of [1]'s
{
p->curr_weapon = THROWINGDYNAMITE_WEAPON;
p->last_weapon = -1;
p->oweapon_pos = p->weapon_pos = 10;
}
break;
case DYNAMITE_WEAPON:
if (p->ammo_amount[DYNAMITE_WEAPON] > 0 && p->gotweapon[DYNAMITE_WEAPON])
fi.addweapon(p, DYNAMITE_WEAPON);
break;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
int doincrements_r(player_struct* p)
{
int snum;
auto pact = p->GetActor();
if (isRRRA())
{
if (WindTime > 0)
WindTime--;
else if ((krand() & 127) == 8)
{
WindTime = 120 + ((krand() & 63) << 2);
WindDir = krand() & 2047;
}
if (BellTime > 0)
{
BellTime--;
if (BellTime == 0 && BellSprite)
BellSprite->spr.picnum++;
}
if (chickenphase > 0)
chickenphase--;
if (p->SeaSick)
{
p->SeaSick--;
if (p->SeaSick == 0)
p->sea_sick_stat = 0;
}
}
snum = p->GetActor()->spr.yvel;
p->player_par++;
if (p->yehaa_timer)
p->yehaa_timer--;
if (p->detonate_count > 0)
{
p->detonate_count++;
p->detonate_time--;
}
p->drink_timer--;
if (p->drink_timer <= 0)
{
p->drink_timer = 1024;
if (p->drink_amt)
{
p->drink_amt--;
}
}
p->eat_timer--;
if (p->eat_timer <= 0)
{
p->eat_timer = 1024;
if (p->eat)
p->eat--;
}
if (p->drink_amt >= 100)
{
if (!S_CheckActorSoundPlaying(pact, 420))
S_PlayActorSound(420, pact);
p->drink_amt -= 9;
p->eat >>= 1;
}
p->eatang = (1647 + p->eat * 8) & 2047;
if (p->eat >= 100)
p->eat = 100;
if (p->eat >= 31 && krand() < p->eat)
{
switch (krand() & 3)
{
case 0:
S_PlayActorSound(404, pact);
break;
case 1:
S_PlayActorSound(422, pact);
break;
case 2:
S_PlayActorSound(423, pact);
break;
case 3:
S_PlayActorSound(424, pact);
break;
}
if (numplayers < 2)
{
p->noise_radius = 16384;
madenoise(screenpeek);
p->vel.X += p->angle.ang.Cos() * (1 << 18);
p->vel.Y += p->angle.ang.Sin() * (1 << 18);
}
p->eat -= 4;
if (p->eat < 0)
p->eat = 0;
}
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->drink_amt > 66 && (p->last_pissed_time % 26) == 0)
p->drink_amt--;
{
if (p->last_pissed_time == 5662)
S_PlayActorSound(434, pact);
else if (p->last_pissed_time == 5567)
S_PlayActorSound(434, pact);
else if (p->last_pissed_time == 5472)
S_PlayActorSound(433, pact);
else if (p->last_pissed_time == 5072)
S_PlayActorSound(435, pact);
else if (p->last_pissed_time == 5014)
S_PlayActorSound(434, pact);
else if (p->last_pissed_time == 4919)
S_PlayActorSound(433, pact);
}
if (p->last_pissed_time == 5668)
{
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);
p->eat = p->drink_amt = 0;
p->eatang = p->drunkang = 1647;
}
if (!(p->steroids_amount & 14))
if (snum == screenpeek || ud.coop == 1)
S_PlayActorSound(DUKE_TAKEPILLS, pact);
}
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)
{
if (p->access_spritenum != nullptr)
{
fi.checkhitswitch(snum, nullptr, p->access_spritenum);
switch (p->access_spritenum->spr.pal)
{
case 0:p->keys[1] = 1; break;
case 21:p->keys[2] = 1; break;
case 23:p->keys[3] = 1; break;
}
p->access_spritenum = nullptr;
}
else
{
fi.checkhitswitch(snum, p->access_wall, nullptr);
switch (p->access_wall->pal)
{
case 0:p->keys[1] = 1; break;
case 21:p->keys[2] = 1; break;
case 23:p->keys[3] = 1; 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)
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)
{
if (!wupass)
{
int snd = currentLevel->rr_startsound ? currentLevel->rr_startsound : 391;
wupass = 1;
S_PlayActorSound(snd, pact);
}
else if (PlayClock > 1024)
if (snum == screenpeek || ud.coop == 1)
{
if (rand() & 1)
S_PlayActorSound(DUKE_CRACK, pact);
else S_PlayActorSound(DUKE_CRACK2, pact);
}
}
else if (p->knuckle_incs == 22 || PlayerInput(snum, SB_FIRE))
p->knuckle_incs = 0;
return 1;
}
return 0;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void checkweapons_r(player_struct* p)
{
static const uint16_t weapon_sprites[MAX_WEAPONS] = { KNEE, FIRSTGUNSPRITE, SHOTGUNSPRITE,
CHAINGUNSPRITE, RPGSPRITE, HEAVYHBOMB, SHRINKERSPRITE, DEVISTATORSPRITE,
POWDERKEG, BOWLINGBALLSPRITE, FREEZEBLAST, HEAVYHBOMB };
if (isRRRA())
{
if (p->OnMotorcycle && numplayers > 1)
{
auto j = spawn(p->GetActor(), 7220);
if (j)
{
j->spr.angle = p->angle.ang;
j->saved_ammo = p->ammo_amount[MOTORCYCLE_WEAPON];
}
p->OnMotorcycle = 0;
p->gotweapon[MOTORCYCLE_WEAPON] = false;
p->horizon.horiz = q16horiz(0);
p->moto_do_bump = 0;
p->MotoSpeed = 0;
p->TiltStatus = 0;
p->moto_drink = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
p->TurbCount = 0;
}
else if (p->OnBoat && numplayers > 1)
{
auto j = spawn(p->GetActor(), 7233);
if (j)
{
j->spr.angle = p->angle.ang;
j->saved_ammo = p->ammo_amount[BOAT_WEAPON];
}
p->OnBoat = 0;
p->gotweapon[BOAT_WEAPON] = false;
p->horizon.horiz = q16horiz(0);
p->moto_do_bump = 0;
p->MotoSpeed = 0;
p->TiltStatus = 0;
p->moto_drink = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
p->TurbCount = 0;
}
}
if (p->curr_weapon > 0)
{
if (krand() & 1)
spawn(p->GetActor(), weapon_sprites[p->curr_weapon]);
else switch (p->curr_weapon)
{
case CHICKEN_WEAPON:
if (!isRRRA()) break;
[[fallthrough]];
case DYNAMITE_WEAPON:
case CROSSBOW_WEAPON:
spawn(p->GetActor(), EXPLOSION2);
break;
}
}
for (int i = 0; i < 5; i++)
{
if (p->keys[i] == 1)
{
auto j = spawn(p->GetActor(), ACCESSCARD);
if (j) switch (i)
{
case 1:
j->spr.lotag = 100;
break;
case 2:
j->spr.lotag = 101;
break;
case 3:
j->spr.lotag = 102;
break;
case 4:
j->spr.lotag = 103;
break;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void onMotorcycle(int snum, ESyncBits &actions)
{
auto p = &ps[snum];
auto pact = p->GetActor();
int rng;
if (p->MotoSpeed < 0)
p->MotoSpeed = 0;
if (p->vehForwardScale != 0)
{
if (p->on_ground)
{
if (p->MotoSpeed == 0 && p->vehBraking)
{
if (!S_CheckActorSoundPlaying(pact, 187))
S_PlayActorSound(187, pact);
}
else if (p->MotoSpeed == 0 && !S_CheckActorSoundPlaying(pact, 214))
{
if (S_CheckActorSoundPlaying(pact, 187))
S_StopSound(187, pact);
S_PlayActorSound(214, pact);
}
else if (p->MotoSpeed >= 50 && !S_CheckActorSoundPlaying(pact, 188))
{
S_PlayActorSound(188, pact);
}
else if (!S_CheckActorSoundPlaying(pact, 188) && !S_CheckActorSoundPlaying(pact, 214))
{
S_PlayActorSound(188, pact);
}
}
}
else
{
if (S_CheckActorSoundPlaying(pact, 214))
{
S_StopSound(214, pact);
if (!S_CheckActorSoundPlaying(pact, 189))
S_PlayActorSound(189, pact);
}
if (S_CheckActorSoundPlaying(pact, 188))
{
S_StopSound(188, pact);
if (!S_CheckActorSoundPlaying(pact, 189))
S_PlayActorSound(189, pact);
}
if (!S_CheckActorSoundPlaying(pact, 189) && !S_CheckActorSoundPlaying(pact, 187))
S_PlayActorSound(187, pact);
}
if (p->drink_amt > 88 && p->moto_drink == 0)
{
rng = krand() & 63;
if (rng == 1)
p->moto_drink = -10;
else if (rng == 2)
p->moto_drink = 10;
}
else if (p->drink_amt > 99 && p->moto_drink == 0)
{
rng = krand() & 31;
if (rng == 1)
p->moto_drink = -20;
else if (rng == 2)
p->moto_drink = 20;
}
if (p->on_ground == 1)
{
if (p->vehBraking && p->MotoSpeed > 0)
{
p->MotoSpeed -= p->moto_on_oil ? 2 : 4;
if (p->MotoSpeed < 0)
p->MotoSpeed = 0;
p->VBumpTarget = -30;
p->moto_do_bump = 1;
}
else if (p->vehForwardScale != 0 && !p->vehBraking)
{
if (p->MotoSpeed < 40)
{
p->VBumpTarget = 70;
p->moto_bump_fast = 1;
}
p->MotoSpeed += 2 * p->vehForwardScale;
p->vehForwardScale = 0;
if (p->MotoSpeed > 120)
p->MotoSpeed = 120;
if (!p->NotOnWater && p->MotoSpeed > 80)
p->MotoSpeed = 80;
}
else if (p->MotoSpeed > 0)
p->MotoSpeed--;
if (p->moto_do_bump && (!p->vehBraking || p->MotoSpeed == 0))
{
p->VBumpTarget = 0;
p->moto_do_bump = 0;
}
if (p->vehReverseScale != 0 && p->MotoSpeed <= 0 && !p->vehBraking)
{
bool temp = p->vehTurnRight;
p->vehTurnRight = p->vehTurnLeft;
p->vehTurnLeft = temp;
p->MotoSpeed = -15 * p->vehReverseScale;
p->vehReverseScale = 0;
}
}
if (p->MotoSpeed != 0 && p->on_ground == 1)
{
if (!p->VBumpNow && (krand() & 3) == 2)
p->VBumpTarget = p->MotoSpeed * (1. / 16.) * ((krand() & 7) - 4);
if (p->vehTurnLeft || p->moto_drink < 0)
{
if (p->moto_drink < 0)
p->moto_drink++;
}
else if (p->vehTurnRight || p->moto_drink > 0)
{
if (p->moto_drink > 0)
p->moto_drink--;
}
}
double horiz = FRACUNIT;
if (p->TurbCount)
{
if (p->TurbCount <= 1)
{
horiz = 0;
p->TurbCount = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
}
else
{
horiz = ((krand() & 15) - 7);
p->TurbCount--;
p->moto_drink = (krand() & 3) - 2;
}
}
else if (p->VBumpTarget > p->VBumpNow)
{
p->VBumpNow += p->moto_bump_fast ? 6 : 1;
if (p->VBumpTarget < p->VBumpNow)
p->VBumpNow = p->VBumpTarget;
horiz = p->VBumpNow * (1. / 3.);
}
else if (p->VBumpTarget < p->VBumpNow)
{
p->VBumpNow -= p->moto_bump_fast ? 6 : 1;
if (p->VBumpTarget > p->VBumpNow)
p->VBumpNow = p->VBumpTarget;
horiz = p->VBumpNow * (1. / 3.);
}
else
{
p->VBumpTarget = 0;
p->moto_bump_fast = 0;
}
if (horiz != FRACUNIT)
{
p->horizon.addadjustment(buildfhoriz(horiz) - p->horizon.horiz);
}
int currSpeed = int(p->MotoSpeed);
int velAdjustment;
if (p->MotoSpeed >= 20 && p->on_ground == 1 && (p->vehTurnLeft || p->vehTurnRight))
{
velAdjustment = p->vehTurnLeft ? -10 : 10;
auto angAdjustment = (velAdjustment < 0 ? 350 : -350) << BAMBITS;
if (p->moto_on_mud || p->moto_on_oil || !p->NotOnWater)
{
currSpeed <<= p->moto_on_oil ? 3 : 2;
if (p->moto_do_bump)
{
currSpeed >>= 5;
angAdjustment >>= 2;
}
else
{
currSpeed >>= 7;
angAdjustment >>= 6;
}
p->moto_on_mud = 0;
p->moto_on_oil = 0;
}
else
{
if (p->moto_do_bump)
{
currSpeed >>= 5;
angAdjustment >>= 4;
if (!S_CheckActorSoundPlaying(pact, 220))
S_PlayActorSound(220, pact);
}
else
{
currSpeed >>= 7;
angAdjustment >>= 7;
}
}
p->vel.X += currSpeed * bcos(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
p->vel.Y += currSpeed * bsin(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
p->angle.addadjustment(deltaangle(p->angle.ang, p->angle.ang - DAngle::fromBam(angAdjustment)));
}
else if (p->MotoSpeed >= 20 && p->on_ground == 1 && (p->moto_on_mud || p->moto_on_oil))
{
rng = krand() & 1;
velAdjustment = rng == 0 ? -10 : 10;
currSpeed = MulScale(currSpeed, p->moto_on_oil ? 10 : 5, 7);
p->vel.X += currSpeed * bcos(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
p->vel.Y += currSpeed * bsin(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
}
p->moto_on_mud = p->moto_on_oil = 0;
p->vehTurnLeft = p->vehTurnRight = p->vehBraking = false;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void onBoat(int snum, ESyncBits &actions)
{
auto p = &ps[snum];
auto pact = p->GetActor();
bool heeltoe;
int rng;
if (p->NotOnWater)
{
if (p->MotoSpeed > 0)
{
if (!S_CheckActorSoundPlaying(pact, 88))
S_PlayActorSound(88, pact);
}
else
{
if (!S_CheckActorSoundPlaying(pact, 87))
S_PlayActorSound(87, pact);
}
}
if (p->MotoSpeed < 0)
p->MotoSpeed = 0;
if (p->vehBraking && (p->vehForwardScale != 0))
{
heeltoe = true;
p->vehBraking = false;
p->vehForwardScale = 0;
}
else
heeltoe = false;
if (p->vehForwardScale != 0)
{
if (p->MotoSpeed == 0 && !S_CheckActorSoundPlaying(pact, 89))
{
if (S_CheckActorSoundPlaying(pact, 87))
S_StopSound(87, pact);
S_PlayActorSound(89, pact);
}
else if (p->MotoSpeed >= 50 && !S_CheckActorSoundPlaying(pact, 88))
S_PlayActorSound(88, pact);
else if (!S_CheckActorSoundPlaying(pact, 88) && !S_CheckActorSoundPlaying(pact, 89))
S_PlayActorSound(88, pact);
}
else
{
if (S_CheckActorSoundPlaying(pact, 89))
{
S_StopSound(89, pact);
if (!S_CheckActorSoundPlaying(pact, 90))
S_PlayActorSound(90, pact);
}
if (S_CheckActorSoundPlaying(pact, 88))
{
S_StopSound(88, pact);
if (!S_CheckActorSoundPlaying(pact, 90))
S_PlayActorSound(90, pact);
}
if (!S_CheckActorSoundPlaying(pact, 90) && !S_CheckActorSoundPlaying(pact, 87))
S_PlayActorSound(87, pact);
}
if (p->vehTurnLeft && !S_CheckActorSoundPlaying(pact, 91) && p->MotoSpeed > 30 && !p->NotOnWater)
S_PlayActorSound(91, pact);
if (p->vehTurnRight && !S_CheckActorSoundPlaying(pact, 91) && p->MotoSpeed > 30 && !p->NotOnWater)
S_PlayActorSound(91, pact);
if (!p->NotOnWater)
{
if (p->drink_amt > 88 && p->moto_drink == 0)
{
rng = krand() & 63;
if (rng == 1)
p->moto_drink = -10;
else if (rng == 2)
p->moto_drink = 10;
}
else if (p->drink_amt > 99 && p->moto_drink == 0)
{
rng = krand() & 31;
if (rng == 1)
p->moto_drink = -20;
else if (rng == 2)
p->moto_drink = 20;
}
}
if (p->on_ground == 1)
{
if (heeltoe)
{
if (p->MotoSpeed <= 25)
{
p->MotoSpeed++;
if (!S_CheckActorSoundPlaying(pact, 182))
S_PlayActorSound(182, pact);
}
else
{
p->MotoSpeed -= 2;
if (p->MotoSpeed < 0)
p->MotoSpeed = 0;
p->VBumpTarget = 30;
p->moto_do_bump = 1;
}
}
else if (p->vehBraking && p->MotoSpeed > 0)
{
p->MotoSpeed -= 2;
if (p->MotoSpeed < 0)
p->MotoSpeed = 0;
p->VBumpTarget = 30;
p->moto_do_bump = 1;
}
else if (p->vehForwardScale != 0)
{
if (p->MotoSpeed < 40 && !p->NotOnWater)
{
p->VBumpTarget = -30;
p->moto_bump_fast = 1;
}
p->MotoSpeed += 1 * p->vehForwardScale;
p->vehForwardScale = 0;
if (p->MotoSpeed > 120)
p->MotoSpeed = 120;
}
else if (p->MotoSpeed > 0)
p->MotoSpeed--;
if (p->moto_do_bump && (!p->vehBraking || p->MotoSpeed == 0))
{
p->VBumpTarget = 0;
p->moto_do_bump = 0;
}
if (p->vehReverseScale != 0 && p->MotoSpeed == 0 && !p->vehBraking)
{
bool temp = p->vehTurnRight;
p->vehTurnRight = p->vehTurnLeft;
p->vehTurnLeft = temp;
p->MotoSpeed = -(!p->NotOnWater ? 25 : 20) * p->vehReverseScale;
p->vehReverseScale = 0;
}
}
if (p->MotoSpeed != 0 && p->on_ground == 1)
{
if (!p->VBumpNow && (krand() & 15) == 14)
p->VBumpTarget = p->MotoSpeed * (1. / 16.) * ((krand() & 3) - 2);
if (p->vehTurnLeft && p->moto_drink < 0)
{
p->moto_drink++;
}
else if (p->vehTurnRight && p->moto_drink > 0)
{
p->moto_drink--;
}
}
double horiz = FRACUNIT;
if (p->TurbCount)
{
if (p->TurbCount <= 1)
{
horiz = 0;
p->TurbCount = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
}
else
{
horiz = ((krand() & 15) - 7);
p->TurbCount--;
p->moto_drink = (krand() & 3) - 2;
}
}
else if (p->VBumpTarget > p->VBumpNow)
{
p->VBumpNow += p->moto_bump_fast ? 6 : 1;
if (p->VBumpTarget < p->VBumpNow)
p->VBumpNow = p->VBumpTarget;
horiz = p->VBumpNow * (1. / 3.);
}
else if (p->VBumpTarget < p->VBumpNow)
{
p->VBumpNow -= p->moto_bump_fast ? 6 : 1;
if (p->VBumpTarget > p->VBumpNow)
p->VBumpNow = p->VBumpTarget;
horiz = p->VBumpNow * (1. / 3.);
}
else
{
p->VBumpTarget = 0;
p->moto_bump_fast = 0;
}
if (horiz != FRACUNIT)
{
p->horizon.addadjustment(buildfhoriz(horiz) - p->horizon.horiz);
}
if (p->MotoSpeed > 0 && p->on_ground == 1 && (p->vehTurnLeft || p->vehTurnRight))
{
int currSpeed = int(p->MotoSpeed * 4.);
int velAdjustment = p->vehTurnLeft ? -10 : 10;
auto angAdjustment = (velAdjustment < 0 ? 350 : -350) << BAMBITS;
if (p->moto_do_bump)
{
currSpeed >>= 6;
angAdjustment >>= 5;
}
else
{
currSpeed >>= 7;
angAdjustment >>= 6;
}
p->vel.X += currSpeed * bcos(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
p->vel.Y += currSpeed * bsin(velAdjustment * -51 + p->angle.ang.Buildang(), 4);
p->angle.addadjustment(deltaangle(p->angle.ang, p->angle.ang - DAngle::fromBam(angAdjustment)));
}
if (p->NotOnWater && p->MotoSpeed > 50)
p->MotoSpeed -= (p->MotoSpeed / 2.);
p->vehTurnLeft = p->vehTurnRight = p->vehBraking = false;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void movement(int snum, ESyncBits actions, sectortype* psect, int fz_, int cz_, int shrunk, int truefdist, int psectlotag)
{
auto p = &ps[snum];
auto pact = p->GetActor();
double floorz = fz_ * zinttoworld;
double ceilingz = cz_ * zinttoworld;
if (p->airleft != 15 * 26)
p->airleft = 15 * 26; //Aprox twenty seconds.
if (p->scuba_on == 1)
p->scuba_on = 0;
double i = 40;
if (psectlotag == ST_1_ABOVE_WATER && p->spritebridge == 0)
{
if (shrunk == 0)
{
i = 34;
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = DAngle::fromBuild(p->pycount).Sin() * 2;
}
else i = 12;
if (shrunk == 0 && truefdist <= gs.int_playerheight)
{
if (p->on_ground == 1)
{
if (p->dummyplayersprite == nullptr)
p->dummyplayersprite = spawn(pact, PLAYERONWATER);
p->footprintcount = 6;
if (p->cursector->floorpicnum == FLOORSLIME)
{
p->footprintpal = 8;
p->footprintshade = 0;
}
else if (isRRRA() && (p->cursector->floorpicnum == RRTILE7756 || p->cursector->floorpicnum == RRTILE7888))
{
p->footprintpal = 0;
p->footprintshade = 40;
}
else
{
p->footprintpal = 0;
p->footprintshade = 0;
}
}
}
}
else if (!p->OnMotorcycle)
{
footprints(snum);
}
if (p->pos.Z < floorz - i) //falling
{
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;
else
{
p->on_ground = 0;
if ((p->OnMotorcycle || p->OnBoat) && floorz - i * 2 > p->pos.Z)
{
if (p->MotoOnGround)
{
p->VBumpTarget = 80;
p->moto_bump_fast = 1;
p->vel.Z -= int(gs.gravity * p->MotoSpeed * (1. / 16.));
p->MotoOnGround = 0;
if (S_CheckActorSoundPlaying(pact, 188))
S_StopSound(188, pact);
S_PlayActorSound(189, pact);
}
else
{
p->vel.Z += gs.gravity - 80 + int(120 - p->MotoSpeed);
if (!S_CheckActorSoundPlaying(pact, 189) && !S_CheckActorSoundPlaying(pact, 190))
S_PlayActorSound(190, pact);
}
}
else
p->vel.Z += (gs.gravity + 80); // (TICSPERFRAME<<6);
if (p->vel.Z >= (4096 + 2048)) p->vel.Z = (4096 + 2048);
if (p->vel.Z > 2400 && p->falling_counter < 255)
{
p->falling_counter++;
if (p->falling_counter == 38 && !S_CheckActorSoundPlaying(pact, DUKE_SCREAM))
S_PlayActorSound(DUKE_SCREAM, pact);
}
if (p->pos.Z + p->vel.Z * zinttoworld >= floorz - i) // hit the ground
{
S_StopSound(DUKE_SCREAM, pact);
if (!p->insector() || p->cursector->lotag != 1)
{
if (isRRRA()) p->MotoOnGround = 1;
if (p->falling_counter > 62 || (isRRRA() && p->falling_counter > 2 && p->insector() && p->cursector->lotag == 802))
quickkill(p);
else if (p->falling_counter > 9)
{
int j = p->falling_counter;
pact->spr.extra -= j - (krand() & 3);
if (pact->spr.extra <= 0)
{
S_PlayActorSound(SQUISHED, pact);
}
else
{
S_PlayActorSound(DUKE_LAND, pact);
S_PlayActorSound(DUKE_LAND_HURT, pact);
}
SetPlayerPal(p, PalEntry(32, 16, 0, 0));
}
else if (p->vel.Z > 2048)
{
if (p->OnMotorcycle)
{
if (S_CheckActorSoundPlaying(pact, 190))
S_StopSound(190, pact);
S_PlayActorSound(191, pact);
p->TurbCount = 12;
}
else S_PlayActorSound(DUKE_LAND, pact);
}
else if (p->vel.Z > 1024 && p->OnMotorcycle)
{
S_PlayActorSound(DUKE_LAND, pact);
p->TurbCount = 12;
}
}
}
}
}
else
{
p->falling_counter = 0;
S_StopSound(-1, pact, CHAN_VOICE);
if (psectlotag != ST_1_ABOVE_WATER && psectlotag != ST_2_UNDERWATER && p->on_ground == 0 && p->vel.Z > (6144 >> 1))
p->hard_landing = p->vel.Z >> 10;
p->on_ground = 1;
if (i == 40)
{
//Smooth on the ground
double k = (floorz - i - p->pos.Z) * 0.5;
if (abs(k) < 1) k = 0;
p->pos.Z += k;
p->vel.Z -= 768;
if (p->vel.Z < 0) p->vel.Z = 0;
}
else if (p->jumping_counter == 0)
{
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)
{
p->pos.Z = floorz - 16;
p->vel.Z >>= 1;
}
}
p->on_warping_sector = 0;
if (((actions & SB_CROUCH) || crouch_toggle) && !p->OnMotorcycle) // FIXME: The crouch_toggle check here is not network safe and needs revision when multiplayer is going.
{
playerCrouch(snum);
}
if ((actions & SB_JUMP) == 0 && !p->OnMotorcycle && p->jumping_toggle == 1)
p->jumping_toggle = 0;
else if ((actions & SB_JUMP) && !p->OnMotorcycle && p->jumping_toggle == 0)
{
playerJump(snum, fz_, cz_);
}
}
if (p->jumping_counter)
{
if ((actions & SB_JUMP) == 0 && !p->OnMotorcycle && p->jumping_toggle == 1)
p->jumping_toggle = 0;
if (p->jumping_counter < 768)
{
if (psectlotag == ST_1_ABOVE_WATER && p->jumping_counter > 768)
{
p->jumping_counter = 0;
p->vel.Z = -512;
}
else
{
p->vel.Z -= bsin(2048 - 128 + p->jumping_counter) / 12;
p->jumping_counter += 180;
p->on_ground = 0;
}
}
else
{
p->jumping_counter = 0;
p->vel.Z = 0;
}
}
p->pos.Z += p->vel.Z * zinttoworld;
if (p->pos.Z < ceilingz + 4)
{
p->jumping_counter = 0;
if (p->vel.Z < 0)
p->vel.X = p->vel.Y = 0;
p->vel.Z = 128;
p->pos.Z = ceilingz + 4;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void underwater(int snum, ESyncBits actions, int fz_, int cz_)
{
auto p = &ps[snum];
auto pact = p->GetActor();
double floorz = fz_ * zinttoworld;
double ceilingz = cz_ * zinttoworld;
p->jumping_counter = 0;
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = DAngle::fromBuild(p->pycount).Sin();
if (!S_CheckActorSoundPlaying(pact, DUKE_UNDERWATER))
S_PlayActorSound(DUKE_UNDERWATER, pact);
if ((actions & SB_JUMP) && !p->OnMotorcycle)
{
if (p->vel.Z > 0) p->vel.Z = 0;
p->vel.Z -= 348;
if (p->vel.Z < -(256 * 6)) p->vel.Z = -(256 * 6);
}
else if ((actions & SB_CROUCH) || p->OnMotorcycle)
{
if (p->vel.Z < 0) p->vel.Z = 0;
p->vel.Z += 348;
if (p->vel.Z > (256 * 6)) p->vel.Z = (256 * 6);
}
else
{
if (p->vel.Z < 0)
{
p->vel.Z += 256;
if (p->vel.Z > 0)
p->vel.Z = 0;
}
if (p->vel.Z > 0)
{
p->vel.Z -= 256;
if (p->vel.Z < 0)
p->vel.Z = 0;
}
}
if (p->vel.Z > 2048)
p->vel.Z >>= 1;
p->pos.Z += p->vel.Z * zinttoworld;
if (p->pos.Z > floorz - 15)
p->pos.Z += (((floorz - 15) - p->pos.Z) * 0.5);
if (p->pos.Z < ceilingz + 4)
{
p->pos.Z = ceilingz + 4;
p->vel.Z = 0;
}
if (p->scuba_on && (krand() & 255) < 8)
{
auto j = spawn(pact, WATERBUBBLE);
if (j)
{
j->add_int_pos({ bcos(p->angle.ang.Buildang() + 64 - (global_random & 128) + 128, -6), bsin(p->angle.ang.Buildang() + 64 - (global_random & 128) + 128, -6), 0 });
j->spr.xrepeat = 3;
j->spr.yrepeat = 2;
j->spr.pos.Z = p->pos.Z + 8;
j->spr.cstat = CSTAT_SPRITE_TRANS_FLIP | CSTAT_SPRITE_TRANSLUCENT;
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void onMotorcycleMove(int snum, walltype* wal)
{
auto p = &ps[snum];
auto pact = p->GetActor();
int angleDelta = abs(p->angle.ang.Buildang() - getangle(wal->delta()));
double damageAmount = p->MotoSpeed * p->MotoSpeed;
p->angle.addadjustment(DAngle::fromBuildf(p->MotoSpeed / (krand() & 1 ? -2 : 2)));
if (angleDelta >= 441 && angleDelta <= 581)
{
damageAmount *= (1. / 256.);
p->MotoSpeed = 0;
if (S_CheckActorSoundPlaying(pact, 238) == 0)
S_PlayActorSound(238, pact);
}
else if (angleDelta >= 311 && angleDelta <= 711)
{
damageAmount *= (1. / 2048.);
p->MotoSpeed -= (p->MotoSpeed / 2.) + (p->MotoSpeed / 4.);
if (S_CheckActorSoundPlaying(pact, 238) == 0)
S_PlayActorSound(238, pact);
}
else if (angleDelta >= 111 && angleDelta <= 911)
{
damageAmount *= (1. / 16384.);
p->MotoSpeed -= p->MotoSpeed / 2.;
if (S_CheckActorSoundPlaying(pact, 239) == 0)
S_PlayActorSound(239, pact);
}
else
{
damageAmount *= (1. / 32768.);
p->MotoSpeed -= p->MotoSpeed / 8.;
if (S_CheckActorSoundPlaying(pact, 240) == 0)
S_PlayActorSound(240, pact);
}
pact->spr.extra -= int(damageAmount);
if (pact->spr.extra <= 0)
{
S_PlayActorSound(SQUISHED, pact);
SetPlayerPal(p, PalEntry(63, 63, 0, 0));
}
else if (damageAmount)
S_PlayActorSound(DUKE_LAND_HURT, pact);
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void onBoatMove(int snum, int psectlotag, walltype* wal)
{
auto p = &ps[snum];
auto pact = p->GetActor();
int angleDelta = abs(p->angle.ang.Buildang() - getangle(wal->delta()));
p->angle.addadjustment(DAngle::fromBuildf(p->MotoSpeed / (krand() & 1 ? -4 : 4)));
if (angleDelta >= 441 && angleDelta <= 581)
{
p->MotoSpeed = ((p->MotoSpeed / 2.) + (p->MotoSpeed / 4.)) / 4.;
if (psectlotag == 1 && S_CheckActorSoundPlaying(pact, 178) == 0)
S_PlayActorSound(178, pact);
}
else if (angleDelta >= 311 && angleDelta <= 711)
{
p->MotoSpeed -= ((p->MotoSpeed / 2.) + (p->MotoSpeed / 4.)) / 8.;
if (psectlotag == 1 && S_CheckActorSoundPlaying(pact, 179) == 0)
S_PlayActorSound(179, pact);
}
else if (angleDelta >= 111 && angleDelta <= 911)
{
p->MotoSpeed -= p->MotoSpeed / 16.;
if (psectlotag == 1 && S_CheckActorSoundPlaying(pact, 180) == 0)
S_PlayActorSound(180, pact);
}
else
{
p->MotoSpeed -= p->MotoSpeed / 64.;
if (psectlotag == 1 && S_CheckActorSoundPlaying(pact, 181) == 0)
S_PlayActorSound(181, pact);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void onMotorcycleHit(int snum, DDukeActor* victim)
{
auto p = &ps[snum];
if (badguy(victim) || victim->spr.picnum == APLAYER)
{
if (victim->spr.picnum != APLAYER)
{
if (numplayers == 1)
{
Collision coll;
int ang = int(p->TiltStatus * 20 + p->angle.ang.Buildang());
movesprite_ex(victim, bcos(ang, -8), bsin(ang, -8), victim->int_zvel(), CLIPMASK0, coll);
}
}
else
victim->SetHitOwner(p->GetActor());
victim->attackertype = MOTOHIT;
victim->hitextra = int(p->MotoSpeed * 0.5);
p->MotoSpeed -= p->MotoSpeed / 4.;
p->TurbCount = 6;
}
else if ((victim->spr.picnum == RRTILE2431 || victim->spr.picnum == RRTILE2443 || victim->spr.picnum == RRTILE2451 || victim->spr.picnum == RRTILE2455)
&& victim->spr.picnum != ACTIVATORLOCKED && p->MotoSpeed > 45)
{
S_PlayActorSound(SQUISHED, victim);
if (victim->spr.picnum == RRTILE2431 || victim->spr.picnum == RRTILE2451)
{
if (victim->spr.lotag != 0)
{
DukeSpriteIterator it;
while (auto act2 = it.Next())
{
if ((act2->spr.picnum == RRTILE2431 || act2->spr.picnum == RRTILE2451) && act2->spr.pal == 4)
{
if (victim->spr.lotag == act2->spr.lotag)
{
act2->spr.xrepeat = 0;
act2->spr.yrepeat = 0;
}
}
}
}
fi.guts(victim, RRTILE2460, 12, myconnectindex);
fi.guts(victim, RRTILE2465, 3, myconnectindex);
}
else
fi.guts(victim, RRTILE2465, 3, myconnectindex);
fi.guts(victim, RRTILE2465, 3, myconnectindex);
victim->spr.xrepeat = 0;
victim->spr.yrepeat = 0;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void onBoatHit(int snum, DDukeActor* victim)
{
auto p = &ps[snum];
if (badguy(victim) || victim->spr.picnum == APLAYER)
{
if (victim->spr.picnum != APLAYER)
{
if (numplayers == 1)
{
Collision coll;
int ang = int(p->TiltStatus * 20 + p->angle.ang.Buildang());
movesprite_ex(victim, bcos(ang, -9), bsin(ang, -9), victim->int_zvel(), CLIPMASK0, coll);
}
}
else
victim->SetHitOwner(p->GetActor());
victim->attackertype = MOTOHIT;
victim->hitextra = int(p->MotoSpeed * 0.25);
p->MotoSpeed -= p->MotoSpeed / 4.;
p->TurbCount = 6;
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void fireweapon(int snum)
{
auto p = &ps[snum];
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
{
if (!isRRRA() && p->curr_weapon >= MOTORCYCLE_WEAPON) return;
switch (p->curr_weapon)
{
case DYNAMITE_WEAPON:
p->hbomb_hold_delay = 0;
if (p->ammo_amount[DYNAMITE_WEAPON] > 0)
p->kickback_pic = 1;
break;
case THROWINGDYNAMITE_WEAPON:
p->hbomb_hold_delay = 0;
p->kickback_pic = 1;
break;
case PISTOL_WEAPON:
if (p->ammo_amount[PISTOL_WEAPON] > 0)
{
p->ammo_amount[PISTOL_WEAPON]--;
p->kickback_pic = 1;
}
break;
case RIFLEGUN_WEAPON:
if (p->ammo_amount[RIFLEGUN_WEAPON] > 0) // && p->random_club_frame == 0)
p->kickback_pic = 1;
break;
case SHOTGUN_WEAPON:
if (p->ammo_amount[SHOTGUN_WEAPON] > 0 && p->random_club_frame == 0)
p->kickback_pic = 1;
break;
case BOWLING_WEAPON:
if (p->ammo_amount[BOWLING_WEAPON] > 0)
p->kickback_pic = 1;
break;
case POWDERKEG_WEAPON:
if (p->ammo_amount[POWDERKEG_WEAPON] > 0)
p->kickback_pic = 1;
break;
case BUZZSAW_WEAPON:
case THROWSAW_WEAPON:
if (p->curr_weapon == BUZZSAW_WEAPON)
{
if (p->ammo_amount[BUZZSAW_WEAPON] > 0)
{
p->kickback_pic = 1;
S_PlayActorSound(431, p->GetActor());
}
}
else if (p->ammo_amount[THROWSAW_WEAPON] > 0)
{
p->kickback_pic = 1;
S_PlayActorSound(SHRINKER_FIRE, p->GetActor());
}
break;
case ALIENBLASTER_WEAPON:
if (p->ammo_amount[ALIENBLASTER_WEAPON] > 0)
p->kickback_pic = 1;
break;
case TIT_WEAPON:
if (p->ammo_amount[TIT_WEAPON] > 0)
{
p->kickback_pic = 1;
p->hbomb_hold_delay = !p->hbomb_hold_delay;
}
break;
case MOTORCYCLE_WEAPON:
if (p->ammo_amount[MOTORCYCLE_WEAPON] > 0)
{
p->kickback_pic = 1;
p->hbomb_hold_delay = !p->hbomb_hold_delay;
}
break;
case BOAT_WEAPON:
if (p->ammo_amount[BOAT_WEAPON] > 0)
p->kickback_pic = 1;
break;
case CROSSBOW_WEAPON:
if (p->ammo_amount[CROSSBOW_WEAPON] > 0)
p->kickback_pic = 1;
break;
case CHICKEN_WEAPON:
if (p->ammo_amount[CHICKEN_WEAPON] > 0)
p->kickback_pic = 1;
break;
case KNEE_WEAPON:
case SLINGBLADE_WEAPON:
if (p->curr_weapon == SLINGBLADE_WEAPON)
{
if (p->ammo_amount[SLINGBLADE_WEAPON] > 0)
if (p->quick_kick == 0)
p->kickback_pic = 1;
}
else if (!isRRRA() || p->ammo_amount[KNEE_WEAPON] > 0)
if (p->quick_kick == 0)
p->kickback_pic = 1;
break;
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
static void operateweapon(int snum, ESyncBits actions, sectortype* psectp)
{
auto p = &ps[snum];
auto pact = p->GetActor();
int i, k;
int psectlotag = psectp ? psectp->lotag : 857;
if (!isRRRA() && p->curr_weapon >= MOTORCYCLE_WEAPON) return;
switch (p->curr_weapon)
{
case DYNAMITE_WEAPON:
if (p->kickback_pic == 1)
S_PlaySound(401);
if (p->kickback_pic == 6 && (actions & SB_FIRE))
p->rapid_fire_hold = 1;
p->kickback_pic++;
if (p->kickback_pic > 19)
{
p->okickback_pic = p->kickback_pic = 0;
p->curr_weapon = THROWINGDYNAMITE_WEAPON;
p->last_weapon = -1;
p->oweapon_pos = p->weapon_pos = 10;
p->detonate_time = 45;
p->detonate_count = 1;
S_PlaySound(402);
}
break;
case THROWINGDYNAMITE_WEAPON:
p->kickback_pic++;
if (p->detonate_time < 0)
{
p->hbomb_on = 0;
}
if (p->kickback_pic == 39)
{
p->hbomb_on = 0;
p->noise_radius = 8192;
madenoise(snum);
}
if (p->kickback_pic == 12)
{
p->ammo_amount[DYNAMITE_WEAPON]--;
if (p->ammo_amount[CROSSBOW_WEAPON])
p->ammo_amount[CROSSBOW_WEAPON]--;
if (p->on_ground && (actions & SB_CROUCH) && !p->OnMotorcycle)
{
k = 15;
i = -MulScale(p->horizon.sum().asq16(), 20, 16);
}
else
{
k = 140;
i = -512 - MulScale(p->horizon.sum().asq16(), 20, 16);
}
auto spawned = EGS(p->cursector,
p->player_int_pos().X + p->angle.ang.Cos() * (1 << 8),
p->player_int_pos().Y + p->angle.ang.Sin() * (1 << 8),
p->player_int_pos().Z, HEAVYHBOMB, -16, 9, 9,
p->angle.ang.Buildang(), (k + (p->hbomb_hold_delay << 5)) * 2, i, pact, 1);
if (spawned)
{
if (k == 15)
{
spawned->spr.yvel = 3;
spawned->spr.pos.Z += 8;
}
k = hits(p->GetActor());
if (k < 512)
{
spawned->add_int_ang(1024);
spawned->mul_int_zvel(1./3.);
spawned->spr.xvel /= 3;
}
p->hbomb_on = 1;
}
}
else if (p->kickback_pic < 12 && (actions & SB_FIRE))
p->hbomb_hold_delay++;
if (p->kickback_pic == 40)
{
p->okickback_pic = p->kickback_pic = 0;
p->curr_weapon = DYNAMITE_WEAPON;
p->last_weapon = -1;
p->detonate_count = 0;
p->detonate_time = 45;
if (p->ammo_amount[DYNAMITE_WEAPON] > 0)
{
fi.addweapon(p, DYNAMITE_WEAPON);
p->oweapon_pos = p->weapon_pos = -9;
}
else checkavailweapon(p);
}
break;
case PISTOL_WEAPON:
if (p->kickback_pic == 1)
{
fi.shoot(pact, SHOTSPARK1);
S_PlayActorSound(PISTOL_FIRE, pact);
p->noise_radius = 8192;
madenoise(snum);
lastvisinc = PlayClock + 32;
p->visibility = 0;
if (psectlotag != 857)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 18);
p->vel.Y -= p->angle.ang.Sin() * (1 << 18);
}
}
else if (p->kickback_pic == 2)
if (p->ammo_amount[PISTOL_WEAPON] <= 0)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
}
p->kickback_pic++;
if (p->kickback_pic >= 22)
{
if (p->ammo_amount[PISTOL_WEAPON] <= 0)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
break;
}
else if ((p->ammo_amount[PISTOL_WEAPON] % 6) == 0)
{
switch (p->kickback_pic)
{
case 24:
S_PlayActorSound(EJECT_CLIP, pact);
break;
case 30:
S_PlayActorSound(INSERT_CLIP, pact);
break;
}
}
else
p->kickback_pic = 38;
}
if (p->kickback_pic == 38)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
}
break;
case SHOTGUN_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 6)
if (p->shotgun_state[0] == 0)
if (p->ammo_amount[SHOTGUN_WEAPON] > 1)
if (actions & SB_FIRE)
p->shotgun_state[1] = 1;
if (p->kickback_pic == 4)
{
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
p->ammo_amount[SHOTGUN_WEAPON]--;
S_PlayActorSound(SHOTGUN_FIRE, pact);
p->noise_radius = 8192;
madenoise(snum);
lastvisinc = PlayClock + 32;
p->visibility = 0;
}
if (p->kickback_pic == 7)
{
if (p->shotgun_state[1])
{
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
fi.shoot(pact, SHOTGUN);
p->ammo_amount[SHOTGUN_WEAPON]--;
S_PlayActorSound(SHOTGUN_FIRE, pact);
if (psectlotag != 857)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 19);
p->vel.Y -= p->angle.ang.Sin() * (1 << 19);
}
}
else if (psectlotag != 857)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 18);
p->vel.Y -= p->angle.ang.Sin() * (1 << 18);
}
}
if (p->shotgun_state[0])
{
switch (p->kickback_pic)
{
case 16:
checkavailweapon(p);
break;
case 17:
S_PlayActorSound(SHOTGUN_COCK, pact);
break;
case 28:
p->okickback_pic = p->kickback_pic = 0;
p->shotgun_state[0] = 0;
p->shotgun_state[1] = 0;
return;
}
}
else if (p->shotgun_state[1])
{
switch (p->kickback_pic)
{
case 26:
checkavailweapon(p);
break;
case 27:
S_PlayActorSound(SHOTGUN_COCK, pact);
break;
case 38:
p->okickback_pic = p->kickback_pic = 0;
p->shotgun_state[0] = 0;
p->shotgun_state[1] = 0;
return;
}
}
else
{
switch (p->kickback_pic)
{
case 16:
checkavailweapon(p);
p->okickback_pic = p->kickback_pic = 0;
p->shotgun_state[0] = 1;
p->shotgun_state[1] = 0;
return;
}
}
break;
case RIFLEGUN_WEAPON:
p->kickback_pic++;
p->horizon.addadjustment(buildhoriz(1));
p->recoil++;
if (p->kickback_pic <= 12)
{
if ((p->kickback_pic % 3) == 0)
{
p->ammo_amount[RIFLEGUN_WEAPON]--;
if ((p->kickback_pic % 3) == 0)
{
auto j = spawn(pact, SHELL);
if (j)
{
j->set_int_ang((j->int_ang() + 1024) & 2047);
j->spr.xvel += 32;
j->spr.pos.Z += 3;
ssp(j, CLIPMASK0);
}
}
S_PlayActorSound(CHAINGUN_FIRE, pact);
fi.shoot(pact, CHAINGUN);
p->noise_radius = 8192;
madenoise(snum);
lastvisinc = PlayClock + 32;
p->visibility = 0;
if (psectlotag != 857)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 18);
p->vel.Y -= p->angle.ang.Sin() * (1 << 18);
}
checkavailweapon(p);
if ((actions & SB_FIRE) == 0)
{
p->okickback_pic = p->kickback_pic = 0;
break;
}
}
}
else if (p->kickback_pic > 10)
{
if (actions & SB_FIRE) p->kickback_pic = 1;
else p->okickback_pic = p->kickback_pic = 0;
}
break;
case BUZZSAW_WEAPON:
if (p->kickback_pic > 3)
{
p->okickback_pic = p->kickback_pic = 0;
fi.shoot(pact, GROWSPARK);
p->noise_radius = 1024;
madenoise(snum);
checkavailweapon(p);
}
else p->kickback_pic++;
break;
case THROWSAW_WEAPON:
if (p->kickback_pic == 1)
{
p->ammo_amount[THROWSAW_WEAPON]--;
fi.shoot(pact, SHRINKSPARK);
checkavailweapon(p);
}
p->kickback_pic++;
if (p->kickback_pic > 20)
p->okickback_pic = p->kickback_pic = 0;
break;
case TIT_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 2 || p->kickback_pic == 4)
{
p->visibility = 0;
lastvisinc = PlayClock + 32;
S_PlayActorSound(CHAINGUN_FIRE, pact);
fi.shoot(pact, SHOTSPARK1);
p->noise_radius = 16384;
madenoise(snum);
p->ammo_amount[TIT_WEAPON]--;
checkavailweapon(p);
}
if (p->kickback_pic == 2)
{
p->angle.addadjustment(DAngle::fromBuild(16));
}
else if (p->kickback_pic == 4)
{
p->angle.addadjustment(DAngle::fromBuild(-16));
}
if (p->kickback_pic > 4)
p->kickback_pic = 1;
if (!(actions & SB_FIRE))
p->okickback_pic = p->kickback_pic = 0;
break;
case MOTORCYCLE_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 2 || p->kickback_pic == 4)
{
p->visibility = 0;
lastvisinc = PlayClock + 32;
S_PlayActorSound(CHAINGUN_FIRE, pact);
fi.shoot(pact, CHAINGUN);
p->noise_radius = 16384;
madenoise(snum);
p->ammo_amount[MOTORCYCLE_WEAPON]--;
if (p->ammo_amount[MOTORCYCLE_WEAPON] <= 0)
p->okickback_pic = p->kickback_pic = 0;
else
checkavailweapon(p);
}
if (p->kickback_pic == 2)
{
p->angle.addadjustment(DAngle::fromBuild(4));
}
else if (p->kickback_pic == 4)
{
p->angle.addadjustment(DAngle::fromBuild(-4));
}
if (p->kickback_pic > 4)
p->kickback_pic = 1;
if (!(actions & SB_FIRE))
p->okickback_pic = p->kickback_pic = 0;
break;
case BOAT_WEAPON:
if (p->kickback_pic == 3)
{
p->MotoSpeed -= 20;
p->ammo_amount[BOAT_WEAPON]--;
fi.shoot(pact, RRTILE1790);
}
p->kickback_pic++;
if (p->kickback_pic > 20)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
}
if (p->ammo_amount[BOAT_WEAPON] <= 0)
p->okickback_pic = p->kickback_pic = 0;
else
checkavailweapon(p);
break;
case ALIENBLASTER_WEAPON:
p->kickback_pic++;
if (p->kickback_pic >= 7 && p->kickback_pic <= 11)
fi.shoot(pact, FIRELASER);
if (p->kickback_pic == 5)
{
S_PlayActorSound(CAT_FIRE, pact);
p->noise_radius = 2048;
madenoise(snum);
}
else if (p->kickback_pic == 9)
{
p->ammo_amount[ALIENBLASTER_WEAPON]--;
p->visibility = 0;
lastvisinc = PlayClock + 32;
checkavailweapon(p);
}
else if (p->kickback_pic == 12)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 18);
p->vel.Y -= p->angle.ang.Sin() * (1 << 18);
p->horizon.addadjustment(buildhoriz(20));
p->recoil += 20;
}
if (p->kickback_pic > 20)
p->okickback_pic = p->kickback_pic = 0;
break;
case POWDERKEG_WEAPON:
if (p->kickback_pic == 3)
{
p->ammo_amount[POWDERKEG_WEAPON]--;
p->gotweapon[POWDERKEG_WEAPON] = false;
if (p->on_ground && (actions & SB_CROUCH) && !p->OnMotorcycle)
{
k = 15;
i = MulScale(p->horizon.sum().asq16(), 20, 16);
}
else
{
k = 32;
i = -512 - MulScale(p->horizon.sum().asq16(), 20, 16);
}
EGS(p->cursector,
p->player_int_pos().X + p->angle.ang.Cos() * (1 << 8),
p->player_int_pos().Y + p->angle.ang.Sin() * (1 << 8),
p->player_int_pos().Z, POWDERKEG, -16, 9, 9,
p->angle.ang.Buildang(), k * 2, i, pact, 1);
}
p->kickback_pic++;
if (p->kickback_pic > 20)
{
p->okickback_pic = p->kickback_pic = 0;
checkavailweapon(p);
}
break;
case BOWLING_WEAPON:
if (p->kickback_pic == 30)
{
p->ammo_amount[BOWLING_WEAPON]--;
S_PlayActorSound(354, pact);
fi.shoot(pact, BOWLINGBALL);
p->noise_radius = 1024;
madenoise(snum);
}
if (p->kickback_pic < 30)
{
p->vel.X -= p->angle.ang.Cos() * (1 << 18);
p->vel.Y -= p->angle.ang.Sin() * (1 << 18);
}
p->kickback_pic++;
if (p->kickback_pic > 40)
{
p->okickback_pic = p->kickback_pic = 0;
p->gotweapon[BOWLING_WEAPON] = false;
checkavailweapon(p);
}
break;
case KNEE_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 3)
S_PlayActorSound(426, pact);
if (p->kickback_pic == 12)
{
fi.shoot(pact, KNEE);
p->noise_radius = 1024;
madenoise(snum);
}
else if (p->kickback_pic == 16)
p->okickback_pic = p->kickback_pic = 0;
if (p->wantweaponfire >= 0)
checkavailweapon(p);
break;
case SLINGBLADE_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 3)
S_PlayActorSound(252, pact);
if (p->kickback_pic == 8)
{
fi.shoot(pact, SLINGBLADE);
p->noise_radius = 1024;
madenoise(snum);
}
else if (p->kickback_pic == 16)
p->okickback_pic = p->kickback_pic = 0;
if (p->wantweaponfire >= 0)
checkavailweapon(p);
break;
case CROSSBOW_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 4)
{
p->ammo_amount[CROSSBOW_WEAPON]--;
if (p->ammo_amount[DYNAMITE_WEAPON])
p->ammo_amount[DYNAMITE_WEAPON]--;
lastvisinc = PlayClock + 32;
p->visibility = 0;
fi.shoot(pact, RPG);
p->noise_radius = 32768;
madenoise(snum);
checkavailweapon(p);
}
else if (p->kickback_pic == 16)
S_PlayActorSound(450, pact);
else if (p->kickback_pic == 34)
p->okickback_pic = p->kickback_pic = 0;
break;
case CHICKEN_WEAPON:
p->kickback_pic++;
if (p->kickback_pic == 4)
{
p->ammo_amount[CHICKEN_WEAPON]--;
lastvisinc = PlayClock + 32;
p->visibility = 0;
fi.shoot(pact, RPG2);
p->noise_radius = 32768;
madenoise(snum);
checkavailweapon(p);
}
else if (p->kickback_pic == 16)
S_PlayActorSound(450, pact);
else if (p->kickback_pic == 34)
p->okickback_pic = p->kickback_pic = 0;
break;
}
}
//---------------------------------------------------------------------------
//
// this function exists because gotos suck. :P
//
//---------------------------------------------------------------------------
static void processweapon(int snum, ESyncBits actions, sectortype* psectp)
{
auto p = &ps[snum];
auto pact = p->GetActor();
int shrunk = (pact->spr.yrepeat < 8);
if (p->detonate_count > 0)
{
if (ud.god)
{
p->detonate_time = 45;
p->detonate_count = 0;
}
else if (p->detonate_time <= 0 && p->kickback_pic < 5)
{
S_PlaySound(14);
quickkill(p);
}
}
if (isRRRA() && (p->curr_weapon == KNEE_WEAPON || p->curr_weapon == SLINGBLADE_WEAPON))
p->random_club_frame += 64;
if (p->curr_weapon == THROWSAW_WEAPON || p->curr_weapon == BUZZSAW_WEAPON)
p->random_club_frame += 64; // Glowing
if (p->curr_weapon == TRIPBOMB_WEAPON || p->curr_weapon == BOWLING_WEAPON)
p->random_club_frame += 64;
if (p->rapid_fire_hold == 1)
{
if (actions & SB_FIRE) return;
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 &&
p->last_weapon == -1 && (p->weapon_pos == 0 || p->holster_weapon == 1))
{
fireweapon(snum);
}
else if (p->kickback_pic)
{
operateweapon(snum, actions, psectp);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void processinput_r(int snum)
{
int i, k, doubvel, fz, cz, truefdist;
Collision chz, clz;
bool shrunk;
int psectlotag;
auto p = &ps[snum];
auto pact = p->GetActor();
p->horizon.resetadjustment();
p->angle.resetadjustment();
ESyncBits& actions = p->sync.actions;
auto sb_fvel = PlayerInputForwardVel(snum);
auto sb_svel = PlayerInputSideVel(snum);
auto sb_avel = PlayerInputAngVel(snum);
auto psectp = p->cursector;
if (p->OnMotorcycle && pact->spr.extra > 0)
{
onMotorcycle(snum, actions);
}
else if (p->OnBoat && pact->spr.extra > 0)
{
onBoat(snum, actions);
}
if (psectp == nullptr)
{
if (pact->spr.extra > 0 && ud.clipping == 0)
{
quickkill(p);
S_PlayActorSound(SQUISHED, pact);
}
psectp = &sector[0];
}
psectlotag = psectp->lotag;
if (psectlotag == 867)
{
DukeSectIterator it(psectp);
while (auto act2 = it.Next())
{
if (act2->spr.picnum == RRTILE380)
if (act2->int_pos().Z - (8 << 8) < p->player_int_pos().Z)
psectlotag = 2;
}
}
else if (psectlotag == 7777 && (currentLevel->gameflags & LEVEL_RR_HULKSPAWN))
lastlevel = 1;
if (psectlotag == 848 && psectp->floorpicnum == WATERTILE2)
psectlotag = 1;
if (psectlotag == 857)
pact->spr.clipdist = 1;
else
pact->spr.clipdist = 64;
p->spritebridge = 0;
shrunk = (pact->spr.yrepeat < 8);
int tempfz;
if (pact->spr.clipdist == 64)
{
getzrange(p->player_int_pos(), psectp, &cz, chz, &fz, clz, 163L, CLIPMASK0);
tempfz = getflorzofslopeptr(psectp, p->player_int_pos().X, p->player_int_pos().Y);
}
else
{
getzrange(p->player_int_pos(), psectp, &cz, chz, &fz, clz, 4L, CLIPMASK0);
tempfz = getflorzofslopeptr(psectp, p->player_int_pos().X, p->player_int_pos().Y);
}
p->truefz = tempfz * zinttoworld;
p->truecz = getceilzofslopeptr(psectp, p->player_int_pos().X, p->player_int_pos().Y) * zinttoworld;
truefdist = abs(p->player_int_pos().Z - tempfz);
if (clz.type == kHitSector && psectlotag == 1 && truefdist > gs.int_playerheight + (16 << 8))
psectlotag = 0;
pact->floorz = fz * zinttoworld;
pact->ceilingz = cz * zinttoworld;
if (SyncInput())
{
p->horizon.backup();
doslopetilting(p);
}
if (chz.type == kHitSprite)
{
if (chz.actor()->spr.statnum == 1 && chz.actor()->spr.extra >= 0)
{
chz.setNone();
cz = p->truecz * zworldtoint;
}
else if (chz.actor()->spr.picnum == LADDER)
{
if (!p->stairs)
{
p->stairs = 10;
if ((actions & SB_JUMP) && !p->OnMotorcycle)
{
chz.setNone();
cz = p->truecz * zworldtoint;
}
}
else
p->stairs--;
}
}
if (clz.type == kHitSprite)
{
auto doVehicleHit = [&]()
{
if (badguy(clz.actor()))
{
clz.actor()->attackertype = MOTOHIT;
clz.actor()->hitextra = int(2 + (p->MotoSpeed * 0.5));
p->MotoSpeed -= p->MotoSpeed * (1. / 16.);
}
};
if ((clz.actor()->spr.cstat & (CSTAT_SPRITE_ALIGNMENT_FLOOR| CSTAT_SPRITE_BLOCK)) == (CSTAT_SPRITE_ALIGNMENT_FLOOR | CSTAT_SPRITE_BLOCK))
{
psectlotag = 0;
p->footprintcount = 0;
p->spritebridge = 1;
}
if (p->OnMotorcycle)
doVehicleHit();
if (p->OnBoat)
{
doVehicleHit();
}
else if (badguy(clz.actor()) && clz.actor()->spr.xrepeat > 24 && abs(pact->int_pos().Z - clz.actor()->int_pos().Z) < (84 << 8))
{
int j = getangle(clz.actor()->int_pos().X - p->player_int_pos().X, clz.actor()->int_pos().Y - p->player_int_pos().Y);
p->vel.X -= bcos(j, 4);
p->vel.Y -= bsin(j, 4);
}
if (clz.actor()->spr.picnum == LADDER)
{
if (!p->stairs)
{
p->stairs = 10;
if ((actions & SB_CROUCH) && !p->OnMotorcycle)
{
cz = clz.actor()->int_pos().Z;
chz.setNone();
fz = clz.actor()->int_pos().Z + (4 << 8);
}
}
else
p->stairs--;
}
else if (clz.actor()->spr.picnum == TOILET || clz.actor()->spr.picnum == RRTILE2121)
{
if ((actions & SB_CROUCH) && !p->OnMotorcycle)
//if (Sound[436].num == 0)
{
S_PlayActorSound(436, pact);
p->last_pissed_time = 4000;
p->eat = 0;
}
}
}
if (pact->spr.extra > 0) fi.incur_damage(p);
else
{
pact->spr.extra = 0;
p->shield_amount = 0;
}
p->last_extra = pact->spr.extra;
if (p->loogcnt > 0)
{
p->oloogcnt = p->loogcnt;
p->loogcnt--;
}
else
{
p->oloogcnt = p->loogcnt = 0;
}
if (p->fist_incs)
{
if (endoflevel(snum)) return;
}
if (p->timebeforeexit > 1 && p->last_extra > 0)
{
if (timedexit(snum))
return;
}
if (pact->spr.extra <= 0 && !ud.god)
{
playerisdead(snum, psectlotag, fz, cz);
return;
}
if (p->GetActor()->spr.xrepeat < 8 && p->jetpack_on == 0)
{
p->ofistsign = p->fistsign;
p->fistsign += p->GetActor()->spr.xvel;
}
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++;
if (p->newOwner != nullptr)
{
p->vel.X = p->vel.Y = pact->spr.xvel = 0;
fi.doincrements(p);
if (p->curr_weapon == THROWINGDYNAMITE_WEAPON) processweapon(snum, actions, psectp);
return;
}
doubvel = TICSPERFRAME;
checklook(snum, actions);
p->apply_seasick(1);
auto oldpos = p->opos;
if (p->on_crane != nullptr)
goto HORIZONLY;
p->playerweaponsway(pact->spr.xvel);
pact->spr.xvel = int(clamp((p->pos.XY() - p->bobpos).Length(), 0., 32.) * worldtoint);
if (p->on_ground) p->bobcounter += p->GetActor()->spr.xvel >> 1;
p->backuppos(ud.clipping == 0 && ((p->insector() && p->cursector->floorpicnum == MIRROR) || !p->insector()));
// Shrinking code
i = 40;
if (psectlotag == ST_17_PLATFORM_UP || (isRRRA() && psectlotag == ST_18_ELEVATOR_DOWN))
{
int tmp;
tmp = getanimationgoal(anim_floorz, p->cursector);
if (tmp >= 0)
{
if (!S_CheckActorSoundPlaying(pact, 432))
S_PlayActorSound(432, pact);
}
else
S_StopSound(432);
}
if (isRRRA() && p->sea_sick_stat)
{
p->pycount += 32;
p->pycount &= 2047;
p->pyoff = DAngle::fromBuild(p->pycount).Sin() * (p->SeaSick? 32 : 1);
}
if (psectlotag == ST_2_UNDERWATER)
{
underwater(snum, actions, fz, cz);
}
else
{
movement(snum, actions, psectp, fz, cz, shrunk, truefdist, psectlotag);
}
p->psectlotag = psectlotag;
//Do the quick lefts and rights
if (movementBlocked(p))
{
doubvel = 0;
p->vel.X = 0;
p->vel.Y = 0;
}
else if (SyncInput())
{
//p->ang += syncangvel * constant
//ENGINE calculates angvel for you
// may still be needed later for demo recording
sb_avel = p->adjustavel(sb_avel);
p->angle.applyinput(sb_avel, &actions);
}
if (p->spritebridge == 0 && pact->insector())
{
int j = pact->sector()->floorpicnum;
k = 0;
if (p->on_ground && truefdist <= gs.int_playerheight + (16 << 8))
{
int whichsound = (gs.tileinfo[j].flags & TFLAG_ELECTRIC) ? 0 : j == FLOORSLIME ? 1 : j == FLOORPLASMA ? 2 :
(isRRRA() && (j == RRTILE7768 || j == RRTILE7820) ? 3 : -1);
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)
{
p->crack_time = CRACK_TIME;
k = bsin(p->bobcounter, -12);
if (isRRRA() && p->spritebridge == 0 && p->on_ground)
{
if (psectlotag == 1)
p->NotOnWater = 0;
else if (p->OnBoat)
{
if (psectlotag == 1234)
p->NotOnWater = 0;
else
p->NotOnWater = 1;
}
else
p->NotOnWater = 1;
}
if (truefdist < gs.int_playerheight + (8 << 8) && (k == 1 || k == 3))
{
if (p->spritebridge == 0 && p->walking_snd_toggle == 0 && p->on_ground)
{
int j;
switch (psectlotag)
{
case 0:
if (clz.type == kHitSprite)
j = clz.actor()->spr.picnum;
else j = psectp->floorpicnum;
break;
case 1:
if ((krand() & 1) == 0)
if (!isRRRA() || (!p->OnBoat && !p->OnMotorcycle && p->cursector->hitag != 321))
S_PlayActorSound(DUKE_ONWATER, pact);
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) << 6);
p->vel.Y += ((sb_svel * doubvel) << 6);
if (!isRRRA() && ((p->curr_weapon == KNEE_WEAPON && p->kickback_pic > 10 && p->on_ground) || (p->on_ground && (actions & SB_CROUCH))))
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction - 0x2000, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction - 0x2000, 16);
}
else
{
if (psectlotag == 2)
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction - 0x1400, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction - 0x1400, 16);
}
else
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction, 16);
}
}
if (isRRRA() && psectp->floorpicnum == RRTILE7888)
{
if (p->OnMotorcycle)
if (p->on_ground)
p->moto_on_oil = 1;
}
else if (isRRRA() && psectp->floorpicnum == RRTILE7889)
{
if (p->OnMotorcycle)
{
if (p->on_ground)
p->moto_on_mud = 1;
}
else if (p->boot_amount > 0)
p->boot_amount--;
else
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction, 16);
}
}
else
if (psectp->floorpicnum == RRTILE3073 || psectp->floorpicnum == RRTILE2702)
{
if (p->OnMotorcycle)
{
if (p->on_ground)
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction - 0x1800, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction - 0x1800, 16);
}
}
else
if (p->boot_amount > 0)
p->boot_amount--;
else
{
p->vel.X = MulScale(p->vel.X, gs.playerfriction - 0x1800, 16);
p->vel.Y = MulScale(p->vel.Y, gs.playerfriction - 0x1800, 16);
}
}
if (abs(p->vel.X) < 2048 && abs(p->vel.Y) < 2048)
p->vel.X = p->vel.Y = 0;
if (shrunk)
{
p->vel.X =
MulScale(p->vel.X, gs.playerfriction - (gs.playerfriction >> 1) + (gs.playerfriction >> 2), 16);
p->vel.Y =
MulScale(p->vel.Y, gs.playerfriction - (gs.playerfriction >> 1) + (gs.playerfriction >> 2), 16);
}
}
HORIZONLY:
if (psectlotag == 1 || p->spritebridge == 1) i = (4L << 8);
else i = (20L << 8);
if (p->insector() && p->cursector->lotag == 2) k = 0;
else k = 1;
Collision clip{};
if (ud.clipping)
{
p->player_add_int_xy({ p->vel.X >> 14, p->vel.Y >> 14 });
updatesector(p->player_int_pos().X, p->player_int_pos().Y, &p->cursector);
ChangeActorSect(pact, p->cursector);
}
else
clipmove(p->pos, &p->cursector, p->vel.X, p->vel.Y, 164, (4 << 8), i, CLIPMASK0, clip);
if (p->jetpack_on == 0 && psectlotag != 2 && psectlotag != 1 && shrunk)
p->pos.Z += 32;
if (clip.type != kHitNone)
checkplayerhurt_r(p, clip);
else if (isRRRA() && p->hurt_delay2 > 0)
p->hurt_delay2--;
if (clip.type == kHitWall)
{
auto wal = clip.hitWall;
if (p->OnMotorcycle)
{
onMotorcycleMove(snum, wal);
}
else if (p->OnBoat)
{
onBoatMove(snum, psectlotag, wal);
}
else
{
if (wal->lotag >= 40 && wal->lotag <= 44)
{
if (wal->lotag < 44)
{
dofurniture(clip.hitWall, p->cursector, snum);
pushmove(p->pos, &p->cursector, 172, (4 << 8), (4 << 8), CLIPMASK0);
}
else
pushmove(p->pos, &p->cursector, 172, (4 << 8), (4 << 8), CLIPMASK0);
}
}
}
if (clip.type == kHitSprite)
{
if (p->OnMotorcycle)
{
onMotorcycleHit(snum, clip.actor());
}
else if (p->OnBoat)
{
onBoatHit(snum, clip.actor());
}
else if (badguy(clip.actor()))
{
if (clip.actor()->spr.statnum != 1)
{
clip.actor()->timetosleep = 0;
if (clip.actor()->spr.picnum == BILLYRAY)
S_PlayActorSound(404, clip.actor());
else
check_fta_sounds_r(clip.actor());
ChangeActorStat(clip.actor(), 1);
}
}
else if (!isRRRA() && clip.actor()->spr.picnum == RRTILE3410)
{
quickkill(p);
S_PlayActorSound(446, pact);
}
if (isRRRA())
{
if (clip.actor()->spr.picnum == RRTILE3410)
{
quickkill(p);
S_PlayActorSound(446, pact);
}
else if (clip.actor()->spr.picnum == RRTILE2443 && clip.actor()->spr.pal == 19)
{
clip.actor()->spr.pal = 0;
p->DrugMode = 5;
ps[snum].GetActor()->spr.extra = gs.max_player_health;
}
}
}
if (p->jetpack_on == 0)
{
if (pact->spr.xvel > 16)
{
if (psectlotag != ST_1_ABOVE_WATER && psectlotag != ST_2_UNDERWATER && p->on_ground && (!isRRRA() || !p->sea_sick_stat))
{
p->pycount += 52;
p->pycount &= 2047;
const double factor = 64. / 1596; // What is 1596?
p->pyoff = abs(pact->spr.xvel * DAngle::fromBuild(p->pycount).Sin()) * factor;
}
}
else if (psectlotag != ST_2_UNDERWATER && psectlotag != 1 && (!isRRRA() || !p->sea_sick_stat))
p->pyoff = 0;
}
// RBG***
SetActor(pact, p->pos.plusZ(gs.playerheight));
if (psectlotag == 800 && (!isRRRA() || !p->lotag800kill))
{
if (isRRRA()) p->lotag800kill = 1;
quickkill(p);
return;
}
if (psectlotag < 3)
{
psectp = pact->sector();
if (ud.clipping == 0 && psectp->lotag == ST_31_TWO_WAY_TRAIN)
{
auto act = barrier_cast<DDukeActor*>(psectp->hitagactor);
if (act && act->spr.xvel && act->temp_data[0] == 0)
{
quickkill(p);
return;
}
}
}
if (truefdist < gs.int_playerheight && p->on_ground && psectlotag != 1 && shrunk == 0 && p->insector() && p->cursector->lotag == 1)
if (!S_CheckActorSoundPlaying(pact, DUKE_ONWATER))
if (!isRRRA() || (!p->OnBoat && !p->OnMotorcycle && p->cursector->hitag != 321))
S_PlayActorSound(DUKE_ONWATER, pact);
if (p->cursector != pact->sector())
ChangeActorSect(pact, p->cursector);
int retry = 0;
while (ud.clipping == 0)
{
int blocked;
if (pact->spr.clipdist == 64)
blocked = (pushmove(p->pos, &p->cursector, 128, (4 << 8), (4 << 8), CLIPMASK0) < 0 && furthestangle(p->GetActor(), 8) < 512);
else
blocked = (pushmove(p->pos, &p->cursector, 16, (4 << 8), (4 << 8), CLIPMASK0) < 0 && furthestangle(p->GetActor(), 8) < 512);
if (fabs(pact->floorz - pact->ceilingz) < 48 || blocked)
{
if (!(pact->sector()->lotag & 0x8000) && (isanunderoperator(pact->sector()->lotag) ||
isanearoperator(pact->sector()->lotag)))
fi.activatebysector(pact->sector(), pact);
if (blocked)
{
if (!retry++)
{
p->pos = p->opos = oldpos;
continue;
}
quickkill(p);
return;
}
}
else if (abs(fz - cz) < (32 << 8) && isanunderoperator(psectp->lotag))
fi.activatebysector(psectp, pact);
break;
}
if (ud.clipping == 0 && (!p->cursector || (p->cursector && p->cursector->ceilingz > (p->cursector->floorz - 12))))
{
quickkill(p);
return;
}
if (actions & SB_CENTERVIEW || p->hard_landing)
{
playerCenterView(snum);
}
else if (actions & SB_LOOK_UP)
{
playerLookUp(snum, actions);
}
else if (actions & SB_LOOK_DOWN)
{
playerLookDown(snum, actions);
}
else if ((actions & SB_AIM_UP) && !p->OnMotorcycle)
{
playerAimUp(snum, actions);
}
else if ((actions & SB_AIM_DOWN) && !p->OnMotorcycle)
{
playerAimDown(snum, actions);
}
if (p->recoil && p->kickback_pic == 0)
{
int d = p->recoil >> 1;
if (!d)
d = 1;
p->recoil -= d;
p->horizon.addadjustment(buildhoriz(-d));
}
if (SyncInput())
{
p->horizon.applyinput(GetPlayerHorizon(snum), &actions);
}
p->checkhardlanding();
//Shooting code/changes
if (p->show_empty_weapon > 0)
p->show_empty_weapon--;
if (p->show_empty_weapon == 1)
{
fi.addweapon(p, p->last_full_weapon);
return;
}
dokneeattack(snum, { FEM10, NAKED1, STATUE });
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;
p->last_weapon = -1;
}
else if (p->holster_weapon == 0)
p->oweapon_pos = p->weapon_pos = 10;
}
else p->weapon_pos--;
}
processweapon(snum, actions, psectp);
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void OnMotorcycle(player_struct *p, DDukeActor* motosprite)
{
if (!p->OnMotorcycle && !(p->cursector->lotag == 2))
{
if (motosprite)
{
p->pos.X = motosprite->spr.pos.X;
p->pos.Y = motosprite->spr.pos.Y;
p->angle.ang = motosprite->spr.angle;
p->ammo_amount[MOTORCYCLE_WEAPON] = motosprite->saved_ammo;
deletesprite(motosprite);
}
p->over_shoulder_on = 0;
p->OnMotorcycle = 1;
p->last_full_weapon = p->curr_weapon;
p->curr_weapon = MOTORCYCLE_WEAPON;
p->gotweapon[MOTORCYCLE_WEAPON] = true;
p->vel.X = 0;
p->vel.Y = 0;
p->horizon.horiz = q16horiz(0);
}
if (!S_CheckActorSoundPlaying(p->GetActor(),186))
S_PlayActorSound(186, p->GetActor());
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void OffMotorcycle(player_struct *p)
{
auto pact = p->GetActor();
if (p->OnMotorcycle)
{
if (S_CheckActorSoundPlaying(pact,188))
S_StopSound(188,pact);
if (S_CheckActorSoundPlaying(pact,187))
S_StopSound(187,pact);
if (S_CheckActorSoundPlaying(pact,186))
S_StopSound(186,pact);
if (S_CheckActorSoundPlaying(pact,214))
S_StopSound(214,pact);
if (!S_CheckActorSoundPlaying(pact,42))
S_PlayActorSound(42, pact);
p->OnMotorcycle = 0;
p->gotweapon[MOTORCYCLE_WEAPON] = false;
p->curr_weapon = p->last_full_weapon;
checkavailweapon(p);
p->horizon.horiz = q16horiz(0);
p->moto_do_bump = 0;
p->MotoSpeed = 0;
p->TiltStatus = 0;
p->moto_drink = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
p->TurbCount = 0;
p->vel.X = 0 - p->angle.ang.Cos() * (1 << 7);
p->vel.Y = 0 - p->angle.ang.Sin() * (1 << 7);
p->moto_underwater = 0;
auto spawned = spawn(p->GetActor(), EMPTYBIKE);
if (spawned)
{
spawned->spr.angle = p->angle.ang;
spawned->spr.xvel += p->angle.ang.Cos() * (1 << 7);
spawned->spr.yvel += p->angle.ang.Sin() * (1 << 7);
spawned->saved_ammo = p->ammo_amount[MOTORCYCLE_WEAPON];
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void OnBoat(player_struct *p, DDukeActor* boat)
{
if (!p->OnBoat)
{
if (boat)
{
p->pos.X = boat->spr.pos.X;
p->pos.Y = boat->spr.pos.Y;
p->angle.ang = boat->spr.angle;
p->ammo_amount[BOAT_WEAPON] = boat->saved_ammo;
deletesprite(boat);
}
p->over_shoulder_on = 0;
p->OnBoat = 1;
p->last_full_weapon = p->curr_weapon;
p->curr_weapon = BOAT_WEAPON;
p->gotweapon[BOAT_WEAPON] = true;
p->vel.X = 0;
p->vel.Y = 0;
p->horizon.horiz = q16horiz(0);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void OffBoat(player_struct *p)
{
if (p->OnBoat)
{
p->OnBoat = 0;
p->gotweapon[BOAT_WEAPON] = false;
p->curr_weapon = p->last_full_weapon;
checkavailweapon(p);
p->horizon.horiz = q16horiz(0);
p->moto_do_bump = 0;
p->MotoSpeed = 0;
p->TiltStatus = 0;
p->moto_drink = 0;
p->VBumpTarget = 0;
p->VBumpNow = 0;
p->TurbCount = 0;
p->vel.X = 0 - p->angle.ang.Cos() * (1 << 7);
p->vel.Y = 0 - p->angle.ang.Sin() * (1 << 7);
p->moto_underwater = 0;
auto spawned = spawn(p->GetActor(), EMPTYBOAT);
if (spawned)
{
spawned->spr.angle = p->angle.ang;
spawned->spr.xvel += p->angle.ang.Cos() * (1 << 7);
spawned->spr.yvel += p->angle.ang.Sin() * (1 << 7);
spawned->saved_ammo = p->ammo_amount[BOAT_WEAPON];
}
}
}
END_DUKE_NS