mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-07 17:00:43 +00:00
66ecb05eb8
All uses of OSD_Printf, initprintf etc. have been renamed.
3145 lines
106 KiB
C++
3145 lines
106 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 2010-2019 EDuke32 developers and contributors
|
|
Copyright (C) 2019 sirlemonhead, Nuke.YKT
|
|
This file is part of PCExhumed.
|
|
PCExhumed is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License version 2
|
|
as published by the Free Software Foundation.
|
|
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
#include "ns.h"
|
|
#include "compat.h"
|
|
#include "player.h"
|
|
#include "runlist.h"
|
|
#include "exhumed.h"
|
|
#include "names.h"
|
|
#include "gun.h"
|
|
#include "items.h"
|
|
#include "engine.h"
|
|
#include "move.h"
|
|
#include "sequence.h"
|
|
#include "lighting.h"
|
|
#include "view.h"
|
|
#include "bubbles.h"
|
|
#include "random.h"
|
|
#include "ra.h"
|
|
#include "ps_input.h"
|
|
#include "light.h"
|
|
#include "status.h"
|
|
#include "sound.h"
|
|
#include "init.h"
|
|
#include "move.h"
|
|
#include "trigdat.h"
|
|
#include "anims.h"
|
|
#include "grenade.h"
|
|
#include "menu.h"
|
|
#include "cd.h"
|
|
#include "map.h"
|
|
#include "sound.h"
|
|
#include "textures.h"
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
BEGIN_PS_NS
|
|
|
|
struct PlayerSave
|
|
{
|
|
int x;
|
|
int y;
|
|
int z;
|
|
short nSector;
|
|
short nAngle;
|
|
};
|
|
|
|
fix16_t lPlayerXVel = 0;
|
|
fix16_t lPlayerYVel = 0;
|
|
fix16_t nPlayerDAng = 0;
|
|
short obobangle = 0, bobangle = 0;
|
|
short bPlayerPan = 0;
|
|
short bLockPan = 0;
|
|
bool g_MyAimMode;
|
|
|
|
static actionSeq ActionSeq[] = {
|
|
{18, 0}, {0, 0}, {9, 0}, {27, 0}, {63, 0},
|
|
{72, 0}, {54, 0}, {45, 0}, {54, 0}, {81, 0},
|
|
{90, 0}, {99, 0}, {108, 0}, {8, 0}, {0, 0},
|
|
{139, 0}, {117, 1}, {119, 1}, {120, 1}, {121, 1},
|
|
{122, 1}
|
|
};
|
|
|
|
static short nHeightTemplate[] = { 0, 0, 0, 0, 0, 0, 7, 7, 7, 9, 9, 9, 9, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
|
|
short nActionEyeLevel[] = {
|
|
-14080, -14080, -14080, -14080, -14080, -14080, -8320,
|
|
-8320, -8320, -8320, -8320, -8320, -8320, -14080,
|
|
-14080, -14080, -14080, -14080, -14080, -14080, -14080
|
|
};
|
|
|
|
uint16_t nGunLotag[] = { 52, 53, 54, 55, 56, 57 };
|
|
uint16_t nGunPicnum[] = { 57, 488, 490, 491, 878, 899, 3455 };
|
|
|
|
int16_t nItemText[] = {
|
|
-1, -1, -1, -1, -1, -1, 18, 20, 19, 13, -1, 10, 1, 0, 2, -1, 3,
|
|
-1, 4, 5, 9, 6, 7, 8, -1, 11, -1, 13, 12, 14, 15, -1, 16, 17,
|
|
-1, -1, -1, 21, 22, -1, -1, -1, -1, -1, -1, 23, 24, 25, 26, 27,
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1
|
|
};
|
|
|
|
|
|
int nLocalPlayer = 0;
|
|
|
|
short nBreathTimer[kMaxPlayers];
|
|
short nPlayerSwear[kMaxPlayers];
|
|
short nPlayerPushSect[kMaxPlayers];
|
|
short nDeathType[kMaxPlayers];
|
|
short nPlayerScore[kMaxPlayers];
|
|
short nPlayerColor[kMaxPlayers];
|
|
int nPlayerDY[kMaxPlayers];
|
|
int nPlayerDX[kMaxPlayers];
|
|
char playerNames[kMaxPlayers][11];
|
|
short nPistolClip[kMaxPlayers];
|
|
int nXDamage[kMaxPlayers];
|
|
int nYDamage[kMaxPlayers];
|
|
short nDoppleSprite[kMaxPlayers];
|
|
short namelen[kMaxPlayers];
|
|
short nPlayerOldWeapon[kMaxPlayers];
|
|
short nPlayerClip[kMaxPlayers];
|
|
short nPlayerPushSound[kMaxPlayers];
|
|
short nTauntTimer[kMaxPlayers];
|
|
short nPlayerTorch[kMaxPlayers];
|
|
uint16_t nPlayerWeapons[kMaxPlayers]; // each set bit represents a weapon the player has
|
|
short nPlayerLives[kMaxPlayers];
|
|
short nPlayerItem[kMaxPlayers];
|
|
Player PlayerList[kMaxPlayers];
|
|
short nPlayerInvisible[kMaxPlayers];
|
|
short nPlayerDouble[kMaxPlayers];
|
|
short nPlayerViewSect[kMaxPlayers];
|
|
short nPlayerFloorSprite[kMaxPlayers];
|
|
PlayerSave sPlayerSave[kMaxPlayers];
|
|
int totalvel[kMaxPlayers] = { 0 };
|
|
int16_t eyelevel[kMaxPlayers], oeyelevel[kMaxPlayers];
|
|
short nNetStartSprite[kMaxPlayers] = { 0 };
|
|
|
|
short nStandHeight;
|
|
|
|
short nPlayerGrenade[kMaxPlayers];
|
|
short nGrenadePlayer[50];
|
|
|
|
short word_D282A[32];
|
|
|
|
|
|
short PlayerCount;
|
|
|
|
short nNetStartSprites;
|
|
short nCurStartSprite;
|
|
|
|
/*
|
|
typedef struct
|
|
{
|
|
fixed dx;
|
|
fixed dy;
|
|
fixed dz;
|
|
fixed dyaw;
|
|
fixed dpitch;
|
|
fixed droll;
|
|
} ControlInfo;
|
|
*/
|
|
|
|
void PlayerInterruptKeys()
|
|
{
|
|
ControlInfo info;
|
|
memset(&info, 0, sizeof(ControlInfo)); // this is done within CONTROL_GetInput() anyway
|
|
CONTROL_GetInput(&info);
|
|
D_ProcessEvents();
|
|
|
|
localInput = {};
|
|
PlayerInput input {};
|
|
|
|
if (PlayerList[nLocalPlayer].nHealth == 0)
|
|
{
|
|
lPlayerYVel = 0;
|
|
lPlayerXVel = 0;
|
|
nPlayerDAng = 0;
|
|
return;
|
|
}
|
|
|
|
// JBF: Run key behaviour is selectable
|
|
int const playerRunning = G_CheckAutorun(buttonMap.ButtonDown(gamefunc_Run));
|
|
int const turnAmount = playerRunning ? 12 : 8;
|
|
int const keyMove = playerRunning ? 12 : 6;
|
|
constexpr int const analogTurnAmount = 12;
|
|
constexpr int const analogExtent = 32767; // KEEPINSYNC sdlayer.cpp
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Strafe))
|
|
{
|
|
static int strafeyaw;
|
|
|
|
input.xVel = -(info.mousex + strafeyaw) >> 6;
|
|
strafeyaw = (info.mousex + strafeyaw) % 64;
|
|
|
|
input.xVel -= info.dyaw * keyMove / analogExtent;
|
|
}
|
|
else
|
|
{
|
|
input.nAngle = fix16_sadd(input.nAngle, fix16_sdiv(fix16_from_int(info.mousex), fix16_from_int(32)));
|
|
input.nAngle = fix16_sadd(input.nAngle, fix16_from_int(info.dyaw / analogExtent * (analogTurnAmount << 1)));
|
|
}
|
|
|
|
g_MyAimMode = in_mousemode || buttonMap.ButtonDown(gamefunc_Mouse_Aiming);
|
|
|
|
if (g_MyAimMode)
|
|
input.horizon = fix16_sadd(input.horizon, fix16_sdiv(fix16_from_int(info.mousey), fix16_from_int(64)));
|
|
else
|
|
input.yVel = -(info.mousey >> 6);
|
|
|
|
if (!in_mouseflip) input.horizon = -input.horizon;
|
|
|
|
input.horizon = fix16_ssub(input.horizon, fix16_from_int(info.dpitch * analogTurnAmount / analogExtent));
|
|
input.xVel -= info.dx * keyMove / analogExtent;
|
|
input.yVel -= info.dz * keyMove / analogExtent;
|
|
|
|
static double lastInputTicks;
|
|
auto const currentHiTicks = timerGetHiTicks();
|
|
double const elapsedInputTicks = currentHiTicks - lastInputTicks;
|
|
|
|
lastInputTicks = currentHiTicks;
|
|
|
|
auto scaleAdjustmentToInterval = [=](double x) { return x * 120 / (1000.0 / elapsedInputTicks); };
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Strafe))
|
|
{
|
|
if (buttonMap.ButtonDown(gamefunc_Turn_Left))
|
|
input.xVel -= -keyMove;
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Turn_Right))
|
|
input.xVel -= keyMove;
|
|
}
|
|
else
|
|
{
|
|
static int turn = 0;
|
|
static int counter = 0;
|
|
// normal, non strafing movement
|
|
if (buttonMap.ButtonDown(gamefunc_Turn_Left))
|
|
{
|
|
turn -= 2;
|
|
|
|
if (turn < -turnAmount)
|
|
turn = -turnAmount;
|
|
}
|
|
else if (buttonMap.ButtonDown(gamefunc_Turn_Right))
|
|
{
|
|
turn += 2;
|
|
|
|
if (turn > turnAmount)
|
|
turn = turnAmount;
|
|
}
|
|
|
|
if (turn < 0)
|
|
{
|
|
turn++;
|
|
if (turn > 0)
|
|
turn = 0;
|
|
}
|
|
|
|
if (turn > 0)
|
|
{
|
|
turn--;
|
|
if (turn < 0)
|
|
turn = 0;
|
|
}
|
|
|
|
//if ((counter++) % 4 == 0) // what was this for???
|
|
input.nAngle = fix16_sadd(input.nAngle, fix16_from_float(scaleAdjustmentToInterval(turn * 2)));
|
|
|
|
}
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Strafe_Left))
|
|
input.xVel += keyMove;
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Strafe_Right))
|
|
input.xVel += -keyMove;
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Move_Forward))
|
|
input.yVel += keyMove;
|
|
|
|
if (buttonMap.ButtonDown(gamefunc_Move_Backward))
|
|
input.yVel += -keyMove;
|
|
|
|
localInput.yVel = clamp(localInput.yVel + input.yVel, -12, 12);
|
|
localInput.xVel = clamp(localInput.xVel + input.xVel, -12, 12);
|
|
|
|
localInput.nAngle = fix16_sadd(localInput.nAngle, input.nAngle);
|
|
PlayerList[nLocalPlayer].q16angle = fix16_sadd(PlayerList[nLocalPlayer].q16angle, input.nAngle) & 0x7FFFFFF;
|
|
|
|
// A horiz diff of 128 equal 45 degrees,
|
|
// so we convert horiz to 1024 angle units
|
|
|
|
float const horizAngle = clamp(atan2f(PlayerList[nLocalPlayer].q16horiz - fix16_from_int(92), fix16_from_int(128)) * (512.f / fPI) + fix16_to_float(input.horizon), -255.f, 255.f);
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_from_int(92) + Blrintf(fix16_from_int(128) * tanf(horizAngle * (fPI / 512.f)));
|
|
|
|
// Look/aim up/down functions.
|
|
if (buttonMap.ButtonDown(gamefunc_Look_Up) || buttonMap.ButtonDown(gamefunc_Aim_Up))
|
|
{
|
|
bLockPan = kFalse;
|
|
if (PlayerList[nLocalPlayer].q16horiz < fix16_from_int(180)) {
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_sadd(PlayerList[nLocalPlayer].q16horiz, fix16_from_float(scaleAdjustmentToInterval(4)));
|
|
}
|
|
|
|
bPlayerPan = kTrue;
|
|
nDestVertPan[nLocalPlayer] = PlayerList[nLocalPlayer].q16horiz;
|
|
}
|
|
else if (buttonMap.ButtonDown(gamefunc_Look_Down) || buttonMap.ButtonDown(gamefunc_Aim_Down))
|
|
{
|
|
bLockPan = kFalse;
|
|
if (PlayerList[nLocalPlayer].q16horiz > fix16_from_int(4)) {
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_ssub(PlayerList[nLocalPlayer].q16horiz, fix16_from_float(scaleAdjustmentToInterval(4)));
|
|
}
|
|
|
|
bPlayerPan = kTrue;
|
|
nDestVertPan[nLocalPlayer] = PlayerList[nLocalPlayer].q16horiz;
|
|
}
|
|
else if (buttonMap.ButtonDown(gamefunc_Look_Straight))
|
|
{
|
|
bLockPan = kFalse;
|
|
bPlayerPan = kFalse;
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_from_int(92);
|
|
nDestVertPan[nLocalPlayer] = fix16_from_int(92);
|
|
}
|
|
|
|
// loc_1C048:
|
|
if (totalvel[nLocalPlayer] > 20) {
|
|
bPlayerPan = kFalse;
|
|
}
|
|
|
|
if (g_MyAimMode)
|
|
bLockPan = kTrue;
|
|
|
|
// loc_1C05E
|
|
fix16_t ecx = nDestVertPan[nLocalPlayer] - PlayerList[nLocalPlayer].q16horiz;
|
|
|
|
if (g_MyAimMode)
|
|
{
|
|
ecx = 0;
|
|
}
|
|
|
|
if (ecx)
|
|
{
|
|
if (ecx / 4 == 0)
|
|
{
|
|
if (ecx >= 0) {
|
|
ecx = 1;
|
|
}
|
|
else
|
|
{
|
|
ecx = -1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ecx /= 4;
|
|
|
|
if (ecx > fix16_from_int(4))
|
|
{
|
|
ecx = fix16_from_int(4);
|
|
}
|
|
else if (ecx < -fix16_from_int(4))
|
|
{
|
|
ecx = -fix16_from_int(4);
|
|
}
|
|
}
|
|
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_sadd(PlayerList[nLocalPlayer].q16horiz, ecx);
|
|
}
|
|
|
|
PlayerList[nLocalPlayer].q16horiz = fix16_clamp(PlayerList[nLocalPlayer].q16horiz, fix16_from_int(0), fix16_from_int(184));
|
|
}
|
|
|
|
void RestoreSavePoint(int nPlayer, int *x, int *y, int *z, short *nSector, short *nAngle)
|
|
{
|
|
*x = sPlayerSave[nPlayer].x;
|
|
*y = sPlayerSave[nPlayer].y;
|
|
*z = sPlayerSave[nPlayer].z;
|
|
*nSector = sPlayerSave[nPlayer].nSector;
|
|
*nAngle = sPlayerSave[nPlayer].nAngle;
|
|
}
|
|
|
|
void SetSavePoint(int nPlayer, int x, int y, int z, short nSector, short nAngle)
|
|
{
|
|
sPlayerSave[nPlayer].x = x;
|
|
sPlayerSave[nPlayer].y = y;
|
|
sPlayerSave[nPlayer].z = z;
|
|
sPlayerSave[nPlayer].nSector = nSector;
|
|
sPlayerSave[nPlayer].nAngle = nAngle;
|
|
}
|
|
|
|
void feebtag(int x, int y, int z, int nSector, short *nSprite, int nVal2, int nVal3)
|
|
{
|
|
*nSprite = -1;
|
|
|
|
int startwall = sector[nSector].wallptr;
|
|
|
|
int nWalls = sector[nSector].wallnum;
|
|
|
|
int var_20 = nVal2 & 2;
|
|
int var_14 = nVal2 & 1;
|
|
|
|
while (1)
|
|
{
|
|
if (nSector != -1)
|
|
{
|
|
short i = headspritesect[nSector];
|
|
|
|
while (i != -1)
|
|
{
|
|
short nNextSprite = nextspritesect[i];
|
|
short nStat = sprite[i].statnum;
|
|
|
|
if (nStat >= 900 && !(sprite[i].cstat & 0x8000))
|
|
{
|
|
int xDiff = sprite[i].x - x;
|
|
int yDiff = sprite[i].y - y;
|
|
int zDiff = sprite[i].z - z;
|
|
|
|
if (zDiff < 5120 && zDiff > -25600)
|
|
{
|
|
int theSqrt = ksqrt(xDiff * xDiff + yDiff * yDiff);
|
|
|
|
if (theSqrt < nVal3 && ((nStat != 950 && nStat != 949) || !(var_14 & 1)) && ((nStat != 912 && nStat != 913) || !(var_20 & 2)))
|
|
{
|
|
nVal3 = theSqrt;
|
|
*nSprite = i;
|
|
}
|
|
}
|
|
}
|
|
|
|
i = nNextSprite;
|
|
}
|
|
}
|
|
|
|
nWalls--;
|
|
if (nWalls < -1)
|
|
return;
|
|
|
|
nSector = wall[startwall].nextsector;
|
|
startwall++;
|
|
}
|
|
}
|
|
|
|
void InitPlayer()
|
|
{
|
|
for (int i = 0; i < kMaxPlayers; i++) {
|
|
PlayerList[i].nSprite = -1;
|
|
}
|
|
}
|
|
|
|
void InitPlayerKeys(short nPlayer)
|
|
{
|
|
PlayerList[nPlayer].keys = 0;
|
|
}
|
|
|
|
void InitPlayerInventory(short nPlayer)
|
|
{
|
|
memset(&PlayerList[nPlayer], 0, sizeof(Player));
|
|
|
|
nPlayerItem[nPlayer] = -1;
|
|
nPlayerSwear[nPlayer] = 4;
|
|
|
|
ResetPlayerWeapons(nPlayer);
|
|
|
|
nPlayerLives[nPlayer] = kDefaultLives;
|
|
|
|
PlayerList[nPlayer].nSprite = -1;
|
|
PlayerList[nPlayer].nRun = -1;
|
|
|
|
nPistolClip[nPlayer] = 6;
|
|
nPlayerClip[nPlayer] = 100;
|
|
|
|
PlayerList[nPlayer].nCurrentWeapon = 0;
|
|
|
|
if (nPlayer == nLocalPlayer) {
|
|
nMapMode = 0;
|
|
}
|
|
|
|
nPlayerScore[nPlayer] = 0;
|
|
|
|
auto pixels = tilePtr(kTile3571 + nPlayer);
|
|
|
|
nPlayerColor[nPlayer] = pixels[tilesiz[nPlayer + kTile3571].x * tilesiz[nPlayer + kTile3571].y / 2];
|
|
}
|
|
|
|
// done
|
|
short GetPlayerFromSprite(short nSprite)
|
|
{
|
|
return RunData[sprite[nSprite].owner].nVal;
|
|
}
|
|
|
|
void RestartPlayer(short nPlayer)
|
|
{
|
|
int nSprite = PlayerList[nPlayer].nSprite;
|
|
int nDopSprite = nDoppleSprite[nPlayer];
|
|
|
|
int floorspr;
|
|
|
|
if (nSprite > -1)
|
|
{
|
|
runlist_DoSubRunRec(sprite[nSprite].owner);
|
|
runlist_FreeRun(sprite[nSprite].lotag - 1);
|
|
|
|
changespritestat(nSprite, 0);
|
|
|
|
PlayerList[nPlayer].nSprite = -1;
|
|
|
|
int nFloorSprite = nPlayerFloorSprite[nPlayer];
|
|
if (nFloorSprite > -1) {
|
|
mydeletesprite(nFloorSprite);
|
|
}
|
|
|
|
if (nDopSprite > -1)
|
|
{
|
|
runlist_DoSubRunRec(sprite[nDopSprite].owner);
|
|
runlist_FreeRun(sprite[nDopSprite].lotag - 1);
|
|
mydeletesprite(nDopSprite);
|
|
}
|
|
}
|
|
|
|
nSprite = GrabBody();
|
|
|
|
mychangespritesect(nSprite, sPlayerSave[nPlayer].nSector);
|
|
changespritestat(nSprite, 100);
|
|
|
|
assert(nSprite >= 0 && nSprite < kMaxSprites);
|
|
|
|
int nDSprite = insertsprite(sprite[nSprite].sectnum, 100);
|
|
nDoppleSprite[nPlayer] = nDSprite;
|
|
|
|
assert(nDSprite >= 0 && nDSprite < kMaxSprites);
|
|
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
int nNStartSprite = nNetStartSprite[nCurStartSprite];
|
|
nCurStartSprite++;
|
|
|
|
if (nCurStartSprite >= nNetStartSprites) {
|
|
nCurStartSprite = 0;
|
|
}
|
|
|
|
sprite[nSprite].x = sprite[nNStartSprite].x;
|
|
sprite[nSprite].y = sprite[nNStartSprite].y;
|
|
sprite[nSprite].z = sprite[nNStartSprite].z;
|
|
mychangespritesect(nSprite, sprite[nNStartSprite].sectnum);
|
|
PlayerList[nPlayer].q16angle = fix16_from_int(sprite[nNStartSprite].ang&kAngleMask);
|
|
sprite[nSprite].ang = fix16_to_int(PlayerList[nPlayer].q16angle);
|
|
|
|
floorspr = insertsprite(sprite[nSprite].sectnum, 0);
|
|
assert(floorspr >= 0 && floorspr < kMaxSprites);
|
|
|
|
sprite[floorspr].x = sprite[nSprite].x;
|
|
sprite[floorspr].y = sprite[nSprite].y;
|
|
sprite[floorspr].z = sprite[nSprite].z;
|
|
sprite[floorspr].yrepeat = 64;
|
|
sprite[floorspr].xrepeat = 64;
|
|
sprite[floorspr].cstat = 32;
|
|
sprite[floorspr].picnum = nPlayer + kTile3571;
|
|
}
|
|
else
|
|
{
|
|
sprite[nSprite].x = sPlayerSave[nPlayer].x;
|
|
sprite[nSprite].y = sPlayerSave[nPlayer].y;
|
|
sprite[nSprite].z = sector[sPlayerSave[nPlayer].nSector].floorz;
|
|
PlayerList[nPlayer].q16angle = fix16_from_int(sPlayerSave[nPlayer].nAngle&kAngleMask);
|
|
sprite[nSprite].ang = fix16_to_int(PlayerList[nPlayer].q16angle);
|
|
|
|
floorspr = -1;
|
|
}
|
|
|
|
PlayerList[nPlayer].opos = sprite[nSprite].pos;
|
|
|
|
nPlayerFloorSprite[nPlayer] = floorspr;
|
|
|
|
sprite[nSprite].cstat = 0x101;
|
|
sprite[nSprite].shade = -12;
|
|
sprite[nSprite].clipdist = 58;
|
|
sprite[nSprite].pal = 0;
|
|
sprite[nSprite].xrepeat = 40;
|
|
sprite[nSprite].yrepeat = 40;
|
|
sprite[nSprite].xoffset = 0;
|
|
sprite[nSprite].yoffset = 0;
|
|
sprite[nSprite].picnum = seq_GetSeqPicnum(kSeqJoe, 18, 0);
|
|
|
|
int nHeight = GetSpriteHeight(nSprite);
|
|
sprite[nSprite].xvel = 0;
|
|
sprite[nSprite].yvel = 0;
|
|
sprite[nSprite].zvel = 0;
|
|
|
|
nStandHeight = nHeight;
|
|
|
|
sprite[nSprite].hitag = 0;
|
|
sprite[nSprite].extra = -1;
|
|
sprite[nSprite].lotag = runlist_HeadRun() + 1;
|
|
|
|
sprite[nDSprite].x = sprite[nSprite].x;
|
|
sprite[nDSprite].y = sprite[nSprite].y;
|
|
sprite[nDSprite].z = sprite[nSprite].z;
|
|
sprite[nDSprite].xrepeat = sprite[nSprite].xrepeat;
|
|
sprite[nDSprite].yrepeat = sprite[nSprite].yrepeat;
|
|
sprite[nDSprite].xoffset = 0;
|
|
sprite[nDSprite].yoffset = 0;
|
|
sprite[nDSprite].shade = sprite[nSprite].shade;
|
|
sprite[nDSprite].ang = sprite[nSprite].ang;
|
|
sprite[nDSprite].cstat = sprite[nSprite].cstat;
|
|
|
|
sprite[nDSprite].lotag = runlist_HeadRun() + 1;
|
|
|
|
PlayerList[nPlayer].nAction = 0;
|
|
PlayerList[nPlayer].nHealth = 800; // TODO - define
|
|
|
|
if (nNetPlayerCount) {
|
|
PlayerList[nPlayer].nHealth = 1600; // TODO - define
|
|
}
|
|
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
PlayerList[nPlayer].nSprite = nSprite;
|
|
PlayerList[nPlayer].bIsMummified = kFalse;
|
|
|
|
if (PlayerList[nPlayer].invincibility >= 0) {
|
|
PlayerList[nPlayer].invincibility = 0;
|
|
}
|
|
|
|
nPlayerTorch[nPlayer] = 0;
|
|
PlayerList[nPlayer].nMaskAmount = 0;
|
|
|
|
SetTorch(nPlayer, 0);
|
|
|
|
nPlayerInvisible[nPlayer] = 0;
|
|
|
|
PlayerList[nPlayer].bIsFiring = 0;
|
|
PlayerList[nPlayer].field_3FOUR = 0;
|
|
nPlayerViewSect[nPlayer] = sPlayerSave[nPlayer].nSector;
|
|
PlayerList[nPlayer].field_3A = 0;
|
|
|
|
nPlayerDouble[nPlayer] = 0;
|
|
|
|
PlayerList[nPlayer].nSeq = kSeqJoe;
|
|
|
|
nPlayerPushSound[nPlayer] = -1;
|
|
|
|
PlayerList[nPlayer].field_38 = -1;
|
|
|
|
if (PlayerList[nPlayer].nCurrentWeapon == 7) {
|
|
PlayerList[nPlayer].nCurrentWeapon = PlayerList[nPlayer].field_3C;
|
|
}
|
|
|
|
PlayerList[nPlayer].field_3C = 0;
|
|
PlayerList[nPlayer].nAir = 100;
|
|
airpages = 0;
|
|
|
|
if (levelnum <= kMap20)
|
|
{
|
|
RestoreMinAmmo(nPlayer);
|
|
}
|
|
else
|
|
{
|
|
ResetPlayerWeapons(nPlayer);
|
|
PlayerList[nPlayer].nMagic = 0;
|
|
}
|
|
|
|
nPlayerGrenade[nPlayer] = -1;
|
|
oeyelevel[nPlayer] = eyelevel[nPlayer] = -14080;
|
|
dVertPan[nPlayer] = 0;
|
|
|
|
nTemperature[nPlayer] = 0;
|
|
|
|
nYDamage[nPlayer] = 0;
|
|
nXDamage[nPlayer] = 0;
|
|
|
|
PlayerList[nPlayer].q16horiz = F16(92);
|
|
nDestVertPan[nPlayer] = F16(92);
|
|
nBreathTimer[nPlayer] = 90;
|
|
|
|
nTauntTimer[nPlayer] = RandomSize(3) + 3;
|
|
|
|
sprite[nDSprite].owner = runlist_AddRunRec(sprite[nDSprite].lotag - 1, nPlayer | 0xA0000);
|
|
sprite[nSprite].owner = runlist_AddRunRec(sprite[nSprite].lotag - 1, nPlayer | 0xA0000);
|
|
|
|
if (PlayerList[nPlayer].nRun < 0) {
|
|
PlayerList[nPlayer].nRun = runlist_AddRunRec(NewRun, nPlayer | 0xA0000);
|
|
}
|
|
|
|
BuildRa(nPlayer);
|
|
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
nLocalSpr = nSprite;
|
|
nPlayerDAng = 0;
|
|
|
|
SetMagicFrame();
|
|
RestoreGreenPal();
|
|
|
|
bPlayerPan = 0;
|
|
bLockPan = 0;
|
|
}
|
|
|
|
sprintf(playerNames[nPlayer], "JOE%d", nPlayer);
|
|
namelen[nPlayer] = strlen(playerNames[nPlayer]);
|
|
|
|
totalvel[nPlayer] = 0;
|
|
|
|
memset(&sPlayerInput[nPlayer], 0, sizeof(PlayerInput));
|
|
sPlayerInput[nPlayer].nItem = -1;
|
|
|
|
nDeathType[nPlayer] = 0;
|
|
nQuake[nPlayer] = 0;
|
|
|
|
if (nPlayer == nLocalPlayer) {
|
|
SetHealthFrame(0);
|
|
}
|
|
}
|
|
|
|
// done
|
|
int GrabPlayer()
|
|
{
|
|
if (PlayerCount >= kMaxPlayers) {
|
|
return -1;
|
|
}
|
|
|
|
return PlayerCount++;
|
|
}
|
|
|
|
// checked OK on 26/03/2019
|
|
void StartDeathSeq(int nPlayer, int nVal)
|
|
{
|
|
FreeRa(nPlayer);
|
|
|
|
short nSprite = PlayerList[nPlayer].nSprite;
|
|
PlayerList[nPlayer].nHealth = 0;
|
|
|
|
short nLotag = sector[sprite[nSprite].sectnum].lotag;
|
|
|
|
if (nLotag > 0) {
|
|
runlist_SignalRun(nLotag - 1, nPlayer | 0x70000);
|
|
}
|
|
|
|
if (nPlayerGrenade[nPlayer] >= 0)
|
|
{
|
|
ThrowGrenade(nPlayer, 0, 0, 0, -10000);
|
|
}
|
|
else
|
|
{
|
|
if (nNetPlayerCount)
|
|
{
|
|
int nWeapon = PlayerList[nPlayer].nCurrentWeapon;
|
|
|
|
if (nWeapon > kWeaponSword && nWeapon <= kWeaponRing)
|
|
{
|
|
short nSector = sprite[nSprite].sectnum;
|
|
if (SectBelow[nSector] > -1) {
|
|
nSector = SectBelow[nSector];
|
|
}
|
|
|
|
int nGunSprite = GrabBodyGunSprite();
|
|
changespritesect(nGunSprite, nSector);
|
|
|
|
sprite[nGunSprite].x = sprite[nSprite].x;
|
|
sprite[nGunSprite].y = sprite[nSprite].y;
|
|
sprite[nGunSprite].z = sector[nSector].floorz - 512;
|
|
|
|
changespritestat(nGunSprite, nGunLotag[nWeapon] + 900);
|
|
|
|
sprite[nGunSprite].picnum = nGunPicnum[nWeapon];
|
|
|
|
BuildItemAnim(nGunSprite);
|
|
}
|
|
}
|
|
}
|
|
|
|
StopFiringWeapon(nPlayer);
|
|
|
|
PlayerList[nPlayer].q16horiz = F16(92);
|
|
oeyelevel[nPlayer] = eyelevel[nPlayer] = -14080;
|
|
nPlayerInvisible[nPlayer] = 0;
|
|
dVertPan[nPlayer] = 15;
|
|
|
|
sprite[nSprite].cstat &= 0x7FFF;
|
|
|
|
SetNewWeaponImmediate(nPlayer, -2);
|
|
|
|
if (SectDamage[sprite[nSprite].sectnum] <= 0)
|
|
{
|
|
nDeathType[nPlayer] = nVal;
|
|
}
|
|
else
|
|
{
|
|
nDeathType[nPlayer] = 2;
|
|
}
|
|
|
|
nVal *= 2;
|
|
|
|
if (nVal || !(SectFlag[sprite[nSprite].sectnum] & kSectUnderwater))
|
|
{
|
|
PlayerList[nPlayer].nAction = nVal + 17;
|
|
}
|
|
else {
|
|
PlayerList[nPlayer].nAction = 16;
|
|
}
|
|
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
|
|
sprite[nSprite].cstat &= 0xFEFE;
|
|
|
|
if (nTotalPlayers == 1)
|
|
{
|
|
short nLives = nPlayerLives[nPlayer];
|
|
|
|
if (nLives > 0) {
|
|
BuildStatusAnim((3 * (nLives - 1)) + 7, 0);
|
|
}
|
|
|
|
if (levelnum > 0) { // if not on the training level
|
|
nPlayerLives[nPlayer]--;
|
|
}
|
|
|
|
if (nPlayerLives[nPlayer] < 0) {
|
|
nPlayerLives[nPlayer] = 0;
|
|
}
|
|
}
|
|
|
|
totalvel[nPlayer] = 0;
|
|
|
|
if (nPlayer == nLocalPlayer) {
|
|
RefreshStatus();
|
|
}
|
|
}
|
|
|
|
int AddAmmo(int nPlayer, int nWeapon, int nAmmoAmount)
|
|
{
|
|
if (!nAmmoAmount) {
|
|
nAmmoAmount = 1;
|
|
}
|
|
|
|
short nCurAmmo = PlayerList[nPlayer].nAmmo[nWeapon];
|
|
|
|
if (nCurAmmo >= 300 && nAmmoAmount > 0) {
|
|
return 0;
|
|
}
|
|
|
|
nAmmoAmount = nCurAmmo + nAmmoAmount;
|
|
if (nAmmoAmount > 300) {
|
|
nAmmoAmount = 300;
|
|
}
|
|
|
|
PlayerList[nPlayer].nAmmo[nWeapon] = nAmmoAmount;
|
|
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nWeapon == nCounterBullet) {
|
|
SetCounter(nAmmoAmount);
|
|
}
|
|
}
|
|
|
|
if (nWeapon == 1)
|
|
{
|
|
if (!nPistolClip[nPlayer]) {
|
|
nPistolClip[nPlayer] = 6;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
void SetPlayerMummified(int nPlayer, int bIsMummified)
|
|
{
|
|
int nSprite = PlayerList[nPlayer].nSprite;
|
|
|
|
sprite[nSprite].yvel = 0;
|
|
sprite[nSprite].xvel = 0;
|
|
|
|
PlayerList[nPlayer].bIsMummified = bIsMummified;
|
|
|
|
if (bIsMummified)
|
|
{
|
|
PlayerList[nPlayer].nAction = 13;
|
|
PlayerList[nPlayer].nSeq = kSeqMummy;
|
|
}
|
|
else
|
|
{
|
|
PlayerList[nPlayer].nAction = 0;
|
|
PlayerList[nPlayer].nSeq = kSeqJoe;
|
|
}
|
|
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
}
|
|
|
|
void ShootStaff(int nPlayer)
|
|
{
|
|
PlayerList[nPlayer].nAction = 15;
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
PlayerList[nPlayer].nSeq = kSeqJoe;
|
|
}
|
|
|
|
void PlayAlert(const char *str)
|
|
{
|
|
StatusMessage(300, str);
|
|
PlayLocalSound(StaticSound[kSound63], 0);
|
|
}
|
|
|
|
void DoKenTest()
|
|
{
|
|
int nPlayerSprite = PlayerList[0].nSprite;
|
|
if ((unsigned int)nPlayerSprite >= kMaxSprites)
|
|
{
|
|
return;
|
|
}
|
|
int nSector = sprite[nPlayerSprite].sectnum;
|
|
if ((unsigned int)nSector >= kMaxSectors)
|
|
{
|
|
Printf("DoKenTest: (unsigned int)nSector >= kMaxSectors\n");
|
|
return;
|
|
}
|
|
|
|
for (int i = headspritesect[nSector]; ; i = nextspritesect[i])
|
|
{
|
|
if (i == -1) {
|
|
return;
|
|
}
|
|
|
|
if (nextspritesect[i] == i) {
|
|
I_Error("ERROR in Ken's linked list!\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
void FuncPlayer(int a, int nDamage, int nRun)
|
|
{
|
|
int var_48 = 0;
|
|
int var_40;
|
|
|
|
short nPlayer = RunData[nRun].nVal;
|
|
assert(nPlayer >= 0 && nPlayer < kMaxPlayers);
|
|
|
|
if (PlayerList[nPlayer].someNetVal == -1)
|
|
return;
|
|
|
|
short nPlayerSprite = PlayerList[nPlayer].nSprite;
|
|
|
|
short nDopple = nDoppleSprite[nPlayer];
|
|
|
|
short nAction = PlayerList[nPlayer].nAction;
|
|
short nActionB = PlayerList[nPlayer].nAction;
|
|
|
|
int nMessage = a & kMessageMask;
|
|
|
|
short nSprite2;
|
|
|
|
PlayerList[nPlayer].opos = sprite[nPlayerSprite].pos;
|
|
oeyelevel[nPlayer] = eyelevel[nPlayer];
|
|
|
|
switch (nMessage)
|
|
{
|
|
case 0x90000:
|
|
{
|
|
seq_PlotSequence(a & 0xFFFF, SeqOffsets[PlayerList[nPlayer].nSeq] + ActionSeq[nAction].a, PlayerList[nPlayer].field_2, ActionSeq[nAction].b);
|
|
return;
|
|
}
|
|
|
|
case 0xA0000:
|
|
{
|
|
if (PlayerList[nPlayer].nHealth <= 0) {
|
|
return;
|
|
}
|
|
|
|
nDamage = runlist_CheckRadialDamage(nPlayerSprite);
|
|
if (!nDamage) {
|
|
return;
|
|
}
|
|
|
|
nSprite2 = nRadialOwner;
|
|
// fall through to case 0x80000
|
|
fallthrough__;
|
|
}
|
|
|
|
case 0x80000:
|
|
{
|
|
// Dunno how to do this otherwise... we fall through from above but don't want to do this check..
|
|
if (nMessage != 0xA0000)
|
|
{
|
|
if (!nDamage) {
|
|
return;
|
|
}
|
|
|
|
nSprite2 = a & 0xFFFF;
|
|
}
|
|
|
|
// ok continue case 0x80000 as normal, loc_1C57C
|
|
if (!PlayerList[nPlayer].nHealth) {
|
|
return;
|
|
}
|
|
|
|
if (!PlayerList[nPlayer].invincibility)
|
|
{
|
|
PlayerList[nPlayer].nHealth -= nDamage;
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
TintPalette(nDamage, 0, 0);
|
|
SetHealthFrame(-1);
|
|
}
|
|
}
|
|
|
|
if (PlayerList[nPlayer].nHealth > 0)
|
|
{
|
|
if (nDamage > 40 || (totalmoves & 0xF) < 2)
|
|
{
|
|
if (PlayerList[nPlayer].invincibility) {
|
|
return;
|
|
}
|
|
|
|
if (SectFlag[sprite[nPlayerSprite].sectnum] & kSectUnderwater)
|
|
{
|
|
if (nAction != 12)
|
|
{
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
PlayerList[nPlayer].nAction = 12;
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (nAction != 4)
|
|
{
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
PlayerList[nPlayer].nAction = 4;
|
|
|
|
if (nSprite2 > -1)
|
|
{
|
|
nPlayerSwear[nPlayer]--;
|
|
if (nPlayerSwear[nPlayer] <= 0)
|
|
{
|
|
D3PlayFX(StaticSound[kSound52], nDopple);
|
|
nPlayerSwear[nPlayer] = RandomSize(3) + 4;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// player has died
|
|
if (nSprite2 > -1 && sprite[nSprite2].statnum == 100)
|
|
{
|
|
short nPlayer2 = GetPlayerFromSprite(nSprite2);
|
|
|
|
if (nPlayer2 == nPlayer) // player caused their own death
|
|
{
|
|
nPlayerScore[nPlayer]--;
|
|
}
|
|
else
|
|
{
|
|
nPlayerScore[nPlayer]++;
|
|
}
|
|
}
|
|
else if (nSprite2 < 0)
|
|
{
|
|
nPlayerScore[nPlayer]--;
|
|
}
|
|
|
|
if (nMessage == 0xA0000)
|
|
{
|
|
for (int i = 122; i <= 131; i++)
|
|
{
|
|
BuildCreatureChunk(nPlayerSprite, seq_GetSeqPicnum(kSeqJoe, i, 0));
|
|
}
|
|
|
|
StartDeathSeq(nPlayer, 1);
|
|
}
|
|
else
|
|
{
|
|
StartDeathSeq(nPlayer, 0);
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
case 0x20000:
|
|
{
|
|
sprite[nPlayerSprite].xvel = sPlayerInput[nPlayer].xVel >> 14;
|
|
sprite[nPlayerSprite].yvel = sPlayerInput[nPlayer].yVel >> 14;
|
|
|
|
if (sPlayerInput[nPlayer].nItem > -1)
|
|
{
|
|
UseItem(nPlayer, sPlayerInput[nPlayer].nItem);
|
|
sPlayerInput[nPlayer].nItem = -1;
|
|
}
|
|
|
|
int var_EC = PlayerList[nPlayer].field_2;
|
|
|
|
sprite[nPlayerSprite].picnum = seq_GetSeqPicnum(PlayerList[nPlayer].nSeq, ActionSeq[nHeightTemplate[nAction]].a, var_EC);
|
|
sprite[nDopple].picnum = sprite[nPlayerSprite].picnum;
|
|
|
|
if (nPlayerTorch[nPlayer] > 0)
|
|
{
|
|
nPlayerTorch[nPlayer]--;
|
|
if (nPlayerTorch[nPlayer] == 0)
|
|
{
|
|
SetTorch(nPlayer, 0);
|
|
}
|
|
else
|
|
{
|
|
if (nPlayer != nLocalPlayer)
|
|
{
|
|
nFlashDepth = 5;
|
|
AddFlash(sprite[nPlayerSprite].sectnum,
|
|
sprite[nPlayerSprite].x,
|
|
sprite[nPlayerSprite].y,
|
|
sprite[nPlayerSprite].z, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (nPlayerDouble[nPlayer] > 0)
|
|
{
|
|
nPlayerDouble[nPlayer]--;
|
|
if (nPlayerDouble[nPlayer] == 150 && nPlayer == nLocalPlayer) {
|
|
PlayAlert("WEAPON POWER IS ABOUT TO EXPIRE");
|
|
}
|
|
}
|
|
|
|
if (nPlayerInvisible[nPlayer] > 0)
|
|
{
|
|
nPlayerInvisible[nPlayer]--;
|
|
if (nPlayerInvisible[nPlayer] == 0)
|
|
{
|
|
sprite[nPlayerSprite].cstat &= 0x7FFF; // set visible
|
|
short nFloorSprite = nPlayerFloorSprite[nPlayerSprite];
|
|
|
|
if (nFloorSprite > -1) {
|
|
sprite[nFloorSprite].cstat &= 0x7FFF; // set visible
|
|
}
|
|
}
|
|
else if (nPlayerInvisible[nPlayer] == 150 && nPlayer == nLocalPlayer)
|
|
{
|
|
PlayAlert("INVISIBILITY IS ABOUT TO EXPIRE");
|
|
}
|
|
}
|
|
|
|
if (PlayerList[nPlayer].invincibility > 0)
|
|
{
|
|
PlayerList[nPlayer].invincibility--;
|
|
if (PlayerList[nPlayer].invincibility == 150 && nPlayer == nLocalPlayer) {
|
|
PlayAlert("INVINCIBILITY IS ABOUT TO EXPIRE");
|
|
}
|
|
}
|
|
|
|
if (nQuake[nPlayer] != 0)
|
|
{
|
|
nQuake[nPlayer] = -nQuake[nPlayer];
|
|
if (nQuake[nPlayer] > 0)
|
|
{
|
|
nQuake[nPlayer] -= 512;
|
|
if (nQuake[nPlayer] < 0)
|
|
nQuake[nPlayer] = 0;
|
|
}
|
|
}
|
|
|
|
// loc_1A494:
|
|
sprite[nPlayerSprite].ang = fix16_to_int(PlayerList[nPlayer].q16angle);
|
|
|
|
// sprite[nPlayerSprite].zvel is modified within Gravity()
|
|
short zVel = sprite[nPlayerSprite].zvel;
|
|
|
|
Gravity(nPlayerSprite);
|
|
|
|
if (sprite[nPlayerSprite].zvel >= 6500 && zVel < 6500)
|
|
{
|
|
D3PlayFX(StaticSound[kSound17], 0);
|
|
}
|
|
|
|
// loc_1A4E6
|
|
short nSector = sprite[nPlayerSprite].sectnum;
|
|
short nSectFlag = SectFlag[nPlayerViewSect[nPlayer]];
|
|
|
|
int playerX = sprite[nPlayerSprite].x;
|
|
int playerY = sprite[nPlayerSprite].y;
|
|
|
|
int x = (sPlayerInput[nPlayer].xVel * 4) >> 2;
|
|
int y = (sPlayerInput[nPlayer].yVel * 4) >> 2;
|
|
int z = (sprite[nPlayerSprite].zvel * 4) >> 2;
|
|
|
|
if (sprite[nPlayerSprite].zvel > 8192)
|
|
sprite[nPlayerSprite].zvel = 8192;
|
|
|
|
if (PlayerList[nPlayer].bIsMummified)
|
|
{
|
|
x /= 2;
|
|
y /= 2;
|
|
}
|
|
|
|
int spr_x = sprite[nPlayerSprite].x;
|
|
int spr_y = sprite[nPlayerSprite].y;
|
|
int spr_z = sprite[nPlayerSprite].z;
|
|
int spr_sectnum = sprite[nPlayerSprite].sectnum;
|
|
|
|
// TODO
|
|
// nSectFlag & kSectUnderwater;
|
|
|
|
zVel = sprite[nPlayerSprite].zvel;
|
|
|
|
int nMove = 0; // TEMP
|
|
|
|
if (bSlipMode)
|
|
{
|
|
nMove = 0;
|
|
|
|
sprite[nPlayerSprite].x += (x >> 14);
|
|
sprite[nPlayerSprite].y += (y >> 14);
|
|
|
|
vec3_t pos = { sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].z };
|
|
setsprite(nPlayerSprite, &pos);
|
|
|
|
sprite[nPlayerSprite].z = sector[sprite[nPlayerSprite].sectnum].floorz;
|
|
}
|
|
else
|
|
{
|
|
nMove = movesprite(nPlayerSprite, x, y, z, 5120, -5120, CLIPMASK0);
|
|
|
|
short var_54 = sprite[nPlayerSprite].sectnum;
|
|
|
|
pushmove_old(&sprite[nPlayerSprite].x, &sprite[nPlayerSprite].y, &sprite[nPlayerSprite].z, &var_54, sprite[nPlayerSprite].clipdist << 2, 5120, -5120, CLIPMASK0);
|
|
if (var_54 != sprite[nPlayerSprite].sectnum) {
|
|
mychangespritesect(nPlayerSprite, var_54);
|
|
}
|
|
}
|
|
|
|
// loc_1A6E4
|
|
if (inside(sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].sectnum) != 1)
|
|
{
|
|
mychangespritesect(nPlayerSprite, spr_sectnum);
|
|
|
|
sprite[nPlayerSprite].x = spr_x;
|
|
sprite[nPlayerSprite].y = spr_y;
|
|
|
|
if (zVel < sprite[nPlayerSprite].zvel) {
|
|
sprite[nPlayerSprite].zvel = zVel;
|
|
}
|
|
}
|
|
|
|
// int _bTouchFloor = bTouchFloor;
|
|
short bUnderwater = SectFlag[sprite[nPlayerSprite].sectnum] & kSectUnderwater;
|
|
|
|
if (bUnderwater)
|
|
{
|
|
nXDamage[nPlayer] /= 2;
|
|
nYDamage[nPlayer] /= 2;
|
|
}
|
|
|
|
// Trigger Ramses?
|
|
if ((SectFlag[sprite[nPlayerSprite].sectnum] & 0x8000) && bTouchFloor)
|
|
{
|
|
if (nTotalPlayers <= 1)
|
|
{
|
|
PlayerList[nPlayer].q16angle = fix16_from_int(GetAngleToSprite(nPlayerSprite, nSpiritSprite) & kAngleMask);
|
|
sprite[nPlayerSprite].ang = fix16_to_int(PlayerList[nPlayer].q16angle);
|
|
|
|
PlayerList[nPlayer].q16horiz = F16(92);
|
|
|
|
lPlayerXVel = 0;
|
|
lPlayerYVel = 0;
|
|
|
|
sprite[nPlayerSprite].xvel = 0;
|
|
sprite[nPlayerSprite].yvel = 0;
|
|
sprite[nPlayerSprite].zvel = 0;
|
|
|
|
nPlayerDAng = 0;
|
|
|
|
if (nFreeze < 1)
|
|
{
|
|
nFreeze = 1;
|
|
StopAllSounds();
|
|
StopLocalSound();
|
|
InitSpiritHead();
|
|
|
|
nDestVertPan[nPlayer] = F16(92);
|
|
|
|
if (levelnum == 11)
|
|
{
|
|
nDestVertPan[nPlayer] += F16(46);
|
|
}
|
|
else
|
|
{
|
|
nDestVertPan[nPlayer] += F16(11);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FinishLevel();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (nMove & 0x3C000)
|
|
{
|
|
if (bTouchFloor)
|
|
{
|
|
// Damage stuff..
|
|
nXDamage[nPlayer] /= 2;
|
|
nYDamage[nPlayer] /= 2;
|
|
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
short zVelB = zVel;
|
|
|
|
if (zVelB < 0) {
|
|
zVelB = -zVelB;
|
|
}
|
|
|
|
if (zVelB > 512 && !bLockPan) {
|
|
nDestVertPan[nPlayer] = F16(92);
|
|
}
|
|
}
|
|
|
|
if (zVel >= 6500)
|
|
{
|
|
sprite[nPlayerSprite].xvel >>= 2;
|
|
sprite[nPlayerSprite].yvel >>= 2;
|
|
|
|
runlist_DamageEnemy(nPlayerSprite, -1, ((zVel - 6500) >> 7) + 10);
|
|
|
|
if (PlayerList[nPlayer].nHealth <= 0)
|
|
{
|
|
sprite[nPlayerSprite].xvel = 0;
|
|
sprite[nPlayerSprite].yvel = 0;
|
|
|
|
StopSpriteSound(nPlayerSprite);
|
|
PlayFXAtXYZ(StaticSound[kSoundJonFDie], sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].z, sprite[nPlayerSprite].sectnum |= 0x4000); // CHECKME
|
|
}
|
|
else
|
|
{
|
|
D3PlayFX(StaticSound[kSound27] | 0x2000, nPlayerSprite);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (((nMove & 0xC000) == 0x4000) || ((nMove & 0xC000) == 0x8000))
|
|
{
|
|
int bx = 0;
|
|
|
|
if ((nMove & 0xC000) == 0x4000)
|
|
{
|
|
bx = nMove & 0x3FFF;
|
|
}
|
|
else if ((nMove & 0xC000) == 0x8000)
|
|
{
|
|
bx = wall[nMove & 0x3FFF].nextsector;
|
|
}
|
|
|
|
if (bx >= 0)
|
|
{
|
|
int var_B4 = bx;
|
|
|
|
if ((sector[bx].hitag == 45) && bTouchFloor)
|
|
{
|
|
int nNormal = GetWallNormal(nMove & 0x3FFF);
|
|
int nDiff = AngleDiff(nNormal, (sprite[nPlayerSprite].ang + 1024) & kAngleMask);
|
|
|
|
if (nDiff < 0) {
|
|
nDiff = -nDiff;
|
|
}
|
|
|
|
if (nDiff <= 256)
|
|
{
|
|
nPlayerPushSect[nPlayer] = bx;
|
|
|
|
int var_F4 = sPlayerInput[nPlayer].xVel;
|
|
int var_F8 = sPlayerInput[nPlayer].yVel;
|
|
int nMyAngle = GetMyAngle(sPlayerInput[nPlayer].xVel, sPlayerInput[nPlayer].yVel);
|
|
|
|
MoveSector(var_B4, nMyAngle, &var_F4, &var_F8);
|
|
|
|
if (nPlayerPushSound[nPlayer] <= -1)
|
|
{
|
|
nPlayerPushSound[nPlayer] = 1;
|
|
short nBlock = sector[nPlayerPushSect[nPlayer]].extra;
|
|
int nBlockSprite = sBlockInfo[nBlock].nSprite;
|
|
|
|
D3PlayFX(StaticSound[kSound23], nBlockSprite | 0x4000);
|
|
}
|
|
else
|
|
{
|
|
sprite[nPlayerSprite].x = spr_x;
|
|
sprite[nPlayerSprite].y = spr_y;
|
|
sprite[nPlayerSprite].z = spr_z;
|
|
|
|
mychangespritesect(nPlayerSprite, spr_sectnum);
|
|
}
|
|
|
|
movesprite(nPlayerSprite, var_F4, var_F8, z, 5120, -5120, CLIPMASK0);
|
|
goto loc_1AB8E;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1AB46:
|
|
if (nPlayerPushSound[nPlayer] > -1)
|
|
{
|
|
if (nPlayerPushSect[nPlayer] > -1)
|
|
{
|
|
StopSpriteSound(sBlockInfo[sector[nPlayerPushSect[nPlayer]].extra].nSprite);
|
|
}
|
|
|
|
nPlayerPushSound[nPlayer] = -1;
|
|
}
|
|
|
|
loc_1AB8E:
|
|
if (!bPlayerPan && !bLockPan)
|
|
{
|
|
fix16_t nPanVal = fix16_from_int(spr_z - sprite[nPlayerSprite].z) / 32 + F16(92);
|
|
|
|
if (nPanVal < F16(0)) {
|
|
nPanVal = F16(0);
|
|
}
|
|
else if (nPanVal > F16(183))
|
|
{
|
|
nPanVal = F16(183);
|
|
}
|
|
|
|
nDestVertPan[nPlayer] = nPanVal;
|
|
}
|
|
|
|
playerX -= sprite[nPlayerSprite].x;
|
|
playerY -= sprite[nPlayerSprite].y;
|
|
|
|
totalvel[nPlayer] = ksqrt((playerY * playerY) + (playerX * playerX));
|
|
|
|
int nViewSect = sprite[nPlayerSprite].sectnum;
|
|
|
|
int EyeZ = eyelevel[nPlayer] + sprite[nPlayerSprite].z + nQuake[nPlayer];
|
|
|
|
while (1)
|
|
{
|
|
int nCeilZ = sector[nViewSect].ceilingz;
|
|
|
|
if (EyeZ >= nCeilZ)
|
|
break;
|
|
|
|
if (SectAbove[nViewSect] <= -1)
|
|
break;
|
|
|
|
nViewSect = SectAbove[nViewSect];
|
|
}
|
|
|
|
// Do underwater sector check
|
|
if (bUnderwater)
|
|
{
|
|
if (nViewSect != sprite[nPlayerSprite].sectnum)
|
|
{
|
|
if ((nMove & 0xC000) == 0x8000)
|
|
{
|
|
int var_C4 = sprite[nPlayerSprite].x;
|
|
int var_D4 = sprite[nPlayerSprite].y;
|
|
int var_C8 = sprite[nPlayerSprite].z;
|
|
|
|
mychangespritesect(nPlayerSprite, nViewSect);
|
|
|
|
sprite[nPlayerSprite].x = spr_x;
|
|
sprite[nPlayerSprite].y = spr_y;
|
|
|
|
int var_FC = sector[nViewSect].floorz + (-5120);
|
|
|
|
sprite[nPlayerSprite].z = var_FC;
|
|
|
|
if ((movesprite(nPlayerSprite, x, y, 0, 5120, 0, CLIPMASK0) & 0xC000) == 0x8000)
|
|
{
|
|
mychangespritesect(nPlayerSprite, sprite[nPlayerSprite].sectnum);
|
|
|
|
sprite[nPlayerSprite].x = var_C4;
|
|
sprite[nPlayerSprite].y = var_D4;
|
|
sprite[nPlayerSprite].z = var_C8;
|
|
}
|
|
else
|
|
{
|
|
sprite[nPlayerSprite].z = var_FC - 256;
|
|
D3PlayFX(StaticSound[kSound42], nPlayerSprite);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1ADAF
|
|
nPlayerViewSect[nPlayer] = nViewSect;
|
|
|
|
nPlayerDX[nPlayer] = sprite[nPlayerSprite].x - spr_x;
|
|
nPlayerDY[nPlayer] = sprite[nPlayerSprite].y - spr_y;
|
|
|
|
int var_5C = SectFlag[nViewSect] & kSectUnderwater;
|
|
|
|
uint16_t buttons = sPlayerInput[nPlayer].buttons;
|
|
|
|
if (buttons & kButtonCheatGodMode) // LOBODEITY cheat
|
|
{
|
|
char strDeity[96]; // TODO - reduce in size?
|
|
|
|
const char *strDMode = NULL;
|
|
|
|
if (PlayerList[nPlayer].invincibility >= 0)
|
|
{
|
|
PlayerList[nPlayer].invincibility = -1;
|
|
strDMode = "ON";
|
|
}
|
|
else
|
|
{
|
|
PlayerList[nPlayer].invincibility = 0;
|
|
strDMode = "OFF";
|
|
}
|
|
|
|
sPlayerInput[nPlayer].buttons &= 0xBF;
|
|
|
|
sprintf(strDeity, "Deity mode %s for player", strDMode);
|
|
StatusMessage(150, strDeity);
|
|
}
|
|
else if (buttons & kButtonCheatGuns) // LOBOCOP cheat
|
|
{
|
|
FillWeapons(nPlayer);
|
|
StatusMessage(150, "All weapons loaded for player");
|
|
}
|
|
else if (buttons & kButtonCheatKeys) // LOBOPICK cheat
|
|
{
|
|
PlayerList[nPlayer].keys = 0xFFFF;
|
|
StatusMessage(150, "All keys loaded for player");
|
|
RefreshStatus();
|
|
}
|
|
else if (buttons & kButtonCheatItems) // LOBOSWAG cheat
|
|
{
|
|
FillItems(nPlayer);
|
|
StatusMessage(150, "All items loaded for player");
|
|
}
|
|
|
|
// loc_1AEF5:
|
|
if (PlayerList[nPlayer].nHealth > 0)
|
|
{
|
|
if (PlayerList[nPlayer].nMaskAmount > 0)
|
|
{
|
|
PlayerList[nPlayer].nMaskAmount--;
|
|
if (PlayerList[nPlayer].nMaskAmount == 150 && nPlayer == nLocalPlayer) {
|
|
PlayAlert("MASK IS ABOUT TO EXPIRE");
|
|
}
|
|
}
|
|
|
|
if (!PlayerList[nPlayer].invincibility)
|
|
{
|
|
// Handle air
|
|
nBreathTimer[nPlayer]--;
|
|
|
|
if (nBreathTimer[nPlayer] <= 0)
|
|
{
|
|
nBreathTimer[nPlayer] = 90;
|
|
|
|
// if underwater
|
|
if (var_5C)
|
|
{
|
|
airpages = 1;
|
|
if (PlayerList[nPlayer].nMaskAmount > 0)
|
|
{
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(132, 0);
|
|
}
|
|
|
|
D3PlayFX(StaticSound[kSound30], nPlayerSprite);
|
|
|
|
PlayerList[nPlayer].nAir = 100;
|
|
}
|
|
else
|
|
{
|
|
PlayerList[nPlayer].nAir -= 25;
|
|
if (PlayerList[nPlayer].nAir > 0)
|
|
{
|
|
D3PlayFX(StaticSound[kSound25], nPlayerSprite);
|
|
}
|
|
else
|
|
{
|
|
PlayerList[nPlayer].nHealth += (PlayerList[nPlayer].nAir << 2);
|
|
if (PlayerList[nPlayer].nHealth <= 0)
|
|
{
|
|
PlayerList[nPlayer].nHealth = 0;
|
|
StartDeathSeq(nPlayer, 0);
|
|
}
|
|
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
SetHealthFrame(-1);
|
|
}
|
|
|
|
PlayerList[nPlayer].nAir = 0;
|
|
|
|
if (PlayerList[nPlayer].nHealth < 300)
|
|
{
|
|
D3PlayFX(StaticSound[kSound79], nPlayerSprite);
|
|
}
|
|
else
|
|
{
|
|
D3PlayFX(StaticSound[kSound19], nPlayerSprite);
|
|
}
|
|
}
|
|
}
|
|
|
|
DoBubbles(nPlayer);
|
|
SetAirFrame();
|
|
}
|
|
else
|
|
{
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
BuildStatusAnim(132, 0);
|
|
}
|
|
|
|
airpages = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1B0B9
|
|
if (var_5C) // if underwater
|
|
{
|
|
if (nPlayerTorch[nPlayer] > 0)
|
|
{
|
|
nPlayerTorch[nPlayer] = 0;
|
|
SetTorch(nPlayer, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int nTmpSectNum = sprite[nPlayerSprite].sectnum;
|
|
|
|
if (totalvel[nPlayer] > 25 && sprite[nPlayerSprite].z > sector[nTmpSectNum].floorz)
|
|
{
|
|
if (SectDepth[nTmpSectNum] && !SectSpeed[nTmpSectNum] && !SectDamage[nTmpSectNum])
|
|
{
|
|
D3PlayFX(StaticSound[kSound42], nPlayerSprite);
|
|
}
|
|
}
|
|
|
|
// CHECKME - wrong place?
|
|
if (nSectFlag & kSectUnderwater)
|
|
{
|
|
if (PlayerList[nPlayer].nAir < 50)
|
|
{
|
|
D3PlayFX(StaticSound[kSound14], nPlayerSprite);
|
|
}
|
|
|
|
nBreathTimer[nPlayer] = 1;
|
|
}
|
|
|
|
airpages = 0;
|
|
|
|
nBreathTimer[nPlayer]--;
|
|
if (nBreathTimer[nPlayer] <= 0)
|
|
{
|
|
nBreathTimer[nPlayer] = 90;
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
// animate lungs
|
|
BuildStatusAnim(132, 0);
|
|
}
|
|
}
|
|
|
|
if (PlayerList[nPlayer].nAir < 100)
|
|
{
|
|
PlayerList[nPlayer].nAir = 100;
|
|
SetAirFrame();
|
|
}
|
|
}
|
|
|
|
// loc_1B1EB
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
int nFloorSprite = nPlayerFloorSprite[nPlayer];
|
|
|
|
sprite[nFloorSprite].x = sprite[nPlayerSprite].x;
|
|
sprite[nFloorSprite].y = sprite[nPlayerSprite].y;
|
|
|
|
if (sprite[nFloorSprite].sectnum != sprite[nPlayerSprite].sectnum)
|
|
{
|
|
mychangespritesect(nFloorSprite, sprite[nPlayerSprite].sectnum);
|
|
}
|
|
|
|
sprite[nFloorSprite].z = sector[sprite[nPlayerSprite].sectnum].floorz;
|
|
}
|
|
|
|
int var_30 = 0;
|
|
|
|
if (PlayerList[nPlayer].nHealth >= 800)
|
|
{
|
|
var_30 = 2;
|
|
}
|
|
|
|
if (PlayerList[nPlayer].nMagic >= 1000)
|
|
{
|
|
var_30 |= 1;
|
|
}
|
|
|
|
// code to handle item pickup?
|
|
short nearTagSector, nearTagWall, nearTagSprite;
|
|
int nearHitDist;
|
|
|
|
short nValB;
|
|
|
|
// neartag finds the nearest sector, wall, and sprite which has its hitag and/or lotag set to a value.
|
|
neartag(sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].z, sprite[nPlayerSprite].sectnum, sprite[nPlayerSprite].ang,
|
|
&nearTagSector, &nearTagWall, &nearTagSprite, (int32_t*)&nearHitDist, 1024, 2, NULL);
|
|
|
|
feebtag(sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].z, sprite[nPlayerSprite].sectnum,
|
|
&nValB, var_30, 768);
|
|
|
|
// Item pickup code
|
|
if (nValB >= 0 && sprite[nValB].statnum >= 900)
|
|
{
|
|
int var_8C = 16;
|
|
int var_88 = 9;
|
|
|
|
int var_70 = sprite[nValB].statnum - 900;
|
|
int var_44 = 0;
|
|
|
|
// item lotags start at 6 (1-5 reserved?) so 0-offset them
|
|
int var_6C = var_70 - 6;
|
|
|
|
if (var_6C <= 54)
|
|
{
|
|
switch (var_6C)
|
|
{
|
|
do_default:
|
|
default:
|
|
{
|
|
// loc_1B3C7
|
|
|
|
// CHECKME - is order of evaluation correct?
|
|
if (levelnum <= 20 || (var_70 >= 25 && (var_70 <= 25 || var_70 == 50)))
|
|
{
|
|
DestroyItemAnim(nValB);
|
|
mydeletesprite(nValB);
|
|
}
|
|
else
|
|
{
|
|
StartRegenerate(nValB);
|
|
}
|
|
do_default_b:
|
|
// loc_1BA74
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 0: // Speed Loader
|
|
{
|
|
if (AddAmmo(nPlayer, 1, sprite[nValB].hitag))
|
|
{
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 1: // Fuel Canister
|
|
{
|
|
if (AddAmmo(nPlayer, 3, sprite[nValB].hitag))
|
|
{
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
goto do_default;
|
|
}
|
|
break;
|
|
}
|
|
case 2: // M - 60 Ammo Belt
|
|
{
|
|
if (AddAmmo(nPlayer, 2, sprite[nValB].hitag))
|
|
{
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
CheckClip(nPlayer);
|
|
goto do_default;
|
|
}
|
|
break;
|
|
}
|
|
case 3: // Grenade
|
|
case 21:
|
|
case 49:
|
|
{
|
|
if (AddAmmo(nPlayer, 4, 1))
|
|
{
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
if (!(nPlayerWeapons[nPlayer] & 0x10))
|
|
{
|
|
nPlayerWeapons[nPlayer] |= 0x10;
|
|
SetNewWeaponIfBetter(nPlayer, 4);
|
|
}
|
|
|
|
if (var_70 == 55)
|
|
{
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
goto do_default;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 4: // Pickable item
|
|
case 9: // Pickable item
|
|
case 10: // Reserved
|
|
case 18:
|
|
case 25:
|
|
case 28:
|
|
case 29:
|
|
case 30:
|
|
case 33:
|
|
case 34:
|
|
case 35:
|
|
case 36:
|
|
case 37:
|
|
case 38:
|
|
case 45:
|
|
case 52:
|
|
{
|
|
goto do_default;
|
|
}
|
|
|
|
case 5: // Map
|
|
{
|
|
GrabMap();
|
|
goto do_default;
|
|
}
|
|
|
|
case 6: // Berry Twig
|
|
{
|
|
if (sprite[nValB].hitag == 0) {
|
|
break;
|
|
}
|
|
|
|
var_88 = 20;
|
|
int edx = 40;
|
|
|
|
if (edx <= 0 || (!(var_30 & 2)))
|
|
{
|
|
if (!PlayerList[nPlayer].invincibility || edx > 0)
|
|
{
|
|
PlayerList[nPlayer].nHealth += edx;
|
|
if (PlayerList[nPlayer].nHealth > 800)
|
|
{
|
|
PlayerList[nPlayer].nHealth = 800;
|
|
}
|
|
else
|
|
{
|
|
if (PlayerList[nPlayer].nHealth < 0)
|
|
{
|
|
var_88 = -1;
|
|
StartDeathSeq(nPlayer, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
SetHealthFrame(1);
|
|
}
|
|
|
|
if (var_70 == 12)
|
|
{
|
|
sprite[nValB].hitag = 0;
|
|
sprite[nValB].picnum++;
|
|
|
|
changespritestat(nValB, 0);
|
|
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (var_70 != 14)
|
|
{
|
|
var_88 = 21;
|
|
}
|
|
else
|
|
{
|
|
var_44 = var_8C;
|
|
var_88 = 22;
|
|
var_8C = 0;
|
|
}
|
|
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 7: // Blood Bowl
|
|
{
|
|
int edx = 160;
|
|
|
|
// Same code as case 6 now till break
|
|
if (edx <= 0 || (!(var_30 & 2)))
|
|
{
|
|
if (!PlayerList[nPlayer].invincibility || edx > 0)
|
|
{
|
|
PlayerList[nPlayer].nHealth += edx;
|
|
if (PlayerList[nPlayer].nHealth > 800)
|
|
{
|
|
PlayerList[nPlayer].nHealth = 800;
|
|
}
|
|
else
|
|
{
|
|
if (PlayerList[nPlayer].nHealth < 0)
|
|
{
|
|
var_88 = -1;
|
|
StartDeathSeq(nPlayer, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
SetHealthFrame(1);
|
|
}
|
|
|
|
if (var_70 == 12)
|
|
{
|
|
sprite[nValB].hitag = 0;
|
|
sprite[nValB].picnum++;
|
|
|
|
changespritestat(nValB, 0);
|
|
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (var_70 != 14)
|
|
{
|
|
var_88 = 21;
|
|
}
|
|
else
|
|
{
|
|
var_44 = var_8C;
|
|
var_88 = 22;
|
|
var_8C = 0;
|
|
}
|
|
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 8: // Cobra Venom Bowl
|
|
{
|
|
int edx = -200;
|
|
|
|
// Same code as case 6 and 7 from now till break
|
|
if (edx <= 0 || (!(var_30 & 2)))
|
|
{
|
|
if (!PlayerList[nPlayer].invincibility || edx > 0)
|
|
{
|
|
PlayerList[nPlayer].nHealth += edx;
|
|
if (PlayerList[nPlayer].nHealth > 800)
|
|
{
|
|
PlayerList[nPlayer].nHealth = 800;
|
|
}
|
|
else
|
|
{
|
|
if (PlayerList[nPlayer].nHealth < 0)
|
|
{
|
|
var_88 = -1;
|
|
StartDeathSeq(nPlayer, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
SetHealthFrame(1);
|
|
}
|
|
|
|
if (var_70 == 12)
|
|
{
|
|
sprite[nValB].hitag = 0;
|
|
sprite[nValB].picnum++;
|
|
|
|
changespritestat(nValB, 0);
|
|
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (var_70 != 14)
|
|
{
|
|
var_88 = 21;
|
|
}
|
|
else
|
|
{
|
|
var_44 = var_8C;
|
|
var_88 = 22;
|
|
var_8C = 0;
|
|
}
|
|
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 11: // Bubble Nest
|
|
{
|
|
PlayerList[nPlayer].nAir += 10;
|
|
if (PlayerList[nPlayer].nAir > 100) {
|
|
PlayerList[nPlayer].nAir = 100; // TODO - constant
|
|
}
|
|
|
|
SetAirFrame();
|
|
|
|
if (nBreathTimer[nPlayer] < 89)
|
|
{
|
|
D3PlayFX(StaticSound[kSound13], nPlayerSprite);
|
|
}
|
|
|
|
nBreathTimer[nPlayer] = 90;
|
|
break;
|
|
}
|
|
|
|
case 12: // Still Beating Heart
|
|
{
|
|
if (GrabItem(nPlayer, kItemHeart)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 13: // Scarab amulet(Invicibility)
|
|
{
|
|
if (GrabItem(nPlayer, kItemInvincibility)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 14: // Severed Slave Hand(double damage)
|
|
{
|
|
if (GrabItem(nPlayer, kItemDoubleDamage)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 15: // Unseen eye(Invisibility)
|
|
{
|
|
if (GrabItem(nPlayer, kItemInvisibility)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 16: // Torch
|
|
{
|
|
if (GrabItem(nPlayer, kItemTorch)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 17: // Sobek Mask
|
|
{
|
|
if (GrabItem(nPlayer, kItemMask)) {
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 19: // Extra Life
|
|
{
|
|
var_88 = -1;
|
|
|
|
if (nPlayerLives[nPlayer] >= kMaxPlayerLives) {
|
|
break;
|
|
}
|
|
|
|
nPlayerLives[nPlayer]++;
|
|
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(146 + ((nPlayerLives[nPlayer] - 1) * 2), 0);
|
|
}
|
|
|
|
var_8C = 32;
|
|
var_44 = 32;
|
|
goto do_default;
|
|
}
|
|
|
|
// FIXME - lots of repeated code from here down!!
|
|
case 20: // sword pickup??
|
|
{
|
|
var_40 = 0;
|
|
int ebx = 0;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 22: // .357 Magnum Revolver
|
|
case 46:
|
|
{
|
|
var_40 = 1;
|
|
int ebx = 6;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 23: // M - 60 Machine Gun
|
|
case 47:
|
|
{
|
|
var_40 = 2;
|
|
int ebx = 24;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 24: // Flame Thrower
|
|
case 48:
|
|
{
|
|
var_40 = 3;
|
|
int ebx = 100;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 26: // Cobra Staff
|
|
case 50:
|
|
{
|
|
var_40 = 5;
|
|
int ebx = 20;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 27: // Eye of Ra Gauntlet
|
|
case 51:
|
|
{
|
|
var_40 = 6;
|
|
int ebx = 2;
|
|
|
|
// loc_1B75D
|
|
int var_18 = 1 << var_40;
|
|
|
|
short weapons = nPlayerWeapons[nPlayer];
|
|
|
|
if (weapons & var_18)
|
|
{
|
|
if (levelnum > 20)
|
|
{
|
|
AddAmmo(nPlayer, WeaponInfo[var_40].nAmmoType, ebx);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
weapons = var_40;
|
|
|
|
SetNewWeaponIfBetter(nPlayer, weapons);
|
|
|
|
nPlayerWeapons[nPlayer] |= var_18;
|
|
|
|
AddAmmo(nPlayer, WeaponInfo[weapons].nAmmoType, ebx);
|
|
|
|
var_88 = StaticSound[kSound72];
|
|
}
|
|
|
|
if (var_40 == 2) {
|
|
CheckClip(nPlayer);
|
|
}
|
|
|
|
if (var_70 <= 50) {
|
|
goto do_default;
|
|
}
|
|
|
|
sprite[nValB].cstat = 0x8000;
|
|
DestroyItemAnim(nValB);
|
|
////
|
|
// loc_1BA74: - repeated block, see in default case
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
if (nItemText[var_70] > -1 && nTotalPlayers == 1)
|
|
{
|
|
StatusMessage(400, gString[nItemTextIndex + nItemText[var_70]]);
|
|
}
|
|
|
|
TintPalette(var_44*4, var_8C*4, 0);
|
|
|
|
if (var_88 > -1)
|
|
{
|
|
PlayLocalSound(var_88, 0);
|
|
}
|
|
}
|
|
|
|
break;
|
|
/////
|
|
}
|
|
|
|
case 31: // Cobra staff ammo
|
|
{
|
|
if (AddAmmo(nPlayer, 5, 1)) {
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 32: // Raw Energy
|
|
{
|
|
if (AddAmmo(nPlayer, 6, sprite[nValB].hitag)) {
|
|
var_88 = StaticSound[kSoundAmmoPickup];
|
|
goto do_default;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
// Lots of repeated code for door key handling
|
|
case 39: // Power key
|
|
{
|
|
int ecx = 4096;
|
|
|
|
var_88 = -1;
|
|
|
|
if (PlayerList[nPlayer].keys != ecx)
|
|
{
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(36, 0);
|
|
}
|
|
|
|
PlayerList[nPlayer].keys |= ecx;
|
|
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
goto do_default_b;
|
|
}
|
|
else
|
|
{
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 40: // Time key
|
|
{
|
|
int ecx = 4096 << 1;
|
|
|
|
var_88 = -1;
|
|
|
|
if (PlayerList[nPlayer].keys != ecx)
|
|
{
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(36 + 2, 0);
|
|
}
|
|
|
|
PlayerList[nPlayer].keys |= ecx;
|
|
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
goto do_default_b;
|
|
}
|
|
else
|
|
{
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 41: // War key
|
|
{
|
|
int ecx = 4096 << 2;
|
|
|
|
var_88 = -1;
|
|
|
|
if (PlayerList[nPlayer].keys != ecx)
|
|
{
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(36 + 4, 0);
|
|
}
|
|
|
|
PlayerList[nPlayer].keys |= ecx;
|
|
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
goto do_default_b;
|
|
}
|
|
else
|
|
{
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
case 42: // Earth key
|
|
{
|
|
int ecx = 4096 << 3;
|
|
|
|
var_88 = -1;
|
|
|
|
if (PlayerList[nPlayer].keys != ecx)
|
|
{
|
|
if (nPlayer == nLocalPlayer) {
|
|
BuildStatusAnim(36 + 6, 0);
|
|
}
|
|
|
|
PlayerList[nPlayer].keys |= ecx;
|
|
|
|
if (nTotalPlayers > 1)
|
|
{
|
|
goto do_default_b;
|
|
}
|
|
else
|
|
{
|
|
goto do_default;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case 43: // Magical Essence
|
|
case 44: // ?
|
|
{
|
|
if (PlayerList[nPlayer].nMagic >= 1000) {
|
|
break;
|
|
}
|
|
|
|
var_88 = StaticSound[kSoundMana1];
|
|
|
|
PlayerList[nPlayer].nMagic += 100;
|
|
if (PlayerList[nPlayer].nMagic >= 1000) {
|
|
PlayerList[nPlayer].nMagic = 1000;
|
|
}
|
|
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
SetMagicFrame();
|
|
}
|
|
|
|
goto do_default;
|
|
}
|
|
|
|
case 53: // Scarab (Checkpoint)
|
|
{
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
short nAnim = sprite[nValB].owner;
|
|
AnimList[nAnim].nSeq++;
|
|
AnimFlags[nAnim] &= 0xEF;
|
|
AnimList[nAnim].field_2 = 0;
|
|
|
|
changespritestat(nValB, 899);
|
|
}
|
|
|
|
SetSavePoint(nPlayer, sprite[nPlayerSprite].x, sprite[nPlayerSprite].y, sprite[nPlayerSprite].z, sprite[nPlayerSprite].sectnum, sprite[nPlayerSprite].ang);
|
|
break;
|
|
}
|
|
|
|
case 54: // Golden Sarcophagus (End Level)
|
|
{
|
|
if (!bInDemo) {
|
|
FinishLevel();
|
|
}
|
|
else {
|
|
//inputState.keySetState(32, 1); // Huh, what?
|
|
}
|
|
|
|
DestroyItemAnim(nValB);
|
|
mydeletesprite(nValB);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// CORRECT ? // loc_1BAF9:
|
|
if (bTouchFloor)
|
|
{
|
|
if (sector[sprite[nPlayerSprite].sectnum].lotag > 0)
|
|
{
|
|
runlist_SignalRun(sector[sprite[nPlayerSprite].sectnum].lotag - 1, nPlayer | 0x50000);
|
|
}
|
|
}
|
|
|
|
if (nSector != sprite[nPlayerSprite].sectnum)
|
|
{
|
|
if (sector[nSector].lotag > 0)
|
|
{
|
|
runlist_SignalRun(sector[nSector].lotag - 1, nPlayer | 0x70000);
|
|
}
|
|
|
|
if (sector[sprite[nPlayerSprite].sectnum].lotag > 0)
|
|
{
|
|
runlist_SignalRun(sector[sprite[nPlayerSprite].sectnum].lotag - 1, nPlayer | 0x60000);
|
|
}
|
|
}
|
|
|
|
if (!PlayerList[nPlayer].bIsMummified)
|
|
{
|
|
if (buttons & kButtonOpen)
|
|
{
|
|
ClearSpaceBar(nPlayer);
|
|
|
|
if (nearTagWall >= 0 && wall[nearTagWall].lotag > 0)
|
|
{
|
|
runlist_SignalRun(wall[nearTagWall].lotag - 1, nPlayer | 0x40000);
|
|
}
|
|
|
|
if (nearTagSector >= 0 && sector[nearTagSector].lotag > 0)
|
|
{
|
|
runlist_SignalRun(sector[nearTagSector].lotag - 1, nPlayer | 0x40000);
|
|
}
|
|
}
|
|
|
|
// was int var_38 = buttons & 0x8
|
|
if (buttons & kButtonFire)
|
|
{
|
|
FireWeapon(nPlayer);
|
|
}
|
|
else
|
|
{
|
|
StopFiringWeapon(nPlayer);
|
|
}
|
|
|
|
// loc_1BC57:
|
|
|
|
// CHECKME - are we finished with 'nSector' variable at this point? if so, maybe set it to sprite[nPlayerSprite].sectnum so we can make this code a bit neater. Don't assume sprite[nPlayerSprite].sectnum == nSector here!!
|
|
if (nStandHeight > (sector[sprite[nPlayerSprite].sectnum].floorz - sector[sprite[nPlayerSprite].sectnum].ceilingz)) {
|
|
var_48 = 1;
|
|
}
|
|
|
|
// Jumping
|
|
if (buttons & kButtonJump)
|
|
{
|
|
if (bUnderwater)
|
|
{
|
|
sprite[nPlayerSprite].zvel = -2048;
|
|
nActionB = 10;
|
|
}
|
|
else if (bTouchFloor)
|
|
{
|
|
if (nAction < 6 || nAction > 8)
|
|
{
|
|
sprite[nPlayerSprite].zvel = -3584;
|
|
nActionB = 3;
|
|
}
|
|
}
|
|
|
|
// goto loc_1BE70:
|
|
}
|
|
else if (buttons & kButtonCrouch)
|
|
{
|
|
if (bUnderwater)
|
|
{
|
|
sprite[nPlayerSprite].zvel = 2048;
|
|
nActionB = 10;
|
|
}
|
|
else
|
|
{
|
|
if (eyelevel[nPlayer] < -8320) {
|
|
eyelevel[nPlayer] += ((-8320 - eyelevel[nPlayer]) >> 1);
|
|
}
|
|
|
|
loc_1BD2E:
|
|
if (totalvel[nPlayer] < 1) {
|
|
nActionB = 6;
|
|
}
|
|
else {
|
|
nActionB = 7;
|
|
}
|
|
}
|
|
|
|
// goto loc_1BE70:
|
|
}
|
|
else
|
|
{
|
|
if (PlayerList[nPlayer].nHealth > 0)
|
|
{
|
|
int var_EC = nActionEyeLevel[nAction];
|
|
eyelevel[nPlayer] += (var_EC - eyelevel[nPlayer]) >> 1;
|
|
|
|
if (bUnderwater)
|
|
{
|
|
if (totalvel[nPlayer] <= 1)
|
|
nActionB = 9;
|
|
else
|
|
nActionB = 10;
|
|
}
|
|
else
|
|
{
|
|
// CHECKME - confirm branching in this area is OK
|
|
if (var_48)
|
|
{
|
|
goto loc_1BD2E;
|
|
}
|
|
else
|
|
{
|
|
if (totalvel[nPlayer] <= 1) {
|
|
nActionB = 0;//bUnderwater; // this is just setting to 0
|
|
}
|
|
else if (totalvel[nPlayer] <= 30) {
|
|
nActionB = 2;
|
|
}
|
|
else
|
|
{
|
|
nActionB = 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// loc_1BE30
|
|
if (buttons & kButtonFire) // was var_38
|
|
{
|
|
if (bUnderwater)
|
|
{
|
|
nActionB = 11;
|
|
}
|
|
else
|
|
{
|
|
if (nActionB != 2 && nActionB != 1)
|
|
{
|
|
nActionB = 5;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1BE70:
|
|
// Handle player pressing number keys to change weapon
|
|
uint8_t var_90 = (buttons >> 13) & 0xF;
|
|
|
|
if (var_90)
|
|
{
|
|
var_90--;
|
|
|
|
if (nPlayerWeapons[nPlayer] & (1 << var_90))
|
|
{
|
|
SetNewWeapon(nPlayer, var_90);
|
|
}
|
|
}
|
|
}
|
|
else // player is mummified
|
|
{
|
|
if (buttons & kButtonFire)
|
|
{
|
|
FireWeapon(nPlayer);
|
|
}
|
|
|
|
if (nAction != 15)
|
|
{
|
|
if (totalvel[nPlayer] <= 1)
|
|
{
|
|
nActionB = 13;
|
|
}
|
|
else
|
|
{
|
|
nActionB = 14;
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1BF09
|
|
if (nActionB != nAction && nAction != 4)
|
|
{
|
|
nAction = nActionB;
|
|
PlayerList[nPlayer].nAction = nActionB;
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
}
|
|
}
|
|
else // else, player's health is less than 0
|
|
{
|
|
// loc_1C0E9
|
|
if (buttons & kButtonOpen)
|
|
{
|
|
ClearSpaceBar(nPlayer);
|
|
|
|
if (nAction >= 16)
|
|
{
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
StopAllSounds();
|
|
StopLocalSound();
|
|
GrabPalette();
|
|
}
|
|
|
|
PlayerList[nPlayer].nCurrentWeapon = nPlayerOldWeapon[nPlayer];
|
|
|
|
if (nPlayerLives[nPlayer] && nNetTime)
|
|
{
|
|
if (nAction != 20)
|
|
{
|
|
sprite[nPlayerSprite].picnum = seq_GetSeqPicnum(kSeqJoe, 120, 0);
|
|
sprite[nPlayerSprite].cstat = 0;
|
|
sprite[nPlayerSprite].z = sector[sprite[nPlayerSprite].sectnum].floorz;
|
|
}
|
|
|
|
// will invalidate nPlayerSprite
|
|
RestartPlayer(nPlayer);
|
|
|
|
nPlayerSprite = PlayerList[nPlayer].nSprite;
|
|
nDopple = nDoppleSprite[nPlayer];
|
|
}
|
|
else
|
|
{
|
|
if (CDplaying()) {
|
|
fadecdaudio();
|
|
}
|
|
|
|
if (levelnum == 20) {
|
|
DoFailedFinalScene();
|
|
}
|
|
else {
|
|
DoGameOverScene();
|
|
}
|
|
|
|
levelnew = 100;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1C201:
|
|
if (nLocalPlayer == nPlayer)
|
|
{
|
|
nLocalEyeSect = nPlayerViewSect[nLocalPlayer];
|
|
CheckAmbience(nLocalEyeSect);
|
|
}
|
|
|
|
int var_AC = SeqOffsets[PlayerList[nPlayer].nSeq] + ActionSeq[nAction].a;
|
|
|
|
seq_MoveSequence(nPlayerSprite, var_AC, PlayerList[nPlayer].field_2);
|
|
PlayerList[nPlayer].field_2++;
|
|
|
|
if (PlayerList[nPlayer].field_2 >= SeqSize[var_AC])
|
|
{
|
|
PlayerList[nPlayer].field_2 = 0;
|
|
|
|
switch (PlayerList[nPlayer].nAction)
|
|
{
|
|
default:
|
|
break;
|
|
|
|
case 3:
|
|
PlayerList[nPlayer].field_2 = SeqSize[var_AC] - 1;
|
|
break;
|
|
case 4:
|
|
PlayerList[nPlayer].nAction = 0;
|
|
break;
|
|
case 16:
|
|
PlayerList[nPlayer].field_2 = SeqSize[var_AC] - 1;
|
|
|
|
if (sprite[nPlayerSprite].z < sector[sprite[nPlayerSprite].sectnum].floorz) {
|
|
sprite[nPlayerSprite].z += 256;
|
|
}
|
|
|
|
if (!RandomSize(5))
|
|
{
|
|
int mouthX, mouthY, mouthZ;
|
|
short mouthSect;
|
|
WheresMyMouth(nPlayer, &mouthX, &mouthY, &mouthZ, &mouthSect);
|
|
|
|
BuildAnim(-1, 71, 0, mouthX, mouthY, sprite[nPlayerSprite].z + 3840, mouthSect, 75, 128);
|
|
}
|
|
break;
|
|
case 17:
|
|
PlayerList[nPlayer].nAction = 18;
|
|
break;
|
|
case 19:
|
|
sprite[nPlayerSprite].cstat |= 0x8000;
|
|
PlayerList[nPlayer].nAction = 20;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// loc_1C3B4:
|
|
if (nPlayer == nLocalPlayer)
|
|
{
|
|
initx = sprite[nPlayerSprite].x;
|
|
inity = sprite[nPlayerSprite].y;
|
|
initz = sprite[nPlayerSprite].z;
|
|
initsect = sprite[nPlayerSprite].sectnum;
|
|
inita = sprite[nPlayerSprite].ang;
|
|
}
|
|
|
|
if (!PlayerList[nPlayer].nHealth)
|
|
{
|
|
nYDamage[nPlayer] = 0;
|
|
nXDamage[nPlayer] = 0;
|
|
|
|
if (eyelevel[nPlayer] >= -2816)
|
|
{
|
|
eyelevel[nPlayer] = -2816;
|
|
dVertPan[nPlayer] = 0;
|
|
}
|
|
else
|
|
{
|
|
if (PlayerList[nPlayer].q16horiz < fix16_from_int(92))
|
|
{
|
|
PlayerList[nPlayer].q16horiz = fix16_from_int(91);
|
|
eyelevel[nPlayer] -= (dVertPan[nPlayer] << 8);
|
|
}
|
|
else
|
|
{
|
|
PlayerList[nPlayer].q16horiz = fix16_sadd(PlayerList[nPlayer].q16horiz, fix16_from_int(dVertPan[nPlayer]));
|
|
if (PlayerList[nPlayer].q16horiz >= fix16_from_int(200))
|
|
{
|
|
PlayerList[nPlayer].q16horiz = fix16_from_int(199);
|
|
}
|
|
else if (PlayerList[nPlayer].q16horiz <= fix16_from_int(92))
|
|
{
|
|
if (!(SectFlag[sprite[nPlayerSprite].sectnum] & kSectUnderwater))
|
|
{
|
|
SetNewWeapon(nPlayer, nDeathType[nPlayer] + 8);
|
|
}
|
|
}
|
|
|
|
dVertPan[nPlayer]--;
|
|
}
|
|
}
|
|
}
|
|
|
|
// loc_1C4E1
|
|
sprite[nDopple].x = sprite[nPlayerSprite].x;
|
|
sprite[nDopple].y = sprite[nPlayerSprite].y;
|
|
sprite[nDopple].z = sprite[nPlayerSprite].z;
|
|
|
|
if (SectAbove[sprite[nPlayerSprite].sectnum] > -1)
|
|
{
|
|
sprite[nDopple].ang = sprite[nPlayerSprite].ang;
|
|
mychangespritesect(nDopple, SectAbove[sprite[nPlayerSprite].sectnum]);
|
|
sprite[nDopple].cstat = 0x101;
|
|
}
|
|
else
|
|
{
|
|
sprite[nDopple].cstat = 0x8000;
|
|
}
|
|
|
|
MoveWeapons(nPlayer);
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static SavegameHelper sgh("player",
|
|
SV(lPlayerXVel),
|
|
SV(lPlayerYVel),
|
|
SV(nPlayerDAng),
|
|
SV(obobangle),
|
|
SV(bobangle),
|
|
SV(bPlayerPan),
|
|
SV(bLockPan),
|
|
SV(g_MyAimMode),
|
|
SV(nStandHeight),
|
|
SV(PlayerCount),
|
|
SV(nNetStartSprites),
|
|
SV(nCurStartSprite),
|
|
SV(nLocalPlayer),
|
|
SA(nBreathTimer),
|
|
SA(nPlayerSwear),
|
|
SA(nPlayerPushSect),
|
|
SA(nDeathType),
|
|
SA(nPlayerScore),
|
|
SA(nPlayerColor),
|
|
SA(nPlayerDY),
|
|
SA(nPlayerDX),
|
|
SA(playerNames),
|
|
SA(nPistolClip),
|
|
SA(nXDamage),
|
|
SA(nYDamage),
|
|
SA(nDoppleSprite),
|
|
SA(namelen),
|
|
SA(nPlayerOldWeapon),
|
|
SA(nPlayerClip),
|
|
SA(nPlayerPushSound),
|
|
SA(nTauntTimer),
|
|
SA(nPlayerTorch),
|
|
SA(nPlayerWeapons),
|
|
SA(nPlayerLives),
|
|
SA(nPlayerItem),
|
|
SA(PlayerList),
|
|
SA(nPlayerInvisible),
|
|
SA(nPlayerDouble),
|
|
SA(nPlayerViewSect),
|
|
SA(nPlayerFloorSprite),
|
|
SA(sPlayerSave),
|
|
SA(totalvel),
|
|
SA(eyelevel),
|
|
SA(nNetStartSprite),
|
|
SA(nPlayerGrenade),
|
|
SA(nGrenadePlayer),
|
|
SA(word_D282A),
|
|
nullptr);
|
|
|
|
END_PS_NS
|