2020-07-06 20:23:18 +00:00
|
|
|
#pragma once
|
|
|
|
|
2021-04-11 06:40:18 +00:00
|
|
|
#include "gamehud.h"
|
2020-07-19 20:34:59 +00:00
|
|
|
#include "global.h"
|
2022-01-21 00:04:08 +00:00
|
|
|
#include "models/modeldata.h"
|
2020-07-06 20:23:18 +00:00
|
|
|
|
|
|
|
// all inline functions.
|
|
|
|
BEGIN_DUKE_NS
|
|
|
|
|
2020-07-19 17:31:31 +00:00
|
|
|
inline int rnd(int X)
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
return ((krand() >> 8) >= (255 - (X)));
|
2020-07-19 17:31:31 +00:00
|
|
|
}
|
|
|
|
|
2022-11-19 08:02:40 +00:00
|
|
|
// internal controller classes can be directly checked for by type
|
|
|
|
inline int ismasterswitch(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
// The STAT_REMOVED check here is important!
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeMasterSwitch && actor->spr.statnum != STAT_REMOVED;
|
|
|
|
}
|
|
|
|
|
2022-11-20 07:10:15 +00:00
|
|
|
inline int issoundcontroller(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeSoundController;
|
|
|
|
}
|
|
|
|
|
2022-11-20 10:58:19 +00:00
|
|
|
inline int isrespawncontroller(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeRespawnController;
|
|
|
|
}
|
|
|
|
|
2022-11-20 11:43:24 +00:00
|
|
|
inline int isactivator(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeActivator;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int islockedactivator(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeActivatorLocked;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int islocator(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->GetClass()->TypeName == NAME_DukeLocator;
|
|
|
|
}
|
|
|
|
|
2022-12-01 19:54:41 +00:00
|
|
|
inline int iseffector(DDukeActor* actor)
|
|
|
|
{
|
|
|
|
return actor->spr.picnum == SECTOREFFECTOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-06 20:23:18 +00:00
|
|
|
inline int badguypic(int const tileNum)
|
|
|
|
{
|
2020-11-29 12:54:58 +00:00
|
|
|
return ((gs.actorinfo[tileNum].flags & (SFLAG_INTERNAL_BADGUY | SFLAG_BADGUY)) != 0);
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2020-09-07 19:38:17 +00:00
|
|
|
inline int bossguypic(int const tileNum)
|
|
|
|
{
|
2020-11-29 12:54:58 +00:00
|
|
|
return ((gs.actorinfo[tileNum].flags & (SFLAG_BOSS)) != 0);
|
2020-09-07 19:38:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-02 09:17:44 +00:00
|
|
|
inline int actorflag(DDukeActor * actor, EDukeFlags1 mask)
|
2020-07-06 16:08:31 +00:00
|
|
|
{
|
2021-12-21 17:19:45 +00:00
|
|
|
return (((gs.actorinfo[actor->spr.picnum].flags) & mask) != 0);
|
2020-07-06 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
2022-01-02 09:17:44 +00:00
|
|
|
inline int actorflag(DDukeActor* actor, EDukeFlags2 mask)
|
|
|
|
{
|
|
|
|
return (((gs.actorinfo[actor->spr.picnum].flags2) & mask) != 0);
|
|
|
|
}
|
2022-11-30 18:39:06 +00:00
|
|
|
|
|
|
|
inline int actorflag(DDukeActor* actor, EDukeFlags3 mask)
|
|
|
|
{
|
|
|
|
return (((gs.actorinfo[actor->spr.picnum].flags3) & mask) != 0);
|
|
|
|
}
|
2022-01-02 09:17:44 +00:00
|
|
|
|
2022-01-26 23:41:33 +00:00
|
|
|
inline int attackerflag(DDukeActor* actor, EDukeFlags1 mask)
|
|
|
|
{
|
|
|
|
return (((gs.actorinfo[actor->attackertype].flags) & mask) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int attackerflag(DDukeActor* actor, EDukeFlags2 mask)
|
|
|
|
{
|
|
|
|
return (((gs.actorinfo[actor->attackertype].flags2) & mask) != 0);
|
|
|
|
}
|
|
|
|
|
2020-10-23 17:02:58 +00:00
|
|
|
inline int actorfella(DDukeActor* actor)
|
2020-07-06 16:08:31 +00:00
|
|
|
{
|
2020-10-23 17:02:58 +00:00
|
|
|
return actorflag(actor, SFLAG_KILLCOUNT);
|
2020-07-06 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
2022-01-02 09:17:44 +00:00
|
|
|
inline void setflag(EDukeFlags1 flag, const std::initializer_list<short>& types)
|
2020-07-06 16:08:31 +00:00
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
for (auto val : types)
|
|
|
|
{
|
2020-11-29 12:54:58 +00:00
|
|
|
gs.actorinfo[val].flags |= flag;
|
2020-07-20 21:21:27 +00:00
|
|
|
}
|
2020-07-06 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
2022-01-02 09:17:44 +00:00
|
|
|
inline void setflag(EDukeFlags2 flag, const std::initializer_list<short>& types)
|
|
|
|
{
|
|
|
|
for (auto val : types)
|
|
|
|
{
|
|
|
|
gs.actorinfo[val].flags2 |= flag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-01 19:11:38 +00:00
|
|
|
inline void setflag(EDukeFlags3 flag, const std::initializer_list<short>& types)
|
|
|
|
{
|
|
|
|
for (auto val : types)
|
|
|
|
{
|
|
|
|
gs.actorinfo[val].flags3 |= flag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-21 21:31:25 +00:00
|
|
|
inline bool inventory(DDukeActor* S)
|
2020-07-06 16:08:31 +00:00
|
|
|
{
|
2022-01-17 23:54:10 +00:00
|
|
|
return actorflag(S, SFLAG_INVENTORY);
|
2020-07-06 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
2022-11-25 16:29:27 +00:00
|
|
|
inline int& tileflags(unsigned int tilenum)
|
2022-11-25 12:45:18 +00:00
|
|
|
{
|
2022-11-25 16:29:27 +00:00
|
|
|
static int sink = 0;
|
|
|
|
if (tilenum >= MAXTILES) return sink;
|
2022-11-25 12:45:18 +00:00
|
|
|
return TileFiles.tiledata[tilenum].tileflags;
|
|
|
|
}
|
2020-07-06 16:08:31 +00:00
|
|
|
|
2020-11-02 22:53:55 +00:00
|
|
|
inline bool wallswitchcheck(DDukeActor* s)
|
2020-07-06 16:08:31 +00:00
|
|
|
{
|
2022-11-25 12:45:18 +00:00
|
|
|
return !!(tileflags(s->spr.picnum) & TFLAG_WALLSWITCH);
|
2020-07-06 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
2022-11-25 16:29:27 +00:00
|
|
|
inline bool isadoorwall(int dapic)
|
|
|
|
{
|
|
|
|
return tileflags(dapic) & TFLAG_DOORWALL;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool isablockdoor(int dapic)
|
|
|
|
{
|
|
|
|
return tileflags(dapic) & TFLAG_BLOCKDOOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-11-18 20:02:59 +00:00
|
|
|
inline int checkcursectnums(sectortype* se)
|
2020-07-06 20:23:18 +00:00
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
int i;
|
|
|
|
for(i=connecthead;i>=0;i=connectpoint2[i])
|
2021-12-30 15:51:56 +00:00
|
|
|
if(ps[i].GetActor() && ps[i].GetActor()->sector() == se ) return i;
|
2020-07-20 21:21:27 +00:00
|
|
|
return -1;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool isIn(int value, int first)
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
return value == first;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
bool isIn(int value, int first, Args... args)
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
return value == first || isIn(value, args...);
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool isIn(int value, const std::initializer_list<int>& list)
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
for (auto v : list) if (v == value) return true;
|
|
|
|
return false;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// these are mainly here to avoid directly accessing the input data so that it can be more easily refactored later.
|
2020-08-27 20:19:24 +00:00
|
|
|
inline bool PlayerInput(int pl, ESyncBits bit)
|
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
return (!!((ps[pl].sync.actions) & bit));
|
2020-08-27 20:19:24 +00:00
|
|
|
}
|
|
|
|
|
2020-08-26 21:02:55 +00:00
|
|
|
inline ESyncBits PlayerInputBits(int pl, ESyncBits bits)
|
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
return (ps[pl].sync.actions & bits);
|
2020-08-26 21:02:55 +00:00
|
|
|
}
|
|
|
|
|
2020-08-27 22:03:35 +00:00
|
|
|
inline void PlayerSetInput(int pl, ESyncBits bit)
|
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
ps[pl].sync.actions |= bit;
|
2020-08-27 22:03:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-08-26 21:02:55 +00:00
|
|
|
inline int PlayerNewWeapon(int pl)
|
|
|
|
{
|
2021-07-17 12:42:58 +00:00
|
|
|
return ps[pl].sync.getNewWeapon();
|
2020-08-26 21:02:55 +00:00
|
|
|
}
|
|
|
|
|
2020-08-27 19:25:09 +00:00
|
|
|
inline void PlayerSetItemUsed(int pl, int num)
|
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
ps[pl].sync.setItemUsed(num - 1);
|
2020-08-27 19:25:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool PlayerUseItem(int pl, int num)
|
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
return ps[pl].sync.isItemUsed(num - 1);
|
2020-08-27 19:25:09 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 23:54:01 +00:00
|
|
|
inline float PlayerInputSideVel(int pl)
|
2020-07-06 20:23:18 +00:00
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
return ps[pl].sync.svel;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2022-09-19 23:54:01 +00:00
|
|
|
inline float PlayerInputForwardVel(int pl)
|
2020-07-06 20:23:18 +00:00
|
|
|
{
|
2020-09-23 14:56:04 +00:00
|
|
|
return ps[pl].sync.fvel;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2020-10-08 03:47:30 +00:00
|
|
|
inline float PlayerInputAngVel(int pl)
|
2020-07-06 20:23:18 +00:00
|
|
|
{
|
2020-10-08 03:47:30 +00:00
|
|
|
return ps[pl].sync.avel;
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2022-12-09 07:09:59 +00:00
|
|
|
inline DAngle GetPlayerHorizon(int pl)
|
2020-08-30 11:04:07 +00:00
|
|
|
{
|
2022-12-09 07:09:59 +00:00
|
|
|
return DAngle::fromDeg(ps[pl].sync.horz);
|
2020-08-30 11:04:07 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 19:10:20 +00:00
|
|
|
inline void clearfriction()
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
for (int i = 0; i != -1; i = connectpoint2[i])
|
|
|
|
{
|
2021-12-22 09:28:51 +00:00
|
|
|
ps[i].fric.X = ps[i].fric.Y = 0;
|
2020-07-20 21:21:27 +00:00
|
|
|
}
|
2020-07-06 19:10:20 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 21:24:35 +00:00
|
|
|
inline void SetPlayerPal(player_struct* p, PalEntry pe)
|
|
|
|
{
|
2020-07-20 21:21:27 +00:00
|
|
|
p->pals = pe;
|
2020-07-06 21:24:35 +00:00
|
|
|
}
|
2020-07-06 19:10:20 +00:00
|
|
|
|
2020-07-21 20:46:26 +00:00
|
|
|
inline bool playrunning()
|
|
|
|
{
|
|
|
|
return (paused == 0 || (paused == 1 && (ud.recstat == 2 || ud.multimode > 1)));
|
|
|
|
}
|
|
|
|
|
2022-11-25 10:18:36 +00:00
|
|
|
inline void doslopetilting(player_struct* p)
|
2021-01-01 13:30:01 +00:00
|
|
|
{
|
2021-12-18 12:14:56 +00:00
|
|
|
bool const canslopetilt = p->on_ground && p->insector() && p->cursector->lotag != ST_2_UNDERWATER && (p->cursector->floorstat & CSTAT_SECTOR_SLOPE);
|
2022-11-25 11:30:40 +00:00
|
|
|
p->Angles.doViewPitch(p->GetActor()->spr.pos.XY(), p->GetActor()->spr.Angles.Yaw, p->aim_mode == 0, canslopetilt, p->cursector);
|
2021-01-01 13:30:01 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 20:23:18 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2020-07-16 11:23:26 +00:00
|
|
|
inline void hud_draw(double x, double y, int tilenum, int shade, int orientation)
|
2020-07-06 20:23:18 +00:00
|
|
|
{
|
2021-11-21 07:56:39 +00:00
|
|
|
int p = ps[screenpeek].cursector->floorpal;
|
2020-07-27 08:39:33 +00:00
|
|
|
hud_drawsprite(x, y, 65536, 0, tilenum, shade, p, 2 | orientation);
|
2020-07-06 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2022-09-07 05:34:55 +00:00
|
|
|
inline void animateshrunken(player_struct* p, double weapon_xoffset, double looking_arc, double look_anghalf, int tilenum, int8_t shade, int o, double interpfrac)
|
2022-06-06 01:04:51 +00:00
|
|
|
{
|
2022-09-15 16:41:01 +00:00
|
|
|
const double fistsign = BobVal(interpolatedvalue<double>(p->ofistsign, p->fistsign, interpfrac)) * 16;
|
|
|
|
if (p->jetpack_on == 0) looking_arc += 32 - (p->GetActor()->vel.X * 8);
|
2022-06-06 01:04:51 +00:00
|
|
|
hud_draw(weapon_xoffset + fistsign + 250 - look_anghalf, looking_arc + 258 - fabs(fistsign * 4), tilenum, shade, o);
|
|
|
|
hud_draw(weapon_xoffset - fistsign + 40 - look_anghalf, looking_arc + 200 + fabs(fistsign * 4), tilenum, shade, o | 4);
|
|
|
|
}
|
|
|
|
|
2021-12-18 17:07:41 +00:00
|
|
|
inline ESpriteFlags randomFlip()
|
|
|
|
{
|
|
|
|
int r = krand() & 12;
|
|
|
|
if (r == 0) return 0;
|
|
|
|
if (r == 4) return CSTAT_SPRITE_XFLIP;
|
|
|
|
if (r == 8) return CSTAT_SPRITE_YFLIP;
|
|
|
|
return CSTAT_SPRITE_XFLIP | CSTAT_SPRITE_YFLIP;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ESpriteFlags randomXFlip()
|
|
|
|
{
|
|
|
|
int r = krand() & 4;
|
|
|
|
if (r == 0) return 0;
|
|
|
|
return CSTAT_SPRITE_XFLIP;
|
|
|
|
}
|
2021-11-18 17:35:59 +00:00
|
|
|
|
2022-09-11 20:57:56 +00:00
|
|
|
inline DAngle randomAngle(double span = 360.)
|
2022-09-11 17:05:14 +00:00
|
|
|
{
|
|
|
|
return DAngle::fromDeg(krandf(span));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline DAngle randomAngle(DAngle span)
|
|
|
|
{
|
|
|
|
return DAngle::fromDeg(krandf(span.Degrees()));
|
|
|
|
}
|
|
|
|
|
2022-09-13 17:52:59 +00:00
|
|
|
inline int angletorotation1(DAngle sprang, DAngle viewang, int shiftv = 8, int andv = 7)
|
|
|
|
{
|
|
|
|
return (((sprang.Buildang() + 3072 + 128 - viewang.Buildang()) & 2047) >> shiftv) & andv;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int angletorotation2(DAngle sprang, DAngle viewang)
|
|
|
|
{
|
|
|
|
return ((sprang.Buildang() + 3072 + 128 - viewang.Buildang()) & 2047) / 170;
|
|
|
|
}
|
|
|
|
|
2022-11-27 20:00:22 +00:00
|
|
|
// 4 (8) frame rotation.
|
2022-01-21 00:04:08 +00:00
|
|
|
inline void applyRotation1(DDukeActor* h, tspritetype* t, DAngle viewang)
|
|
|
|
{
|
|
|
|
if (hw_models && modelManager.CheckModel(h->spr.picnum, h->spr.pal))
|
|
|
|
{
|
|
|
|
t->cstat &= ~CSTAT_SPRITE_XFLIP;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-25 12:13:50 +00:00
|
|
|
int k = angletorotation1(t->Angles.Yaw, viewang);
|
2022-01-21 00:04:08 +00:00
|
|
|
if (k > 4)
|
|
|
|
{
|
|
|
|
k = 8 - k;
|
|
|
|
t->cstat |= CSTAT_SPRITE_XFLIP;
|
|
|
|
}
|
|
|
|
else t->cstat &= ~CSTAT_SPRITE_XFLIP;
|
|
|
|
t->picnum = h->spr.picnum + k;
|
|
|
|
}
|
|
|
|
|
2022-11-27 20:00:22 +00:00
|
|
|
// 6 (12) frame rotation.
|
|
|
|
inline void applyRotation2(DDukeActor* h, tspritetype* t, DAngle viewang)
|
|
|
|
{
|
|
|
|
if (hw_models && modelManager.CheckModel(h->spr.picnum, h->spr.pal))
|
|
|
|
{
|
|
|
|
t->cstat &= ~CSTAT_SPRITE_XFLIP;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int k = angletorotation2(t->Angles.Yaw, viewang);
|
|
|
|
if (k > 6)
|
|
|
|
{
|
|
|
|
k = 12 - k;
|
|
|
|
t->cstat |= CSTAT_SPRITE_XFLIP;
|
|
|
|
}
|
|
|
|
else t->cstat &= ~CSTAT_SPRITE_XFLIP;
|
|
|
|
t->picnum = h->spr.picnum + k;
|
|
|
|
}
|
|
|
|
|
2022-11-28 17:57:50 +00:00
|
|
|
inline int monsterCheatCheck(DDukeActor* self)
|
|
|
|
{
|
|
|
|
if (ud.multimode < 2)
|
|
|
|
{
|
|
|
|
if (actor_tog == 1)
|
|
|
|
{
|
|
|
|
self->spr.cstat = CSTAT_SPRITE_INVISIBLE;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (actor_tog == 2) self->spr.cstat = CSTAT_SPRITE_BLOCK_ALL;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-12-06 08:09:24 +00:00
|
|
|
inline void processinputvel(int snum)
|
|
|
|
{
|
|
|
|
const auto p = &ps[snum];
|
|
|
|
const auto velvect = DVector2(p->sync.fvel, p->sync.svel).Rotated(p->GetActor()->spr.Angles.Yaw) + p->fric;
|
|
|
|
p->sync.fvel = (float)velvect.X;
|
|
|
|
p->sync.svel = (float)velvect.Y;
|
|
|
|
}
|
|
|
|
|
2020-07-06 20:23:18 +00:00
|
|
|
END_DUKE_NS
|