2020-05-05 09:58:39 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 1996, 2003 - 3D Realms Entertainment
|
|
|
|
Copyright (C) 2000, 2003 - Matt Saettler (EDuke Enhancements)
|
|
|
|
Copyright (C) 2017-2019 - Nuke.YKT
|
|
|
|
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.
|
|
|
|
|
|
|
|
This file is a combination of code from the following sources:
|
|
|
|
- EDuke 2 by Matt Saettler
|
|
|
|
- JFDuke by Jonathon Fowler (jf@jonof.id.au),
|
|
|
|
- DukeGDX and RedneckGDX by Alexander Makarov-[M210] (m210-2007@mail.ru)
|
|
|
|
- Redneck Rampage reconstructed source by Nuke.YKT
|
|
|
|
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "ns.h"
|
|
|
|
#include "global.h"
|
2020-05-07 20:30:19 +00:00
|
|
|
#include "namesdyn.h"
|
2020-05-05 09:58:39 +00:00
|
|
|
|
|
|
|
BEGIN_DUKE_NS
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
bool ceilingspace_d(int sectnum);
|
|
|
|
bool ceilingspace_r(int sectnum);
|
|
|
|
bool floorspace_d(int sectnum);
|
|
|
|
bool floorspace_r(int sectnum);
|
|
|
|
void addweapon_d(struct player_struct *p, int weapon);
|
|
|
|
void addweapon_r(struct player_struct *p, int weapon);
|
|
|
|
void hitradius_d(short i, int r, int hp1, int hp2, int hp3, int hp4);
|
|
|
|
void hitradius_r(short i, int r, int hp1, int hp2, int hp3, int hp4);
|
|
|
|
int movesprite_d(short spritenum, int xchange, int ychange, int zchange, unsigned int cliptype);
|
|
|
|
int movesprite_r(short spritenum, int xchange, int ychange, int zchange, unsigned int cliptype);
|
|
|
|
void lotsofmoney_d(spritetype *s, short n);
|
|
|
|
void lotsofmail_d(spritetype *s, short n);
|
|
|
|
void lotsofpaper_d(spritetype *s, short n);
|
|
|
|
void lotsoffeathers_r(spritetype *s, short n);
|
|
|
|
void guts_d(spritetype* s, short gtype, short n, short p);
|
|
|
|
void guts_r(spritetype* s, short gtype, short n, short p);
|
|
|
|
void gutsdir_d(spritetype* s, short gtype, short n, short p);
|
|
|
|
void gutsdir_r(spritetype* s, short gtype, short n, short p);
|
|
|
|
int ifhitsectors_d(int sectnum);
|
|
|
|
int ifhitsectors_r(int sectnum);
|
|
|
|
int ifhitbyweapon_r(int sn);
|
|
|
|
int ifhitbyweapon_d(int sn);
|
|
|
|
|
|
|
|
bool ceilingspace(int sectnum)
|
2020-05-07 12:38:01 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
return isRR()? ceilingspace_r(sectnum) : ceilingspace_d(sectnum);
|
|
|
|
}
|
2020-05-07 12:38:01 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
bool floorspace(int sectnum)
|
|
|
|
{
|
|
|
|
return isRR()? floorspace_r(sectnum) : floorspace_d(sectnum);
|
|
|
|
}
|
2020-05-07 12:38:01 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void addweapon(struct player_struct *p, int weapon)
|
|
|
|
{
|
|
|
|
if (isRR()) addweapon_r(p, weapon);
|
|
|
|
else addweapon_d(p, weapon);
|
|
|
|
}
|
2020-05-05 13:25:59 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void hitradius(short i, int r, int hp1, int hp2, int hp3, int hp4)
|
2020-05-05 13:25:59 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (isRR()) hitradius_r(i, r, hp1, hp2, hp3, hp4);
|
|
|
|
else hitradius_d(i, r, hp1, hp2, hp3, hp4);
|
|
|
|
}
|
2020-05-05 16:47:07 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
int movesprite(short spritenum, int xchange, int ychange, int zchange, unsigned int cliptype)
|
|
|
|
{
|
|
|
|
if (isRR()) return movesprite_r(spritenum, xchange, ychange, zchange, cliptype);
|
|
|
|
else return movesprite_d(spritenum, xchange, ychange, zchange, cliptype);
|
2020-05-05 13:25:59 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void lotsofmoney(spritetype *s, short n)
|
|
|
|
{
|
|
|
|
if (isRR()) lotsoffeathers_r(s, n);
|
|
|
|
else lotsofmoney_d(s, n);
|
|
|
|
}
|
2020-05-05 13:25:59 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void lotsofmail(spritetype *s, short n)
|
2020-05-05 13:25:59 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (isRR()) lotsoffeathers_r(s, n);
|
|
|
|
else lotsofmail_d(s, n);
|
|
|
|
}
|
2020-05-05 16:47:07 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void lotsofpaper(spritetype *s, short n)
|
|
|
|
{
|
|
|
|
if (isRR()) lotsoffeathers_r(s, n);
|
|
|
|
else lotsofpaper_d(s, n);
|
2020-05-05 14:33:23 +00:00
|
|
|
}
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void guts(spritetype* s, short gtype, short n, short p)
|
|
|
|
{
|
|
|
|
if (isRR()) guts_r(s, gtype, n, p);
|
|
|
|
else guts_d(s, gtype, n, p);
|
|
|
|
}
|
2020-05-05 16:47:07 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void gutsdir(spritetype* s, short gtype, short n, short p)
|
2020-05-05 14:33:23 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (isRR()) gutsdir_r(s, gtype, n, p);
|
|
|
|
else gutsdir_d(s, gtype, n, p);
|
|
|
|
}
|
2020-05-05 14:33:23 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
int ifhitsectors(int sectnum)
|
|
|
|
{
|
|
|
|
return isRR()? ifhitsectors_r(sectnum) : ifhitsectors_d(sectnum);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ifhitbyweapon(int sectnum)
|
|
|
|
{
|
|
|
|
return isRR()? ifhitbyweapon_r(sectnum) : ifhitbyweapon_d(sectnum);
|
2020-05-05 14:33:23 +00:00
|
|
|
}
|
2020-05-05 13:25:59 +00:00
|
|
|
|
2020-05-05 14:33:23 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void addammo(short weapon, struct player_struct* p, short amount)
|
2020-05-05 14:33:23 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
p->ammo_amount[weapon] += amount;
|
2020-05-05 14:33:23 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (p->ammo_amount[weapon] > max_ammo_amount[weapon])
|
|
|
|
p->ammo_amount[weapon] = max_ammo_amount[weapon];
|
2020-05-05 13:25:59 +00:00
|
|
|
}
|
|
|
|
|
2020-05-05 20:07:54 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void checkavailinven(struct player_struct* p)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (p->firstaid_amount > 0)
|
|
|
|
p->inven_icon = ICON_FIRSTAID;
|
|
|
|
else if (p->steroids_amount > 0)
|
|
|
|
p->inven_icon = ICON_STEROIDS;
|
|
|
|
else if (p->holoduke_amount > 0)
|
|
|
|
p->inven_icon = ICON_HOLODUKE;
|
|
|
|
else if (p->jetpack_amount > 0)
|
|
|
|
p->inven_icon = ICON_JETPACK;
|
|
|
|
else if (p->heat_amount > 0)
|
|
|
|
p->inven_icon = ICON_HEATS;
|
|
|
|
else if (p->scuba_amount > 0)
|
|
|
|
p->inven_icon = ICON_SCUBA;
|
|
|
|
else if (p->boot_amount > 0)
|
|
|
|
p->inven_icon = ICON_BOOTS;
|
|
|
|
else p->inven_icon = ICON_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void checkavailweapon(struct player_struct* p)
|
|
|
|
{
|
|
|
|
short i, snum;
|
|
|
|
int weap;
|
|
|
|
|
|
|
|
if (p->wantweaponfire >= 0)
|
|
|
|
{
|
|
|
|
weap = p->wantweaponfire;
|
|
|
|
p->wantweaponfire = -1;
|
|
|
|
|
|
|
|
if (weap == p->curr_weapon) return;
|
|
|
|
else if (p->gotweapon[weap] && p->ammo_amount[weap] > 0)
|
|
|
|
{
|
|
|
|
addweapon(p, weap);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
weap = p->curr_weapon;
|
|
|
|
if (p->gotweapon[weap] && p->ammo_amount[weap] > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
snum = sprite[p->i].yvel;
|
|
|
|
|
|
|
|
// Note: RedNukem has this restriction, but the original source and RedneckGDX do not.
|
|
|
|
#if 1 // TRANSITIONAL
|
2020-05-06 14:10:44 +00:00
|
|
|
int max = ((isRR()) ? DEVISTATOR_WEAPON : FREEZE_WEAPON);
|
2020-05-05 20:07:54 +00:00
|
|
|
#else
|
|
|
|
int max = FREEZE_WEAPON;
|
|
|
|
#endif
|
|
|
|
for (i = 0; i < 10; i++)
|
|
|
|
{
|
|
|
|
weap = ud.wchoice[snum][i];
|
|
|
|
if ((g_gameType & GAMEFLAG_SHAREWARE) && weap > 6) continue;
|
|
|
|
|
|
|
|
if (weap == 0) weap = max;
|
|
|
|
else weap--;
|
|
|
|
|
|
|
|
if (weap == KNEE_WEAPON || (p->gotweapon[weap] && p->ammo_amount[weap] > 0))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == HANDREMOTE_WEAPON) weap = KNEE_WEAPON;
|
|
|
|
|
|
|
|
// Found the weapon
|
|
|
|
|
|
|
|
p->last_weapon = p->curr_weapon;
|
|
|
|
p->random_club_frame = 0;
|
|
|
|
p->curr_weapon = weap;
|
2020-05-06 14:10:44 +00:00
|
|
|
if (isWW2GI())
|
2020-05-05 20:07:54 +00:00
|
|
|
{
|
|
|
|
SetGameVarID(g_iWeaponVarID, p->curr_weapon, p->i, snum);
|
|
|
|
if (p->curr_weapon >= 0)
|
|
|
|
{
|
|
|
|
SetGameVarID(g_iWorksLikeVarID, aplWeaponWorksLike[p->curr_weapon][snum], p->i, snum);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SetGameVarID(g_iWorksLikeVarID, -1, p->i, snum);
|
|
|
|
}
|
|
|
|
OnEvent(EVENT_CHANGEWEAPON, p->i, snum, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
p->kickback_pic = 0;
|
|
|
|
if (p->holster_weapon == 1)
|
|
|
|
{
|
|
|
|
p->holster_weapon = 0;
|
|
|
|
p->weapon_pos = 10;
|
|
|
|
}
|
|
|
|
else p->weapon_pos = -1;
|
|
|
|
}
|
|
|
|
|
2020-05-05 22:08:08 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void clearcamera(player_struct* ps)
|
|
|
|
{
|
|
|
|
ps->newowner = -1;
|
|
|
|
ps->posx = ps->oposx;
|
|
|
|
ps->posy = ps->oposy;
|
|
|
|
ps->posz = ps->oposz;
|
|
|
|
ps->q16ang = ps->oq16ang;
|
|
|
|
updatesector(ps->posx, ps->posy, &ps->cursectnum);
|
|
|
|
setpal(ps);
|
|
|
|
|
|
|
|
int k = headspritestat[1];
|
|
|
|
while (k >= 0)
|
|
|
|
{
|
|
|
|
if (sprite[k].picnum == CAMERA1)
|
|
|
|
sprite[k].yvel = 0;
|
|
|
|
k = nextspritestat[k];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
int ssp(short i, unsigned int cliptype) //The set sprite function
|
2020-05-05 22:08:08 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
spritetype* s;
|
|
|
|
int movetype;
|
2020-05-05 22:08:08 +00:00
|
|
|
|
|
|
|
s = &sprite[i];
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
movetype = movesprite(i,
|
|
|
|
(s->xvel * (sintable[(s->ang + 512) & 2047])) >> 14,
|
|
|
|
(s->xvel * (sintable[s->ang & 2047])) >> 14, s->zvel,
|
|
|
|
cliptype);
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
return (movetype == 0);
|
|
|
|
}
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void insertspriteq(int i)
|
|
|
|
{
|
|
|
|
if (spriteqamount > 0)
|
|
|
|
{
|
|
|
|
if (spriteq[spriteqloc] >= 0)
|
|
|
|
sprite[spriteq[spriteqloc]].xrepeat = 0;
|
|
|
|
spriteq[spriteqloc] = i;
|
|
|
|
spriteqloc = (spriteqloc + 1) % spriteqamount;
|
2020-05-05 22:08:08 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
else sprite[i].xrepeat = sprite[i].yrepeat = 0;
|
|
|
|
}
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// consolidation of several nearly identical functions
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void lotsofstuff(spritetype* s, short n, int spawntype)
|
|
|
|
{
|
|
|
|
short i, j;
|
|
|
|
for (i = n; i > 0; i--)
|
2020-05-05 22:08:08 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
short r1 = krand(), r2 = krand(); // using the RANDCORRECT version from RR.
|
|
|
|
// TRANSITIONAL RedNukem sets the spawner as owner.
|
|
|
|
j = EGS(s->sectnum, s->x, s->y, s->z - (r2 % (47 << 8)), spawntype, -32, 8, 8, r1 & 2047, 0, 0, 0, 5);
|
|
|
|
sprite[j].cstat = krand() & 12;
|
|
|
|
}
|
|
|
|
}
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// movesector - why is this in actors.cpp?
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void ms(short i)
|
|
|
|
{
|
|
|
|
//T1,T2 and T3 are used for all the sector moving stuff!!!
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
short startwall, endwall, x;
|
|
|
|
int tx, ty;
|
|
|
|
spritetype* s;
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
s = &sprite[i];
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
s->x += (s->xvel * (sintable[(s->ang + 512) & 2047])) >> 14;
|
|
|
|
s->y += (s->xvel * (sintable[s->ang & 2047])) >> 14;
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
int j = hittype[i].temp_data[1];
|
|
|
|
int k = hittype[i].temp_data[2];
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
startwall = sector[s->sectnum].wallptr;
|
|
|
|
endwall = startwall + sector[s->sectnum].wallnum;
|
|
|
|
for (x = startwall; x < endwall; x++)
|
|
|
|
{
|
|
|
|
rotatepoint(
|
|
|
|
0, 0,
|
|
|
|
msx[j], msy[j],
|
|
|
|
k & 2047, &tx, &ty);
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
dragpoint(x, s->x + tx, s->y + ty);
|
2020-05-05 22:08:08 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
j++;
|
|
|
|
}
|
|
|
|
}
|
2020-05-06 14:10:44 +00:00
|
|
|
|
2020-05-06 19:11:36 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void movecyclers(void)
|
|
|
|
{
|
|
|
|
short q, j, x, t, s, * c;
|
|
|
|
walltype* wal;
|
|
|
|
char cshade;
|
|
|
|
|
|
|
|
for (q = numcyclers - 1; q >= 0; q--)
|
|
|
|
{
|
|
|
|
|
|
|
|
c = &cyclers[q][0];
|
|
|
|
s = c[0];
|
|
|
|
|
|
|
|
t = c[3];
|
|
|
|
j = t + (sintable[c[1] & 2047] >> 10);
|
|
|
|
cshade = c[2];
|
|
|
|
|
|
|
|
if (j < cshade) j = cshade;
|
|
|
|
else if (j > t) j = t;
|
|
|
|
|
|
|
|
c[1] += sector[s].extra;
|
|
|
|
if (c[5])
|
|
|
|
{
|
|
|
|
wal = &wall[sector[s].wallptr];
|
|
|
|
for (x = sector[s].wallnum; x > 0; x--, wal++)
|
|
|
|
if (wal->hitag != 1)
|
|
|
|
{
|
|
|
|
wal->shade = j;
|
|
|
|
|
|
|
|
if ((wal->cstat & 2) && wal->nextwall >= 0)
|
|
|
|
wall[wal->nextwall].shade = j;
|
|
|
|
|
|
|
|
}
|
|
|
|
sector[s].floorshade = sector[s].ceilingshade = j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void movedummyplayers(void)
|
|
|
|
{
|
|
|
|
short i, p, nexti;
|
|
|
|
|
|
|
|
i = headspritestat[STAT_DUMMYPLAYER];
|
|
|
|
while (i >= 0)
|
|
|
|
{
|
|
|
|
nexti = nextspritestat[i];
|
|
|
|
|
|
|
|
p = sprite[sprite[i].owner].yvel;
|
|
|
|
|
|
|
|
if ((!isRR() && ps[p].on_crane >= 0) || sector[ps[p].cursectnum].lotag != 1 || sprite[ps[p].i].extra <= 0)
|
|
|
|
{
|
|
|
|
ps[p].dummyplayersprite = -1;
|
|
|
|
deletesprite(i);
|
|
|
|
i = nexti;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (ps[p].on_ground && ps[p].on_warping_sector == 1 && sector[ps[p].cursectnum].lotag == 1)
|
|
|
|
{
|
|
|
|
sprite[i].cstat = 257;
|
|
|
|
sprite[i].z = sector[sprite[i].sectnum].ceilingz + (27 << 8);
|
|
|
|
sprite[i].ang = ps[p].q16ang >> FRACBITS;
|
|
|
|
if (hittype[i].temp_data[0] == 8)
|
|
|
|
hittype[i].temp_data[0] = 0;
|
|
|
|
else hittype[i].temp_data[0]++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (sector[sprite[i].sectnum].lotag != 2) sprite[i].z = sector[sprite[i].sectnum].floorz;
|
|
|
|
sprite[i].cstat = (short)32768;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sprite[i].x += (ps[p].posx - ps[p].oposx);
|
|
|
|
sprite[i].y += (ps[p].posy - ps[p].oposy);
|
|
|
|
setsprite(i, sprite[i].x, sprite[i].y, sprite[i].z);
|
|
|
|
i = nexti;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
int otherp;
|
|
|
|
void moveplayers(void) //Players
|
|
|
|
{
|
|
|
|
short i, nexti;
|
|
|
|
int otherx;
|
|
|
|
spritetype* s;
|
|
|
|
struct player_struct* p;
|
|
|
|
|
|
|
|
i = headspritestat[STAT_PLAYER];
|
|
|
|
while (i >= 0)
|
|
|
|
{
|
|
|
|
nexti = nextspritestat[i];
|
|
|
|
|
|
|
|
s = &sprite[i];
|
|
|
|
p = &ps[s->yvel];
|
|
|
|
if (s->owner >= 0)
|
|
|
|
{
|
|
|
|
if (p->newowner >= 0) //Looking thru the camera
|
|
|
|
{
|
|
|
|
s->x = p->oposx;
|
|
|
|
s->y = p->oposy;
|
|
|
|
hittype[i].bposz = s->z = p->oposz + PHEIGHT;
|
|
|
|
s->ang = p->oq16ang >> FRACBITS;
|
|
|
|
setsprite(i, s->x, s->y, s->z);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (ud.multimode > 1)
|
|
|
|
otherp = findotherplayer(s->yvel, &otherx);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
otherp = s->yvel;
|
|
|
|
otherx = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
execute(i, s->yvel, otherx);
|
|
|
|
|
2020-05-07 08:28:31 +00:00
|
|
|
p->oq16ang = p->q16ang;
|
|
|
|
|
2020-05-06 19:11:36 +00:00
|
|
|
if (ud.multimode > 1)
|
|
|
|
if (sprite[ps[otherp].i].extra > 0)
|
|
|
|
{
|
|
|
|
if (s->yrepeat > 32 && sprite[ps[otherp].i].yrepeat < 32)
|
|
|
|
{
|
|
|
|
if (otherx < 1400 && p->knee_incs == 0)
|
|
|
|
{
|
|
|
|
p->knee_incs = 1;
|
|
|
|
p->weapon_pos = -1;
|
|
|
|
p->actorsqu = ps[otherp].i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ud.god)
|
|
|
|
{
|
|
|
|
s->extra = p->max_player_health;
|
|
|
|
s->cstat = 257;
|
|
|
|
if (!isWW2GI() && !isRR())
|
|
|
|
p->jetpack_amount = 1599;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (s->extra > 0)
|
|
|
|
{
|
|
|
|
// currently alive...
|
|
|
|
|
|
|
|
hittype[i].owner = i;
|
|
|
|
|
|
|
|
if (ud.god == 0)
|
|
|
|
if (ceilingspace(s->sectnum) || floorspace(s->sectnum))
|
|
|
|
quickkill(p);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
|
|
|
|
p->posx = s->x;
|
|
|
|
p->posy = s->y;
|
|
|
|
p->posz = s->z - (20 << 8);
|
|
|
|
|
|
|
|
p->newowner = -1;
|
|
|
|
|
|
|
|
if (p->wackedbyactor >= 0 && sprite[p->wackedbyactor].statnum < MAXSTATUS)
|
|
|
|
{
|
|
|
|
int ang = p->q16ang >> FRACBITS;
|
|
|
|
ang += getincangle(ang, getangle(sprite[p->wackedbyactor].x - p->posx, sprite[p->wackedbyactor].y - p->posy)) >> 1;
|
|
|
|
ang &= 2047;
|
|
|
|
p->q16ang = ang << FRACBITS;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
s->ang = p->q16ang >> FRACBITS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (p->holoduke_on == -1)
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
i = nexti;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
hittype[i].bposx = s->x;
|
|
|
|
hittype[i].bposy = s->y;
|
|
|
|
hittype[i].bposz = s->z;
|
|
|
|
|
|
|
|
s->cstat = 0;
|
|
|
|
|
|
|
|
if (s->xrepeat < 42)
|
|
|
|
{
|
|
|
|
s->xrepeat += 4;
|
|
|
|
s->cstat |= 2;
|
|
|
|
}
|
|
|
|
else s->xrepeat = 42;
|
|
|
|
if (s->yrepeat < 36)
|
|
|
|
s->yrepeat += 4;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s->yrepeat = 36;
|
|
|
|
if (sector[s->sectnum].lotag != ST_2_UNDERWATER)
|
|
|
|
makeitfall(i);
|
|
|
|
if (s->zvel == 0 && sector[s->sectnum].lotag == ST_1_ABOVE_WATER)
|
|
|
|
s->z += (32 << 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->extra < 8)
|
|
|
|
{
|
|
|
|
s->xvel = 128;
|
|
|
|
s->ang = p->q16ang >> FRACBITS;
|
|
|
|
s->extra++;
|
|
|
|
//IFMOVING; // JBF 20040825: is really "if (ssp(i,CLIPMASK0)) ;" which is probably
|
|
|
|
ssp(i, CLIPMASK0); // not the safest of ideas because a zealous optimiser probably sees
|
|
|
|
// it as redundant, so I'll call the "ssp(i,CLIPMASK0)" explicitly.
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s->ang = 2047 - (p->q16ang >> FRACBITS);
|
|
|
|
setsprite(i, s->x, s->y, s->z);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sector[s->sectnum].ceilingstat & 1)
|
|
|
|
s->shade += (sector[s->sectnum].ceilingshade - s->shade) >> 1;
|
|
|
|
else
|
|
|
|
s->shade += (sector[s->sectnum].floorshade - s->shade) >> 1;
|
|
|
|
|
|
|
|
i = nexti;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-07 07:49:05 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void movefx(void)
|
|
|
|
{
|
|
|
|
short i, j, nexti, p;
|
|
|
|
int x, ht;
|
|
|
|
spritetype* s;
|
|
|
|
|
|
|
|
i = headspritestat[STAT_FX];
|
|
|
|
while (i >= 0)
|
|
|
|
{
|
|
|
|
s = &sprite[i];
|
|
|
|
|
|
|
|
nexti = nextspritestat[i];
|
|
|
|
|
|
|
|
switch (s->picnum)
|
|
|
|
{
|
|
|
|
case RESPAWN:
|
|
|
|
if (sprite[i].extra == 66)
|
2020-05-07 20:30:19 +00:00
|
|
|
{
|
|
|
|
j = spawn(i, sprite[i].hitag);
|
|
|
|
if (isRRRA())
|
|
|
|
{
|
|
|
|
respawn_rrra(i, j);
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (sprite[i].extra > (66 - 13))
|
|
|
|
sprite[i].extra++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MUSICANDSFX:
|
|
|
|
|
|
|
|
ht = s->hitag;
|
|
|
|
|
|
|
|
if (hittype[i].temp_data[1] != (int)SoundEnabled())
|
|
|
|
{
|
|
|
|
hittype[i].temp_data[1] = SoundEnabled();
|
|
|
|
hittype[i].temp_data[0] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->lotag >= 1000 && s->lotag < 2000)
|
|
|
|
{
|
|
|
|
x = ldist(&sprite[ps[screenpeek].i], s);
|
|
|
|
if (x < ht && hittype[i].temp_data[0] == 0)
|
|
|
|
{
|
|
|
|
FX_SetReverb(s->lotag - 1000);
|
|
|
|
hittype[i].temp_data[0] = 1;
|
|
|
|
}
|
|
|
|
if (x >= ht && hittype[i].temp_data[0] == 1)
|
|
|
|
{
|
|
|
|
FX_SetReverb(0);
|
|
|
|
FX_SetReverbDelay(0);
|
|
|
|
hittype[i].temp_data[0] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (s->lotag < 999 && (unsigned)sector[s->sectnum].lotag < ST_9_SLIDING_ST_DOOR && snd_ambience && sector[sprite[i].sectnum].floorz != sector[sprite[i].sectnum].ceilingz)
|
|
|
|
{
|
|
|
|
auto flags = S_GetUserFlags(s->lotag);
|
|
|
|
if (flags & SF_MSFX)
|
|
|
|
{
|
|
|
|
int x = dist(&sprite[ps[screenpeek].i], s);
|
|
|
|
|
|
|
|
if (x < ht && hittype[i].temp_data[0] == 0)
|
|
|
|
{
|
|
|
|
// Start playing an ambience sound.
|
|
|
|
A_PlaySound(s->lotag, i, CHAN_AUTO, CHANF_LOOP);
|
|
|
|
hittype[i].temp_data[0] = 1; // AMBIENT_SFX_PLAYING
|
|
|
|
}
|
|
|
|
else if (x >= ht && hittype[i].temp_data[0] == 1)
|
|
|
|
{
|
|
|
|
// Stop playing ambience sound because we're out of its range.
|
|
|
|
S_StopEnvSound(s->lotag, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((flags & (SF_GLOBAL | SF_DTAG)) == SF_GLOBAL)
|
|
|
|
{
|
|
|
|
if (hittype[i].temp_data[4] > 0) hittype[i].temp_data[4]--;
|
|
|
|
else for (p = connecthead; p >= 0; p = connectpoint2[p])
|
|
|
|
if (p == myconnectindex && ps[p].cursectnum == s->sectnum)
|
|
|
|
{
|
|
|
|
S_PlaySound(s->lotag + (unsigned)global_random % (s->hitag + 1));
|
|
|
|
hittype[i].temp_data[4] = 26 * 40 + (global_random % (26 * 40));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
i = nexti;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// split out of movestandables
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movecrane(int i, int crane)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
auto s = &sprite[i];
|
|
|
|
int sect = s->sectnum;
|
|
|
|
int x;
|
|
|
|
|
|
|
|
//t[0] = state
|
|
|
|
//t[1] = checking sector number
|
|
|
|
|
|
|
|
if (s->xvel) getglobalz(i);
|
|
|
|
|
|
|
|
if (t[0] == 0) //Waiting to check the sector
|
|
|
|
{
|
|
|
|
int j = headspritesect[t[1]];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
|
|
|
int nextj = nextspritesect[j];
|
|
|
|
switch (sprite[j].statnum)
|
|
|
|
{
|
|
|
|
case STAT_ACTOR:
|
|
|
|
case STAT_ZOMBIEACTOR:
|
|
|
|
case STAT_STANDABLE:
|
|
|
|
case STAT_PLAYER:
|
|
|
|
s->ang = getangle(msx[t[4] + 1] - s->x, msy[t[4] + 1] - s->y);
|
|
|
|
setsprite(j, msx[t[4] + 1], msy[t[4] + 1], sprite[j].z);
|
|
|
|
t[0]++;
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
j = nextj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t[0] == 1)
|
|
|
|
{
|
|
|
|
if (s->xvel < 184)
|
|
|
|
{
|
|
|
|
s->picnum = crane + 1;
|
|
|
|
s->xvel += 8;
|
|
|
|
}
|
|
|
|
//IFMOVING; // JBF 20040825: see my rant above about this
|
|
|
|
ssp(i, CLIPMASK0);
|
|
|
|
if (sect == t[1])
|
|
|
|
t[0]++;
|
|
|
|
}
|
|
|
|
else if (t[0] == 2 || t[0] == 7)
|
|
|
|
{
|
|
|
|
s->z += (1024 + 512);
|
|
|
|
|
|
|
|
if (t[0] == 2)
|
|
|
|
{
|
|
|
|
if ((sector[sect].floorz - s->z) < (64 << 8))
|
|
|
|
if (s->picnum > crane) s->picnum--;
|
|
|
|
|
|
|
|
if ((sector[sect].floorz - s->z) < (4096 + 1024))
|
|
|
|
t[0]++;
|
|
|
|
}
|
|
|
|
if (t[0] == 7)
|
|
|
|
{
|
|
|
|
if ((sector[sect].floorz - s->z) < (64 << 8))
|
|
|
|
{
|
|
|
|
if (s->picnum > crane) s->picnum--;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (s->owner == -2)
|
|
|
|
{
|
|
|
|
auto p = findplayer(s, &x);
|
|
|
|
spritesound(isRR() ? 390 : DUKE_GRUNT, ps[p].i);
|
|
|
|
if (ps[p].on_crane == i)
|
|
|
|
ps[p].on_crane = -1;
|
|
|
|
}
|
|
|
|
t[0]++;
|
|
|
|
s->owner = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (t[0] == 3)
|
|
|
|
{
|
|
|
|
s->picnum++;
|
|
|
|
if (s->picnum == (crane + 2))
|
|
|
|
{
|
|
|
|
auto p = checkcursectnums(t[1]);
|
|
|
|
if (p >= 0 && ps[p].on_ground)
|
|
|
|
{
|
|
|
|
s->owner = -2;
|
|
|
|
ps[p].on_crane = i;
|
|
|
|
spritesound(isRR() ? 390 : DUKE_GRUNT, ps[p].i);
|
|
|
|
ps[p].q16ang = (s->ang + 1024) << FRACBITS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int j = headspritesect[t[1]];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
|
|
|
switch (sprite[j].statnum)
|
|
|
|
{
|
|
|
|
case 1:
|
|
|
|
case 6:
|
|
|
|
s->owner = j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
j = nextspritesect[j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
t[0]++;//Grabbed the sprite
|
|
|
|
t[2] = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (t[0] == 4) //Delay before going up
|
|
|
|
{
|
|
|
|
t[2]++;
|
|
|
|
if (t[2] > 10)
|
|
|
|
t[0]++;
|
|
|
|
}
|
|
|
|
else if (t[0] == 5 || t[0] == 8)
|
|
|
|
{
|
|
|
|
if (t[0] == 8 && s->picnum < (crane + 2))
|
|
|
|
if ((sector[sect].floorz - s->z) > 8192)
|
|
|
|
s->picnum++;
|
|
|
|
|
|
|
|
if (s->z < msx[t[4] + 2])
|
|
|
|
{
|
|
|
|
t[0]++;
|
|
|
|
s->xvel = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
s->z -= (1024 + 512);
|
|
|
|
}
|
|
|
|
else if (t[0] == 6)
|
|
|
|
{
|
|
|
|
if (s->xvel < 192)
|
|
|
|
s->xvel += 8;
|
|
|
|
s->ang = getangle(msx[t[4]] - s->x, msy[t[4]] - s->y);
|
|
|
|
//IFMOVING; // JBF 20040825: see my rant above about this
|
|
|
|
ssp(i, CLIPMASK0);
|
|
|
|
if (((s->x - msx[t[4]]) * (s->x - msx[t[4]]) + (s->y - msy[t[4]]) * (s->y - msy[t[4]])) < (128 * 128))
|
|
|
|
t[0]++;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (t[0] == 9)
|
|
|
|
t[0] = 0;
|
|
|
|
|
|
|
|
setsprite(msy[t[4] + 2], s->x, s->y, s->z - (34 << 8));
|
|
|
|
|
|
|
|
if (s->owner != -1)
|
|
|
|
{
|
|
|
|
auto p = findplayer(s, &x);
|
|
|
|
|
|
|
|
int j = ifhitbyweapon(i);
|
|
|
|
if (j >= 0)
|
|
|
|
{
|
|
|
|
if (s->owner == -2)
|
|
|
|
if (ps[p].on_crane == i)
|
|
|
|
ps[p].on_crane = -1;
|
|
|
|
s->owner = -1;
|
|
|
|
s->picnum = crane;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->owner >= 0)
|
|
|
|
{
|
|
|
|
setsprite(s->owner, s->x, s->y, s->z);
|
|
|
|
|
|
|
|
hittype[s->owner].bposx = s->x;
|
|
|
|
hittype[s->owner].bposy = s->y;
|
|
|
|
hittype[s->owner].bposz = s->z;
|
|
|
|
|
|
|
|
s->zvel = 0;
|
|
|
|
}
|
|
|
|
else if (s->owner == -2)
|
|
|
|
{
|
|
|
|
auto ang = ps[p].q16ang >> FRACBITS;
|
|
|
|
ps[p].oposx = ps[p].posx = s->x - (sintable[(ang + 512) & 2047] >> 6);
|
|
|
|
ps[p].oposy = ps[p].posy = s->y - (sintable[ang & 2047] >> 6);
|
|
|
|
ps[p].oposz = ps[p].posz = s->z + (2 << 8);
|
|
|
|
setsprite(ps[p].i, ps[p].posx, ps[p].posy, ps[p].posz);
|
|
|
|
ps[p].cursectnum = sprite[ps[p].i].sectnum;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movefountain(int i, int fountain)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
auto s = &sprite[i];
|
|
|
|
int x;
|
|
|
|
if (t[0] > 0)
|
|
|
|
{
|
|
|
|
if (t[0] < 20)
|
|
|
|
{
|
|
|
|
t[0]++;
|
|
|
|
|
|
|
|
s->picnum++;
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->picnum == fountain + 3)
|
|
|
|
s->picnum = fountain + 1;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
findplayer(s, &x);
|
|
|
|
|
|
|
|
if (x > 512)
|
|
|
|
{
|
|
|
|
t[0] = 0;
|
2020-05-07 20:30:19 +00:00
|
|
|
s->picnum = fountain;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
|
|
|
else t[0] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void moveflammable(int i, int tire, int box, int pool)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
int j;
|
|
|
|
if (hittype[i].temp_data[0] == 1)
|
|
|
|
{
|
|
|
|
hittype[i].temp_data[1]++;
|
|
|
|
if ((hittype[i].temp_data[1] & 3) > 0) return;
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (!isRR() && s->picnum == tire && hittype[i].temp_data[1] == 32)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
s->cstat = 0;
|
2020-05-07 20:30:19 +00:00
|
|
|
j = spawn(i, pool);
|
2020-05-07 07:49:05 +00:00
|
|
|
sprite[j].shade = 127;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (s->shade < 64) s->shade++;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
j = s->xrepeat - (krand() & 7);
|
|
|
|
if (j < 10)
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
s->xrepeat = j;
|
|
|
|
|
|
|
|
j = s->yrepeat - (krand() & 7);
|
|
|
|
if (j < 4)
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
s->yrepeat = j;
|
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
if (box >= 0 && s->picnum == box)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
makeitfall(i);
|
|
|
|
hittype[i].ceilingz = sector[s->sectnum].ceilingz;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void detonate(int i, int explosion)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
earthquaketime = 16;
|
|
|
|
|
|
|
|
int j = headspritestat[STAT_EFFECTOR];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->hitag == sprite[j].hitag)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (sprite[j].lotag == SE_13_EXPLOSIVE)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (hittype[j].temp_data[2] == 0)
|
|
|
|
hittype[j].temp_data[2] = 1;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
else if (sprite[j].lotag == SE_8_UP_OPEN_DOOR_LIGHTS)
|
|
|
|
hittype[j].temp_data[4] = 1;
|
|
|
|
else if (sprite[j].lotag == SE_18_INCREMENTAL_SECTOR_RISE_FALL)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (hittype[j].temp_data[0] == 0)
|
|
|
|
hittype[j].temp_data[0] = 1;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
else if (sprite[j].lotag == SE_21_DROP_FLOOR)
|
|
|
|
hittype[j].temp_data[0] = 1;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
j = nextspritestat[j];
|
|
|
|
}
|
|
|
|
|
|
|
|
s->z -= (32 << 8);
|
|
|
|
|
|
|
|
if ((t[3] == 1 && s->xrepeat) || s->lotag == -99)
|
|
|
|
{
|
|
|
|
int x = s->extra;
|
|
|
|
spawn(i, explosion);
|
|
|
|
hitradius(i, seenineblastradius, x >> 2, x - (x >> 1), x - (x >> 2), x);
|
|
|
|
spritesound(PIPEBOMB_EXPLODE, i);
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
|
|
|
|
if (s->xrepeat)
|
|
|
|
for (int x = 0; x < 8; x++) RANDOMSCRAP(s, i);
|
|
|
|
|
|
|
|
deletesprite(i);
|
|
|
|
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movemasterswitch(int i, int spectype1, int spectype2)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
if (s->yvel == 1)
|
|
|
|
{
|
|
|
|
s->hitag--;
|
|
|
|
if (s->hitag <= 0)
|
|
|
|
{
|
|
|
|
operatesectors(s->sectnum, i);
|
|
|
|
|
|
|
|
int j = headspritesect[s->sectnum];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
|
|
|
if (sprite[j].statnum == 3)
|
|
|
|
{
|
|
|
|
switch (sprite[j].lotag)
|
|
|
|
{
|
|
|
|
case SE_2_EARTHQUAKE:
|
|
|
|
case SE_21_DROP_FLOOR:
|
|
|
|
case SE_31_FLOOR_RISE_FALL:
|
|
|
|
case SE_32_CEILING_RISE_FALL:
|
|
|
|
case SE_36_PROJ_SHOOTER:
|
|
|
|
hittype[j].temp_data[0] = 1;
|
|
|
|
break;
|
|
|
|
case SE_3_RANDOM_LIGHTS_AFTER_SHOT_OUT:
|
|
|
|
hittype[j].temp_data[4] = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (sprite[j].statnum == 6)
|
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (sprite[j].picnum == spectype1 || sprite[j].picnum == spectype2) // SEENINE and OOZFILTER
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
sprite[j].shade = -31;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
j = nextspritesect[j];
|
|
|
|
}
|
|
|
|
deletesprite(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movetrash(int i)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
if (s->xvel == 0) s->xvel = 1;
|
|
|
|
if (ssp(i, CLIPMASK0))
|
|
|
|
{
|
|
|
|
makeitfall(i);
|
|
|
|
if (krand() & 1) s->zvel -= 256;
|
|
|
|
if (klabs(s->xvel) < 48)
|
|
|
|
s->xvel += (krand() & 3);
|
|
|
|
}
|
|
|
|
else deletesprite(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movewaterdrip(int i, int drip)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
int sect = s->sectnum;
|
|
|
|
|
|
|
|
if (t[1])
|
|
|
|
{
|
|
|
|
t[1]--;
|
|
|
|
if (t[1] == 0)
|
|
|
|
s->cstat &= 32767;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
makeitfall(i);
|
|
|
|
ssp(i, CLIPMASK0);
|
|
|
|
if (s->xvel > 0) s->xvel -= 2;
|
|
|
|
|
|
|
|
if (s->zvel == 0)
|
|
|
|
{
|
|
|
|
s->cstat |= 32768;
|
|
|
|
|
|
|
|
if (s->pal != 2 && (isRR() || s->hitag == 0))
|
|
|
|
spritesound(SOMETHING_DRIPPING, i);
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (sprite[s->owner].picnum != drip)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hittype[i].bposz = s->z = t[0];
|
|
|
|
t[1] = 48 + (krand() & 31);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movedoorshock(int i)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
int sect = s->sectnum;
|
|
|
|
int j = abs(sector[sect].ceilingz - sector[sect].floorz) >> 9;
|
|
|
|
s->yrepeat = j + 4;
|
|
|
|
s->xrepeat = 16;
|
|
|
|
s->z = sector[sect].floorz;
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movetouchplate(int i, int plate)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
int sect = s->sectnum;
|
|
|
|
int x;
|
|
|
|
int p;
|
|
|
|
|
|
|
|
if (t[1] == 1 && s->hitag >= 0) //Move the sector floor
|
|
|
|
{
|
|
|
|
x = sector[sect].floorz;
|
|
|
|
|
|
|
|
if (t[3] == 1)
|
|
|
|
{
|
|
|
|
if (x >= t[2])
|
|
|
|
{
|
|
|
|
sector[sect].floorz = x;
|
|
|
|
t[1] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sector[sect].floorz += sector[sect].extra;
|
|
|
|
p = checkcursectnums(sect);
|
|
|
|
if (p >= 0) ps[p].posz += sector[sect].extra;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (x <= s->z)
|
|
|
|
{
|
|
|
|
sector[sect].floorz = s->z;
|
|
|
|
t[1] = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sector[sect].floorz -= sector[sect].extra;
|
|
|
|
p = checkcursectnums(sect);
|
|
|
|
if (p >= 0)
|
|
|
|
ps[p].posz -= sector[sect].extra;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (t[5] == 1) return;
|
|
|
|
|
|
|
|
p = checkcursectnums(sect);
|
|
|
|
if (p >= 0 && (ps[p].on_ground || s->ang == 512))
|
|
|
|
{
|
|
|
|
if (t[0] == 0 && !check_activator_motion(s->lotag))
|
|
|
|
{
|
|
|
|
t[0] = 1;
|
|
|
|
t[1] = 1;
|
|
|
|
t[3] = !t[3];
|
|
|
|
operatemasterswitches(s->lotag);
|
|
|
|
operateactivators(s->lotag, p);
|
|
|
|
if (s->hitag > 0)
|
|
|
|
{
|
|
|
|
s->hitag--;
|
|
|
|
if (s->hitag == 0) t[5] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else t[0] = 0;
|
|
|
|
|
|
|
|
if (t[1] == 1)
|
|
|
|
{
|
|
|
|
int j = headspritestat[STAT_STANDABLE];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (j != i && sprite[j].picnum == plate && sprite[j].lotag == s->lotag)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
hittype[j].temp_data[1] = 1;
|
|
|
|
hittype[j].temp_data[3] = t[3];
|
|
|
|
}
|
|
|
|
j = nextspritestat[j];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void moveooz(int i, int seenine, int seeninedead, int ooz, int explosion)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
2020-05-07 20:30:19 +00:00
|
|
|
auto t = &hittype[i].temp_data[0];
|
|
|
|
int j;
|
|
|
|
if (s->shade != -32 && s->shade != -33)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->xrepeat)
|
|
|
|
j = (ifhitbyweapon(i) >= 0);
|
|
|
|
else
|
|
|
|
j = 0;
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (j || s->shade == -31)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (j) s->lotag = 0;
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
t[3] = 1;
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
j = headspritestat[STAT_STANDABLE];
|
|
|
|
while (j >= 0)
|
|
|
|
{
|
|
|
|
if (s->hitag == sprite[j].hitag && (sprite[j].picnum == seenine || sprite[j].picnum == ooz))
|
|
|
|
sprite[j].shade = -32;
|
|
|
|
j = nextspritestat[j];
|
|
|
|
}
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (s->shade == -32)
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->lotag > 0)
|
|
|
|
{
|
|
|
|
s->lotag -= 3;
|
|
|
|
if (s->lotag <= 0) s->lotag = -99;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
s->shade = -33;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
else
|
2020-05-07 07:49:05 +00:00
|
|
|
{
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->xrepeat > 0)
|
|
|
|
{
|
|
|
|
hittype[i].temp_data[2]++;
|
|
|
|
if (hittype[i].temp_data[2] == 3)
|
|
|
|
{
|
|
|
|
if (s->picnum == ooz)
|
|
|
|
{
|
|
|
|
hittype[i].temp_data[2] = 0;
|
|
|
|
detonate(i, explosion);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (s->picnum != (seeninedead + 1))
|
|
|
|
{
|
|
|
|
hittype[i].temp_data[2] = 0;
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->picnum == seeninedead) s->picnum++;
|
|
|
|
else if (s->picnum == seenine)
|
|
|
|
s->picnum = seeninedead;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
detonate(i, explosion);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
detonate(i, explosion);
|
|
|
|
return;
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
2020-05-07 20:30:19 +00:00
|
|
|
}
|
|
|
|
}
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movecanwithsomething(int i)
|
|
|
|
{
|
|
|
|
auto s = &sprite[i];
|
|
|
|
makeitfall(i);
|
|
|
|
int j = ifhitbyweapon(i);
|
|
|
|
if (j >= 0)
|
|
|
|
{
|
|
|
|
spritesound(VENT_BUST, i);
|
|
|
|
for (j = 0; j < 10; j++)
|
|
|
|
RANDOMSCRAP(s, i);
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
if (s->lotag) spawn(i, s->lotag);
|
2020-05-07 07:49:05 +00:00
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
deletesprite(i);
|
2020-05-07 07:49:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-07 12:38:01 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void bounce(int i)
|
2020-05-07 12:38:01 +00:00
|
|
|
{
|
|
|
|
int k, l, daang, dax, day, daz, xvect, yvect, zvect;
|
|
|
|
short hitsect;
|
|
|
|
spritetype* s = &sprite[i];
|
|
|
|
|
|
|
|
xvect = mulscale10(s->xvel, sintable[(s->ang + 512) & 2047]);
|
|
|
|
yvect = mulscale10(s->xvel, sintable[s->ang & 2047]);
|
|
|
|
zvect = s->zvel;
|
|
|
|
|
|
|
|
hitsect = s->sectnum;
|
|
|
|
|
|
|
|
k = sector[hitsect].wallptr; l = wall[k].point2;
|
|
|
|
daang = getangle(wall[l].x - wall[k].x, wall[l].y - wall[k].y);
|
|
|
|
|
|
|
|
if (s->z < (hittype[i].floorz + hittype[i].ceilingz) >> 1)
|
|
|
|
k = sector[hitsect].ceilingheinum;
|
|
|
|
else
|
|
|
|
k = sector[hitsect].floorheinum;
|
|
|
|
|
|
|
|
dax = mulscale14(k, sintable[(daang) & 2047]);
|
|
|
|
day = mulscale14(k, sintable[(daang + 1536) & 2047]);
|
|
|
|
daz = 4096;
|
|
|
|
|
|
|
|
k = xvect * dax + yvect * day + zvect * daz;
|
|
|
|
l = dax * dax + day * day + daz * daz;
|
|
|
|
if ((abs(k) >> 14) < l)
|
|
|
|
{
|
|
|
|
k = divscale17(k, l);
|
|
|
|
xvect -= mulscale16(dax, k);
|
|
|
|
yvect -= mulscale16(day, k);
|
|
|
|
zvect -= mulscale16(daz, k);
|
|
|
|
}
|
|
|
|
|
|
|
|
s->zvel = zvect;
|
|
|
|
s->xvel = ksqrt(dmulscale8(xvect, xvect, yvect, yvect));
|
|
|
|
s->ang = getangle(xvect, yvect);
|
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
2020-05-07 20:30:19 +00:00
|
|
|
// taken out of moveweapon
|
2020-05-07 12:38:01 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-05-07 20:30:19 +00:00
|
|
|
void movetongue(int i, int tongue, int jaw)
|
2020-05-07 12:38:01 +00:00
|
|
|
{
|
|
|
|
spritetype* s = &sprite[i];
|
|
|
|
|
|
|
|
hittype[i].temp_data[0] = sintable[(hittype[i].temp_data[1]) & 2047] >> 9;
|
|
|
|
hittype[i].temp_data[1] += 32;
|
|
|
|
if (hittype[i].temp_data[1] > 2047)
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sprite[s->owner].statnum == MAXSTATUS)
|
|
|
|
if (badguy(&sprite[s->owner]) == 0)
|
|
|
|
{
|
|
|
|
deletesprite(i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
s->ang = sprite[s->owner].ang;
|
|
|
|
s->x = sprite[s->owner].x;
|
|
|
|
s->y = sprite[s->owner].y;
|
|
|
|
if (sprite[s->owner].picnum == APLAYER)
|
|
|
|
s->z = sprite[s->owner].z - (34 << 8);
|
|
|
|
for (int k = 0; k < hittype[i].temp_data[0]; k++)
|
|
|
|
{
|
|
|
|
int q = EGS(s->sectnum,
|
|
|
|
s->x + ((k * sintable[(s->ang + 512) & 2047]) >> 9),
|
|
|
|
s->y + ((k * sintable[s->ang & 2047]) >> 9),
|
2020-05-07 20:30:19 +00:00
|
|
|
s->z + ((k * ksgn(s->zvel)) * abs(s->zvel / 12)), tongue, -40 + (k << 1),
|
2020-05-07 12:38:01 +00:00
|
|
|
8, 8, 0, 0, 0, i, 5);
|
|
|
|
sprite[q].cstat = 128;
|
|
|
|
sprite[q].pal = 8;
|
|
|
|
}
|
|
|
|
int k = hittype[i].temp_data[0]; // do not depend on the above loop counter.
|
|
|
|
int q = EGS(s->sectnum,
|
|
|
|
s->x + ((k * sintable[(s->ang + 512) & 2047]) >> 9),
|
|
|
|
s->y + ((k * sintable[s->ang & 2047]) >> 9),
|
2020-05-07 20:30:19 +00:00
|
|
|
s->z + ((k * ksgn(s->zvel)) * abs(s->zvel / 12)), jaw, -40,
|
2020-05-07 12:38:01 +00:00
|
|
|
32, 32, 0, 0, 0, i, 5);
|
|
|
|
sprite[q].cstat = 128;
|
|
|
|
if (hittype[i].temp_data[1] > 512 && hittype[i].temp_data[1] < (1024))
|
2020-05-07 20:30:19 +00:00
|
|
|
sprite[q].picnum = jaw + 1;
|
2020-05-07 12:38:01 +00:00
|
|
|
}
|
|
|
|
|
2020-05-06 19:11:36 +00:00
|
|
|
|
2020-05-05 09:58:39 +00:00
|
|
|
|
|
|
|
END_DUKE_NS
|