mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-23 00:41:03 +00:00
8f19dc12d8
* a large number of code pointer records were removed because none of these functions ever gets assigned to a pointer * instead of looking up entries by index, do it by name. This is far less fragile and will survive deeper refactoring. The old storage by table index will break as soon as a single entry gets removed. Since the old savegames got broken due to this problem recently it was a good time to change the setup.
698 lines
17 KiB
C++
698 lines
17 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
Shadow Warrior 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
#include "ns.h"
|
|
#include "build.h"
|
|
|
|
#include "names2.h"
|
|
#include "panel.h"
|
|
#include "game.h"
|
|
#include "network.h"
|
|
#include "tags.h"
|
|
#include "sector.h"
|
|
#include "interpolate.h"
|
|
#include "misc.h"
|
|
#include "sprite.h"
|
|
#include "weapon.h"
|
|
#include "quotemgr.h"
|
|
|
|
BEGIN_SW_NS
|
|
|
|
short DoVatorMatch(PLAYERp pp, short match);
|
|
bool TestVatorMatchActive(short match);
|
|
void InterpSectorSprites(short sectnum, bool state);
|
|
int InitBloodSpray(short, bool, short);
|
|
|
|
void ReverseVator(short SpriteNum)
|
|
{
|
|
USERp u = User[SpriteNum].Data();
|
|
SPRITEp sp = u->SpriteP;
|
|
|
|
// if paused go ahead and start it up again
|
|
if (u->Tics)
|
|
{
|
|
u->Tics = 0;
|
|
SetVatorActive(SpriteNum);
|
|
return;
|
|
}
|
|
|
|
// moving toward to OFF pos
|
|
if (u->z_tgt == u->oz)
|
|
{
|
|
if (sp->z == u->oz)
|
|
u->z_tgt = u->sz;
|
|
else if (u->sz == u->oz)
|
|
u->z_tgt = sp->z;
|
|
}
|
|
else if (u->z_tgt == u->sz)
|
|
{
|
|
if (sp->z == u->oz)
|
|
u->z_tgt = sp->z;
|
|
else if (u->sz == u->oz)
|
|
u->z_tgt = u->sz;
|
|
}
|
|
|
|
u->vel_rate = -u->vel_rate;
|
|
}
|
|
|
|
bool
|
|
VatorSwitch(short match, short setting)
|
|
{
|
|
SPRITEp sp;
|
|
int i;
|
|
bool found = false;
|
|
|
|
StatIterator it(STAT_DEFAULT);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
sp = &sprite[i];
|
|
|
|
if (sp->lotag == TAG_SPRITE_SWITCH_VATOR && sp->hitag == match)
|
|
{
|
|
found = true;
|
|
AnimateSwitch(sp, setting);
|
|
}
|
|
}
|
|
|
|
return found;
|
|
}
|
|
|
|
void SetVatorActive(short SpriteNum)
|
|
{
|
|
USERp u = User[SpriteNum].Data();
|
|
SPRITEp sp = u->SpriteP;
|
|
SECTORp sectp = §or[sp->sectnum];
|
|
|
|
if (TEST(sp->cstat, CSTAT_SPRITE_YFLIP))
|
|
StartInterpolation(sp->sectnum, Interp_Sect_Ceilingz);
|
|
else
|
|
StartInterpolation(sp->sectnum, Interp_Sect_Floorz);
|
|
|
|
InterpSectorSprites(sp->sectnum, true);
|
|
|
|
// play activate sound
|
|
DoSoundSpotMatch(SP_TAG2(sp), 1, SOUND_OBJECT_TYPE);
|
|
|
|
SET(u->Flags, SPR_ACTIVE);
|
|
u->Tics = 0;
|
|
|
|
// moving to the ON position
|
|
if (u->z_tgt == sp->z)
|
|
VatorSwitch(SP_TAG2(sp), ON);
|
|
else
|
|
// moving to the OFF position
|
|
if (u->z_tgt == u->sz)
|
|
VatorSwitch(SP_TAG2(sp), OFF);
|
|
}
|
|
|
|
void SetVatorInactive(short SpriteNum)
|
|
{
|
|
USERp u = User[SpriteNum].Data();
|
|
SPRITEp sp = u->SpriteP;
|
|
SECTORp sectp = §or[sp->sectnum];
|
|
|
|
if (TEST(sp->cstat, CSTAT_SPRITE_YFLIP))
|
|
StopInterpolation(sp->sectnum, Interp_Sect_Ceilingz);
|
|
else
|
|
StopInterpolation(sp->sectnum, Interp_Sect_Floorz);
|
|
|
|
InterpSectorSprites(sp->sectnum, false);
|
|
|
|
// play inactivate sound
|
|
DoSoundSpotMatch(SP_TAG2(sp), 2, SOUND_OBJECT_TYPE);
|
|
|
|
RESET(u->Flags, SPR_ACTIVE);
|
|
}
|
|
|
|
// called for operation from the space bar
|
|
short DoVatorOperate(PLAYERp pp, short sectnum)
|
|
{
|
|
SPRITEp fsp;
|
|
short match;
|
|
int i;
|
|
|
|
SectIterator it(sectnum);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
fsp = &sprite[i];
|
|
|
|
if (fsp->statnum == STAT_VATOR && SP_TAG1(fsp) == SECT_VATOR && SP_TAG3(fsp) == 0)
|
|
{
|
|
sectnum = fsp->sectnum;
|
|
|
|
// single play only vator
|
|
// bool 8 must be set for message to display
|
|
if (TEST_BOOL4(fsp) && (gNet.MultiGameType == MULTI_GAME_COMMBAT || gNet.MultiGameType == MULTI_GAME_AI_BOTS))
|
|
{
|
|
if (pp && TEST_BOOL11(fsp)) PutStringInfo(pp,"This only opens in single play.");
|
|
continue;
|
|
}
|
|
|
|
match = SP_TAG2(fsp);
|
|
if (match > 0)
|
|
{
|
|
if (TestVatorMatchActive(match))
|
|
return -1;
|
|
else
|
|
return DoVatorMatch(pp, match);
|
|
}
|
|
|
|
if (pp && SectUser[sectnum].Data() && SectUser[sectnum]->stag == SECT_LOCK_DOOR && SectUser[sectnum]->number)
|
|
{
|
|
short key_num;
|
|
|
|
key_num = SectUser[sectnum]->number;
|
|
|
|
#if 0
|
|
if (pp->HasKey[key_num - 1])
|
|
{
|
|
int i;
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (SectUser[i] && SectUser[i]->stag == SECT_LOCK_DOOR && SectUser[i]->number == key_num)
|
|
SectUser[i]->number = 0; // unlock all doors of this type
|
|
}
|
|
UnlockKeyLock(key_num);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
PutStringInfo(pp, quoteMgr.GetQuote(QUOTE_DOORMSG + key_num - 1));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
SetVatorActive(i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
// called from switches and triggers
|
|
// returns first vator found
|
|
short
|
|
DoVatorMatch(PLAYERp pp, short match)
|
|
{
|
|
USERp fu;
|
|
SPRITEp fsp;
|
|
short sectnum;
|
|
short first_vator = -1;
|
|
|
|
int i;
|
|
|
|
//VatorSwitch(match, ON);
|
|
|
|
StatIterator it(STAT_VATOR);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
fsp = &sprite[i];
|
|
|
|
if (SP_TAG1(fsp) == SECT_VATOR && SP_TAG2(fsp) == match)
|
|
{
|
|
fu = User[i].Data();
|
|
|
|
if (first_vator == -1)
|
|
first_vator = i;
|
|
|
|
// single play only vator
|
|
// bool 8 must be set for message to display
|
|
if (TEST_BOOL4(fsp) && (gNet.MultiGameType == MULTI_GAME_COMMBAT || gNet.MultiGameType == MULTI_GAME_AI_BOTS))
|
|
{
|
|
if (pp && TEST_BOOL11(fsp)) PutStringInfo(pp, GStrings("TXTS_SPONLY"));
|
|
continue;
|
|
}
|
|
|
|
// lock code
|
|
sectnum = fsp->sectnum;
|
|
if (pp && SectUser[sectnum].Data() && SectUser[sectnum]->stag == SECT_LOCK_DOOR && SectUser[sectnum]->number)
|
|
{
|
|
short key_num;
|
|
|
|
key_num = SectUser[sectnum]->number;
|
|
|
|
#if 0
|
|
if (pp->HasKey[key_num - 1])
|
|
{
|
|
int i;
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (SectUser[i] && SectUser[i]->stag == SECT_LOCK_DOOR && SectUser[i]->number == key_num)
|
|
SectUser[i]->number = 0; // unlock all doors of this type
|
|
}
|
|
UnlockKeyLock(key_num);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
PutStringInfo(pp, quoteMgr.GetQuote(QUOTE_DOORMSG + key_num - 1));
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
// remember the player than activated it
|
|
fu->PlayerP = pp;
|
|
|
|
if (TEST(fu->Flags, SPR_ACTIVE))
|
|
{
|
|
ReverseVator(i);
|
|
continue;
|
|
}
|
|
|
|
SetVatorActive(i);
|
|
}
|
|
}
|
|
|
|
return first_vator;
|
|
}
|
|
|
|
|
|
bool
|
|
TestVatorMatchActive(short match)
|
|
{
|
|
USERp fu;
|
|
SPRITEp fsp;
|
|
|
|
int i;
|
|
|
|
StatIterator it(STAT_VATOR);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
fsp = &sprite[i];
|
|
|
|
if (SP_TAG1(fsp) == SECT_VATOR && SP_TAG2(fsp) == match)
|
|
{
|
|
fu = User[i].Data();
|
|
|
|
// Does not have to be inactive to be operated
|
|
if (TEST_BOOL6(fsp))
|
|
continue;
|
|
|
|
if (TEST(fu->Flags, SPR_ACTIVE) || fu->Tics)
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void InterpSectorSprites(short sectnum, bool state)
|
|
{
|
|
SPRITEp sp;
|
|
int i;
|
|
|
|
SectIterator it(sectnum);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
sp = &sprite[i];
|
|
|
|
if (User[i].Data())
|
|
{
|
|
if (TEST(User[i]->Flags, SPR_SKIP4) && sp->statnum <= STAT_SKIP4_INTERP_END)
|
|
continue;
|
|
|
|
if (TEST(User[i]->Flags, SPR_SKIP2) && sp->statnum <= STAT_SKIP2_INTERP_END)
|
|
continue;
|
|
}
|
|
|
|
if (state)
|
|
StartInterpolation(i, Interp_Sprite_Z);
|
|
else
|
|
StopInterpolation(i, Interp_Sprite_Z);
|
|
}
|
|
}
|
|
|
|
void MoveSpritesWithSector(short sectnum, int z_amt, bool type)
|
|
{
|
|
SPRITEp sp;
|
|
int i;
|
|
bool both = false;
|
|
|
|
if (SectUser[sectnum].Data())
|
|
both = !!TEST(SectUser[sectnum]->flags, SECTFU_VATOR_BOTH);
|
|
|
|
SectIterator it(sectnum);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
sp = &sprite[i];
|
|
|
|
if (User[i].Data())
|
|
{
|
|
switch (sp->statnum)
|
|
{
|
|
case STAT_ITEM:
|
|
case STAT_NO_STATE:
|
|
case STAT_MINE_STUCK:
|
|
case STAT_WALLBLOOD_QUEUE:
|
|
case STAT_FLOORBLOOD_QUEUE:
|
|
case STAT_STATIC_FIRE:
|
|
break;
|
|
default:
|
|
goto cont;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch (sp->statnum)
|
|
{
|
|
case STAT_STAR_QUEUE:
|
|
case STAT_HOLE_QUEUE:
|
|
// case STAT_WALLBLOOD_QUEUE:
|
|
// case STAT_FLOORBLOOD_QUEUE:
|
|
goto cont;
|
|
}
|
|
}
|
|
|
|
if (TEST(sp->extra, SPRX_STAY_PUT_VATOR))
|
|
continue;
|
|
|
|
if (both)
|
|
{
|
|
// sprite started close to floor
|
|
if (TEST(sp->cstat, CSTAT_SPRITE_CLOSE_FLOOR))
|
|
{
|
|
// this is a ceiling
|
|
if (type == 1)
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
// this is a floor
|
|
if (type == 0)
|
|
continue;
|
|
}
|
|
}
|
|
|
|
sp->z += z_amt;
|
|
|
|
cont:
|
|
continue;
|
|
}
|
|
}
|
|
|
|
int DoVatorMove(short SpriteNum, int *lptr)
|
|
{
|
|
USERp u = User[SpriteNum].Data();
|
|
int zval;
|
|
int move_amt;
|
|
|
|
zval = *lptr;
|
|
|
|
// if LESS THAN goal
|
|
if (zval < u->z_tgt)
|
|
{
|
|
// move it DOWN
|
|
zval += (synctics * u->jump_speed);
|
|
|
|
u->jump_speed += u->vel_rate * synctics;
|
|
|
|
// if the other way make it equal
|
|
if (zval > u->z_tgt)
|
|
zval = u->z_tgt;
|
|
}
|
|
|
|
// if GREATER THAN goal
|
|
if (zval > u->z_tgt)
|
|
{
|
|
// move it UP
|
|
zval -= (synctics * u->jump_speed);
|
|
|
|
u->jump_speed += u->vel_rate * synctics;
|
|
|
|
if (zval < u->z_tgt)
|
|
zval = u->z_tgt;
|
|
}
|
|
|
|
move_amt = zval - *lptr;
|
|
*lptr = zval;
|
|
|
|
return move_amt;
|
|
}
|
|
|
|
|
|
int DoVator(DSWActor* actor)
|
|
{
|
|
USER* u = actor->u();
|
|
int SpriteNum = u->SpriteNum;
|
|
SPRITEp sp = u->SpriteP;
|
|
SECTORp sectp = §or[sp->sectnum];
|
|
int *lptr;
|
|
int amt;
|
|
|
|
// u->sz - where the sector z started
|
|
// u->z_tgt - current target z
|
|
// u->oz - original z - where it initally starts off
|
|
// sp->z - z of the sprite
|
|
// u->vel_rate - velocity
|
|
|
|
if (TEST(sp->cstat, CSTAT_SPRITE_YFLIP))
|
|
{
|
|
lptr = §p->ceilingz;
|
|
amt = DoVatorMove(SpriteNum, lptr);
|
|
MoveSpritesWithSector(sp->sectnum, amt, true); // ceiling
|
|
}
|
|
else
|
|
{
|
|
lptr = §p->floorz;
|
|
amt = DoVatorMove(SpriteNum, lptr);
|
|
MoveSpritesWithSector(sp->sectnum, amt, false); // floor
|
|
}
|
|
|
|
// EQUAL this entry has finished
|
|
if (*lptr == u->z_tgt)
|
|
{
|
|
// in the ON position
|
|
if (u->z_tgt == sp->z)
|
|
{
|
|
// change target
|
|
u->z_tgt = u->sz;
|
|
u->vel_rate = -u->vel_rate;
|
|
|
|
SetVatorInactive(SpriteNum);
|
|
|
|
// if tag6 and nothing blocking door
|
|
if (SP_TAG6(sp) && !TEST_BOOL8(sp))
|
|
DoMatchEverything(u->PlayerP, SP_TAG6(sp), -1);
|
|
}
|
|
else
|
|
// in the OFF position
|
|
if (u->z_tgt == u->sz)
|
|
{
|
|
short match = SP_TAG2(sp);
|
|
|
|
// change target
|
|
u->jump_speed = u->vel_tgt;
|
|
u->vel_rate = short(abs(u->vel_rate));
|
|
u->z_tgt = sp->z;
|
|
|
|
RESET_BOOL8(sp);
|
|
SetVatorInactive(SpriteNum);
|
|
|
|
// set owner swith back to OFF
|
|
// only if ALL vators are inactive
|
|
if (!TestVatorMatchActive(match))
|
|
{
|
|
//VatorSwitch(match, OFF);
|
|
}
|
|
|
|
if (SP_TAG6(sp) && TEST_BOOL5(sp))
|
|
DoMatchEverything(u->PlayerP, SP_TAG6(sp), -1);
|
|
}
|
|
|
|
// operate only once
|
|
if (TEST_BOOL2(sp))
|
|
{
|
|
SetVatorInactive(SpriteNum);
|
|
KillSprite(SpriteNum);
|
|
return 0;
|
|
}
|
|
|
|
// setup to go back to the original z
|
|
if (*lptr != u->oz)
|
|
{
|
|
if (u->WaitTics)
|
|
u->Tics = u->WaitTics;
|
|
}
|
|
}
|
|
else // if (*lptr == u->z_tgt)
|
|
{
|
|
// if heading for the OFF (original) position and should NOT CRUSH
|
|
if (TEST_BOOL3(sp) && u->z_tgt == u->oz)
|
|
{
|
|
int i;
|
|
SPRITEp bsp;
|
|
USERp bu;
|
|
bool found = false;
|
|
|
|
SectIterator it(sp->sectnum);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
bsp = &sprite[i];
|
|
bu = User[i].Data();
|
|
|
|
if (bsp->statnum == STAT_ENEMY)
|
|
{
|
|
if (labs(sectp->ceilingz - sectp->floorz) < SPRITEp_SIZE_Z(bsp))
|
|
{
|
|
InitBloodSpray(i, true, -1);
|
|
UpdateSinglePlayKills(i);
|
|
KillSprite(i);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (bu && TEST(bsp->cstat, CSTAT_SPRITE_BLOCK) && TEST(bsp->extra, SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
// found something blocking so reverse to ON position
|
|
ReverseVator(SpriteNum);
|
|
SET_BOOL8(sp); // tell vator that something blocking door
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
{
|
|
short pnum;
|
|
PLAYERp pp;
|
|
// go ahead and look for players clip box bounds
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
pp = Player + pnum;
|
|
|
|
if (pp->lo_sectp == §or[sp->sectnum] ||
|
|
pp->hi_sectp == §or[sp->sectnum])
|
|
{
|
|
ReverseVator(SpriteNum);
|
|
|
|
u->vel_rate = -u->vel_rate;
|
|
found = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int i;
|
|
SPRITEp bsp;
|
|
|
|
SectIterator it(sp->sectnum);
|
|
while ((i = it.NextIndex()) >= 0)
|
|
{
|
|
bsp = &sprite[i];
|
|
|
|
if (bsp->statnum == STAT_ENEMY)
|
|
{
|
|
if (labs(sectp->ceilingz - sectp->floorz) < SPRITEp_SIZE_Z(bsp))
|
|
{
|
|
InitBloodSpray(i, true, -1);
|
|
UpdateSinglePlayKills(i);
|
|
KillSprite(i);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoVatorAuto(DSWActor* actor)
|
|
{
|
|
USER* u = actor->u();
|
|
int SpriteNum = u->SpriteNum;
|
|
SPRITEp sp = u->SpriteP;
|
|
SECTORp sectp = §or[sp->sectnum];
|
|
int *lptr;
|
|
int amt;
|
|
|
|
if (TEST(sp->cstat, CSTAT_SPRITE_YFLIP))
|
|
{
|
|
lptr = §p->ceilingz;
|
|
amt = DoVatorMove(SpriteNum, lptr);
|
|
MoveSpritesWithSector(sp->sectnum, amt, true); // ceiling
|
|
}
|
|
else
|
|
{
|
|
lptr = §p->floorz;
|
|
amt = DoVatorMove(SpriteNum, lptr);
|
|
MoveSpritesWithSector(sp->sectnum, amt, false); // floor
|
|
}
|
|
|
|
// EQUAL this entry has finished
|
|
if (*lptr == u->z_tgt)
|
|
{
|
|
// in the UP position
|
|
if (u->z_tgt == sp->z)
|
|
{
|
|
// change target
|
|
u->z_tgt = u->sz;
|
|
u->vel_rate = -u->vel_rate;
|
|
u->Tics = u->WaitTics;
|
|
|
|
if (SP_TAG6(sp))
|
|
DoMatchEverything(u->PlayerP, SP_TAG6(sp), -1);
|
|
}
|
|
else
|
|
// in the DOWN position
|
|
if (u->z_tgt == u->sz)
|
|
{
|
|
// change target
|
|
u->jump_speed = u->vel_tgt;
|
|
u->vel_rate = short(abs(u->vel_rate));
|
|
u->z_tgt = sp->z;
|
|
u->Tics = u->WaitTics;
|
|
|
|
if (SP_TAG6(sp) && TEST_BOOL5(sp))
|
|
DoMatchEverything(nullptr, SP_TAG6(sp), -1);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
#include "saveable.h"
|
|
|
|
static saveable_code saveable_vator_code[] =
|
|
{
|
|
SAVE_CODE(DoVator),
|
|
SAVE_CODE(DoVatorAuto),
|
|
};
|
|
|
|
saveable_module saveable_vator =
|
|
{
|
|
// code
|
|
saveable_vator_code,
|
|
SIZ(saveable_vator_code),
|
|
|
|
// data
|
|
nullptr,0
|
|
};
|
|
END_SW_NS
|