7e6f74e429
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@11738 a1c6a512-1295-4272-9138-f99709370657
3255 lines
95 KiB
C
3255 lines
95 KiB
C
/* Emacs style mode select -*- C++ -*-
|
|
*-----------------------------------------------------------------------------
|
|
*
|
|
*
|
|
* PrBoom a Doom port merged with LxDoom and LSDLDoom
|
|
* based on BOOM, a modified and improved DOOM engine
|
|
* Copyright (C) 1999 by
|
|
* id Software, Chi Hoang, Lee Killough, Jim Flynn, Rand Phares, Ty Halderman
|
|
* Copyright (C) 1999-2000 by
|
|
* Jess Haas, Nicolas Kalkhof, Colin Phipps, Florian Schulze
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
* DESCRIPTION:
|
|
* -Loads and initializes texture and flat animation sequences
|
|
* -Implements utility functions for all linedef/sector special handlers
|
|
* -Dispatches walkover and gun line triggers
|
|
* -Initializes and implements special sector types
|
|
* -Implements donut linedef triggers
|
|
* -Initializes and implements BOOM linedef triggers for
|
|
* Scrollers/Conveyors
|
|
* Friction
|
|
* Wind/Current
|
|
*
|
|
*-----------------------------------------------------------------------------*/
|
|
|
|
#include "doomstat.h"
|
|
#include "p_spec.h"
|
|
#include "p_tick.h"
|
|
#include "p_setup.h"
|
|
#include "m_random.h"
|
|
#include "d_englsh.h"
|
|
#include "m_argv.h"
|
|
#include "w_wad.h"
|
|
#include "r_main.h"
|
|
#include "p_maputl.h"
|
|
#include "p_map.h"
|
|
#include "g_game.h"
|
|
#include "p_inter.h"
|
|
#include "m_swap.h"
|
|
#include "s_sound.h"
|
|
#include "sounds.h"
|
|
#include "m_bbox.h" // phares 3/20/98
|
|
#include "d_deh.h"
|
|
#include "r_plane.h"
|
|
#include "i_system.h"
|
|
#include "rockmacros.h"
|
|
//
|
|
// Animating textures and planes
|
|
// There is another anim_t used in wi_stuff, unrelated.
|
|
//
|
|
typedef struct
|
|
{
|
|
boolean istexture;
|
|
int picnum;
|
|
int basepic;
|
|
int numpics;
|
|
int speed;
|
|
|
|
} anim_t;
|
|
|
|
//
|
|
// source animation definition
|
|
//
|
|
//
|
|
#ifdef _MSC_VER // proff: This is the same as __attribute__ ((packed)) in GNUC
|
|
#pragma pack(push)
|
|
#pragma pack(1)
|
|
#endif //_MSC_VER
|
|
|
|
#if defined(__MWERKS__)
|
|
#pragma options align=packed
|
|
#endif
|
|
|
|
typedef struct
|
|
{
|
|
signed char istexture; //jff 3/23/98 make char for comparison // cph - make signed
|
|
char endname[9]; // if false, it is a flat
|
|
char startname[9];
|
|
int speed;
|
|
} PACKEDATTR animdef_t; //jff 3/23/98 pack to read from memory
|
|
|
|
#define MAXANIMS 32 // no longer a strict limit -- killough
|
|
|
|
anim_t* lastanim;
|
|
anim_t* anims; // new structure w/o limits -- killough
|
|
static size_t maxanims;
|
|
|
|
// killough 3/7/98: Initialize generalized scrolling
|
|
static void P_SpawnScrollers(void);
|
|
|
|
static void P_SpawnFriction(void); // phares 3/16/98
|
|
static void P_SpawnPushers(void); // phares 3/20/98
|
|
|
|
extern int allow_pushers;
|
|
extern int variable_friction; // phares 3/20/98
|
|
|
|
//
|
|
// P_InitPicAnims
|
|
//
|
|
// Load the table of animation definitions, checking for existence of
|
|
// the start and end of each frame. If the start doesn't exist the sequence
|
|
// is skipped, if the last doesn't exist, BOOM exits.
|
|
//
|
|
// Wall/Flat animation sequences, defined by name of first and last frame,
|
|
// The full animation sequence is given using all lumps between the start
|
|
// and end entry, in the order found in the WAD file.
|
|
//
|
|
// This routine modified to read its data from a predefined lump or
|
|
// PWAD lump called ANIMATED rather than a static table in this module to
|
|
// allow wad designers to insert or modify animation sequences.
|
|
//
|
|
// Lump format is an array of byte packed animdef_t structures, terminated
|
|
// by a structure with istexture == -1. The lump can be generated from a
|
|
// text source file using SWANTBLS.EXE, distributed with the BOOM utils.
|
|
// The standard list of switches and animations is contained in the example
|
|
// source text file DEFSWANI.DAT also in the BOOM util distribution.
|
|
//
|
|
//
|
|
void P_InitPicAnims (void)
|
|
{
|
|
int i;
|
|
const animdef_t *animdefs; //jff 3/23/98 pointer to animation lump
|
|
int lump = W_GetNumForName("ANIMATED"); // cph - new wad lump handling
|
|
// Init animation
|
|
|
|
//jff 3/23/98 read from predefined or wad lump instead of table
|
|
animdefs = (const animdef_t *)W_CacheLumpNum(lump);
|
|
|
|
lastanim = anims;
|
|
for (i=0 ; animdefs[i].istexture != -1 ; i++)
|
|
{
|
|
// 1/11/98 killough -- removed limit by array-doubling
|
|
if (lastanim >= anims + maxanims)
|
|
{
|
|
size_t newmax = maxanims ? maxanims*2 : MAXANIMS;
|
|
anims = realloc(anims, newmax*sizeof(*anims)); // killough
|
|
lastanim = anims + maxanims;
|
|
maxanims = newmax;
|
|
}
|
|
|
|
if (animdefs[i].istexture)
|
|
{
|
|
// different episode ?
|
|
if (R_CheckTextureNumForName(animdefs[i].startname) == -1)
|
|
continue;
|
|
|
|
lastanim->picnum = R_TextureNumForName (animdefs[i].endname);
|
|
lastanim->basepic = R_TextureNumForName (animdefs[i].startname);
|
|
}
|
|
else
|
|
{
|
|
if ((W_CheckNumForName)(animdefs[i].startname, ns_flats) == -1) // killough 4/17/98
|
|
continue;
|
|
|
|
lastanim->picnum = R_FlatNumForName (animdefs[i].endname);
|
|
lastanim->basepic = R_FlatNumForName (animdefs[i].startname);
|
|
}
|
|
|
|
lastanim->istexture = animdefs[i].istexture;
|
|
lastanim->numpics = lastanim->picnum - lastanim->basepic + 1;
|
|
|
|
if (lastanim->numpics < 2)
|
|
I_Error ("P_InitPicAnims: bad cycle from %s to %s",
|
|
animdefs[i].startname,
|
|
animdefs[i].endname);
|
|
|
|
lastanim->speed = LONG(animdefs[i].speed); // killough 5/5/98: add LONG()
|
|
lastanim++;
|
|
}
|
|
W_UnlockLumpNum(lump);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////
|
|
//
|
|
// Linedef and Sector Special Implementation Utility Functions
|
|
//
|
|
///////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// getSide()
|
|
//
|
|
// Will return a side_t*
|
|
// given the number of the current sector,
|
|
// the line number, and the side (0/1) that you want.
|
|
//
|
|
// Note: if side=1 is specified, it must exist or results undefined
|
|
//
|
|
side_t* getSide
|
|
( int currentSector,
|
|
int line,
|
|
int side )
|
|
{
|
|
return &sides[ (sectors[currentSector].lines[line])->sidenum[side] ];
|
|
}
|
|
|
|
|
|
//
|
|
// getSector()
|
|
//
|
|
// Will return a sector_t*
|
|
// given the number of the current sector,
|
|
// the line number and the side (0/1) that you want.
|
|
//
|
|
// Note: if side=1 is specified, it must exist or results undefined
|
|
//
|
|
sector_t* getSector
|
|
( int currentSector,
|
|
int line,
|
|
int side )
|
|
{
|
|
return sides[ (sectors[currentSector].lines[line])->sidenum[side] ].sector;
|
|
}
|
|
|
|
|
|
//
|
|
// twoSided()
|
|
//
|
|
// Given the sector number and the line number,
|
|
// it will tell you whether the line is two-sided or not.
|
|
//
|
|
// modified to return actual two-sidedness rather than presence
|
|
// of 2S flag unless compatibility optioned
|
|
//
|
|
int twoSided
|
|
( int sector,
|
|
int line )
|
|
{
|
|
//jff 1/26/98 return what is actually needed, whether the line
|
|
//has two sidedefs, rather than whether the 2S flag is set
|
|
|
|
return comp[comp_model] ?
|
|
(sectors[sector].lines[line])->flags & ML_TWOSIDED
|
|
:
|
|
(sectors[sector].lines[line])->sidenum[1] != -1;
|
|
}
|
|
|
|
|
|
//
|
|
// getNextSector()
|
|
//
|
|
// Return sector_t * of sector next to current across line.
|
|
//
|
|
// Note: returns NULL if not two-sided line, or both sides refer to sector
|
|
//
|
|
sector_t* getNextSector
|
|
( line_t* line,
|
|
sector_t* sec )
|
|
{
|
|
//jff 1/26/98 check unneeded since line->backsector already
|
|
//returns NULL if the line is not two sided, and does so from
|
|
//the actual two-sidedness of the line, rather than its 2S flag
|
|
|
|
if (comp[comp_model])
|
|
{
|
|
if (!(line->flags & ML_TWOSIDED))
|
|
return NULL;
|
|
}
|
|
|
|
if (line->frontsector == sec) {
|
|
if (comp[comp_model] || line->backsector!=sec)
|
|
return line->backsector; //jff 5/3/98 don't retn sec unless compatibility
|
|
else // fixes an intra-sector line breaking functions
|
|
return NULL; // like floor->highest floor
|
|
}
|
|
return line->frontsector;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindLowestFloorSurrounding()
|
|
//
|
|
// Returns the fixed point value of the lowest floor height
|
|
// in the sector passed or its surrounding sectors.
|
|
//
|
|
fixed_t P_FindLowestFloorSurrounding(sector_t* sec)
|
|
{
|
|
int i;
|
|
line_t* check;
|
|
sector_t* other;
|
|
fixed_t floor = sec->floorheight;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
{
|
|
check = sec->lines[i];
|
|
other = getNextSector(check,sec);
|
|
|
|
if (!other)
|
|
continue;
|
|
|
|
if (other->floorheight < floor)
|
|
floor = other->floorheight;
|
|
}
|
|
return floor;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindHighestFloorSurrounding()
|
|
//
|
|
// Passed a sector, returns the fixed point value of the largest
|
|
// floor height in the surrounding sectors, not including that passed
|
|
//
|
|
// NOTE: if no surrounding sector exists -32000*FRACUINT is returned
|
|
// if compatibility then -500*FRACUNIT is the smallest return possible
|
|
//
|
|
fixed_t P_FindHighestFloorSurrounding(sector_t *sec)
|
|
{
|
|
int i;
|
|
line_t* check;
|
|
sector_t* other;
|
|
fixed_t floor = -500*FRACUNIT;
|
|
|
|
//jff 1/26/98 Fix initial value for floor to not act differently
|
|
//in sections of wad that are below -500 units
|
|
if (!comp[comp_model]) /* jff 3/12/98 avoid ovf */
|
|
floor = -32000*FRACUNIT; // in height calculations
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
{
|
|
check = sec->lines[i];
|
|
other = getNextSector(check,sec);
|
|
|
|
if (!other)
|
|
continue;
|
|
|
|
if (other->floorheight > floor)
|
|
floor = other->floorheight;
|
|
}
|
|
return floor;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindNextHighestFloor()
|
|
//
|
|
// Passed a sector and a floor height, returns the fixed point value
|
|
// of the smallest floor height in a surrounding sector larger than
|
|
// the floor height passed. If no such height exists the floorheight
|
|
// passed is returned.
|
|
//
|
|
// Rewritten by Lee Killough to avoid fixed array and to be faster
|
|
//
|
|
fixed_t P_FindNextHighestFloor(sector_t *sec, int currentheight)
|
|
{
|
|
sector_t *other;
|
|
int i;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->floorheight > currentheight)
|
|
{
|
|
int height = other->floorheight;
|
|
while (++i < sec->linecount)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->floorheight < height &&
|
|
other->floorheight > currentheight)
|
|
height = other->floorheight;
|
|
return height;
|
|
}
|
|
return currentheight;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindNextLowestFloor()
|
|
//
|
|
// Passed a sector and a floor height, returns the fixed point value
|
|
// of the largest floor height in a surrounding sector smaller than
|
|
// the floor height passed. If no such height exists the floorheight
|
|
// passed is returned.
|
|
//
|
|
// jff 02/03/98 Twiddled Lee's P_FindNextHighestFloor to make this
|
|
//
|
|
fixed_t P_FindNextLowestFloor(sector_t *sec, int currentheight)
|
|
{
|
|
sector_t *other;
|
|
int i;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->floorheight < currentheight)
|
|
{
|
|
int height = other->floorheight;
|
|
while (++i < sec->linecount)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->floorheight > height &&
|
|
other->floorheight < currentheight)
|
|
height = other->floorheight;
|
|
return height;
|
|
}
|
|
return currentheight;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindNextLowestCeiling()
|
|
//
|
|
// Passed a sector and a ceiling height, returns the fixed point value
|
|
// of the largest ceiling height in a surrounding sector smaller than
|
|
// the ceiling height passed. If no such height exists the ceiling height
|
|
// passed is returned.
|
|
//
|
|
// jff 02/03/98 Twiddled Lee's P_FindNextHighestFloor to make this
|
|
//
|
|
fixed_t P_FindNextLowestCeiling(sector_t *sec, int currentheight)
|
|
{
|
|
sector_t *other;
|
|
int i;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->ceilingheight < currentheight)
|
|
{
|
|
int height = other->ceilingheight;
|
|
while (++i < sec->linecount)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->ceilingheight > height &&
|
|
other->ceilingheight < currentheight)
|
|
height = other->ceilingheight;
|
|
return height;
|
|
}
|
|
return currentheight;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindNextHighestCeiling()
|
|
//
|
|
// Passed a sector and a ceiling height, returns the fixed point value
|
|
// of the smallest ceiling height in a surrounding sector larger than
|
|
// the ceiling height passed. If no such height exists the ceiling height
|
|
// passed is returned.
|
|
//
|
|
// jff 02/03/98 Twiddled Lee's P_FindNextHighestFloor to make this
|
|
//
|
|
fixed_t P_FindNextHighestCeiling(sector_t *sec, int currentheight)
|
|
{
|
|
sector_t *other;
|
|
int i;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->ceilingheight > currentheight)
|
|
{
|
|
int height = other->ceilingheight;
|
|
while (++i < sec->linecount)
|
|
if ((other = getNextSector(sec->lines[i],sec)) &&
|
|
other->ceilingheight < height &&
|
|
other->ceilingheight > currentheight)
|
|
height = other->ceilingheight;
|
|
return height;
|
|
}
|
|
return currentheight;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindLowestCeilingSurrounding()
|
|
//
|
|
// Passed a sector, returns the fixed point value of the smallest
|
|
// ceiling height in the surrounding sectors, not including that passed
|
|
//
|
|
// NOTE: if no surrounding sector exists 32000*FRACUINT is returned
|
|
// but if compatibility then INT_MAX is the return
|
|
//
|
|
fixed_t P_FindLowestCeilingSurrounding(sector_t* sec)
|
|
{
|
|
int i;
|
|
line_t* check;
|
|
sector_t* other;
|
|
fixed_t height = INT_MAX;
|
|
|
|
/* jff 3/12/98 avoid ovf in height calculations */
|
|
if (!comp[comp_model]) height = 32000*FRACUNIT;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
{
|
|
check = sec->lines[i];
|
|
other = getNextSector(check,sec);
|
|
|
|
if (!other)
|
|
continue;
|
|
|
|
if (other->ceilingheight < height)
|
|
height = other->ceilingheight;
|
|
}
|
|
return height;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindHighestCeilingSurrounding()
|
|
//
|
|
// Passed a sector, returns the fixed point value of the largest
|
|
// ceiling height in the surrounding sectors, not including that passed
|
|
//
|
|
// NOTE: if no surrounding sector exists -32000*FRACUINT is returned
|
|
// but if compatibility then 0 is the smallest return possible
|
|
//
|
|
fixed_t P_FindHighestCeilingSurrounding(sector_t* sec)
|
|
{
|
|
int i;
|
|
line_t* check;
|
|
sector_t* other;
|
|
fixed_t height = 0;
|
|
|
|
/* jff 1/26/98 Fix initial value for floor to not act differently
|
|
* in sections of wad that are below 0 units
|
|
* jff 3/12/98 avoid ovf in height calculations */
|
|
if (!comp[comp_model]) height = -32000*FRACUNIT;
|
|
|
|
for (i=0 ;i < sec->linecount ; i++)
|
|
{
|
|
check = sec->lines[i];
|
|
other = getNextSector(check,sec);
|
|
|
|
if (!other)
|
|
continue;
|
|
|
|
if (other->ceilingheight > height)
|
|
height = other->ceilingheight;
|
|
}
|
|
return height;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindShortestTextureAround()
|
|
//
|
|
// Passed a sector number, returns the shortest lower texture on a
|
|
// linedef bounding the sector.
|
|
//
|
|
// Note: If no lower texture exists 32000*FRACUNIT is returned.
|
|
// but if compatibility then INT_MAX is returned
|
|
//
|
|
// jff 02/03/98 Add routine to find shortest lower texture
|
|
//
|
|
fixed_t P_FindShortestTextureAround(int secnum)
|
|
{
|
|
int minsize = INT_MAX;
|
|
side_t* side;
|
|
int i;
|
|
sector_t *sec = §ors[secnum];
|
|
|
|
if (!comp[comp_model])
|
|
minsize = 32000<<FRACBITS; //jff 3/13/98 prevent overflow in height calcs
|
|
|
|
for (i = 0; i < sec->linecount; i++)
|
|
{
|
|
if (twoSided(secnum, i))
|
|
{
|
|
side = getSide(secnum,i,0);
|
|
if (side->bottomtexture > 0) //jff 8/14/98 texture 0 is a placeholder
|
|
if (textureheight[side->bottomtexture] < minsize)
|
|
minsize = textureheight[side->bottomtexture];
|
|
side = getSide(secnum,i,1);
|
|
if (side->bottomtexture > 0) //jff 8/14/98 texture 0 is a placeholder
|
|
if (textureheight[side->bottomtexture] < minsize)
|
|
minsize = textureheight[side->bottomtexture];
|
|
}
|
|
}
|
|
return minsize;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindShortestUpperAround()
|
|
//
|
|
// Passed a sector number, returns the shortest upper texture on a
|
|
// linedef bounding the sector.
|
|
//
|
|
// Note: If no upper texture exists 32000*FRACUNIT is returned.
|
|
// but if compatibility then INT_MAX is returned
|
|
//
|
|
// jff 03/20/98 Add routine to find shortest upper texture
|
|
//
|
|
fixed_t P_FindShortestUpperAround(int secnum)
|
|
{
|
|
int minsize = INT_MAX;
|
|
side_t* side;
|
|
int i;
|
|
sector_t *sec = §ors[secnum];
|
|
|
|
if (!comp[comp_model])
|
|
minsize = 32000<<FRACBITS; //jff 3/13/98 prevent overflow
|
|
// in height calcs
|
|
for (i = 0; i < sec->linecount; i++)
|
|
{
|
|
if (twoSided(secnum, i))
|
|
{
|
|
side = getSide(secnum,i,0);
|
|
if (side->toptexture > 0) //jff 8/14/98 texture 0 is a placeholder
|
|
if (textureheight[side->toptexture] < minsize)
|
|
minsize = textureheight[side->toptexture];
|
|
side = getSide(secnum,i,1);
|
|
if (side->toptexture > 0) //jff 8/14/98 texture 0 is a placeholder
|
|
if (textureheight[side->toptexture] < minsize)
|
|
minsize = textureheight[side->toptexture];
|
|
}
|
|
}
|
|
return minsize;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindModelFloorSector()
|
|
//
|
|
// Passed a floor height and a sector number, return a pointer to a
|
|
// a sector with that floor height across the lowest numbered two sided
|
|
// line surrounding the sector.
|
|
//
|
|
// Note: If no sector at that height bounds the sector passed, return NULL
|
|
//
|
|
// jff 02/03/98 Add routine to find numeric model floor
|
|
// around a sector specified by sector number
|
|
// jff 3/14/98 change first parameter to plain height to allow call
|
|
// from routine not using floormove_t
|
|
//
|
|
sector_t *P_FindModelFloorSector(fixed_t floordestheight,int secnum)
|
|
{
|
|
int i;
|
|
sector_t *sec=NULL;
|
|
int linecount;
|
|
|
|
sec = §ors[secnum]; //jff 3/2/98 woops! better do this
|
|
//jff 5/23/98 don't disturb sec->linecount while searching
|
|
// but allow early exit in old demos
|
|
linecount = sec->linecount;
|
|
for (i = 0; i < (demo_compatibility && sec->linecount<linecount?
|
|
sec->linecount : linecount); i++)
|
|
{
|
|
if ( twoSided(secnum, i) )
|
|
{
|
|
if (getSide(secnum,i,0)->sector-sectors == secnum)
|
|
sec = getSector(secnum,i,1);
|
|
else
|
|
sec = getSector(secnum,i,0);
|
|
|
|
if (sec->floorheight == floordestheight)
|
|
return sec;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
//
|
|
// P_FindModelCeilingSector()
|
|
//
|
|
// Passed a ceiling height and a sector number, return a pointer to a
|
|
// a sector with that ceiling height across the lowest numbered two sided
|
|
// line surrounding the sector.
|
|
//
|
|
// Note: If no sector at that height bounds the sector passed, return NULL
|
|
//
|
|
// jff 02/03/98 Add routine to find numeric model ceiling
|
|
// around a sector specified by sector number
|
|
// used only from generalized ceiling types
|
|
// jff 3/14/98 change first parameter to plain height to allow call
|
|
// from routine not using ceiling_t
|
|
//
|
|
sector_t *P_FindModelCeilingSector(fixed_t ceildestheight,int secnum)
|
|
{
|
|
int i;
|
|
sector_t *sec=NULL;
|
|
int linecount;
|
|
|
|
sec = §ors[secnum]; //jff 3/2/98 woops! better do this
|
|
//jff 5/23/98 don't disturb sec->linecount while searching
|
|
// but allow early exit in old demos
|
|
linecount = sec->linecount;
|
|
for (i = 0; i < (demo_compatibility && sec->linecount<linecount?
|
|
sec->linecount : linecount); i++)
|
|
{
|
|
if ( twoSided(secnum, i) )
|
|
{
|
|
if (getSide(secnum,i,0)->sector-sectors == secnum)
|
|
sec = getSector(secnum,i,1);
|
|
else
|
|
sec = getSector(secnum,i,0);
|
|
|
|
if (sec->ceilingheight == ceildestheight)
|
|
return sec;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// RETURN NEXT SECTOR # THAT LINE TAG REFERS TO
|
|
//
|
|
|
|
// Find the next sector with the same tag as a linedef.
|
|
// Rewritten by Lee Killough to use chained hashing to improve speed
|
|
|
|
int P_FindSectorFromLineTag(const line_t *line, int start)
|
|
{
|
|
start = start >= 0 ? sectors[start].nexttag :
|
|
sectors[(unsigned) line->tag % (unsigned) numsectors].firsttag;
|
|
while (start >= 0 && sectors[start].tag != line->tag)
|
|
start = sectors[start].nexttag;
|
|
return start;
|
|
}
|
|
|
|
// killough 4/16/98: Same thing, only for linedefs
|
|
|
|
int P_FindLineFromLineTag(const line_t *line, int start)
|
|
{
|
|
start = start >= 0 ? lines[start].nexttag :
|
|
lines[(unsigned) line->tag % (unsigned) numlines].firsttag;
|
|
while (start >= 0 && lines[start].tag != line->tag)
|
|
start = lines[start].nexttag;
|
|
return start;
|
|
}
|
|
|
|
// Hash the sector tags across the sectors and linedefs.
|
|
static void P_InitTagLists(void)
|
|
{
|
|
register int i;
|
|
|
|
for (i=numsectors; --i>=0; ) // Initially make all slots empty.
|
|
sectors[i].firsttag = -1;
|
|
for (i=numsectors; --i>=0; ) // Proceed from last to first sector
|
|
{ // so that lower sectors appear first
|
|
int j = (unsigned) sectors[i].tag % (unsigned) numsectors; // Hash func
|
|
sectors[i].nexttag = sectors[j].firsttag; // Prepend sector to chain
|
|
sectors[j].firsttag = i;
|
|
}
|
|
|
|
// killough 4/17/98: same thing, only for linedefs
|
|
|
|
for (i=numlines; --i>=0; ) // Initially make all slots empty.
|
|
lines[i].firsttag = -1;
|
|
for (i=numlines; --i>=0; ) // Proceed from last to first linedef
|
|
{ // so that lower linedefs appear first
|
|
int j = (unsigned) lines[i].tag % (unsigned) numlines; // Hash func
|
|
lines[i].nexttag = lines[j].firsttag; // Prepend linedef to chain
|
|
lines[j].firsttag = i;
|
|
}
|
|
}
|
|
|
|
//
|
|
// P_FindMinSurroundingLight()
|
|
//
|
|
// Passed a sector and a light level, returns the smallest light level
|
|
// in a surrounding sector less than that passed. If no smaller light
|
|
// level exists, the light level passed is returned.
|
|
//
|
|
int P_FindMinSurroundingLight
|
|
( sector_t* sector,
|
|
int max )
|
|
{
|
|
int i;
|
|
int min;
|
|
line_t* line;
|
|
sector_t* check;
|
|
|
|
min = max;
|
|
for (i=0 ; i < sector->linecount ; i++)
|
|
{
|
|
line = sector->lines[i];
|
|
check = getNextSector(line,sector);
|
|
|
|
if (!check)
|
|
continue;
|
|
|
|
if (check->lightlevel < min)
|
|
min = check->lightlevel;
|
|
}
|
|
return min;
|
|
}
|
|
|
|
|
|
//
|
|
// P_CanUnlockGenDoor()
|
|
//
|
|
// Passed a generalized locked door linedef and a player, returns whether
|
|
// the player has the keys necessary to unlock that door.
|
|
//
|
|
// Note: The linedef passed MUST be a generalized locked door type
|
|
// or results are undefined.
|
|
//
|
|
// jff 02/05/98 routine added to test for unlockability of
|
|
// generalized locked doors
|
|
//
|
|
boolean P_CanUnlockGenDoor
|
|
( line_t* line,
|
|
player_t* player)
|
|
{
|
|
// does this line special distinguish between skulls and keys?
|
|
int skulliscard = (line->special & LockedNKeys)>>LockedNKeysShift;
|
|
|
|
// determine for each case of lock type if player's keys are adequate
|
|
switch((line->special & LockedKey)>>LockedKeyShift)
|
|
{
|
|
case AnyKey:
|
|
if
|
|
(
|
|
!player->cards[it_redcard] &&
|
|
!player->cards[it_redskull] &&
|
|
!player->cards[it_bluecard] &&
|
|
!player->cards[it_blueskull] &&
|
|
!player->cards[it_yellowcard] &&
|
|
!player->cards[it_yellowskull]
|
|
)
|
|
{
|
|
player->message = s_PD_ANY; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case RCard:
|
|
if
|
|
(
|
|
!player->cards[it_redcard] &&
|
|
(!skulliscard || !player->cards[it_redskull])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_REDK : s_PD_REDC; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case BCard:
|
|
if
|
|
(
|
|
!player->cards[it_bluecard] &&
|
|
(!skulliscard || !player->cards[it_blueskull])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_BLUEK : s_PD_BLUEC; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case YCard:
|
|
if
|
|
(
|
|
!player->cards[it_yellowcard] &&
|
|
(!skulliscard || !player->cards[it_yellowskull])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_YELLOWK : s_PD_YELLOWC; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case RSkull:
|
|
if
|
|
(
|
|
!player->cards[it_redskull] &&
|
|
(!skulliscard || !player->cards[it_redcard])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_REDK : s_PD_REDS; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case BSkull:
|
|
if
|
|
(
|
|
!player->cards[it_blueskull] &&
|
|
(!skulliscard || !player->cards[it_bluecard])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_BLUEK : s_PD_BLUES; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case YSkull:
|
|
if
|
|
(
|
|
!player->cards[it_yellowskull] &&
|
|
(!skulliscard || !player->cards[it_yellowcard])
|
|
)
|
|
{
|
|
player->message = skulliscard? s_PD_YELLOWK : s_PD_YELLOWS; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
case AllKeys:
|
|
if
|
|
(
|
|
!skulliscard &&
|
|
(
|
|
!player->cards[it_redcard] ||
|
|
!player->cards[it_redskull] ||
|
|
!player->cards[it_bluecard] ||
|
|
!player->cards[it_blueskull] ||
|
|
!player->cards[it_yellowcard] ||
|
|
!player->cards[it_yellowskull]
|
|
)
|
|
)
|
|
{
|
|
player->message = s_PD_ALL6; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
if
|
|
(
|
|
skulliscard &&
|
|
(
|
|
(!player->cards[it_redcard] &&
|
|
!player->cards[it_redskull]) ||
|
|
(!player->cards[it_bluecard] &&
|
|
!player->cards[it_blueskull]) ||
|
|
(!player->cards[it_yellowcard] &&
|
|
!player->cards[it_yellowskull])
|
|
)
|
|
)
|
|
{
|
|
player->message = s_PD_ALL3; // Ty 03/27/98 - externalized
|
|
S_StartSound(player->mo,sfx_oof); // killough 3/20/98
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
//
|
|
// P_SectorActive()
|
|
//
|
|
// Passed a linedef special class (floor, ceiling, lighting) and a sector
|
|
// returns whether the sector is already busy with a linedef special of the
|
|
// same class. If old demo compatibility true, all linedef special classes
|
|
// are the same.
|
|
//
|
|
// jff 2/23/98 added to prevent old demos from
|
|
// succeeding in starting multiple specials on one sector
|
|
//
|
|
boolean P_SectorActive(special_e t,sector_t *sec)
|
|
{
|
|
if (demo_compatibility) // return whether any thinker is active
|
|
return sec->floordata || sec->ceilingdata || sec->lightingdata;
|
|
else
|
|
switch (t) // return whether thinker of same type is active
|
|
{
|
|
case floor_special:
|
|
return (sec->floordata!=NULL);
|
|
case ceiling_special:
|
|
return (sec->ceilingdata!=NULL);
|
|
case lighting_special:
|
|
return (sec->lightingdata!=NULL);
|
|
}
|
|
return true; // don't know which special, must be active, shouldn't be here
|
|
}
|
|
|
|
|
|
//
|
|
// P_CheckTag()
|
|
//
|
|
// Passed a line, returns true if the tag is non-zero or the line special
|
|
// allows no tag without harm. If compatibility, all linedef specials are
|
|
// allowed to have zero tag.
|
|
//
|
|
// Note: Only line specials activated by walkover, pushing, or shooting are
|
|
// checked by this routine.
|
|
//
|
|
// jff 2/27/98 Added to check for zero tag allowed for regular special types
|
|
//
|
|
int P_CheckTag(line_t *line)
|
|
{
|
|
/* tag not zero, allowed, or
|
|
* killough 11/98: compatibility option */
|
|
if (comp[comp_zerotags] || line->tag)
|
|
return 1;
|
|
|
|
switch(line->special)
|
|
{
|
|
case 1: // Manual door specials
|
|
case 26:
|
|
case 27:
|
|
case 28:
|
|
case 31:
|
|
case 32:
|
|
case 33:
|
|
case 34:
|
|
case 117:
|
|
case 118:
|
|
|
|
case 139: // Lighting specials
|
|
case 170:
|
|
case 79:
|
|
case 35:
|
|
case 138:
|
|
case 171:
|
|
case 81:
|
|
case 13:
|
|
case 192:
|
|
case 169:
|
|
case 80:
|
|
case 12:
|
|
case 194:
|
|
case 173:
|
|
case 157:
|
|
case 104:
|
|
case 193:
|
|
case 172:
|
|
case 156:
|
|
case 17:
|
|
|
|
case 195: // Thing teleporters
|
|
case 174:
|
|
case 97:
|
|
case 39:
|
|
case 126:
|
|
case 125:
|
|
case 210:
|
|
case 209:
|
|
case 208:
|
|
case 207:
|
|
|
|
case 11: // Exits
|
|
case 52:
|
|
case 197:
|
|
case 51:
|
|
case 124:
|
|
case 198:
|
|
|
|
case 48: // Scrolling walls
|
|
case 85:
|
|
return 1; // zero tag allowed
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return 0; // zero tag not allowed
|
|
}
|
|
|
|
|
|
//
|
|
// P_IsSecret()
|
|
//
|
|
// Passed a sector, returns if the sector secret type is still active, i.e.
|
|
// secret type is set and the secret has not yet been obtained.
|
|
//
|
|
// jff 3/14/98 added to simplify checks for whether sector is secret
|
|
// in automap and other places
|
|
//
|
|
boolean P_IsSecret(sector_t *sec)
|
|
{
|
|
return (sec->special==9 || (sec->special&SECRET_MASK));
|
|
}
|
|
|
|
|
|
//
|
|
// P_WasSecret()
|
|
//
|
|
// Passed a sector, returns if the sector secret type is was active, i.e.
|
|
// secret type was set and the secret has been obtained already.
|
|
//
|
|
// jff 3/14/98 added to simplify checks for whether sector is secret
|
|
// in automap and other places
|
|
//
|
|
boolean P_WasSecret(sector_t *sec)
|
|
{
|
|
return (sec->oldspecial==9 || (sec->oldspecial&SECRET_MASK));
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Events
|
|
//
|
|
// Events are operations triggered by using, crossing,
|
|
// or shooting special lines, or by timed thinkers.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// P_CrossSpecialLine - Walkover Trigger Dispatcher
|
|
//
|
|
// Called every time a thing origin is about
|
|
// to cross a line with a non 0 special, whether a walkover type or not.
|
|
//
|
|
// jff 02/12/98 all W1 lines were fixed to check the result from the EV_
|
|
// function before clearing the special. This avoids losing the function
|
|
// of the line, should the sector already be active when the line is
|
|
// crossed. Change is qualified by demo_compatibility.
|
|
//
|
|
// CPhipps - take a line_t pointer instead of a line number, as in MBF
|
|
void P_CrossSpecialLine(line_t *line, int side, mobj_t *thing)
|
|
{
|
|
int ok;
|
|
|
|
// Things that should never trigger lines
|
|
if (!thing->player)
|
|
{
|
|
// Things that should NOT trigger specials...
|
|
switch(thing->type)
|
|
{
|
|
case MT_ROCKET:
|
|
case MT_PLASMA:
|
|
case MT_BFG:
|
|
case MT_TROOPSHOT:
|
|
case MT_HEADSHOT:
|
|
case MT_BRUISERSHOT:
|
|
return;
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
//jff 02/04/98 add check here for generalized lindef types
|
|
if (!demo_compatibility) // generalized types not recognized if old demo
|
|
{
|
|
// pointer to line function is NULL by default, set non-null if
|
|
// line special is walkover generalized linedef type
|
|
int (*linefunc)(line_t *line)=NULL;
|
|
|
|
// check each range of generalized linedefs
|
|
if ((unsigned)line->special >= GenEnd)
|
|
{
|
|
// Out of range for GenFloors
|
|
}
|
|
else if ((unsigned)line->special >= GenFloorBase)
|
|
{
|
|
if (!thing->player)
|
|
if ((line->special & FloorChange) || !(line->special & FloorModel))
|
|
return; // FloorModel is "Allow Monsters" if FloorChange is 0
|
|
if (!line->tag) //jff 2/27/98 all walk generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenFloor;
|
|
}
|
|
else if ((unsigned)line->special >= GenCeilingBase)
|
|
{
|
|
if (!thing->player)
|
|
if ((line->special & CeilingChange) || !(line->special & CeilingModel))
|
|
return; // CeilingModel is "Allow Monsters" if CeilingChange is 0
|
|
if (!line->tag) //jff 2/27/98 all walk generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenCeiling;
|
|
}
|
|
else if ((unsigned)line->special >= GenDoorBase)
|
|
{
|
|
if (!thing->player)
|
|
{
|
|
if (!(line->special & DoorMonster))
|
|
return; // monsters disallowed from this door
|
|
if (line->flags & ML_SECRET) // they can't open secret doors either
|
|
return;
|
|
}
|
|
if (!line->tag) //3/2/98 move outside the monster check
|
|
return;
|
|
linefunc = EV_DoGenDoor;
|
|
}
|
|
else if ((unsigned)line->special >= GenLockedBase)
|
|
{
|
|
if (!thing->player)
|
|
return; // monsters disallowed from unlocking doors
|
|
if (((line->special&TriggerType)==WalkOnce) || ((line->special&TriggerType)==WalkMany))
|
|
{ //jff 4/1/98 check for being a walk type before reporting door type
|
|
if (!P_CanUnlockGenDoor(line,thing->player))
|
|
return;
|
|
}
|
|
else
|
|
return;
|
|
linefunc = EV_DoGenLockedDoor;
|
|
}
|
|
else if ((unsigned)line->special >= GenLiftBase)
|
|
{
|
|
if (!thing->player)
|
|
if (!(line->special & LiftMonster))
|
|
return; // monsters disallowed
|
|
if (!line->tag) //jff 2/27/98 all walk generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenLift;
|
|
}
|
|
else if ((unsigned)line->special >= GenStairsBase)
|
|
{
|
|
if (!thing->player)
|
|
if (!(line->special & StairMonster))
|
|
return; // monsters disallowed
|
|
if (!line->tag) //jff 2/27/98 all walk generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenStairs;
|
|
}
|
|
|
|
if (linefunc) // if it was a valid generalized type
|
|
switch((line->special & TriggerType) >> TriggerTypeShift)
|
|
{
|
|
case WalkOnce:
|
|
if (linefunc(line))
|
|
line->special = 0; // clear special if a walk once type
|
|
return;
|
|
case WalkMany:
|
|
linefunc(line);
|
|
return;
|
|
default: // if not a walk type, do nothing here
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!thing->player)
|
|
{
|
|
ok = 0;
|
|
switch(line->special)
|
|
{
|
|
case 39: // teleport trigger
|
|
case 97: // teleport retrigger
|
|
case 125: // teleport monsteronly trigger
|
|
case 126: // teleport monsteronly retrigger
|
|
case 4: // raise door
|
|
case 10: // plat down-wait-up-stay trigger
|
|
case 88: // plat down-wait-up-stay retrigger
|
|
//jff 3/5/98 add ability of monsters etc. to use teleporters
|
|
case 208: //silent thing teleporters
|
|
case 207:
|
|
case 243: //silent line-line teleporter
|
|
case 244: //jff 3/6/98 make fit within DCK's 256 linedef types
|
|
case 262: //jff 4/14/98 add monster only
|
|
case 263: //jff 4/14/98 silent thing,line,line rev types
|
|
case 264: //jff 4/14/98 plus player/monster silent line
|
|
case 265: // reversed types
|
|
case 266:
|
|
case 267:
|
|
case 268:
|
|
case 269:
|
|
ok = 1;
|
|
break;
|
|
}
|
|
if (!ok)
|
|
return;
|
|
}
|
|
|
|
if (!P_CheckTag(line)) //jff 2/27/98 disallow zero tag on some types
|
|
return;
|
|
|
|
// Dispatch on the line special value to the line's action routine
|
|
// If a once only function, and successful, clear the line special
|
|
|
|
switch (line->special)
|
|
{
|
|
// Regular walk once triggers
|
|
|
|
case 2:
|
|
// Open Door
|
|
if (EV_DoDoor(line,p_open) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 3:
|
|
// Close Door
|
|
if (EV_DoDoor(line,p_close) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 4:
|
|
// Raise Door
|
|
if (EV_DoDoor(line,normal) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 5:
|
|
// Raise Floor
|
|
if (EV_DoFloor(line,raiseFloor) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 6:
|
|
// Fast Ceiling Crush & Raise
|
|
if (EV_DoCeiling(line,fastCrushAndRaise) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 8:
|
|
// Build Stairs
|
|
if (EV_BuildStairs(line,build8) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 10:
|
|
// PlatDownWaitUp
|
|
if (EV_DoPlat(line,downWaitUpStay,0) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 12:
|
|
// Light Turn On - brightest near
|
|
if (EV_LightTurnOn(line,0) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 13:
|
|
// Light Turn On 255
|
|
if (EV_LightTurnOn(line,255) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 16:
|
|
// Close Door 30
|
|
if (EV_DoDoor(line,close30ThenOpen) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 17:
|
|
// Start Light Strobing
|
|
if (EV_StartLightStrobing(line) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 19:
|
|
// Lower Floor
|
|
if (EV_DoFloor(line,lowerFloor) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 22:
|
|
// Raise floor to nearest height and change texture
|
|
if (EV_DoPlat(line,raiseToNearestAndChange,0) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 25:
|
|
// Ceiling Crush and Raise
|
|
if (EV_DoCeiling(line,crushAndRaise) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 30:
|
|
// Raise floor to shortest texture height
|
|
// on either side of lines.
|
|
if (EV_DoFloor(line,raiseToTexture) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 35:
|
|
// Lights Very Dark
|
|
if (EV_LightTurnOn(line,35) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 36:
|
|
// Lower Floor (TURBO)
|
|
if (EV_DoFloor(line,turboLower) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 37:
|
|
// LowerAndChange
|
|
if (EV_DoFloor(line,lowerAndChange) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 38:
|
|
// Lower Floor To Lowest
|
|
if (EV_DoFloor(line, lowerFloorToLowest) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 39:
|
|
// TELEPORT! //jff 02/09/98 fix using up with wrong side crossing
|
|
if (EV_Teleport(line, side, thing) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 40:
|
|
// RaiseCeilingLowerFloor
|
|
if (demo_compatibility)
|
|
{
|
|
EV_DoCeiling( line, raiseToHighest );
|
|
EV_DoFloor( line, lowerFloorToLowest ); //jff 02/12/98 doesn't work
|
|
line->special = 0;
|
|
}
|
|
else
|
|
if (EV_DoCeiling(line, raiseToHighest))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 44:
|
|
// Ceiling Crush
|
|
if (EV_DoCeiling(line, lowerAndCrush) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 52:
|
|
// EXIT!
|
|
// killough 10/98: prevent zombies from exiting levels
|
|
if (!(thing->player && thing->player->health <= 0 && !comp[comp_zombie]))
|
|
G_ExitLevel ();
|
|
break;
|
|
|
|
case 53:
|
|
// Perpetual Platform Raise
|
|
if (EV_DoPlat(line,perpetualRaise,0) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 54:
|
|
// Platform Stop
|
|
if (EV_StopPlat(line) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 56:
|
|
// Raise Floor Crush
|
|
if (EV_DoFloor(line,raiseFloorCrush) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 57:
|
|
// Ceiling Crush Stop
|
|
if (EV_CeilingCrushStop(line) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 58:
|
|
// Raise Floor 24
|
|
if (EV_DoFloor(line,raiseFloor24) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 59:
|
|
// Raise Floor 24 And Change
|
|
if (EV_DoFloor(line,raiseFloor24AndChange) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 100:
|
|
// Build Stairs Turbo 16
|
|
if (EV_BuildStairs(line,turbo16) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 104:
|
|
// Turn lights off in sector(tag)
|
|
if (EV_TurnTagLightsOff(line) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 108:
|
|
// Blazing Door Raise (faster than TURBO!)
|
|
if (EV_DoDoor(line,blazeRaise) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 109:
|
|
// Blazing Door Open (faster than TURBO!)
|
|
if (EV_DoDoor (line,blazeOpen) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 110:
|
|
// Blazing Door Close (faster than TURBO!)
|
|
if (EV_DoDoor (line,blazeClose) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 119:
|
|
// Raise floor to nearest surr. floor
|
|
if (EV_DoFloor(line,raiseFloorToNearest) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 121:
|
|
// Blazing PlatDownWaitUpStay
|
|
if (EV_DoPlat(line,blazeDWUS,0) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 124:
|
|
// Secret EXIT
|
|
// killough 10/98: prevent zombies from exiting levels
|
|
// CPhipps - change for lxdoom's compatibility handling
|
|
if (!(thing->player && thing->player->health <= 0 && !comp[comp_zombie]))
|
|
G_SecretExitLevel ();
|
|
break;
|
|
|
|
case 125:
|
|
// TELEPORT MonsterONLY
|
|
if (!thing->player &&
|
|
(EV_Teleport(line, side, thing) || demo_compatibility))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 130:
|
|
// Raise Floor Turbo
|
|
if (EV_DoFloor(line,raiseFloorTurbo) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 141:
|
|
// Silent Ceiling Crush & Raise
|
|
if (EV_DoCeiling(line,silentCrushAndRaise) || demo_compatibility)
|
|
line->special = 0;
|
|
break;
|
|
|
|
// Regular walk many retriggerable
|
|
|
|
case 72:
|
|
// Ceiling Crush
|
|
EV_DoCeiling( line, lowerAndCrush );
|
|
break;
|
|
|
|
case 73:
|
|
// Ceiling Crush and Raise
|
|
EV_DoCeiling(line,crushAndRaise);
|
|
break;
|
|
|
|
case 74:
|
|
// Ceiling Crush Stop
|
|
EV_CeilingCrushStop(line);
|
|
break;
|
|
|
|
case 75:
|
|
// Close Door
|
|
EV_DoDoor(line,p_close);
|
|
break;
|
|
|
|
case 76:
|
|
// Close Door 30
|
|
EV_DoDoor(line,close30ThenOpen);
|
|
break;
|
|
|
|
case 77:
|
|
// Fast Ceiling Crush & Raise
|
|
EV_DoCeiling(line,fastCrushAndRaise);
|
|
break;
|
|
|
|
case 79:
|
|
// Lights Very Dark
|
|
EV_LightTurnOn(line,35);
|
|
break;
|
|
|
|
case 80:
|
|
// Light Turn On - brightest near
|
|
EV_LightTurnOn(line,0);
|
|
break;
|
|
|
|
case 81:
|
|
// Light Turn On 255
|
|
EV_LightTurnOn(line,255);
|
|
break;
|
|
|
|
case 82:
|
|
// Lower Floor To Lowest
|
|
EV_DoFloor( line, lowerFloorToLowest );
|
|
break;
|
|
|
|
case 83:
|
|
// Lower Floor
|
|
EV_DoFloor(line,lowerFloor);
|
|
break;
|
|
|
|
case 84:
|
|
// LowerAndChange
|
|
EV_DoFloor(line,lowerAndChange);
|
|
break;
|
|
|
|
case 86:
|
|
// Open Door
|
|
EV_DoDoor(line,p_open);
|
|
break;
|
|
|
|
case 87:
|
|
// Perpetual Platform Raise
|
|
EV_DoPlat(line,perpetualRaise,0);
|
|
break;
|
|
|
|
case 88:
|
|
// PlatDownWaitUp
|
|
EV_DoPlat(line,downWaitUpStay,0);
|
|
break;
|
|
|
|
case 89:
|
|
// Platform Stop
|
|
EV_StopPlat(line);
|
|
break;
|
|
|
|
case 90:
|
|
// Raise Door
|
|
EV_DoDoor(line,normal);
|
|
break;
|
|
|
|
case 91:
|
|
// Raise Floor
|
|
EV_DoFloor(line,raiseFloor);
|
|
break;
|
|
|
|
case 92:
|
|
// Raise Floor 24
|
|
EV_DoFloor(line,raiseFloor24);
|
|
break;
|
|
|
|
case 93:
|
|
// Raise Floor 24 And Change
|
|
EV_DoFloor(line,raiseFloor24AndChange);
|
|
break;
|
|
|
|
case 94:
|
|
// Raise Floor Crush
|
|
EV_DoFloor(line,raiseFloorCrush);
|
|
break;
|
|
|
|
case 95:
|
|
// Raise floor to nearest height
|
|
// and change texture.
|
|
EV_DoPlat(line,raiseToNearestAndChange,0);
|
|
break;
|
|
|
|
case 96:
|
|
// Raise floor to shortest texture height
|
|
// on either side of lines.
|
|
EV_DoFloor(line,raiseToTexture);
|
|
break;
|
|
|
|
case 97:
|
|
// TELEPORT!
|
|
EV_Teleport( line, side, thing );
|
|
break;
|
|
|
|
case 98:
|
|
// Lower Floor (TURBO)
|
|
EV_DoFloor(line,turboLower);
|
|
break;
|
|
|
|
case 105:
|
|
// Blazing Door Raise (faster than TURBO!)
|
|
EV_DoDoor (line,blazeRaise);
|
|
break;
|
|
|
|
case 106:
|
|
// Blazing Door Open (faster than TURBO!)
|
|
EV_DoDoor (line,blazeOpen);
|
|
break;
|
|
|
|
case 107:
|
|
// Blazing Door Close (faster than TURBO!)
|
|
EV_DoDoor (line,blazeClose);
|
|
break;
|
|
|
|
case 120:
|
|
// Blazing PlatDownWaitUpStay.
|
|
EV_DoPlat(line,blazeDWUS,0);
|
|
break;
|
|
|
|
case 126:
|
|
// TELEPORT MonsterONLY.
|
|
if (!thing->player)
|
|
EV_Teleport( line, side, thing );
|
|
break;
|
|
|
|
case 128:
|
|
// Raise To Nearest Floor
|
|
EV_DoFloor(line,raiseFloorToNearest);
|
|
break;
|
|
|
|
case 129:
|
|
// Raise Floor Turbo
|
|
EV_DoFloor(line,raiseFloorTurbo);
|
|
break;
|
|
|
|
// Extended walk triggers
|
|
|
|
// jff 1/29/98 added new linedef types to fill all functions out so that
|
|
// all have varieties SR, S1, WR, W1
|
|
|
|
// killough 1/31/98: "factor out" compatibility test, by
|
|
// adding inner switch qualified by compatibility flag.
|
|
// relax test to demo_compatibility
|
|
|
|
// killough 2/16/98: Fix problems with W1 types being cleared too early
|
|
|
|
default:
|
|
if (!demo_compatibility)
|
|
switch (line->special)
|
|
{
|
|
// Extended walk once triggers
|
|
|
|
case 142:
|
|
// Raise Floor 512
|
|
// 142 W1 EV_DoFloor(raiseFloor512)
|
|
if (EV_DoFloor(line,raiseFloor512))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 143:
|
|
// Raise Floor 24 and change
|
|
// 143 W1 EV_DoPlat(raiseAndChange,24)
|
|
if (EV_DoPlat(line,raiseAndChange,24))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 144:
|
|
// Raise Floor 32 and change
|
|
// 144 W1 EV_DoPlat(raiseAndChange,32)
|
|
if (EV_DoPlat(line,raiseAndChange,32))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 145:
|
|
// Lower Ceiling to Floor
|
|
// 145 W1 EV_DoCeiling(lowerToFloor)
|
|
if (EV_DoCeiling( line, lowerToFloor ))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 146:
|
|
// Lower Pillar, Raise Donut
|
|
// 146 W1 EV_DoDonut()
|
|
if (EV_DoDonut(line))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 199:
|
|
// Lower ceiling to lowest surrounding ceiling
|
|
// 199 W1 EV_DoCeiling(lowerToLowest)
|
|
if (EV_DoCeiling(line,lowerToLowest))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 200:
|
|
// Lower ceiling to highest surrounding floor
|
|
// 200 W1 EV_DoCeiling(lowerToMaxFloor)
|
|
if (EV_DoCeiling(line,lowerToMaxFloor))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 207:
|
|
// killough 2/16/98: W1 silent teleporter (normal kind)
|
|
if (EV_SilentTeleport(line, side, thing))
|
|
line->special = 0;
|
|
break;
|
|
|
|
//jff 3/16/98 renumber 215->153
|
|
case 153: //jff 3/15/98 create texture change no motion type
|
|
// Texture/Type Change Only (Trig)
|
|
// 153 W1 Change Texture/Type Only
|
|
if (EV_DoChange(line,trigChangeOnly))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 239: //jff 3/15/98 create texture change no motion type
|
|
// Texture/Type Change Only (Numeric)
|
|
// 239 W1 Change Texture/Type Only
|
|
if (EV_DoChange(line,numChangeOnly))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 219:
|
|
// Lower floor to next lower neighbor
|
|
// 219 W1 Lower Floor Next Lower Neighbor
|
|
if (EV_DoFloor(line,lowerFloorToNearest))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 227:
|
|
// Raise elevator next floor
|
|
// 227 W1 Raise Elevator next floor
|
|
if (EV_DoElevator(line,elevateUp))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 231:
|
|
// Lower elevator next floor
|
|
// 231 W1 Lower Elevator next floor
|
|
if (EV_DoElevator(line,elevateDown))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 235:
|
|
// Elevator to current floor
|
|
// 235 W1 Elevator to current floor
|
|
if (EV_DoElevator(line,elevateCurrent))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 243: //jff 3/6/98 make fit within DCK's 256 linedef types
|
|
// killough 2/16/98: W1 silent teleporter (linedef-linedef kind)
|
|
if (EV_SilentLineTeleport(line, side, thing, false))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 262: //jff 4/14/98 add silent line-line reversed
|
|
if (EV_SilentLineTeleport(line, side, thing, true))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 264: //jff 4/14/98 add monster-only silent line-line reversed
|
|
if (!thing->player &&
|
|
EV_SilentLineTeleport(line, side, thing, true))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 266: //jff 4/14/98 add monster-only silent line-line
|
|
if (!thing->player &&
|
|
EV_SilentLineTeleport(line, side, thing, false))
|
|
line->special = 0;
|
|
break;
|
|
|
|
case 268: //jff 4/14/98 add monster-only silent
|
|
if (!thing->player && EV_SilentTeleport(line, side, thing))
|
|
line->special = 0;
|
|
break;
|
|
|
|
//jff 1/29/98 end of added W1 linedef types
|
|
|
|
// Extended walk many retriggerable
|
|
|
|
//jff 1/29/98 added new linedef types to fill all functions
|
|
//out so that all have varieties SR, S1, WR, W1
|
|
|
|
case 147:
|
|
// Raise Floor 512
|
|
// 147 WR EV_DoFloor(raiseFloor512)
|
|
EV_DoFloor(line,raiseFloor512);
|
|
break;
|
|
|
|
case 148:
|
|
// Raise Floor 24 and Change
|
|
// 148 WR EV_DoPlat(raiseAndChange,24)
|
|
EV_DoPlat(line,raiseAndChange,24);
|
|
break;
|
|
|
|
case 149:
|
|
// Raise Floor 32 and Change
|
|
// 149 WR EV_DoPlat(raiseAndChange,32)
|
|
EV_DoPlat(line,raiseAndChange,32);
|
|
break;
|
|
|
|
case 150:
|
|
// Start slow silent crusher
|
|
// 150 WR EV_DoCeiling(silentCrushAndRaise)
|
|
EV_DoCeiling(line,silentCrushAndRaise);
|
|
break;
|
|
|
|
case 151:
|
|
// RaiseCeilingLowerFloor
|
|
// 151 WR EV_DoCeiling(raiseToHighest),
|
|
// EV_DoFloor(lowerFloortoLowest)
|
|
EV_DoCeiling( line, raiseToHighest );
|
|
EV_DoFloor( line, lowerFloorToLowest );
|
|
break;
|
|
|
|
case 152:
|
|
// Lower Ceiling to Floor
|
|
// 152 WR EV_DoCeiling(lowerToFloor)
|
|
EV_DoCeiling( line, lowerToFloor );
|
|
break;
|
|
|
|
//jff 3/16/98 renumber 153->256
|
|
case 256:
|
|
// Build stairs, step 8
|
|
// 256 WR EV_BuildStairs(build8)
|
|
EV_BuildStairs(line,build8);
|
|
break;
|
|
|
|
//jff 3/16/98 renumber 154->257
|
|
case 257:
|
|
// Build stairs, step 16
|
|
// 257 WR EV_BuildStairs(turbo16)
|
|
EV_BuildStairs(line,turbo16);
|
|
break;
|
|
|
|
case 155:
|
|
// Lower Pillar, Raise Donut
|
|
// 155 WR EV_DoDonut()
|
|
EV_DoDonut(line);
|
|
break;
|
|
|
|
case 156:
|
|
// Start lights strobing
|
|
// 156 WR Lights EV_StartLightStrobing()
|
|
EV_StartLightStrobing(line);
|
|
break;
|
|
|
|
case 157:
|
|
// Lights to dimmest near
|
|
// 157 WR Lights EV_TurnTagLightsOff()
|
|
EV_TurnTagLightsOff(line);
|
|
break;
|
|
|
|
case 201:
|
|
// Lower ceiling to lowest surrounding ceiling
|
|
// 201 WR EV_DoCeiling(lowerToLowest)
|
|
EV_DoCeiling(line,lowerToLowest);
|
|
break;
|
|
|
|
case 202:
|
|
// Lower ceiling to highest surrounding floor
|
|
// 202 WR EV_DoCeiling(lowerToMaxFloor)
|
|
EV_DoCeiling(line,lowerToMaxFloor);
|
|
break;
|
|
|
|
case 208:
|
|
// killough 2/16/98: WR silent teleporter (normal kind)
|
|
EV_SilentTeleport(line, side, thing);
|
|
break;
|
|
|
|
case 212: //jff 3/14/98 create instant toggle floor type
|
|
// Toggle floor between C and F instantly
|
|
// 212 WR Instant Toggle Floor
|
|
EV_DoPlat(line,toggleUpDn,0);
|
|
break;
|
|
|
|
//jff 3/16/98 renumber 216->154
|
|
case 154: //jff 3/15/98 create texture change no motion type
|
|
// Texture/Type Change Only (Trigger)
|
|
// 154 WR Change Texture/Type Only
|
|
EV_DoChange(line,trigChangeOnly);
|
|
break;
|
|
|
|
case 240: //jff 3/15/98 create texture change no motion type
|
|
// Texture/Type Change Only (Numeric)
|
|
// 240 WR Change Texture/Type Only
|
|
EV_DoChange(line,numChangeOnly);
|
|
break;
|
|
|
|
case 220:
|
|
// Lower floor to next lower neighbor
|
|
// 220 WR Lower Floor Next Lower Neighbor
|
|
EV_DoFloor(line,lowerFloorToNearest);
|
|
break;
|
|
|
|
case 228:
|
|
// Raise elevator next floor
|
|
// 228 WR Raise Elevator next floor
|
|
EV_DoElevator(line,elevateUp);
|
|
break;
|
|
|
|
case 232:
|
|
// Lower elevator next floor
|
|
// 232 WR Lower Elevator next floor
|
|
EV_DoElevator(line,elevateDown);
|
|
break;
|
|
|
|
case 236:
|
|
// Elevator to current floor
|
|
// 236 WR Elevator to current floor
|
|
EV_DoElevator(line,elevateCurrent);
|
|
break;
|
|
|
|
case 244: //jff 3/6/98 make fit within DCK's 256 linedef types
|
|
// killough 2/16/98: WR silent teleporter (linedef-linedef kind)
|
|
EV_SilentLineTeleport(line, side, thing, false);
|
|
break;
|
|
|
|
case 263: //jff 4/14/98 add silent line-line reversed
|
|
EV_SilentLineTeleport(line, side, thing, true);
|
|
break;
|
|
|
|
case 265: //jff 4/14/98 add monster-only silent line-line reversed
|
|
if (!thing->player)
|
|
EV_SilentLineTeleport(line, side, thing, true);
|
|
break;
|
|
|
|
case 267: //jff 4/14/98 add monster-only silent line-line
|
|
if (!thing->player)
|
|
EV_SilentLineTeleport(line, side, thing, false);
|
|
break;
|
|
|
|
case 269: //jff 4/14/98 add monster-only silent
|
|
if (!thing->player)
|
|
EV_SilentTeleport(line, side, thing);
|
|
break;
|
|
|
|
//jff 1/29/98 end of added WR linedef types
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// P_ShootSpecialLine - Gun trigger special dispatcher
|
|
//
|
|
// Called when a thing shoots a special line with bullet, shell, saw, or fist.
|
|
//
|
|
// jff 02/12/98 all G1 lines were fixed to check the result from the EV_
|
|
// function before clearing the special. This avoids losing the function
|
|
// of the line, should the sector already be in motion when the line is
|
|
// impacted. Change is qualified by demo_compatibility.
|
|
//
|
|
void P_ShootSpecialLine
|
|
( mobj_t* thing,
|
|
line_t* line )
|
|
{
|
|
//jff 02/04/98 add check here for generalized linedef
|
|
if (!demo_compatibility)
|
|
{
|
|
// pointer to line function is NULL by default, set non-null if
|
|
// line special is gun triggered generalized linedef type
|
|
int (*linefunc)(line_t *line)=NULL;
|
|
|
|
// check each range of generalized linedefs
|
|
if ((unsigned)line->special >= GenEnd)
|
|
{
|
|
// Out of range for GenFloors
|
|
}
|
|
else if ((unsigned)line->special >= GenFloorBase)
|
|
{
|
|
if (!thing->player)
|
|
if ((line->special & FloorChange) || !(line->special & FloorModel))
|
|
return; // FloorModel is "Allow Monsters" if FloorChange is 0
|
|
if (!line->tag) //jff 2/27/98 all gun generalized types require tag
|
|
return;
|
|
|
|
linefunc = EV_DoGenFloor;
|
|
}
|
|
else if ((unsigned)line->special >= GenCeilingBase)
|
|
{
|
|
if (!thing->player)
|
|
if ((line->special & CeilingChange) || !(line->special & CeilingModel))
|
|
return; // CeilingModel is "Allow Monsters" if CeilingChange is 0
|
|
if (!line->tag) //jff 2/27/98 all gun generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenCeiling;
|
|
}
|
|
else if ((unsigned)line->special >= GenDoorBase)
|
|
{
|
|
if (!thing->player)
|
|
{
|
|
if (!(line->special & DoorMonster))
|
|
return; // monsters disallowed from this door
|
|
if (line->flags & ML_SECRET) // they can't open secret doors either
|
|
return;
|
|
}
|
|
if (!line->tag) //jff 3/2/98 all gun generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenDoor;
|
|
}
|
|
else if ((unsigned)line->special >= GenLockedBase)
|
|
{
|
|
if (!thing->player)
|
|
return; // monsters disallowed from unlocking doors
|
|
if (((line->special&TriggerType)==GunOnce) || ((line->special&TriggerType)==GunMany))
|
|
{ //jff 4/1/98 check for being a gun type before reporting door type
|
|
if (!P_CanUnlockGenDoor(line,thing->player))
|
|
return;
|
|
}
|
|
else
|
|
return;
|
|
if (!line->tag) //jff 2/27/98 all gun generalized types require tag
|
|
return;
|
|
|
|
linefunc = EV_DoGenLockedDoor;
|
|
}
|
|
else if ((unsigned)line->special >= GenLiftBase)
|
|
{
|
|
if (!thing->player)
|
|
if (!(line->special & LiftMonster))
|
|
return; // monsters disallowed
|
|
linefunc = EV_DoGenLift;
|
|
}
|
|
else if ((unsigned)line->special >= GenStairsBase)
|
|
{
|
|
if (!thing->player)
|
|
if (!(line->special & StairMonster))
|
|
return; // monsters disallowed
|
|
if (!line->tag) //jff 2/27/98 all gun generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenStairs;
|
|
}
|
|
else if ((unsigned)line->special >= GenCrusherBase)
|
|
{
|
|
if (!thing->player)
|
|
if (!(line->special & StairMonster))
|
|
return; // monsters disallowed
|
|
if (!line->tag) //jff 2/27/98 all gun generalized types require tag
|
|
return;
|
|
linefunc = EV_DoGenCrusher;
|
|
}
|
|
|
|
if (linefunc)
|
|
switch((line->special & TriggerType) >> TriggerTypeShift)
|
|
{
|
|
case GunOnce:
|
|
if (linefunc(line))
|
|
P_ChangeSwitchTexture(line,0);
|
|
return;
|
|
case GunMany:
|
|
if (linefunc(line))
|
|
P_ChangeSwitchTexture(line,1);
|
|
return;
|
|
default: // if not a gun type, do nothing here
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Impacts that other things can activate.
|
|
if (!thing->player)
|
|
{
|
|
int ok = 0;
|
|
switch(line->special)
|
|
{
|
|
case 46:
|
|
// 46 GR Open door on impact weapon is monster activatable
|
|
ok = 1;
|
|
break;
|
|
}
|
|
if (!ok)
|
|
return;
|
|
}
|
|
|
|
if (!P_CheckTag(line)) //jff 2/27/98 disallow zero tag on some types
|
|
return;
|
|
|
|
switch(line->special)
|
|
{
|
|
case 24:
|
|
// 24 G1 raise floor to highest adjacent
|
|
if (EV_DoFloor(line,raiseFloor) || demo_compatibility)
|
|
P_ChangeSwitchTexture(line,0);
|
|
break;
|
|
|
|
case 46:
|
|
// 46 GR open door, stay open
|
|
EV_DoDoor(line,p_open);
|
|
P_ChangeSwitchTexture(line,1);
|
|
break;
|
|
|
|
case 47:
|
|
// 47 G1 raise floor to nearest and change texture and type
|
|
if (EV_DoPlat(line,raiseToNearestAndChange,0) || demo_compatibility)
|
|
P_ChangeSwitchTexture(line,0);
|
|
break;
|
|
|
|
//jff 1/30/98 added new gun linedefs here
|
|
// killough 1/31/98: added demo_compatibility check, added inner switch
|
|
|
|
default:
|
|
if (!demo_compatibility)
|
|
switch (line->special)
|
|
{
|
|
case 197:
|
|
// Exit to next level
|
|
// killough 10/98: prevent zombies from exiting levels
|
|
if(thing->player && thing->player->health<=0 && !comp[comp_zombie])
|
|
break;
|
|
P_ChangeSwitchTexture(line,0);
|
|
G_ExitLevel();
|
|
break;
|
|
|
|
case 198:
|
|
// Exit to secret level
|
|
// killough 10/98: prevent zombies from exiting levels
|
|
if(thing->player && thing->player->health<=0 && !comp[comp_zombie])
|
|
break;
|
|
P_ChangeSwitchTexture(line,0);
|
|
G_SecretExitLevel();
|
|
break;
|
|
//jff end addition of new gun linedefs
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// P_PlayerInSpecialSector()
|
|
//
|
|
// Called every tick frame
|
|
// that the player origin is in a special sector
|
|
//
|
|
// Changed to ignore sector types the engine does not recognize
|
|
//
|
|
void P_PlayerInSpecialSector (player_t* player)
|
|
{
|
|
sector_t* sector;
|
|
|
|
sector = player->mo->subsector->sector;
|
|
|
|
// Falling, not all the way down yet?
|
|
// Sector specials don't apply in mid-air
|
|
if (player->mo->z != sector->floorheight)
|
|
return;
|
|
|
|
// Has hit ground.
|
|
//jff add if to handle old vs generalized types
|
|
if (sector->special<32) // regular sector specials
|
|
{
|
|
switch (sector->special)
|
|
{
|
|
case 5:
|
|
// 5/10 unit damage per 31 ticks
|
|
if (!player->powers[pw_ironfeet])
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 10);
|
|
break;
|
|
|
|
case 7:
|
|
// 2/5 unit damage per 31 ticks
|
|
if (!player->powers[pw_ironfeet])
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 5);
|
|
break;
|
|
|
|
case 16:
|
|
// 10/20 unit damage per 31 ticks
|
|
case 4:
|
|
// 10/20 unit damage plus blinking light (light already spawned)
|
|
if (!player->powers[pw_ironfeet]
|
|
|| (P_Random(pr_slimehurt)<5) ) // even with suit, take damage
|
|
{
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 20);
|
|
}
|
|
break;
|
|
|
|
case 9:
|
|
// Tally player in secret sector, clear secret special
|
|
player->secretcount++;
|
|
sector->special = 0;
|
|
break;
|
|
|
|
case 11:
|
|
// Exit on health < 11, take 10/20 damage per 31 ticks
|
|
if (comp[comp_god]) /* killough 2/21/98: add compatibility switch */
|
|
player->cheats &= ~CF_GODMODE; // on godmode cheat clearing
|
|
// does not affect invulnerability
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 20);
|
|
|
|
if (player->health <= 10)
|
|
G_ExitLevel();
|
|
break;
|
|
|
|
default:
|
|
//jff 1/24/98 Don't exit as DOOM2 did, just ignore
|
|
break;
|
|
};
|
|
}
|
|
else //jff 3/14/98 handle extended sector types for secrets and damage
|
|
{
|
|
switch ((sector->special&DAMAGE_MASK)>>DAMAGE_SHIFT)
|
|
{
|
|
case 0: // no damage
|
|
break;
|
|
case 1: // 2/5 damage per 31 ticks
|
|
if (!player->powers[pw_ironfeet])
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 5);
|
|
break;
|
|
case 2: // 5/10 damage per 31 ticks
|
|
if (!player->powers[pw_ironfeet])
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 10);
|
|
break;
|
|
case 3: // 10/20 damage per 31 ticks
|
|
if (!player->powers[pw_ironfeet]
|
|
|| (P_Random(pr_slimehurt)<5)) // take damage even with suit
|
|
{
|
|
if (!(leveltime&0x1f))
|
|
P_DamageMobj (player->mo, NULL, NULL, 20);
|
|
}
|
|
break;
|
|
}
|
|
if (sector->special&SECRET_MASK)
|
|
{
|
|
player->secretcount++;
|
|
sector->special &= ~SECRET_MASK;
|
|
if (sector->special<32) // if all extended bits clear,
|
|
sector->special=0; // sector is not special anymore
|
|
}
|
|
|
|
// phares 3/19/98:
|
|
//
|
|
// If FRICTION_MASK or PUSH_MASK is set, we don't care at this
|
|
// point, since the code to deal with those situations is
|
|
// handled by Thinkers.
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
// P_UpdateSpecials()
|
|
//
|
|
// Check level timer, frag counter,
|
|
// animate flats, scroll walls,
|
|
// change button textures
|
|
//
|
|
// Reads and modifies globals:
|
|
// levelTimer, levelTimeCount,
|
|
// levelFragLimit, levelFragLimitCount
|
|
//
|
|
|
|
boolean levelTimer;
|
|
int levelTimeCount;
|
|
boolean levelFragLimit; // Ty 03/18/98 Added -frags support
|
|
int levelFragLimitCount; // Ty 03/18/98 Added -frags support
|
|
|
|
void P_UpdateSpecials (void)
|
|
{
|
|
anim_t* anim;
|
|
int pic;
|
|
int i;
|
|
|
|
// Downcount level timer, exit level if elapsed
|
|
if (levelTimer == true)
|
|
{
|
|
levelTimeCount--;
|
|
if (!levelTimeCount)
|
|
G_ExitLevel();
|
|
}
|
|
|
|
// Check frag counters, if frag limit reached, exit level // Ty 03/18/98
|
|
// Seems like the total frags should be kept in a simple
|
|
// array somewhere, but until they are...
|
|
if (levelFragLimit == true) // we used -frags so compare count
|
|
{
|
|
int k,m,fragcount,exitflag=false;
|
|
for (k=0;k<MAXPLAYERS;k++)
|
|
{
|
|
if (!playeringame[k]) continue;
|
|
fragcount = 0;
|
|
for (m=0;m<MAXPLAYERS;m++)
|
|
{
|
|
if (!playeringame[m]) continue;
|
|
fragcount += (m!=k)? players[k].frags[m] : -players[k].frags[m];
|
|
}
|
|
if (fragcount >= levelFragLimitCount) exitflag = true;
|
|
if (exitflag == true) break; // skip out of the loop--we're done
|
|
}
|
|
if (exitflag == true)
|
|
G_ExitLevel();
|
|
}
|
|
|
|
// Animate flats and textures globally
|
|
for (anim = anims ; anim < lastanim ; anim++)
|
|
{
|
|
for (i=anim->basepic ; i<anim->basepic+anim->numpics ; i++)
|
|
{
|
|
pic = anim->basepic + ( (leveltime/anim->speed + i)%anim->numpics );
|
|
if (anim->istexture)
|
|
texturetranslation[i] = pic;
|
|
else
|
|
flattranslation[i] = pic;
|
|
}
|
|
}
|
|
|
|
// Check buttons (retriggerable switches) and change texture on timeout
|
|
for (i = 0; i < MAXBUTTONS; i++)
|
|
if (buttonlist[i].btimer)
|
|
{
|
|
buttonlist[i].btimer--;
|
|
if (!buttonlist[i].btimer)
|
|
{
|
|
switch(buttonlist[i].where)
|
|
{
|
|
case top:
|
|
sides[buttonlist[i].line->sidenum[0]].toptexture =
|
|
buttonlist[i].btexture;
|
|
break;
|
|
|
|
case middle:
|
|
sides[buttonlist[i].line->sidenum[0]].midtexture =
|
|
buttonlist[i].btexture;
|
|
break;
|
|
|
|
case bottom:
|
|
sides[buttonlist[i].line->sidenum[0]].bottomtexture =
|
|
buttonlist[i].btexture;
|
|
break;
|
|
}
|
|
S_StartSound((mobj_t *)&buttonlist[i].soundorg,sfx_swtchn);
|
|
memset(&buttonlist[i],0,sizeof(button_t));
|
|
}
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Sector and Line special thinker spawning at level startup
|
|
//
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// P_SpawnSpecials
|
|
// After the map has been loaded,
|
|
// scan for specials that spawn thinkers
|
|
//
|
|
|
|
// Parses command line parameters.
|
|
void P_SpawnSpecials (void)
|
|
{
|
|
sector_t* sector;
|
|
int i;
|
|
int episode;
|
|
|
|
episode = 1;
|
|
if (W_CheckNumForName("texture2") >= 0)
|
|
episode = 2;
|
|
|
|
// See if -timer needs to be used.
|
|
levelTimer = false;
|
|
|
|
i = M_CheckParm("-avg"); // Austin Virtual Gaming 20 min timer on DM play
|
|
if (i && deathmatch)
|
|
{
|
|
levelTimer = true;
|
|
levelTimeCount = 20 * 60 * TICRATE;
|
|
}
|
|
|
|
i = M_CheckParm("-timer"); // user defined timer on game play
|
|
if (i && deathmatch)
|
|
{
|
|
int time;
|
|
time = atoi(myargv[i+1]) * 60 * TICRATE;
|
|
levelTimer = true;
|
|
levelTimeCount = time;
|
|
}
|
|
|
|
// See if -frags has been used
|
|
levelFragLimit = false;
|
|
i = M_CheckParm("-frags"); // Ty 03/18/98 Added -frags support
|
|
if (i && deathmatch)
|
|
{
|
|
int frags;
|
|
frags = atoi(myargv[i+1]);
|
|
if (frags <= 0) frags = 10; // default 10 if no count provided
|
|
levelFragLimit = true;
|
|
levelFragLimitCount = frags;
|
|
}
|
|
|
|
|
|
// Init special sectors.
|
|
sector = sectors;
|
|
for (i=0 ; i<numsectors ; i++, sector++)
|
|
{
|
|
if (!sector->special)
|
|
continue;
|
|
|
|
if (sector->special&SECRET_MASK) //jff 3/15/98 count extended
|
|
totalsecret++; // secret sectors too
|
|
|
|
switch (sector->special&31)
|
|
{
|
|
case 1:
|
|
// random off
|
|
P_SpawnLightFlash (sector);
|
|
break;
|
|
|
|
case 2:
|
|
// strobe fast
|
|
P_SpawnStrobeFlash(sector,FASTDARK,0);
|
|
break;
|
|
|
|
case 3:
|
|
// strobe slow
|
|
P_SpawnStrobeFlash(sector,SLOWDARK,0);
|
|
break;
|
|
|
|
case 4:
|
|
// strobe fast/death slime
|
|
P_SpawnStrobeFlash(sector,FASTDARK,0);
|
|
sector->special |= 3<<DAMAGE_SHIFT; //jff 3/14/98 put damage bits in
|
|
break;
|
|
|
|
case 8:
|
|
// glowing light
|
|
P_SpawnGlowingLight(sector);
|
|
break;
|
|
case 9:
|
|
// secret sector
|
|
if (sector->special<32) //jff 3/14/98 bits don't count unless not
|
|
totalsecret++; // a generalized sector type
|
|
break;
|
|
|
|
case 10:
|
|
// door close in 30 seconds
|
|
P_SpawnDoorCloseIn30 (sector);
|
|
break;
|
|
|
|
case 12:
|
|
// sync strobe slow
|
|
P_SpawnStrobeFlash (sector, SLOWDARK, 1);
|
|
break;
|
|
|
|
case 13:
|
|
// sync strobe fast
|
|
P_SpawnStrobeFlash (sector, FASTDARK, 1);
|
|
break;
|
|
|
|
case 14:
|
|
// door raise in 5 minutes
|
|
P_SpawnDoorRaiseIn5Mins (sector, i);
|
|
break;
|
|
|
|
case 17:
|
|
// fire flickering
|
|
P_SpawnFireFlicker(sector);
|
|
break;
|
|
}
|
|
}
|
|
|
|
P_RemoveAllActiveCeilings(); // jff 2/22/98 use killough's scheme
|
|
|
|
P_RemoveAllActivePlats(); // killough
|
|
|
|
for (i = 0;i < MAXBUTTONS;i++)
|
|
memset(&buttonlist[i],0,sizeof(button_t));
|
|
|
|
// P_InitTagLists() must be called before P_FindSectorFromLineTag()
|
|
// or P_FindLineFromLineTag() can be called.
|
|
|
|
P_InitTagLists(); // killough 1/30/98: Create xref tables for tags
|
|
|
|
P_SpawnScrollers(); // killough 3/7/98: Add generalized scrollers
|
|
|
|
P_SpawnFriction(); // phares 3/12/98: New friction model using linedefs
|
|
|
|
P_SpawnPushers(); // phares 3/20/98: New pusher model using linedefs
|
|
|
|
for (i=0; i<numlines; i++)
|
|
switch (lines[i].special)
|
|
{
|
|
int s, sec;
|
|
|
|
// killough 3/7/98:
|
|
// support for drawn heights coming from different sector
|
|
case 242:
|
|
sec = sides[*lines[i].sidenum].sector-sectors;
|
|
for (s = -1; (s = P_FindSectorFromLineTag(lines+i,s)) >= 0;)
|
|
sectors[s].heightsec = sec;
|
|
break;
|
|
|
|
// killough 3/16/98: Add support for setting
|
|
// floor lighting independently (e.g. lava)
|
|
case 213:
|
|
sec = sides[*lines[i].sidenum].sector-sectors;
|
|
for (s = -1; (s = P_FindSectorFromLineTag(lines+i,s)) >= 0;)
|
|
sectors[s].floorlightsec = sec;
|
|
break;
|
|
|
|
// killough 4/11/98: Add support for setting
|
|
// ceiling lighting independently
|
|
case 261:
|
|
sec = sides[*lines[i].sidenum].sector-sectors;
|
|
for (s = -1; (s = P_FindSectorFromLineTag(lines+i,s)) >= 0;)
|
|
sectors[s].ceilinglightsec = sec;
|
|
break;
|
|
|
|
// killough 10/98:
|
|
//
|
|
// Support for sky textures being transferred from sidedefs.
|
|
// Allows scrolling and other effects (but if scrolling is
|
|
// used, then the same sector tag needs to be used for the
|
|
// sky sector, the sky-transfer linedef, and the scroll-effect
|
|
// linedef). Still requires user to use F_SKY1 for the floor
|
|
// or ceiling texture, to distinguish floor and ceiling sky.
|
|
|
|
case 271: // Regular sky
|
|
case 272: // Same, only flipped
|
|
for (s = -1; (s = P_FindSectorFromLineTag(lines+i,s)) >= 0;)
|
|
sectors[s].sky = i | PL_SKYFLAT;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// killough 2/28/98:
|
|
//
|
|
// This function, with the help of r_plane.c and r_bsp.c, supports generalized
|
|
// scrolling floors and walls, with optional mobj-carrying properties, e.g.
|
|
// conveyor belts, rivers, etc. A linedef with a special type affects all
|
|
// tagged sectors the same way, by creating scrolling and/or object-carrying
|
|
// properties. Multiple linedefs may be used on the same sector and are
|
|
// cumulative, although the special case of scrolling a floor and carrying
|
|
// things on it, requires only one linedef. The linedef's direction determines
|
|
// the scrolling direction, and the linedef's length determines the scrolling
|
|
// speed. This was designed so that an edge around the sector could be used to
|
|
// control the direction of the sector's scrolling, which is usually what is
|
|
// desired.
|
|
//
|
|
// Process the active scrollers.
|
|
//
|
|
// This is the main scrolling code
|
|
// killough 3/7/98
|
|
|
|
void T_Scroll(scroll_t *s)
|
|
{
|
|
fixed_t dx = s->dx, dy = s->dy;
|
|
|
|
if (s->control != -1)
|
|
{ // compute scroll amounts based on a sector's height changes
|
|
fixed_t height = sectors[s->control].floorheight +
|
|
sectors[s->control].ceilingheight;
|
|
fixed_t delta = height - s->last_height;
|
|
s->last_height = height;
|
|
dx = FixedMul(dx, delta);
|
|
dy = FixedMul(dy, delta);
|
|
}
|
|
|
|
// killough 3/14/98: Add acceleration
|
|
if (s->accel)
|
|
{
|
|
s->vdx = dx += s->vdx;
|
|
s->vdy = dy += s->vdy;
|
|
}
|
|
|
|
if (!(dx | dy)) // no-op if both (x,y) offsets 0
|
|
return;
|
|
|
|
switch (s->type)
|
|
{
|
|
side_t *side;
|
|
sector_t *sec;
|
|
fixed_t height, waterheight; // killough 4/4/98: add waterheight
|
|
msecnode_t *node;
|
|
mobj_t *thing;
|
|
|
|
case sc_side: // killough 3/7/98: Scroll wall texture
|
|
side = sides + s->affectee;
|
|
side->textureoffset += dx;
|
|
side->rowoffset += dy;
|
|
break;
|
|
|
|
case sc_floor: // killough 3/7/98: Scroll floor texture
|
|
sec = sectors + s->affectee;
|
|
sec->floor_xoffs += dx;
|
|
sec->floor_yoffs += dy;
|
|
break;
|
|
|
|
case sc_ceiling: // killough 3/7/98: Scroll ceiling texture
|
|
sec = sectors + s->affectee;
|
|
sec->ceiling_xoffs += dx;
|
|
sec->ceiling_yoffs += dy;
|
|
break;
|
|
|
|
case sc_carry:
|
|
|
|
// killough 3/7/98: Carry things on floor
|
|
// killough 3/20/98: use new sector list which reflects true members
|
|
// killough 3/27/98: fix carrier bug
|
|
// killough 4/4/98: Underwater, carry things even w/o gravity
|
|
|
|
sec = sectors + s->affectee;
|
|
height = sec->floorheight;
|
|
waterheight = sec->heightsec != -1 &&
|
|
sectors[sec->heightsec].floorheight > height ?
|
|
sectors[sec->heightsec].floorheight : INT_MIN;
|
|
|
|
for (node = sec->touching_thinglist; node; node = node->m_snext)
|
|
if (!((thing = node->m_thing)->flags & MF_NOCLIP) &&
|
|
(!(thing->flags & MF_NOGRAVITY || thing->z > height) ||
|
|
thing->z < waterheight))
|
|
{
|
|
// Move objects only if on floor or underwater,
|
|
// non-floating, and clipped.
|
|
thing->momx += dx;
|
|
thing->momy += dy;
|
|
}
|
|
break;
|
|
|
|
case sc_carry_ceiling: // to be added later
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Add_Scroller()
|
|
//
|
|
// Add a generalized scroller to the thinker list.
|
|
//
|
|
// type: the enumerated type of scrolling: floor, ceiling, floor carrier,
|
|
// wall, floor carrier & scroller
|
|
//
|
|
// (dx,dy): the direction and speed of the scrolling or its acceleration
|
|
//
|
|
// control: the sector whose heights control this scroller's effect
|
|
// remotely, or -1 if no control sector
|
|
//
|
|
// affectee: the index of the affected object (sector or sidedef)
|
|
//
|
|
// accel: non-zero if this is an accelerative effect
|
|
//
|
|
|
|
static void Add_Scroller(int type, fixed_t dx, fixed_t dy,
|
|
int control, int affectee, int accel)
|
|
{
|
|
scroll_t *s = Z_Malloc(sizeof *s, PU_LEVSPEC, 0);
|
|
s->thinker.function = T_Scroll;
|
|
s->type = type;
|
|
s->dx = dx;
|
|
s->dy = dy;
|
|
s->accel = accel;
|
|
s->vdx = s->vdy = 0;
|
|
if ((s->control = control) != -1)
|
|
s->last_height =
|
|
sectors[control].floorheight + sectors[control].ceilingheight;
|
|
s->affectee = affectee;
|
|
P_AddThinker(&s->thinker);
|
|
}
|
|
|
|
// Adds wall scroller. Scroll amount is rotated with respect to wall's
|
|
// linedef first, so that scrolling towards the wall in a perpendicular
|
|
// direction is translated into vertical motion, while scrolling along
|
|
// the wall in a parallel direction is translated into horizontal motion.
|
|
//
|
|
// killough 5/25/98: cleaned up arithmetic to avoid drift due to roundoff
|
|
//
|
|
// killough 10/98:
|
|
// fix scrolling aliasing problems, caused by long linedefs causing overflowing
|
|
|
|
static void Add_WallScroller(fixed_t dx, fixed_t dy, const line_t *l,
|
|
int control, int accel)
|
|
{
|
|
fixed_t x = D_abs(l->dx), y = D_abs(l->dy), d;
|
|
if (y > x)
|
|
d = x, x = y, y = d;
|
|
d = FixedDiv(x, finesine[(tantoangle[FixedDiv(y,x) >> DBITS] + ANG90)
|
|
>> ANGLETOFINESHIFT]);
|
|
|
|
// CPhipps - Import scroller calc overflow fix, compatibility optioned
|
|
if (compatibility_level >= lxdoom_1_compatibility) {
|
|
x = (fixed_t)(((int_64_t)dy * -(int_64_t)l->dy - (int_64_t)dx * (int_64_t)l->dx) / (int_64_t)d); // killough 10/98:
|
|
y = (fixed_t)(((int_64_t)dy * (int_64_t)l->dx - (int_64_t)dx * (int_64_t)l->dy) / (int_64_t)d); // Use long long arithmetic
|
|
} else {
|
|
x = -FixedDiv(FixedMul(dy, l->dy) + FixedMul(dx, l->dx), d);
|
|
y = -FixedDiv(FixedMul(dx, l->dy) - FixedMul(dy, l->dx), d);
|
|
}
|
|
Add_Scroller(sc_side, x, y, control, *l->sidenum, accel);
|
|
}
|
|
|
|
// Amount (dx,dy) vector linedef is shifted right to get scroll amount
|
|
#define SCROLL_SHIFT 5
|
|
|
|
// Factor to scale scrolling effect into mobj-carrying properties = 3/32.
|
|
// (This is so scrolling floors and objects on them can move at same speed.)
|
|
#define CARRYFACTOR ((fixed_t)(FRACUNIT*.09375))
|
|
|
|
// Initialize the scrollers
|
|
static void P_SpawnScrollers(void)
|
|
{
|
|
int i;
|
|
line_t *l = lines;
|
|
|
|
for (i=0;i<numlines;i++,l++)
|
|
{
|
|
fixed_t dx = l->dx >> SCROLL_SHIFT; // direction and speed of scrolling
|
|
fixed_t dy = l->dy >> SCROLL_SHIFT;
|
|
int control = -1, accel = 0; // no control sector or acceleration
|
|
int special = l->special;
|
|
|
|
// killough 3/7/98: Types 245-249 are same as 250-254 except that the
|
|
// first side's sector's heights cause scrolling when they change, and
|
|
// this linedef controls the direction and speed of the scrolling. The
|
|
// most complicated linedef since donuts, but powerful :)
|
|
//
|
|
// killough 3/15/98: Add acceleration. Types 214-218 are the same but
|
|
// are accelerative.
|
|
|
|
if (special >= 245 && special <= 249) // displacement scrollers
|
|
{
|
|
special += 250-245;
|
|
control = sides[*l->sidenum].sector - sectors;
|
|
}
|
|
else
|
|
if (special >= 214 && special <= 218) // accelerative scrollers
|
|
{
|
|
accel = 1;
|
|
special += 250-214;
|
|
control = sides[*l->sidenum].sector - sectors;
|
|
}
|
|
|
|
switch (special)
|
|
{
|
|
register int s;
|
|
|
|
case 250: // scroll effect ceiling
|
|
for (s=-1; (s = P_FindSectorFromLineTag(l,s)) >= 0;)
|
|
Add_Scroller(sc_ceiling, -dx, dy, control, s, accel);
|
|
break;
|
|
|
|
case 251: // scroll effect floor
|
|
case 253: // scroll and carry objects on floor
|
|
for (s=-1; (s = P_FindSectorFromLineTag(l,s)) >= 0;)
|
|
Add_Scroller(sc_floor, -dx, dy, control, s, accel);
|
|
if (special != 253)
|
|
break;
|
|
|
|
case 252: // carry objects on floor
|
|
dx = FixedMul(dx,CARRYFACTOR);
|
|
dy = FixedMul(dy,CARRYFACTOR);
|
|
for (s=-1; (s = P_FindSectorFromLineTag(l,s)) >= 0;)
|
|
Add_Scroller(sc_carry, dx, dy, control, s, accel);
|
|
break;
|
|
|
|
// killough 3/1/98: scroll wall according to linedef
|
|
// (same direction and speed as scrolling floors)
|
|
case 254:
|
|
for (s=-1; (s = P_FindLineFromLineTag(l,s)) >= 0;)
|
|
if (s != i)
|
|
Add_WallScroller(dx, dy, lines+s, control, accel);
|
|
break;
|
|
|
|
case 255: // killough 3/2/98: scroll according to sidedef offsets
|
|
s = lines[i].sidenum[0];
|
|
Add_Scroller(sc_side, -sides[s].textureoffset,
|
|
sides[s].rowoffset, -1, s, accel);
|
|
break;
|
|
|
|
case 48: // scroll first side
|
|
Add_Scroller(sc_side, FRACUNIT, 0, -1, lines[i].sidenum[0], accel);
|
|
break;
|
|
|
|
case 85: // jff 1/30/98 2-way scroll
|
|
Add_Scroller(sc_side, -FRACUNIT, 0, -1, lines[i].sidenum[0], accel);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// killough 3/7/98 -- end generalized scroll effects
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// FRICTION EFFECTS
|
|
//
|
|
// phares 3/12/98: Start of friction effects
|
|
//
|
|
// As the player moves, friction is applied by decreasing the x and y
|
|
// momentum values on each tic. By varying the percentage of decrease,
|
|
// we can simulate muddy or icy conditions. In mud, the player slows
|
|
// down faster. In ice, the player slows down more slowly.
|
|
//
|
|
// The amount of friction change is controlled by the length of a linedef
|
|
// with type 223. A length < 100 gives you mud. A length > 100 gives you ice.
|
|
//
|
|
// Also, each sector where these effects are to take place is given a
|
|
// new special type _______. Changing the type value at runtime allows
|
|
// these effects to be turned on or off.
|
|
//
|
|
// Sector boundaries present problems. The player should experience these
|
|
// friction changes only when his feet are touching the sector floor. At
|
|
// sector boundaries where floor height changes, the player can find
|
|
// himself still 'in' one sector, but with his feet at the floor level
|
|
// of the next sector (steps up or down). To handle this, Thinkers are used
|
|
// in icy/muddy sectors. These thinkers examine each object that is touching
|
|
// their sectors, looking for players whose feet are at the same level as
|
|
// their floors. Players satisfying this condition are given new friction
|
|
// values that are applied by the player movement code later.
|
|
//
|
|
// killough 8/28/98:
|
|
//
|
|
// Completely redid code, which did not need thinkers, and which put a heavy
|
|
// drag on CPU. Friction is now a property of sectors, NOT objects inside
|
|
// them. All objects, not just players, are affected by it, if they touch
|
|
// the sector's floor. Code simpler and faster, only calling on friction
|
|
// calculations when an object needs friction considered, instead of doing
|
|
// friction calculations on every sector during every tic.
|
|
//
|
|
// Although this -might- ruin Boom demo sync involving friction, it's the only
|
|
// way, short of code explosion, to fix the original design bug. Fixing the
|
|
// design bug in Boom's original friction code, while maintaining demo sync
|
|
// under every conceivable circumstance, would double or triple code size, and
|
|
// would require maintenance of buggy legacy code which is only useful for old
|
|
// demos. Doom demos, which are more important IMO, are not affected by this
|
|
// change.
|
|
//
|
|
/////////////////////////////
|
|
//
|
|
// Initialize the sectors where friction is increased or decreased
|
|
|
|
static void P_SpawnFriction(void)
|
|
{
|
|
int i;
|
|
line_t *l = lines;
|
|
|
|
// killough 8/28/98: initialize all sectors to normal friction first
|
|
for (i = 0; i < numsectors; i++)
|
|
{
|
|
sectors[i].friction = ORIG_FRICTION;
|
|
sectors[i].movefactor = ORIG_FRICTION_FACTOR;
|
|
}
|
|
|
|
for (i = 0 ; i < numlines ; i++,l++)
|
|
if (l->special == 223)
|
|
{
|
|
int length = P_AproxDistance(l->dx,l->dy)>>FRACBITS;
|
|
int friction = (0x1EB8*length)/0x80 + 0xD000;
|
|
int movefactor, s;
|
|
|
|
// The following check might seem odd. At the time of movement,
|
|
// the move distance is multiplied by 'friction/0x10000', so a
|
|
// higher friction value actually means 'less friction'.
|
|
|
|
if (friction > ORIG_FRICTION) // ice
|
|
movefactor = ((0x10092 - friction)*(0x70))/0x158;
|
|
else
|
|
movefactor = ((friction - 0xDB34)*(0xA))/0x80;
|
|
|
|
if (mbf_features)
|
|
{ // killough 8/28/98: prevent odd situations
|
|
if (friction > FRACUNIT)
|
|
friction = FRACUNIT;
|
|
if (friction < 0)
|
|
friction = 0;
|
|
if (movefactor < 32)
|
|
movefactor = 32;
|
|
}
|
|
|
|
for (s = -1; (s = P_FindSectorFromLineTag(l,s)) >= 0 ; )
|
|
{
|
|
// killough 8/28/98:
|
|
//
|
|
// Instead of spawning thinkers, which are slow and expensive,
|
|
// modify the sector's own friction values. Friction should be
|
|
// a property of sectors, not objects which reside inside them.
|
|
// Original code scanned every object in every friction sector
|
|
// on every tic, adjusting its friction, putting unnecessary
|
|
// drag on CPU. New code adjusts friction of sector only once
|
|
// at level startup, and then uses this friction value.
|
|
|
|
sectors[s].friction = friction;
|
|
sectors[s].movefactor = movefactor;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// phares 3/12/98: End of friction effects
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// PUSH/PULL EFFECT
|
|
//
|
|
// phares 3/20/98: Start of push/pull effects
|
|
//
|
|
// This is where push/pull effects are applied to objects in the sectors.
|
|
//
|
|
// There are four kinds of push effects
|
|
//
|
|
// 1) Pushing Away
|
|
//
|
|
// Pushes you away from a point source defined by the location of an
|
|
// MT_PUSH Thing. The force decreases linearly with distance from the
|
|
// source. This force crosses sector boundaries and is felt w/in a circle
|
|
// whose center is at the MT_PUSH. The force is felt only if the point
|
|
// MT_PUSH can see the target object.
|
|
//
|
|
// 2) Pulling toward
|
|
//
|
|
// Same as Pushing Away except you're pulled toward an MT_PULL point
|
|
// source. This force crosses sector boundaries and is felt w/in a circle
|
|
// whose center is at the MT_PULL. The force is felt only if the point
|
|
// MT_PULL can see the target object.
|
|
//
|
|
// 3) Wind
|
|
//
|
|
// Pushes you in a constant direction. Full force above ground, half
|
|
// force on the ground, nothing if you're below it (water).
|
|
//
|
|
// 4) Current
|
|
//
|
|
// Pushes you in a constant direction. No force above ground, full
|
|
// force if on the ground or below it (water).
|
|
//
|
|
// The magnitude of the force is controlled by the length of a controlling
|
|
// linedef. The force vector for types 3 & 4 is determined by the angle
|
|
// of the linedef, and is constant.
|
|
//
|
|
// For each sector where these effects occur, the sector special type has
|
|
// to have the PUSH_MASK bit set. If this bit is turned off by a switch
|
|
// at run-time, the effect will not occur. The controlling sector for
|
|
// types 1 & 2 is the sector containing the MT_PUSH/MT_PULL Thing.
|
|
|
|
|
|
#define PUSH_FACTOR 7
|
|
|
|
/////////////////////////////
|
|
//
|
|
// Add a push thinker to the thinker list
|
|
|
|
static void Add_Pusher(int type, int x_mag, int y_mag, mobj_t* source, int affectee)
|
|
{
|
|
pusher_t *p = Z_Malloc(sizeof *p, PU_LEVSPEC, 0);
|
|
|
|
p->thinker.function = T_Pusher;
|
|
p->source = source;
|
|
p->type = type;
|
|
p->x_mag = x_mag>>FRACBITS;
|
|
p->y_mag = y_mag>>FRACBITS;
|
|
p->magnitude = P_AproxDistance(p->x_mag,p->y_mag);
|
|
if (source) // point source exist?
|
|
{
|
|
p->radius = (p->magnitude)<<(FRACBITS+1); // where force goes to zero
|
|
p->x = p->source->x;
|
|
p->y = p->source->y;
|
|
}
|
|
p->affectee = affectee;
|
|
P_AddThinker(&p->thinker);
|
|
}
|
|
|
|
/////////////////////////////
|
|
//
|
|
// PIT_PushThing determines the angle and magnitude of the effect.
|
|
// The object's x and y momentum values are changed.
|
|
//
|
|
// tmpusher belongs to the point source (MT_PUSH/MT_PULL).
|
|
//
|
|
// killough 10/98: allow to affect things besides players
|
|
|
|
pusher_t* tmpusher; // pusher structure for blockmap searches
|
|
|
|
boolean PIT_PushThing(mobj_t* thing)
|
|
{
|
|
/* killough 10/98: made more general */
|
|
if (!mbf_features ?
|
|
thing->player && !(thing->flags & (MF_NOCLIP | MF_NOGRAVITY)) :
|
|
(sentient(thing) || thing->flags & MF_SHOOTABLE) &&
|
|
!(thing->flags & MF_NOCLIP))
|
|
{
|
|
angle_t pushangle;
|
|
fixed_t speed;
|
|
fixed_t sx = tmpusher->x;
|
|
fixed_t sy = tmpusher->y;
|
|
|
|
speed = (tmpusher->magnitude -
|
|
((P_AproxDistance(thing->x - sx,thing->y - sy)
|
|
>>FRACBITS)>>1))<<(FRACBITS-PUSH_FACTOR-1);
|
|
|
|
// killough 10/98: make magnitude decrease with square
|
|
// of distance, making it more in line with real nature,
|
|
// so long as it's still in range with original formula.
|
|
//
|
|
// Removes angular distortion, and makes effort required
|
|
// to stay close to source, grow increasingly hard as you
|
|
// get closer, as expected. Still, it doesn't consider z :(
|
|
|
|
if (speed > 0 && mbf_features)
|
|
{
|
|
int x = (thing->x-sx) >> FRACBITS;
|
|
int y = (thing->y-sy) >> FRACBITS;
|
|
speed = (int)(((uint_64_t) tmpusher->magnitude << 23) / (x*x+y*y+1));
|
|
}
|
|
|
|
// If speed <= 0, you're outside the effective radius. You also have
|
|
// to be able to see the push/pull source point.
|
|
|
|
if (speed > 0 && P_CheckSight(thing,tmpusher->source))
|
|
{
|
|
pushangle = R_PointToAngle2(thing->x,thing->y,sx,sy);
|
|
if (tmpusher->source->type == MT_PUSH)
|
|
pushangle += ANG180; // away
|
|
pushangle >>= ANGLETOFINESHIFT;
|
|
thing->momx += FixedMul(speed,finecosine[pushangle]);
|
|
thing->momy += FixedMul(speed,finesine[pushangle]);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/////////////////////////////
|
|
//
|
|
// T_Pusher looks for all objects that are inside the radius of
|
|
// the effect.
|
|
//
|
|
|
|
void T_Pusher(pusher_t *p)
|
|
{
|
|
sector_t *sec;
|
|
mobj_t *thing;
|
|
msecnode_t* node;
|
|
int xspeed,yspeed;
|
|
int xl,xh,yl,yh,bx,by;
|
|
int radius;
|
|
int ht = 0;
|
|
|
|
if (!allow_pushers)
|
|
return;
|
|
|
|
sec = sectors + p->affectee;
|
|
|
|
// Be sure the special sector type is still turned on. If so, proceed.
|
|
// Else, bail out; the sector type has been changed on us.
|
|
|
|
if (!(sec->special & PUSH_MASK))
|
|
return;
|
|
|
|
// For constant pushers (wind/current) there are 3 situations:
|
|
//
|
|
// 1) Affected Thing is above the floor.
|
|
//
|
|
// Apply the full force if wind, no force if current.
|
|
//
|
|
// 2) Affected Thing is on the ground.
|
|
//
|
|
// Apply half force if wind, full force if current.
|
|
//
|
|
// 3) Affected Thing is below the ground (underwater effect).
|
|
//
|
|
// Apply no force if wind, full force if current.
|
|
|
|
if (p->type == p_push)
|
|
{
|
|
|
|
// Seek out all pushable things within the force radius of this
|
|
// point pusher. Crosses sectors, so use blockmap.
|
|
|
|
tmpusher = p; // MT_PUSH/MT_PULL point source
|
|
radius = p->radius; // where force goes to zero
|
|
tmbbox[BOXTOP] = p->y + radius;
|
|
tmbbox[BOXBOTTOM] = p->y - radius;
|
|
tmbbox[BOXRIGHT] = p->x + radius;
|
|
tmbbox[BOXLEFT] = p->x - radius;
|
|
|
|
xl = (tmbbox[BOXLEFT] - bmaporgx - MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
xh = (tmbbox[BOXRIGHT] - bmaporgx + MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
yl = (tmbbox[BOXBOTTOM] - bmaporgy - MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
yh = (tmbbox[BOXTOP] - bmaporgy + MAXRADIUS)>>MAPBLOCKSHIFT;
|
|
for (bx=xl ; bx<=xh ; bx++)
|
|
for (by=yl ; by<=yh ; by++)
|
|
P_BlockThingsIterator(bx,by,PIT_PushThing);
|
|
return;
|
|
}
|
|
|
|
// constant pushers p_wind and p_current
|
|
|
|
if (sec->heightsec != -1) // special water sector?
|
|
ht = sectors[sec->heightsec].floorheight;
|
|
node = sec->touching_thinglist; // things touching this sector
|
|
for ( ; node ; node = node->m_snext)
|
|
{
|
|
thing = node->m_thing;
|
|
if (!thing->player || (thing->flags & (MF_NOGRAVITY | MF_NOCLIP)))
|
|
continue;
|
|
if (p->type == p_wind)
|
|
{
|
|
if (sec->heightsec == -1) // NOT special water sector
|
|
if (thing->z > thing->floorz) // above ground
|
|
{
|
|
xspeed = p->x_mag; // full force
|
|
yspeed = p->y_mag;
|
|
}
|
|
else // on ground
|
|
{
|
|
xspeed = (p->x_mag)>>1; // half force
|
|
yspeed = (p->y_mag)>>1;
|
|
}
|
|
else // special water sector
|
|
{
|
|
if (thing->z > ht) // above ground
|
|
{
|
|
xspeed = p->x_mag; // full force
|
|
yspeed = p->y_mag;
|
|
}
|
|
else if (thing->player->viewz < ht) // underwater
|
|
xspeed = yspeed = 0; // no force
|
|
else // wading in water
|
|
{
|
|
xspeed = (p->x_mag)>>1; // half force
|
|
yspeed = (p->y_mag)>>1;
|
|
}
|
|
}
|
|
}
|
|
else // p_current
|
|
{
|
|
if (sec->heightsec == -1) // NOT special water sector
|
|
if (thing->z > sec->floorheight) // above ground
|
|
xspeed = yspeed = 0; // no force
|
|
else // on ground
|
|
{
|
|
xspeed = p->x_mag; // full force
|
|
yspeed = p->y_mag;
|
|
}
|
|
else // special water sector
|
|
if (thing->z > ht) // above ground
|
|
xspeed = yspeed = 0; // no force
|
|
else // underwater
|
|
{
|
|
xspeed = p->x_mag; // full force
|
|
yspeed = p->y_mag;
|
|
}
|
|
}
|
|
thing->momx += xspeed<<(FRACBITS-PUSH_FACTOR);
|
|
thing->momy += yspeed<<(FRACBITS-PUSH_FACTOR);
|
|
}
|
|
}
|
|
|
|
/////////////////////////////
|
|
//
|
|
// P_GetPushThing() returns a pointer to an MT_PUSH or MT_PULL thing,
|
|
// NULL otherwise.
|
|
|
|
mobj_t* P_GetPushThing(int s)
|
|
{
|
|
mobj_t* thing;
|
|
sector_t* sec;
|
|
|
|
sec = sectors + s;
|
|
thing = sec->thinglist;
|
|
while (thing)
|
|
{
|
|
switch(thing->type)
|
|
{
|
|
case MT_PUSH:
|
|
case MT_PULL:
|
|
return thing;
|
|
default:
|
|
break;
|
|
}
|
|
thing = thing->snext;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/////////////////////////////
|
|
//
|
|
// Initialize the sectors where pushers are present
|
|
//
|
|
|
|
static void P_SpawnPushers(void)
|
|
{
|
|
int i;
|
|
line_t *l = lines;
|
|
register int s;
|
|
mobj_t* thing;
|
|
|
|
for (i = 0 ; i < numlines ; i++,l++)
|
|
switch(l->special)
|
|
{
|
|
case 224: // wind
|
|
for (s = -1; (s = P_FindSectorFromLineTag(l,s)) >= 0 ; )
|
|
Add_Pusher(p_wind,l->dx,l->dy,NULL,s);
|
|
break;
|
|
case 225: // current
|
|
for (s = -1; (s = P_FindSectorFromLineTag(l,s)) >= 0 ; )
|
|
Add_Pusher(p_current,l->dx,l->dy,NULL,s);
|
|
break;
|
|
case 226: // push/pull
|
|
for (s = -1; (s = P_FindSectorFromLineTag(l,s)) >= 0 ; )
|
|
{
|
|
thing = P_GetPushThing(s);
|
|
if (thing) // No MT_P* means no effect
|
|
Add_Pusher(p_push,l->dx,l->dy,thing,s);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// phares 3/20/98: End of Pusher effects
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|