2014-03-15 16:59:03 +00:00
// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
// Copyright (C) 1993-1996 by id Software, Inc.
// Copyright (C) 1998-2000 by DooM Legacy Team.
2023-03-31 12:53:31 +00:00
// Copyright (C) 1999-2023 by Sonic Team Junior.
2014-03-15 16:59:03 +00:00
//
// This program is free software distributed under the
// terms of the GNU General Public License, version 2.
// See the 'LICENSE' file for more details.
//-----------------------------------------------------------------------------
/// \file p_spec.c
/// \brief Implements special effects:
/// Texture animation, height or lighting changes
/// according to adjacent sectors, respective
/// utility functions, etc.
/// Line Tag handling. Line and Sector triggers.
2020-04-25 07:20:53 +00:00
# include "dehacked.h"
2014-03-15 16:59:03 +00:00
# include "doomdef.h"
# include "g_game.h"
# include "p_local.h"
# include "p_setup.h" // levelflats for flat animation
# include "r_data.h"
2022-04-13 01:45:49 +00:00
# include "r_fps.h"
2020-01-07 15:35:10 +00:00
# include "r_textures.h"
2014-03-15 16:59:03 +00:00
# include "m_random.h"
# include "p_mobj.h"
# include "i_system.h"
# include "s_sound.h"
# include "w_wad.h"
# include "z_zone.h"
# include "r_main.h" //Two extra includes.
# include "r_sky.h"
# include "p_polyobj.h"
2015-10-10 16:57:35 +00:00
# include "p_slopes.h"
2014-03-15 16:59:03 +00:00
# include "hu_stuff.h"
2018-03-26 22:53:09 +00:00
# include "v_video.h" // V_AUTOFADEOUT|V_ALLOWLOWERCASE
2014-03-15 16:59:03 +00:00
# include "m_misc.h"
# include "m_cond.h" //unlock triggers
2020-11-17 12:14:45 +00:00
# include "lua_hook.h" // LUA_HookLinedefExecute
2018-11-03 14:54:47 +00:00
# include "f_finale.h" // control text prompt
2020-03-09 13:54:56 +00:00
# include "r_skins.h" // skins
2014-03-15 16:59:03 +00:00
# ifdef HW3SOUND
# include "hardware/hw3sound.h"
# endif
// Not sure if this is necessary, but it was in w_wad.c, so I'm putting it here too -Shadow Hog
# include <errno.h>
2017-04-26 17:16:01 +00:00
mobj_t * skyboxmo [ 2 ] ; // current skybox mobjs: 0 = viewpoint, 1 = centerpoint
mobj_t * skyboxviewpnts [ 16 ] ; // array of MT_SKYBOX viewpoint mobjs
mobj_t * skyboxcenterpnts [ 16 ] ; // array of MT_SKYBOX centerpoint mobjs
2014-03-15 16:59:03 +00:00
2023-08-23 20:49:29 +00:00
size_t secportalcount ;
size_t secportalcapacity ;
sectorportal_t * secportals ;
2014-03-15 16:59:03 +00:00
/** Animated texture descriptor
* This keeps track of an animated texture or an animated flat .
* \ sa P_UpdateSpecials , P_InitPicAnims , animdef_t
*/
typedef struct
{
2024-01-15 05:40:43 +00:00
INT32 picnum ; ///< The end texture number
INT32 basepic ; ///< The start texture number
2014-03-15 16:59:03 +00:00
INT32 numpics ; ///< Number of frames in the animation
tic_t speed ; ///< Number of tics for which each frame is shown
} anim_t ;
/** Animated texture definition.
2017-10-07 22:18:25 +00:00
* Used for loading an ANIMDEFS lump from a wad .
2014-03-15 16:59:03 +00:00
*
* Animations are defined by the first and last frame ( i . e . , flat or texture ) .
* The animation sequence uses all flats between the start and end entry , in
* the order found in the wad .
*
* \ sa anim_t
*/
typedef struct
{
SINT8 istexture ; ///< True for a texture, false for a flat.
char endname [ 9 ] ; ///< Name of the last frame, null-terminated.
char startname [ 9 ] ; ///< Name of the first frame, null-terminated.
2024-01-15 05:40:43 +00:00
INT32 speed ; ///< Number of tics for which each frame is shown.
2014-03-15 16:59:03 +00:00
} ATTRPACK animdef_t ;
typedef struct
{
UINT32 count ;
thinker_t * * thinkers ;
} thinkerlist_t ;
static void P_SpawnScrollers ( void ) ;
static void P_SpawnFriction ( void ) ;
static void P_SpawnPushers ( void ) ;
2021-07-01 18:05:23 +00:00
static void Add_Pusher ( pushertype_e type , fixed_t x_mag , fixed_t y_mag , fixed_t z_mag , INT32 affectee , INT32 referrer , INT32 exclusive , INT32 slider ) ; //SoM: 3/9/2000
2014-03-15 16:59:03 +00:00
static void Add_MasterDisappearer ( tic_t appeartime , tic_t disappeartime , tic_t offset , INT32 line , INT32 sourceline ) ;
2018-08-18 09:55:49 +00:00
static void P_ResetFakeFloorFader ( ffloor_t * rover , fade_t * data , boolean finalize ) ;
# define P_RemoveFakeFloorFader(l) P_ResetFakeFloorFader(l, NULL, false);
2018-09-11 14:28:24 +00:00
static boolean P_FadeFakeFloor ( ffloor_t * rover , INT16 sourcevalue , INT16 destvalue , INT16 speed , boolean ticbased , INT32 * timer ,
2018-09-12 20:57:35 +00:00
boolean doexists , boolean dotranslucent , boolean dolighting , boolean docolormap ,
boolean docollision , boolean doghostfade , boolean exactalpha ) ;
2018-08-17 19:13:05 +00:00
static void P_AddFakeFloorFader ( ffloor_t * rover , size_t sectornum , size_t ffloornum ,
2018-09-10 13:48:24 +00:00
INT16 destvalue , INT16 speed , boolean ticbased , boolean relative ,
2018-09-12 20:57:35 +00:00
boolean doexists , boolean dotranslucent , boolean dolighting , boolean docolormap ,
boolean docollision , boolean doghostfade , boolean exactalpha ) ;
static void P_ResetColormapFader ( sector_t * sector ) ;
static void Add_ColormapFader ( sector_t * sector , extracolormap_t * source_exc , extracolormap_t * dest_exc ,
2018-09-13 14:38:15 +00:00
boolean ticbased , INT32 duration ) ;
2014-03-15 16:59:03 +00:00
static void P_AddBlockThinker ( sector_t * sec , line_t * sourceline ) ;
2020-04-17 22:26:49 +00:00
static void P_AddFloatThinker ( sector_t * sec , UINT16 tag , line_t * sourceline ) ;
2016-03-30 18:05:07 +00:00
//static void P_AddBridgeThinker(line_t *sourceline, sector_t *sec);
2022-01-04 18:33:17 +00:00
static void P_AddFakeFloorsByLine ( size_t line , INT32 alpha , UINT8 blendmode , ffloortype_e ffloorflags , thinkerlist_t * secthinkers ) ;
2014-11-12 00:55:07 +00:00
static void P_ProcessLineSpecial ( line_t * line , mobj_t * mo , sector_t * callsec ) ;
2014-03-15 16:59:03 +00:00
static void Add_Friction ( INT32 friction , INT32 movefactor , INT32 affectee , INT32 referrer ) ;
2017-04-11 21:33:04 +00:00
static void P_AddPlaneDisplaceThinker ( INT32 type , fixed_t speed , INT32 control , INT32 affectee , UINT8 reverse ) ;
2014-03-15 16:59:03 +00:00
//SoM: 3/7/2000: New sturcture without limits.
static anim_t * lastanim ;
static anim_t * anims = NULL ; /// \todo free leak
static size_t maxanims ;
// Animating line specials
// Init animated textures
// - now called at level loading P_SetupLevel()
static animdef_t * animdefs = NULL ;
// A prototype; here instead of p_spec.h, so they're "private"
2016-10-19 07:17:36 +00:00
void P_ParseANIMDEFSLump ( INT32 wadNum , UINT16 lumpnum ) ;
void P_ParseAnimationDefintion ( SINT8 istexture ) ;
2014-03-15 16:59:03 +00:00
2024-01-15 05:40:43 +00:00
static boolean P_FindTextureForAnimation ( anim_t * anim , animdef_t * animdef )
{
if ( R_CheckTextureNumForName ( animdef - > startname ) = = - 1 )
return false ;
anim - > picnum = R_TextureNumForName ( animdef - > endname ) ;
anim - > basepic = R_TextureNumForName ( animdef - > startname ) ;
return true ;
}
static boolean P_FindFlatForAnimation ( anim_t * anim , animdef_t * animdef )
{
if ( R_CheckFlatNumForName ( animdef - > startname ) = = - 1 )
return false ;
anim - > picnum = R_CheckFlatNumForName ( animdef - > endname ) ;
anim - > basepic = R_CheckFlatNumForName ( animdef - > startname ) ;
return true ;
}
2014-03-15 16:59:03 +00:00
/** Sets up texture and flat animations.
*
2017-10-07 22:18:25 +00:00
* Converts an : : animdef_t array loaded from a lump into
2014-03-15 16:59:03 +00:00
* : : anim_t format .
*
* Issues an error if any animation cycles are invalid .
*
2016-10-19 07:17:36 +00:00
* \ author Steven McGranahan ( original ) , Shadow Hog ( had to rewrite it to handle multiple WADs ) , JTE ( had to rewrite it to handle multiple WADs _correctly_ )
2014-03-15 16:59:03 +00:00
*/
void P_InitPicAnims ( void )
{
// Init animation
INT32 w ; // WAD
2016-10-19 07:17:36 +00:00
size_t i ;
I_Assert ( animdefs = = NULL ) ;
2017-11-03 03:13:13 +00:00
2017-10-07 22:18:25 +00:00
maxanims = 0 ;
2014-03-15 16:59:03 +00:00
2017-10-26 15:58:18 +00:00
for ( w = numwadfiles - 1 ; w > = 0 ; w - - )
2014-03-15 16:59:03 +00:00
{
2017-10-26 15:58:18 +00:00
UINT16 animdefsLumpNum ;
// Find ANIMDEFS lump in the WAD
animdefsLumpNum = W_CheckNumForNamePwad ( " ANIMDEFS " , w , 0 ) ;
while ( animdefsLumpNum ! = INT16_MAX )
2014-03-15 16:59:03 +00:00
{
2017-10-26 15:58:18 +00:00
P_ParseANIMDEFSLump ( w , animdefsLumpNum ) ;
animdefsLumpNum = W_CheckNumForNamePwad ( " ANIMDEFS " , ( UINT16 ) w , animdefsLumpNum + 1 ) ;
2014-03-15 16:59:03 +00:00
}
}
2017-11-03 03:13:13 +00:00
2017-10-07 22:18:25 +00:00
// Define the last one
animdefs [ maxanims ] . istexture = - 1 ;
strncpy ( animdefs [ maxanims ] . endname , " " , 9 ) ;
strncpy ( animdefs [ maxanims ] . startname , " " , 9 ) ;
animdefs [ maxanims ] . speed = 0 ;
2014-03-15 16:59:03 +00:00
if ( anims )
free ( anims ) ;
anims = ( anim_t * ) malloc ( sizeof ( * anims ) * ( maxanims + 1 ) ) ;
if ( ! anims )
2017-10-07 22:18:25 +00:00
I_Error ( " Not enough free memory for ANIMDEFS data " ) ;
2014-03-15 16:59:03 +00:00
lastanim = anims ;
for ( i = 0 ; animdefs [ i ] . istexture ! = - 1 ; i + + )
{
2024-01-15 05:40:43 +00:00
animdef_t * animdef = & animdefs [ i ] ;
// If this animation is for a texture, look for one first, THEN look for a flat
2014-03-15 16:59:03 +00:00
if ( animdefs [ i ] . istexture )
{
2024-01-15 05:40:43 +00:00
if ( ! P_FindTextureForAnimation ( lastanim , animdef ) )
{
if ( ! P_FindFlatForAnimation ( lastanim , animdef ) )
continue ;
}
2014-03-15 16:59:03 +00:00
}
2024-01-15 05:40:43 +00:00
// Else, if this animation is for a flat, look for one first, THEN look for a texture
2014-03-15 16:59:03 +00:00
else
{
2024-01-15 05:40:43 +00:00
if ( ! P_FindFlatForAnimation ( lastanim , animdef ) )
{
if ( ! P_FindTextureForAnimation ( lastanim , animdef ) )
continue ;
}
2014-03-15 16:59:03 +00:00
}
lastanim - > numpics = lastanim - > picnum - lastanim - > basepic + 1 ;
if ( lastanim - > numpics < 2 )
{
free ( anims ) ;
I_Error ( " P_InitPicAnims: bad cycle from %s to %s " ,
2024-01-15 05:40:43 +00:00
animdef - > startname , animdef - > endname ) ;
2014-03-15 16:59:03 +00:00
}
2024-01-15 05:40:43 +00:00
lastanim - > speed = animdef - > speed ;
2014-03-15 16:59:03 +00:00
lastanim + + ;
}
R_ClearTextureNumCache ( false ) ;
2016-10-19 07:17:36 +00:00
// Clear animdefs now that we're done with it.
// We'll only be using anims from now on.
2017-10-07 22:18:25 +00:00
Z_Free ( animdefs ) ;
2016-10-19 07:17:36 +00:00
animdefs = NULL ;
2014-03-15 16:59:03 +00:00
}
2016-10-19 07:17:36 +00:00
void P_ParseANIMDEFSLump ( INT32 wadNum , UINT16 lumpnum )
2014-03-15 16:59:03 +00:00
{
char * animdefsLump ;
size_t animdefsLumpLength ;
char * animdefsText ;
char * animdefsToken ;
2016-10-19 07:17:36 +00:00
char * p ;
2014-03-15 16:59:03 +00:00
// Since lumps AREN'T \0-terminated like I'd assumed they should be, I'll
// need to make a space of memory where I can ensure that it will terminate
// correctly. Start by loading the relevant data from the WAD.
animdefsLump = ( char * ) W_CacheLumpNumPwad ( wadNum , lumpnum , PU_STATIC ) ;
// If that didn't exist, we have nothing to do here.
if ( animdefsLump = = NULL ) return ;
// If we're still here, then it DOES exist; figure out how long it is, and allot memory accordingly.
animdefsLumpLength = W_LumpLengthPwad ( wadNum , lumpnum ) ;
animdefsText = ( char * ) Z_Malloc ( ( animdefsLumpLength + 1 ) * sizeof ( char ) , PU_STATIC , NULL ) ;
// Now move the contents of the lump into this new location.
memmove ( animdefsText , animdefsLump , animdefsLumpLength ) ;
// Make damn well sure the last character in our new memory location is \0.
animdefsText [ animdefsLumpLength ] = ' \0 ' ;
// Finally, free up the memory from the first data load, because we really
// don't need it.
Z_Free ( animdefsLump ) ;
// Now, let's start parsing this thing
2016-10-19 07:17:36 +00:00
p = animdefsText ;
animdefsToken = M_GetToken ( p ) ;
2014-03-15 16:59:03 +00:00
while ( animdefsToken ! = NULL )
{
if ( stricmp ( animdefsToken , " TEXTURE " ) = = 0 )
{
Z_Free ( animdefsToken ) ;
2016-10-19 07:17:36 +00:00
P_ParseAnimationDefintion ( 1 ) ;
2014-03-15 16:59:03 +00:00
}
else if ( stricmp ( animdefsToken , " FLAT " ) = = 0 )
{
Z_Free ( animdefsToken ) ;
2016-10-19 07:17:36 +00:00
P_ParseAnimationDefintion ( 0 ) ;
2014-03-15 16:59:03 +00:00
}
else if ( stricmp ( animdefsToken , " OSCILLATE " ) = = 0 )
{
// This probably came off the tail of an earlier definition. It's technically legal syntax, but we don't support it.
I_Error ( " Error parsing ANIMDEFS lump: Animation definitions utilizing \" OSCILLATE \" (the animation plays in reverse when it reaches the end) are not supported by SRB2 " ) ;
}
else
{
I_Error ( " Error parsing ANIMDEFS lump: Expected \" TEXTURE \" or \" FLAT \" , got \" %s \" " , animdefsToken ) ;
}
2016-10-19 07:17:36 +00:00
// parse next line
while ( * p ! = ' \0 ' & & * p ! = ' \n ' ) + + p ;
if ( * p = = ' \n ' ) + + p ;
animdefsToken = M_GetToken ( p ) ;
2014-03-15 16:59:03 +00:00
}
Z_Free ( animdefsToken ) ;
Z_Free ( ( void * ) animdefsText ) ;
}
2016-10-19 07:17:36 +00:00
void P_ParseAnimationDefintion ( SINT8 istexture )
2014-03-15 16:59:03 +00:00
{
char * animdefsToken ;
2016-05-22 04:44:12 +00:00
size_t animdefsTokenLength ;
2014-03-15 16:59:03 +00:00
char * endPos ;
INT32 animSpeed ;
2016-10-19 07:17:36 +00:00
size_t i ;
2014-03-15 16:59:03 +00:00
// Startname
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where start texture/flat name should be " ) ;
}
if ( stricmp ( animdefsToken , " OPTIONAL " ) = = 0 )
{
// This is meaningful to ZDoom - it tells the program NOT to bomb out
// if the textures can't be found - but it's useless in SRB2, so we'll
// just smile, nod, and carry on
Z_Free ( animdefsToken ) ;
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where start texture/flat name should be " ) ;
}
else if ( stricmp ( animdefsToken , " RANGE " ) = = 0 )
{
// Oh. Um. Apparently "OPTIONAL" is a texture name. Naughty.
// I should probably handle this more gracefully, but right now
// I can't be bothered; especially since ZDoom doesn't handle this
// condition at all.
I_Error ( " Error parsing ANIMDEFS lump: \" OPTIONAL \" is a keyword; you cannot use it as the startname of an animation " ) ;
}
}
animdefsTokenLength = strlen ( animdefsToken ) ;
if ( animdefsTokenLength > 8 )
{
I_Error ( " Error parsing ANIMDEFS lump: lump name \" %s \" exceeds 8 characters " , animdefsToken ) ;
}
2016-10-19 07:17:36 +00:00
// Search for existing animdef
for ( i = 0 ; i < maxanims ; i + + )
2017-06-25 17:17:05 +00:00
if ( animdefs [ i ] . istexture = = istexture // Check if it's the same type!
& & stricmp ( animdefsToken , animdefs [ i ] . startname ) = = 0 )
2016-10-19 07:17:36 +00:00
{
//CONS_Alert(CONS_NOTICE, "Duplicate animation: %s\n", animdefsToken);
// If we weren't parsing in reverse order, we would `break` here and parse the new data into the existing slot we found.
// Instead, we're just going to skip parsing the rest of this line entirely.
Z_Free ( animdefsToken ) ;
return ;
}
// Not found
if ( i = = maxanims )
{
// Increase the size to make room for the new animation definition
2024-01-15 05:40:43 +00:00
maxanims + + ;
animdefs = ( animdef_t * ) Z_Realloc ( animdefs , sizeof ( animdef_t ) * ( maxanims + 1 ) , PU_STATIC , NULL ) ;
2024-02-20 21:30:15 +00:00
strncpy ( animdefs [ i ] . startname , animdefsToken , sizeof ( animdefs [ i ] . startname ) - 1 ) ;
2016-10-19 07:17:36 +00:00
}
// animdefs[i].startname is now set to animdefsToken either way.
2014-03-15 16:59:03 +00:00
Z_Free ( animdefsToken ) ;
2016-10-19 07:17:36 +00:00
// set texture type
animdefs [ i ] . istexture = istexture ;
2014-03-15 16:59:03 +00:00
// "RANGE"
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where \" RANGE \" after \" %s \" 's startname should be " , animdefs [ i ] . startname ) ;
2014-03-15 16:59:03 +00:00
}
if ( stricmp ( animdefsToken , " ALLOWDECALS " ) = = 0 )
{
// Another ZDoom keyword, ho-hum. Skip it, move on to the next token.
Z_Free ( animdefsToken ) ;
animdefsToken = M_GetToken ( NULL ) ;
}
if ( stricmp ( animdefsToken , " PIC " ) = = 0 )
{
// This is technically legitimate ANIMDEFS syntax, but SRB2 doesn't support it.
I_Error ( " Error parsing ANIMDEFS lump: Animation definitions utilizing \" PIC \" (specific frames instead of a consecutive range) are not supported by SRB2 " ) ;
}
if ( stricmp ( animdefsToken , " RANGE " ) ! = 0 )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Expected \" RANGE \" after \" %s \" 's startname, got \" %s \" " , animdefs [ i ] . startname , animdefsToken ) ;
2014-03-15 16:59:03 +00:00
}
Z_Free ( animdefsToken ) ;
// Endname
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where \" %s \" 's end texture/flat name should be " , animdefs [ i ] . startname ) ;
2014-03-15 16:59:03 +00:00
}
animdefsTokenLength = strlen ( animdefsToken ) ;
if ( animdefsTokenLength > 8 )
{
I_Error ( " Error parsing ANIMDEFS lump: lump name \" %s \" exceeds 8 characters " , animdefsToken ) ;
}
2016-10-19 07:17:36 +00:00
strncpy ( animdefs [ i ] . endname , animdefsToken , 9 ) ;
2014-03-15 16:59:03 +00:00
Z_Free ( animdefsToken ) ;
// "TICS"
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where \" %s \" 's \" TICS \" should be " , animdefs [ i ] . startname ) ;
2014-03-15 16:59:03 +00:00
}
if ( stricmp ( animdefsToken , " RAND " ) = = 0 )
{
// This is technically legitimate ANIMDEFS syntax, but SRB2 doesn't support it.
I_Error ( " Error parsing ANIMDEFS lump: Animation definitions utilizing \" RAND \" (random duration per frame) are not supported by SRB2 " ) ;
}
if ( stricmp ( animdefsToken , " TICS " ) ! = 0 )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Expected \" TICS \" in animation definition for \" %s \" , got \" %s \" " , animdefs [ i ] . startname , animdefsToken ) ;
2014-03-15 16:59:03 +00:00
}
Z_Free ( animdefsToken ) ;
// Speed
animdefsToken = M_GetToken ( NULL ) ;
if ( animdefsToken = = NULL )
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Unexpected end of file where \" %s \" 's animation speed should be " , animdefs [ i ] . startname ) ;
2014-03-15 16:59:03 +00:00
}
endPos = NULL ;
# ifndef AVOID_ERRNO
errno = 0 ;
# endif
animSpeed = strtol ( animdefsToken , & endPos , 10 ) ;
if ( endPos = = animdefsToken // Empty string
| | * endPos ! = ' \0 ' // Not end of string
# ifndef AVOID_ERRNO
| | errno = = ERANGE // Number out-of-range
# endif
| | animSpeed < 0 ) // Number is not positive
{
2016-10-19 07:17:36 +00:00
I_Error ( " Error parsing ANIMDEFS lump: Expected a positive integer for \" %s \" 's animation speed, got \" %s \" " , animdefs [ i ] . startname , animdefsToken ) ;
2014-03-15 16:59:03 +00:00
}
2016-10-19 07:17:36 +00:00
animdefs [ i ] . speed = animSpeed ;
2014-03-15 16:59:03 +00:00
Z_Free ( animdefsToken ) ;
}
//
// UTILITIES
//
2016-03-30 18:05:07 +00:00
#if 0
2014-03-15 16:59:03 +00:00
/** Gets a side from a sector line.
*
* \ param currentSector Sector the line is in .
* \ param line Index of the line within the sector .
* \ param side 0 for front , 1 for back .
* \ return Pointer to the side_t of the side you want .
* \ sa getSector , twoSided , getNextSector
*/
static inline side_t * getSide ( INT32 currentSector , INT32 line , INT32 side )
{
return & sides [ ( sectors [ currentSector ] . lines [ line ] ) - > sidenum [ side ] ] ;
}
/** Gets a sector from a sector line.
*
* \ param currentSector Sector the line is in .
* \ param line Index of the line within the sector .
* \ param side 0 for front , 1 for back .
* \ return Pointer to the : : sector_t of the sector on that side .
* \ sa getSide , twoSided , getNextSector
*/
static inline sector_t * getSector ( INT32 currentSector , INT32 line , INT32 side )
{
return sides [ ( sectors [ currentSector ] . lines [ line ] ) - > sidenum [ side ] ] . sector ;
}
/** Determines whether a sector line is two-sided.
* Uses the Boom method , checking if the line ' s back side is set to - 1 , rather
* than looking for : : ML_TWOSIDED .
*
* \ param sector The sector .
* \ param line Line index within the sector .
* \ return 1 if the sector is two - sided , 0 otherwise .
* \ sa getSide , getSector , getNextSector
*/
static inline boolean twoSided ( INT32 sector , INT32 line )
{
2023-09-21 05:06:06 +00:00
return ( sectors [ sector ] . lines [ line ] ) - > sidenum [ 1 ] ! = NO_SIDEDEF ;
2014-03-15 16:59:03 +00:00
}
2016-03-30 18:05:07 +00:00
# endif
2014-03-15 16:59:03 +00:00
/** Finds sector next to current.
*
* \ param line Pointer to the line to cross .
* \ param sec Pointer to the current sector .
* \ return Pointer to a : : sector_t of the adjacent sector , or NULL if the line
* is one - sided .
* \ sa getSide , getSector , twoSided
* \ author Steven McGranahan
*/
static sector_t * getNextSector ( line_t * line , sector_t * sec )
{
if ( line - > frontsector = = sec )
{
if ( line - > backsector ! = sec )
return line - > backsector ;
else
return NULL ;
}
return line - > frontsector ;
}
/** Finds lowest floor in adjacent sectors.
*
* \ param sec Sector to start in .
* \ return Lowest floor height in an adjacent sector .
* \ sa P_FindHighestFloorSurrounding , P_FindNextLowestFloor ,
* P_FindLowestCeilingSurrounding
*/
fixed_t P_FindLowestFloorSurrounding ( sector_t * sec )
{
size_t i ;
line_t * check ;
sector_t * other ;
fixed_t floorh ;
floorh = sec - > floorheight ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
check = sec - > lines [ i ] ;
other = getNextSector ( check , sec ) ;
if ( ! other )
continue ;
if ( other - > floorheight < floorh )
floorh = other - > floorheight ;
}
return floorh ;
}
/** Finds highest floor in adjacent sectors.
*
* \ param sec Sector to start in .
* \ return Highest floor height in an adjacent sector .
* \ sa P_FindLowestFloorSurrounding , P_FindNextHighestFloor ,
* P_FindHighestCeilingSurrounding
*/
fixed_t P_FindHighestFloorSurrounding ( sector_t * sec )
{
size_t i ;
line_t * check ;
sector_t * other ;
fixed_t floorh = - 500 * FRACUNIT ;
INT32 foundsector = 0 ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
check = sec - > lines [ i ] ;
other = getNextSector ( check , sec ) ;
if ( ! other )
continue ;
if ( other - > floorheight > floorh | | ! foundsector )
floorh = other - > floorheight ;
if ( ! foundsector )
foundsector = 1 ;
}
return floorh ;
}
/** Finds next highest floor in adjacent sectors.
*
* \ param sec Sector to start in .
* \ param currentheight Height to start at .
* \ return Next highest floor height in an adjacent sector , or currentheight
* if there are none higher .
* \ sa P_FindHighestFloorSurrounding , P_FindNextLowestFloor ,
* P_FindNextHighestCeiling
* \ author Lee Killough
*/
fixed_t P_FindNextHighestFloor ( sector_t * sec , fixed_t currentheight )
{
sector_t * other ;
size_t i ;
fixed_t height ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > floorheight > currentheight )
{
height = other - > floorheight ;
while ( + + i < sec - > linecount )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & &
other - > floorheight < height & &
other - > floorheight > currentheight )
height = other - > floorheight ;
}
return height ;
}
}
return currentheight ;
}
////////////////////////////////////////////////////
// SoM: Start new Boom functions
////////////////////////////////////////////////////
/** Finds next lowest floor in adjacent sectors.
*
* \ param sec Sector to start in .
* \ param currentheight Height to start at .
* \ return Next lowest floor height in an adjacent sector , or currentheight
* if there are none lower .
* \ sa P_FindLowestFloorSurrounding , P_FindNextHighestFloor ,
* P_FindNextLowestCeiling
* \ author Lee Killough
*/
fixed_t P_FindNextLowestFloor ( sector_t * sec , fixed_t currentheight )
{
sector_t * other ;
size_t i ;
fixed_t height ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > floorheight < currentheight )
{
height = other - > floorheight ;
while ( + + i < sec - > linecount )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > floorheight > height
& & other - > floorheight < currentheight )
height = other - > floorheight ;
}
return height ;
}
}
return currentheight ;
}
#if 0
/** Finds next lowest ceiling in adjacent sectors.
*
* \ param sec Sector to start in .
* \ param currentheight Height to start at .
* \ return Next lowest ceiling height in an adjacent sector , or currentheight
* if there are none lower .
* \ sa P_FindLowestCeilingSurrounding , P_FindNextHighestCeiling ,
* P_FindNextLowestFloor
* \ author Lee Killough
*/
static fixed_t P_FindNextLowestCeiling ( sector_t * sec , fixed_t currentheight )
{
sector_t * other ;
size_t i ;
fixed_t height ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > ceilingheight < currentheight )
{
height = other - > ceilingheight ;
while ( + + i < sec - > linecount )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > ceilingheight > height
& & other - > ceilingheight < currentheight )
height = other - > ceilingheight ;
}
return height ;
}
}
return currentheight ;
}
/** Finds next highest ceiling in adjacent sectors.
*
* \ param sec Sector to start in .
* \ param currentheight Height to start at .
* \ return Next highest ceiling height in an adjacent sector , or currentheight
* if there are none higher .
* \ sa P_FindHighestCeilingSurrounding , P_FindNextLowestCeiling ,
* P_FindNextHighestFloor
* \ author Lee Killough
*/
static fixed_t P_FindNextHighestCeiling ( sector_t * sec , fixed_t currentheight )
{
sector_t * other ;
size_t i ;
fixed_t height ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > ceilingheight > currentheight )
{
height = other - > ceilingheight ;
while ( + + i < sec - > linecount )
{
other = getNextSector ( sec - > lines [ i ] , sec ) ;
if ( other & & other - > ceilingheight < height
& & other - > ceilingheight > currentheight )
height = other - > ceilingheight ;
}
return height ;
}
}
return currentheight ;
}
# endif
////////////////////////////
// End New Boom functions
////////////////////////////
/** Finds lowest ceiling in adjacent sectors.
*
* \ param sec Sector to start in .
* \ return Lowest ceiling height in an adjacent sector .
* \ sa P_FindHighestCeilingSurrounding , P_FindNextLowestCeiling ,
* P_FindLowestFloorSurrounding
*/
fixed_t P_FindLowestCeilingSurrounding ( sector_t * sec )
{
size_t i ;
line_t * check ;
sector_t * other ;
fixed_t height = 32000 * FRACUNIT ; //SoM: 3/7/2000: Remove ovf
INT32 foundsector = 0 ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
check = sec - > lines [ i ] ;
other = getNextSector ( check , sec ) ;
if ( ! other )
continue ;
if ( other - > ceilingheight < height | | ! foundsector )
height = other - > ceilingheight ;
if ( ! foundsector )
foundsector = 1 ;
}
return height ;
}
/** Finds Highest ceiling in adjacent sectors.
*
* \ param sec Sector to start in .
* \ return Highest ceiling height in an adjacent sector .
* \ sa P_FindLowestCeilingSurrounding , P_FindNextHighestCeiling ,
* P_FindHighestFloorSurrounding
*/
fixed_t P_FindHighestCeilingSurrounding ( sector_t * sec )
{
size_t i ;
line_t * check ;
sector_t * other ;
fixed_t height = 0 ;
INT32 foundsector = 0 ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
check = sec - > lines [ i ] ;
other = getNextSector ( check , sec ) ;
if ( ! other )
continue ;
if ( other - > ceilingheight > height | | ! foundsector )
height = other - > ceilingheight ;
if ( ! foundsector )
foundsector = 1 ;
}
return height ;
}
#if 0
//SoM: 3/7/2000: UTILS.....
//
// P_FindShortestTextureAround()
//
// Passed a sector number, returns the shortest lower texture on a
// linedef bounding the sector.
//
//
static fixed_t P_FindShortestTextureAround ( INT32 secnum )
{
fixed_t minsize = 32000 < < FRACBITS ;
side_t * side ;
size_t i ;
sector_t * sec = & sectors [ secnum ] ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
if ( twoSided ( secnum , i ) )
{
side = getSide ( secnum , i , 0 ) ;
if ( side - > bottomtexture > 0 )
if ( textureheight [ side - > bottomtexture ] < minsize )
minsize = textureheight [ side - > bottomtexture ] ;
side = getSide ( secnum , i , 1 ) ;
if ( side - > bottomtexture > 0 )
if ( textureheight [ side - > bottomtexture ] < minsize )
minsize = textureheight [ side - > bottomtexture ] ;
}
}
return minsize ;
}
//SoM: 3/7/2000: Stuff.... (can you tell I'm getting tired? It's 12 : 30!)
//
// P_FindShortestUpperAround()
//
// Passed a sector number, returns the shortest upper texture on a
// linedef bounding the sector.
//
//
static fixed_t P_FindShortestUpperAround ( INT32 secnum )
{
fixed_t minsize = 32000 < < FRACBITS ;
side_t * side ;
size_t i ;
sector_t * sec = & sectors [ secnum ] ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
if ( twoSided ( secnum , i ) )
{
side = getSide ( secnum , i , 0 ) ;
if ( side - > toptexture > 0 )
if ( textureheight [ side - > toptexture ] < minsize )
minsize = textureheight [ side - > toptexture ] ;
side = getSide ( secnum , i , 1 ) ;
if ( side - > toptexture > 0 )
if ( textureheight [ side - > toptexture ] < minsize )
minsize = textureheight [ side - > toptexture ] ;
}
}
return minsize ;
}
//SoM: 3/7/2000
//
// 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
//
//
static sector_t * P_FindModelFloorSector ( fixed_t floordestheight , INT32 secnum )
{
size_t i ;
sector_t * sec = & sectors [ secnum ] ;
for ( i = 0 ; i < sec - > 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
//
static sector_t * P_FindModelCeilingSector ( fixed_t ceildestheight , INT32 secnum )
{
size_t i ;
sector_t * sec = & sectors [ secnum ] ;
for ( i = 0 ; i < sec - > 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 ;
}
# endif
// Parses arguments for parameterized polyobject door types
static boolean PolyDoor ( line_t * line )
{
polydoordata_t pdd ;
2021-07-02 15:18:55 +00:00
pdd . polyObjNum = line - > args [ 0 ] ; // polyobject id
2014-03-15 16:59:03 +00:00
switch ( line - > special )
{
case 480 : // Polyobj_DoorSlide
pdd . doorType = POLY_DOOR_SLIDE ;
2021-07-02 15:18:55 +00:00
pdd . speed = line - > args [ 1 ] < < ( FRACBITS - 3 ) ;
2014-03-15 16:59:03 +00:00
pdd . angle = R_PointToAngle2 ( line - > v1 - > x , line - > v1 - > y , line - > v2 - > x , line - > v2 - > y ) ; // angle of motion
2021-07-02 15:18:55 +00:00
pdd . distance = line - > args [ 2 ] < < FRACBITS ;
pdd . delay = line - > args [ 3 ] ; // delay in tics
2014-03-15 16:59:03 +00:00
break ;
case 481 : // Polyobj_DoorSwing
pdd . doorType = POLY_DOOR_SWING ;
2021-07-02 15:18:55 +00:00
pdd . speed = line - > args [ 1 ] ; // angular speed
pdd . distance = line - > args [ 2 ] ; // angular distance
pdd . delay = line - > args [ 3 ] ; // delay in tics
2014-03-15 16:59:03 +00:00
break ;
default :
return 0 ; // ???
}
return EV_DoPolyDoor ( & pdd ) ;
}
2021-07-04 10:54:48 +00:00
// Parses arguments for parameterized polyobject move special
2014-03-15 16:59:03 +00:00
static boolean PolyMove ( line_t * line )
{
polymovedata_t pmd ;
2021-07-04 10:54:48 +00:00
pmd . polyObjNum = line - > args [ 0 ] ;
pmd . speed = line - > args [ 1 ] < < ( FRACBITS - 3 ) ;
2014-03-15 16:59:03 +00:00
pmd . angle = R_PointToAngle2 ( line - > v1 - > x , line - > v1 - > y , line - > v2 - > x , line - > v2 - > y ) ;
2021-07-04 10:54:48 +00:00
pmd . distance = line - > args [ 2 ] < < FRACBITS ;
2014-03-15 16:59:03 +00:00
2021-07-04 10:54:48 +00:00
pmd . overRide = ! ! line - > args [ 3 ] ; // Polyobj_OR_Move
2014-03-15 16:59:03 +00:00
return EV_DoPolyObjMove ( & pmd ) ;
}
2021-07-04 11:45:41 +00:00
static void PolySetVisibilityTangibility ( line_t * line )
2014-03-15 16:59:03 +00:00
{
2021-07-04 11:45:41 +00:00
INT32 polyObjNum = line - > args [ 0 ] ;
polyobj_t * po ;
2014-03-15 16:59:03 +00:00
if ( ! ( po = Polyobj_GetForNum ( polyObjNum ) ) )
{
2021-07-04 11:45:41 +00:00
CONS_Debug ( DBG_POLYOBJ , " PolySetVisibilityTangibility: bad polyobj %d \n " , polyObjNum ) ;
2014-03-15 16:59:03 +00:00
return ;
}
// don't allow line actions to affect bad polyobjects
if ( po - > isBad )
return ;
2021-07-04 11:45:41 +00:00
if ( line - > args [ 1 ] = = TMPV_VISIBLE )
2014-03-15 16:59:03 +00:00
{
2021-07-04 11:45:41 +00:00
po - > flags & = ~ POF_NOSPECIALS ;
po - > flags | = ( po - > spawnflags & POF_RENDERALL ) ;
}
else if ( line - > args [ 1 ] = = TMPV_INVISIBLE )
{
po - > flags | = POF_NOSPECIALS ;
po - > flags & = ~ POF_RENDERALL ;
2014-03-15 16:59:03 +00:00
}
2021-07-04 11:45:41 +00:00
if ( line - > args [ 2 ] = = TMPT_TANGIBLE )
2014-03-15 16:59:03 +00:00
po - > flags | = POF_SOLID ;
2021-07-04 11:45:41 +00:00
else if ( line - > args [ 2 ] = = TMPT_INTANGIBLE )
po - > flags & = ~ POF_SOLID ;
2014-03-15 16:59:03 +00:00
}
// Sets the translucency of a polyobject
static void PolyTranslucency ( line_t * line )
{
2021-07-04 13:37:42 +00:00
INT32 polyObjNum = line - > args [ 0 ] ;
2014-03-15 16:59:03 +00:00
polyobj_t * po ;
if ( ! ( po = Polyobj_GetForNum ( polyObjNum ) ) )
{
2021-07-04 13:37:42 +00:00
CONS_Debug ( DBG_POLYOBJ , " PolyTranslucency: bad polyobj %d \n " , polyObjNum ) ;
2014-03-15 16:59:03 +00:00
return ;
}
// don't allow line actions to affect bad polyobjects
if ( po - > isBad )
return ;
2021-07-04 13:37:42 +00:00
if ( lines - > args [ 2 ] ) // relative calc
po - > translucency + = line - > args [ 1 ] ;
2018-09-13 04:04:50 +00:00
else
2021-07-04 13:37:42 +00:00
po - > translucency = line - > args [ 1 ] ;
2020-05-04 08:29:35 +00:00
po - > translucency = max ( min ( po - > translucency , NUMTRANSMAPS ) , 0 ) ;
2014-03-15 16:59:03 +00:00
}
2018-09-07 19:27:18 +00:00
// Makes a polyobject translucency fade and applies tangibility
2018-09-09 05:08:02 +00:00
static boolean PolyFade ( line_t * line )
2018-09-07 19:27:18 +00:00
{
2021-07-04 17:28:14 +00:00
INT32 polyObjNum = line - > args [ 0 ] ;
2018-09-07 19:27:18 +00:00
polyobj_t * po ;
2018-09-15 04:56:27 +00:00
polyfadedata_t pfd ;
2018-09-07 19:27:18 +00:00
if ( ! ( po = Polyobj_GetForNum ( polyObjNum ) ) )
{
2018-09-18 11:06:59 +00:00
CONS_Debug ( DBG_POLYOBJ , " PolyFade: bad polyobj %d \n " , polyObjNum ) ;
2018-09-09 05:08:02 +00:00
return 0 ;
2018-09-07 19:27:18 +00:00
}
// don't allow line actions to affect bad polyobjects
if ( po - > isBad )
2018-09-09 05:08:02 +00:00
return 0 ;
2018-09-07 19:27:18 +00:00
2018-09-18 11:06:59 +00:00
// Prevent continuous execs from interfering on an existing fade
2021-07-04 17:28:14 +00:00
if ( ! ( line - > args [ 3 ] & TMPF_OVERRIDE )
2018-09-18 11:06:59 +00:00
& & po - > thinker
& & po - > thinker - > function . acp1 = = ( actionf_p1 ) T_PolyObjFade )
{
CONS_Debug ( DBG_POLYOBJ , " Line type 492 Executor: Fade PolyObject thinker already exists \n " ) ;
return 0 ;
}
2018-09-13 04:04:50 +00:00
pfd . polyObjNum = polyObjNum ;
2021-07-04 17:28:14 +00:00
if ( line - > args [ 3 ] & TMPF_RELATIVE ) // relative calc
pfd . destvalue = po - > translucency + line - > args [ 1 ] ;
2018-09-13 04:04:50 +00:00
else
2021-07-04 17:28:14 +00:00
pfd . destvalue = line - > args [ 1 ] ;
2020-05-04 08:29:35 +00:00
pfd . destvalue = max ( min ( pfd . destvalue , NUMTRANSMAPS ) , 0 ) ;
2018-09-13 04:04:50 +00:00
2018-09-09 05:08:02 +00:00
// already equal, nothing to do
2018-09-13 04:04:50 +00:00
if ( po - > translucency = = pfd . destvalue )
2018-09-09 05:08:02 +00:00
return 1 ;
2018-09-07 19:27:18 +00:00
2021-07-04 17:28:14 +00:00
pfd . docollision = ! ( line - > args [ 3 ] & TMPF_IGNORECOLLISION ) ; // do not handle collision flags
pfd . doghostfade = ( line - > args [ 3 ] & TMPF_GHOSTFADE ) ; // do ghost fade (no collision flags during fade)
pfd . ticbased = ( line - > args [ 3 ] & TMPF_TICBASED ) ; // Speed = Tic Duration
2018-09-07 19:27:18 +00:00
2021-07-04 17:28:14 +00:00
pfd . speed = line - > args [ 2 ] ;
2018-09-07 19:27:18 +00:00
return EV_DoPolyObjFade ( & pfd ) ;
2014-03-15 16:59:03 +00:00
}
// Parses arguments for parameterized polyobject waypoint movement
static boolean PolyWaypoint ( line_t * line )
{
polywaypointdata_t pwd ;
2021-07-04 11:20:10 +00:00
pwd . polyObjNum = line - > args [ 0 ] ;
pwd . speed = line - > args [ 1 ] < < ( FRACBITS - 3 ) ;
pwd . sequence = line - > args [ 2 ] ;
pwd . returnbehavior = line - > args [ 3 ] ;
pwd . flags = line - > args [ 4 ] ;
2014-03-15 16:59:03 +00:00
return EV_DoPolyObjWaypoint ( & pwd ) ;
}
2021-07-04 10:54:48 +00:00
// Parses arguments for parameterized polyobject rotate special
2014-03-15 16:59:03 +00:00
static boolean PolyRotate ( line_t * line )
{
polyrotdata_t prd ;
2021-07-04 10:54:48 +00:00
prd . polyObjNum = line - > args [ 0 ] ;
prd . speed = line - > args [ 1 ] ; // angular speed
prd . distance = abs ( line - > args [ 2 ] ) ; // angular distance
prd . direction = ( line - > args [ 2 ] < 0 ) ? - 1 : 1 ;
prd . flags = line - > args [ 3 ] ;
2014-11-12 00:55:07 +00:00
2014-03-15 16:59:03 +00:00
return EV_DoPolyObjRotate ( & prd ) ;
}
2020-05-04 07:58:27 +00:00
// Parses arguments for polyobject flag waving special
static boolean PolyFlag ( line_t * line )
{
polyflagdata_t pfd ;
2021-07-04 12:10:11 +00:00
pfd . polyObjNum = line - > args [ 0 ] ;
pfd . speed = line - > args [ 1 ] ;
2021-08-24 08:13:18 +00:00
pfd . angle = line - > angle > > ANGLETOFINESHIFT ;
2021-07-04 12:10:11 +00:00
pfd . momx = line - > args [ 2 ] ;
2020-05-04 07:58:27 +00:00
return EV_DoPolyObjFlag ( & pfd ) ;
}
2014-11-12 00:55:07 +00:00
// Parses arguments for parameterized polyobject move-by-sector-heights specials
static boolean PolyDisplace ( line_t * line )
{
polydisplacedata_t pdd ;
2021-07-04 12:41:37 +00:00
fixed_t length = R_PointToDist2 ( line - > v2 - > x , line - > v2 - > y , line - > v1 - > x , line - > v1 - > y ) ;
fixed_t speed = line - > args [ 1 ] < < FRACBITS ;
2014-11-12 00:55:07 +00:00
2021-07-04 12:41:37 +00:00
pdd . polyObjNum = line - > args [ 0 ] ;
2014-11-12 00:55:07 +00:00
pdd . controlSector = line - > frontsector ;
2021-07-04 12:41:37 +00:00
pdd . dx = FixedMul ( FixedDiv ( line - > dx , length ) , speed ) > > 8 ;
pdd . dy = FixedMul ( FixedDiv ( line - > dy , length ) , speed ) > > 8 ;
2014-11-12 00:55:07 +00:00
return EV_DoPolyObjDisplace ( & pdd ) ;
}
2019-04-05 09:50:59 +00:00
2020-05-04 08:01:44 +00:00
// Parses arguments for parameterized polyobject rotate-by-sector-heights specials
2019-04-05 09:50:59 +00:00
static boolean PolyRotDisplace ( line_t * line )
{
polyrotdisplacedata_t pdd ;
fixed_t anginter , distinter ;
2021-07-04 12:41:37 +00:00
pdd . polyObjNum = line - > args [ 0 ] ;
2019-04-05 09:50:59 +00:00
pdd . controlSector = line - > frontsector ;
// Rotate 'anginter' interval for each 'distinter' interval from the control sector.
2021-07-04 12:41:37 +00:00
anginter = line - > args [ 2 ] < < FRACBITS ;
distinter = line - > args [ 1 ] < < FRACBITS ;
2019-04-05 09:50:59 +00:00
pdd . rotscale = FixedDiv ( anginter , distinter ) ;
// Same behavior as other rotators when carrying things.
2021-07-04 12:41:37 +00:00
pdd . turnobjs = line - > args [ 3 ] ;
2019-04-05 09:50:59 +00:00
return EV_DoPolyObjRotDisplace ( & pdd ) ;
}
2014-03-15 16:59:03 +00:00
2018-03-31 14:11:27 +00:00
//
// P_RunNightserizeExecutors
//
void P_RunNightserizeExecutors ( mobj_t * actor )
{
size_t i ;
for ( i = 0 ; i < numlines ; i + + )
{
2021-09-23 08:21:53 +00:00
if ( lines [ i ] . special = = 323 )
2018-03-31 14:11:27 +00:00
P_RunTriggerLinedef ( & lines [ i ] , actor , NULL ) ;
}
}
//
// P_RunDeNightserizeExecutors
//
void P_RunDeNightserizeExecutors ( mobj_t * actor )
{
size_t i ;
for ( i = 0 ; i < numlines ; i + + )
{
2021-09-23 08:21:53 +00:00
if ( lines [ i ] . special = = 325 )
2018-03-31 14:11:27 +00:00
P_RunTriggerLinedef ( & lines [ i ] , actor , NULL ) ;
}
}
//
// P_RunNightsLapExecutors
//
void P_RunNightsLapExecutors ( mobj_t * actor )
{
size_t i ;
for ( i = 0 ; i < numlines ; i + + )
{
2021-09-23 08:21:53 +00:00
if ( lines [ i ] . special = = 327 )
2018-03-31 14:11:27 +00:00
P_RunTriggerLinedef ( & lines [ i ] , actor , NULL ) ;
}
}
2018-04-01 12:23:59 +00:00
//
2018-08-14 16:00:37 +00:00
// P_RunNightsCapsuleTouchExecutors
2018-04-01 12:23:59 +00:00
//
2018-08-14 16:00:37 +00:00
void P_RunNightsCapsuleTouchExecutors ( mobj_t * actor , boolean entering , boolean enoughspheres )
2018-04-01 12:23:59 +00:00
{
size_t i ;
for ( i = 0 ; i < numlines ; i + + )
{
2021-09-23 08:21:53 +00:00
if ( lines [ i ] . special ! = 329 )
continue ;
if ( ! ! ( lines [ i ] . args [ 7 ] & TMI_ENTER ) ! = entering )
continue ;
if ( lines [ i ] . args [ 6 ] = = TMS_IFENOUGH & & ! enoughspheres )
continue ;
if ( lines [ i ] . args [ 6 ] = = TMS_IFNOTENOUGH & & enoughspheres )
continue ;
P_RunTriggerLinedef ( & lines [ i ] , actor , NULL ) ;
2018-04-01 12:23:59 +00:00
}
}
2014-03-15 16:59:03 +00:00
/** Finds minimum light from an adjacent sector.
*
* \ param sector Sector to start in .
* \ param max Maximum value to return .
* \ return Minimum light value from an adjacent sector , or max if the minimum
* light value is greater than max .
*/
INT32 P_FindMinSurroundingLight ( sector_t * sector , INT32 max )
{
size_t i ;
INT32 min = max ;
line_t * line ;
sector_t * check ;
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 ;
}
void T_ExecutorDelay ( executor_t * e )
{
if ( - - e - > timer < = 0 )
{
if ( e - > caller & & P_MobjWasRemoved ( e - > caller ) ) // If the mobj died while we were delaying
P_SetTarget ( & e - > caller , NULL ) ; // Call with no mobj!
2014-11-12 00:55:07 +00:00
P_ProcessLineSpecial ( e - > line , e - > caller , e - > sector ) ;
2014-03-15 16:59:03 +00:00
P_SetTarget ( & e - > caller , NULL ) ; // Let the mobj know it can be removed now.
P_RemoveThinker ( & e - > thinker ) ;
}
}
2014-11-12 00:55:07 +00:00
static void P_AddExecutorDelay ( line_t * line , mobj_t * mobj , sector_t * sector )
2014-03-15 16:59:03 +00:00
{
executor_t * e ;
2020-05-03 18:41:37 +00:00
INT32 delay ;
2014-03-15 16:59:03 +00:00
2020-05-03 18:41:37 +00:00
if ( udmf )
delay = line - > executordelay ;
else
{
if ( ! line - > backsector )
I_Error ( " P_AddExecutorDelay: Line has no backsector! \n " ) ;
2014-03-15 16:59:03 +00:00
2020-05-03 18:41:37 +00:00
delay = ( line - > backsector - > ceilingheight > > FRACBITS ) + ( line - > backsector - > floorheight > > FRACBITS ) ;
}
2014-03-15 16:59:03 +00:00
e = Z_Calloc ( sizeof ( * e ) , PU_LEVSPEC , NULL ) ;
e - > thinker . function . acp1 = ( actionf_p1 ) T_ExecutorDelay ;
e - > line = line ;
2014-11-12 00:55:07 +00:00
e - > sector = sector ;
2020-05-03 18:41:37 +00:00
e - > timer = delay ;
2014-03-15 16:59:03 +00:00
P_SetTarget ( & e - > caller , mobj ) ; // Use P_SetTarget to make sure the mobj doesn't get freed while we're delaying.
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & e - > thinker ) ;
2014-03-15 16:59:03 +00:00
}
2018-04-02 12:11:28 +00:00
/** Used by P_RunTriggerLinedef to check a NiGHTS trigger linedef's conditions
*
* \ param triggerline Trigger linedef to check conditions for ; should NEVER be NULL .
* \ param actor Object initiating the action ; should not be NULL .
* \ sa P_RunTriggerLinedef
*/
static boolean P_CheckNightsTriggerLine ( line_t * triggerline , mobj_t * actor )
{
INT16 specialtype = triggerline - > special ;
size_t i ;
2021-09-23 08:21:53 +00:00
UINT8 inputmare = max ( 0 , min ( 255 , triggerline - > args [ 1 ] ) ) ;
UINT8 inputlap = max ( 0 , min ( 255 , triggerline - > args [ 2 ] ) ) ;
2018-04-02 12:11:28 +00:00
2021-09-23 08:21:53 +00:00
textmapcomparison_t marecomp = triggerline - > args [ 3 ] ;
textmapcomparison_t lapcomp = triggerline - > args [ 4 ] ;
textmapnightsplayer_t checkplayer = triggerline - > args [ 5 ] ;
2018-04-02 12:11:28 +00:00
2021-09-23 08:21:53 +00:00
boolean lapfrombonustime ;
2018-04-02 12:11:28 +00:00
2021-09-23 08:21:53 +00:00
boolean donomares = ( specialtype = = 323 ) & & ( triggerline - > args [ 7 ] & TMN_LEVELCOMPLETION ) ; // nightserize: run at end of level (no mares)
2018-04-02 12:11:28 +00:00
UINT8 currentmare = UINT8_MAX ;
UINT8 currentlap = UINT8_MAX ;
2021-09-23 08:21:53 +00:00
// Set lapfrombonustime
switch ( specialtype )
{
case 323 :
lapfrombonustime = ! ! ( triggerline - > args [ 7 ] & TMN_BONUSLAPS ) ;
break ;
case 325 :
lapfrombonustime = ! ! ( triggerline - > args [ 7 ] ) ;
break ;
case 327 :
lapfrombonustime = ! ! ( triggerline - > args [ 6 ] ) ;
break ;
case 329 :
lapfrombonustime = ! ! ( triggerline - > args [ 7 ] & TMI_BONUSLAPS ) ;
break ;
default :
lapfrombonustime = false ;
break ;
}
2018-04-02 12:11:28 +00:00
// Do early returns for Nightserize
2021-09-23 08:21:53 +00:00
if ( specialtype = = 323 )
2018-04-02 12:11:28 +00:00
{
// run only when no mares are found
if ( donomares & & P_FindLowestMare ( ) ! = UINT8_MAX )
return false ;
2018-04-02 12:13:53 +00:00
// run only if there is a mare present
if ( ! donomares & & P_FindLowestMare ( ) = = UINT8_MAX )
return false ;
2018-04-02 12:11:28 +00:00
// run only if player is nightserizing from non-nights
2021-09-23 08:21:53 +00:00
if ( triggerline - > args [ 6 ] = = TMN_FROMNONIGHTS )
2018-04-02 12:11:28 +00:00
{
if ( ! actor - > player )
return false ;
else if ( actor - > player - > powers [ pw_carry ] = = CR_NIGHTSMODE )
return false ;
}
2018-04-07 11:33:07 +00:00
// run only if player is nightserizing from nights
2021-09-23 08:21:53 +00:00
else if ( triggerline - > args [ 6 ] = = TMN_FROMNIGHTS )
2018-04-07 11:33:07 +00:00
{
if ( ! actor - > player )
return false ;
else if ( actor - > player - > powers [ pw_carry ] ! = CR_NIGHTSMODE )
return false ;
}
2018-04-02 12:11:28 +00:00
}
// Get current mare and lap (and check early return for DeNightserize)
2021-09-23 08:21:53 +00:00
if ( checkplayer ! = TMNP_TRIGGERER
| | ( specialtype = = 325 & & triggerline - > args [ 6 ] ! = TMD_ALWAYS ) )
2018-04-02 12:11:28 +00:00
{
2018-04-07 11:33:07 +00:00
UINT8 playersarenights = 0 ;
2018-04-02 12:11:28 +00:00
for ( i = 0 ; i < MAXPLAYERS ; i + + )
{
2018-09-04 16:04:24 +00:00
UINT8 lap ;
2018-04-02 12:11:28 +00:00
if ( ! playeringame [ i ] | | players [ i ] . spectator )
continue ;
// denightserize: run only if all players are not nights
2021-09-23 08:21:53 +00:00
if ( specialtype = = 325 & & triggerline - > args [ 6 ] = = TMD_NOBODYNIGHTS
2018-04-02 12:11:28 +00:00
& & players [ i ] . powers [ pw_carry ] = = CR_NIGHTSMODE )
return false ;
2018-04-07 11:33:07 +00:00
// count number of nights players for denightserize return
2021-09-23 08:21:53 +00:00
if ( specialtype = = 325 & & triggerline - > args [ 6 ] = = TMD_SOMEBODYNIGHTS
2018-04-07 11:33:07 +00:00
& & players [ i ] . powers [ pw_carry ] = = CR_NIGHTSMODE )
playersarenights + + ;
2018-09-04 16:04:24 +00:00
lap = lapfrombonustime ? players [ i ] . marebonuslap : players [ i ] . marelap ;
2018-04-02 12:11:28 +00:00
// get highest mare/lap of players
2021-09-23 08:21:53 +00:00
if ( checkplayer = = TMNP_FASTEST )
2018-04-02 12:11:28 +00:00
{
if ( players [ i ] . mare > currentmare | | currentmare = = UINT8_MAX )
{
currentmare = players [ i ] . mare ;
currentlap = UINT8_MAX ;
}
2018-08-14 14:48:50 +00:00
if ( players [ i ] . mare = = currentmare
2018-04-02 12:11:28 +00:00
& & ( lap > currentlap | | currentlap = = UINT8_MAX ) )
currentlap = lap ;
}
// get lowest mare/lap of players
2021-09-23 08:21:53 +00:00
else if ( checkplayer = = TMNP_SLOWEST )
2018-04-02 12:11:28 +00:00
{
if ( players [ i ] . mare < currentmare | | currentmare = = UINT8_MAX )
{
currentmare = players [ i ] . mare ;
currentlap = UINT8_MAX ;
}
2018-08-14 14:48:50 +00:00
if ( players [ i ] . mare = = currentmare
2018-04-02 12:11:28 +00:00
& & ( lap < currentlap | | currentlap = = UINT8_MAX ) )
currentlap = lap ;
}
}
2018-04-07 11:33:07 +00:00
// denightserize: run only if >0 players are nights
2021-09-23 08:21:53 +00:00
if ( specialtype = = 325 & & triggerline - > args [ 6 ] = = TMD_SOMEBODYNIGHTS
2018-04-07 11:33:07 +00:00
& & playersarenights < 1 )
return false ;
2018-04-02 12:11:28 +00:00
}
// get current mare/lap from triggering player
2021-09-23 08:21:53 +00:00
else if ( checkplayer = = TMNP_TRIGGERER )
2018-04-02 12:11:28 +00:00
{
if ( ! actor - > player )
return false ;
currentmare = actor - > player - > mare ;
currentlap = lapfrombonustime ? actor - > player - > marebonuslap : actor - > player - > marelap ;
}
2018-08-14 20:08:21 +00:00
if ( lapfrombonustime & & ! currentlap )
return false ; // special case: player->marebonuslap is 0 until passing through on bonus time. Don't trigger lines looking for inputlap 0.
2018-04-02 12:11:28 +00:00
// Compare current mare/lap to input mare/lap based on rules
2021-09-23 08:21:53 +00:00
if ( ! donomares // don't return false if donomares and we got this far
& & ( ( marecomp = = TMC_LTE & & currentmare > inputmare )
| | ( marecomp = = TMC_GTE & & currentmare < inputmare )
| | ( marecomp = = TMC_EQUAL & & currentmare ! = inputmare )
| | ( lapcomp = = TMC_LTE & & currentlap > inputlap )
| | ( lapcomp = = TMC_GTE & & currentlap < inputlap )
| | ( lapcomp = = TMC_EQUAL & & currentlap ! = inputlap ) )
2018-04-02 12:11:28 +00:00
)
return false ;
return true ;
}
2021-12-09 19:57:44 +00:00
static boolean P_CheckPlayerMareOld ( line_t * triggerline )
2014-03-15 16:59:03 +00:00
{
2021-12-09 06:10:16 +00:00
UINT8 mare ;
2021-12-09 18:17:16 +00:00
INT32 targetmare = P_AproxDistance ( triggerline - > dx , triggerline - > dy ) > > FRACBITS ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
if ( ! ( maptol & TOL_NIGHTS ) )
return false ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
mare = P_FindLowestMare ( ) ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
if ( triggerline - > flags & ML_NOCLIMB )
return mare < = targetmare ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
if ( triggerline - > flags & ML_BLOCKMONSTERS )
return mare > = targetmare ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
return mare = = targetmare ;
}
2014-03-15 16:59:03 +00:00
2021-12-09 19:57:44 +00:00
static boolean P_CheckPlayerMare ( line_t * triggerline )
{
UINT8 mare ;
INT32 targetmare = triggerline - > args [ 1 ] ;
if ( ! ( maptol & TOL_NIGHTS ) )
return false ;
mare = P_FindLowestMare ( ) ;
switch ( triggerline - > args [ 2 ] )
{
case TMC_EQUAL :
default :
return mare = = targetmare ;
case TMC_GTE :
return mare > = targetmare ;
case TMC_LTE :
return mare < = targetmare ;
}
}
2021-12-09 06:47:01 +00:00
static boolean P_CheckPlayerRings ( line_t * triggerline , mobj_t * actor )
2021-12-09 06:10:16 +00:00
{
INT32 rings = 0 ;
2021-12-09 06:47:01 +00:00
INT32 targetrings = triggerline - > args [ 1 ] ;
2021-12-09 06:10:16 +00:00
size_t i ;
2014-03-15 16:59:03 +00:00
2021-12-09 06:47:01 +00:00
// Count all players' rings.
if ( triggerline - > args [ 3 ] )
2021-12-09 06:10:16 +00:00
{
for ( i = 0 ; i < MAXPLAYERS ; i + + )
2014-03-15 16:59:03 +00:00
{
2021-12-09 06:10:16 +00:00
if ( ! playeringame [ i ] | | players [ i ] . spectator )
continue ;
if ( ! players [ i ] . mo | | ( ( maptol & TOL_NIGHTS ) ? players [ i ] . spheres : players [ i ] . rings ) < = 0 )
continue ;
rings + = ( maptol & TOL_NIGHTS ) ? players [ i ] . spheres : players [ i ] . rings ;
2014-08-04 03:49:33 +00:00
}
}
2021-12-09 06:10:16 +00:00
else
2014-08-04 03:49:33 +00:00
{
2021-12-09 06:10:16 +00:00
if ( ! ( actor & & actor - > player ) )
return false ; // no player to count rings from here, sorry
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
rings = ( maptol & TOL_NIGHTS ) ? actor - > player - > spheres : actor - > player - > rings ;
}
2014-03-15 16:59:03 +00:00
2021-12-09 06:47:01 +00:00
switch ( triggerline - > args [ 2 ] )
{
case TMC_EQUAL :
default :
return rings = = targetrings ;
case TMC_GTE :
return rings > = targetrings ;
case TMC_LTE :
return rings < = targetrings ;
}
2021-12-09 06:10:16 +00:00
}
2021-12-09 06:56:15 +00:00
static boolean P_CheckPushables ( line_t * triggerline , sector_t * caller )
2021-12-09 06:10:16 +00:00
{
msecnode_t * node ;
mobj_t * mo ;
INT32 numpushables = 0 ;
2021-12-09 06:56:15 +00:00
INT32 targetpushables = triggerline - > args [ 1 ] ;
2021-12-09 06:10:16 +00:00
if ( ! caller )
return false ; // we need a calling sector to find pushables in, silly!
// Count the pushables in this sector
for ( node = caller - > touching_thinglist ; node ; node = node - > m_thinglist_next )
{
mo = node - > m_thing ;
if ( ( mo - > flags & MF_PUSHABLE ) | | ( ( mo - > info - > flags & MF_PUSHABLE ) & & mo - > fuse ) )
numpushables + + ;
}
2021-12-09 06:56:15 +00:00
switch ( triggerline - > args [ 2 ] )
{
case TMC_EQUAL :
default :
return numpushables = = targetpushables ;
case TMC_GTE :
return numpushables > = targetpushables ;
case TMC_LTE :
return numpushables < = targetpushables ;
}
2021-12-09 06:10:16 +00:00
}
2021-12-09 19:37:39 +00:00
static boolean P_CheckEmeralds ( INT32 checktype , UINT16 target )
{
switch ( checktype )
{
case TMF_HASALL :
default :
return ( emeralds & target ) = = target ;
case TMF_HASANY :
return ! ! ( emeralds & target ) ;
case TMF_HASEXACTLY :
return emeralds = = target ;
case TMF_DOESNTHAVEALL :
return ( emeralds & target ) ! = target ;
case TMF_DOESNTHAVEANY :
return ! ( emeralds & target ) ;
}
}
2021-12-09 06:10:16 +00:00
static void P_ActivateLinedefExecutor ( line_t * line , mobj_t * actor , sector_t * caller )
{
if ( line - > special < 400 | | line - > special > = 500 )
return ;
if ( line - > executordelay )
P_AddExecutorDelay ( line , actor , caller ) ;
else
P_ProcessLineSpecial ( line , actor , caller ) ;
}
static boolean P_ActivateLinedefExecutorsInSector ( line_t * triggerline , mobj_t * actor , sector_t * caller )
{
sector_t * ctlsector = triggerline - > frontsector ;
size_t sectori = ( size_t ) ( ctlsector - sectors ) ;
size_t linecnt = ctlsector - > linecount ;
size_t i ;
2022-01-09 12:58:27 +00:00
if ( ! udmf & & triggerline - > flags & ML_WRAPMIDTEX ) // disregard order for efficiency
2021-12-09 06:10:16 +00:00
{
for ( i = 0 ; i < linecnt ; i + + )
P_ActivateLinedefExecutor ( ctlsector - > lines [ i ] , actor , caller ) ;
}
else // walk around the sector in a defined order
{
boolean backwards = false ;
size_t j , masterlineindex = ( size_t ) - 1 ;
for ( i = 0 ; i < linecnt ; i + + )
if ( ctlsector - > lines [ i ] = = triggerline )
{
masterlineindex = i ;
break ;
}
# ifdef PARANOIA
if ( masterlineindex = = ( size_t ) - 1 )
2014-08-04 03:49:33 +00:00
{
2021-12-09 06:10:16 +00:00
const size_t li = ( size_t ) ( ctlsector - > lines [ i ] - lines ) ;
I_Error ( " Line %s isn't linked into its front sector " , sizeu1 ( li ) ) ;
2014-08-04 03:49:33 +00:00
}
2021-12-09 06:10:16 +00:00
# endif
// i == masterlineindex
for ( ; ; )
2014-08-04 03:49:33 +00:00
{
2021-12-09 06:10:16 +00:00
if ( backwards ) // v2 to v1
{
for ( j = 0 ; j < linecnt ; j + + )
{
if ( i = = j )
continue ;
if ( ctlsector - > lines [ i ] - > v1 = = ctlsector - > lines [ j ] - > v2 )
{
i = j ;
break ;
}
if ( ctlsector - > lines [ i ] - > v1 = = ctlsector - > lines [ j ] - > v1 )
{
i = j ;
backwards = false ;
break ;
}
}
if ( j = = linecnt )
{
const size_t vertexei = ( size_t ) ( ctlsector - > lines [ i ] - > v1 - vertexes ) ;
CONS_Debug ( DBG_GAMELOGIC , " Warning: Sector %s is not closed at vertex %s (%d, %d) \n " ,
sizeu1 ( sectori ) , sizeu2 ( vertexei ) , ctlsector - > lines [ i ] - > v1 - > x , ctlsector - > lines [ i ] - > v1 - > y ) ;
return false ; // abort
}
}
else // v1 to v2
{
for ( j = 0 ; j < linecnt ; j + + )
{
if ( i = = j )
continue ;
if ( ctlsector - > lines [ i ] - > v2 = = ctlsector - > lines [ j ] - > v1 )
{
i = j ;
break ;
}
if ( ctlsector - > lines [ i ] - > v2 = = ctlsector - > lines [ j ] - > v2 )
{
i = j ;
backwards = true ;
break ;
}
}
if ( j = = linecnt )
{
const size_t vertexei = ( size_t ) ( ctlsector - > lines [ i ] - > v1 - vertexes ) ;
CONS_Debug ( DBG_GAMELOGIC , " Warning: Sector %s is not closed at vertex %s (%d, %d) \n " ,
sizeu1 ( sectori ) , sizeu2 ( vertexei ) , ctlsector - > lines [ i ] - > v2 - > x , ctlsector - > lines [ i ] - > v2 - > y ) ;
return false ; // abort
}
}
if ( i = = masterlineindex )
break ;
P_ActivateLinedefExecutor ( ctlsector - > lines [ i ] , actor , caller ) ;
2014-08-04 03:49:33 +00:00
}
}
2021-12-09 06:10:16 +00:00
return true ;
}
/** Used by P_LinedefExecute to check a trigger linedef's conditions
* The linedef executor specials in the trigger linedef ' s sector are run if all conditions are met .
*
* \ param triggerline Trigger linedef to check conditions for ; should NEVER be NULL .
* \ param actor Object initiating the action ; should not be NULL .
* \ param caller Sector in which the action was started . May be NULL .
* \ sa P_ProcessLineSpecial , P_LinedefExecute
*/
2023-03-16 10:47:21 +00:00
void P_RunTriggerLinedef ( line_t * triggerline , mobj_t * actor , sector_t * caller )
2021-12-09 06:10:16 +00:00
{
INT16 specialtype = triggerline - > special ;
////////////////////////
// Trigger conditions //
////////////////////////
2021-12-31 10:39:34 +00:00
if ( caller & & ! udmf )
2014-08-04 03:49:33 +00:00
{
2021-12-31 10:39:34 +00:00
if ( caller - > triggerer = = TO_PLAYEREMERALDS )
2014-08-04 03:49:33 +00:00
{
if ( ! ( ALL7EMERALDS ( emeralds ) ) )
2023-03-16 10:47:21 +00:00
return ;
2014-08-04 03:49:33 +00:00
}
2021-12-31 10:39:34 +00:00
else if ( caller - > triggerer = = TO_PLAYERNIGHTS )
2014-08-04 03:49:33 +00:00
{
2021-12-09 19:57:44 +00:00
if ( ! P_CheckPlayerMareOld ( triggerline ) )
2023-03-16 10:47:21 +00:00
return ;
2014-03-15 16:59:03 +00:00
}
2014-08-04 03:49:33 +00:00
}
2014-03-15 16:59:03 +00:00
2014-08-04 03:49:33 +00:00
switch ( specialtype )
{
2021-12-09 06:47:01 +00:00
case 303 :
if ( ! P_CheckPlayerRings ( triggerline , actor ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 06:10:16 +00:00
break ;
2021-12-09 18:17:16 +00:00
case 305 :
if ( ! ( actor & & actor - > player & & actor - > player - > charability = = triggerline - > args [ 1 ] ) )
2023-03-16 10:47:21 +00:00
return ;
2014-08-04 03:49:33 +00:00
break ;
2021-12-09 07:49:50 +00:00
case 309 :
// Only red/blue team members can activate this.
if ( ! ( actor & & actor - > player ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 07:49:50 +00:00
if ( actor - > player - > ctfteam ! = ( ( triggerline - > args [ 1 ] = = TMT_RED ) ? 1 : 2 ) )
2023-03-16 10:47:21 +00:00
return ;
2014-08-04 03:49:33 +00:00
break ;
2021-12-09 06:56:15 +00:00
case 314 :
if ( ! P_CheckPushables ( triggerline , caller ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 06:10:16 +00:00
break ;
2021-12-09 07:08:04 +00:00
case 317 :
2014-08-04 03:49:33 +00:00
{ // Unlockable triggers required
2021-12-09 07:08:04 +00:00
INT32 trigid = triggerline - > args [ 1 ] ;
2014-08-04 03:49:33 +00:00
2022-10-12 05:18:02 +00:00
if ( trigid < 0 | | trigid > 31 ) // limited by 32 bit variable
2014-03-15 16:59:03 +00:00
{
2023-10-25 16:16:10 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Unlockable trigger (sidedef %u): bad trigger ID %d \n " , triggerline - > sidenum [ 0 ] , trigid ) ;
2023-03-16 10:47:21 +00:00
return ;
2014-03-15 16:59:03 +00:00
}
2014-08-04 03:49:33 +00:00
else if ( ! ( unlocktriggers & ( 1 < < trigid ) ) )
2023-03-16 10:47:21 +00:00
return ;
2014-08-04 03:49:33 +00:00
}
break ;
2021-12-09 07:08:04 +00:00
case 319 :
2014-08-04 03:49:33 +00:00
{ // An unlockable itself must be unlocked!
2021-12-09 07:08:04 +00:00
INT32 unlockid = triggerline - > args [ 1 ] ;
2014-08-04 03:49:33 +00:00
2023-05-24 10:54:19 +00:00
if ( unlockid < = 0 | | unlockid > MAXUNLOCKABLES ) // limited by unlockable count
2014-03-15 16:59:03 +00:00
{
2023-10-25 16:16:10 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Unlockable check (sidedef %u): bad unlockable ID %d \n " , triggerline - > sidenum [ 0 ] , unlockid ) ;
2023-03-16 10:47:21 +00:00
return ;
2014-03-15 16:59:03 +00:00
}
2022-10-12 05:18:02 +00:00
else if ( ! ( serverGamedata - > unlocked [ unlockid - 1 ] ) )
2023-03-16 10:47:21 +00:00
return ;
2014-03-15 16:59:03 +00:00
}
2014-08-04 03:49:33 +00:00
break ;
2021-12-09 17:15:27 +00:00
case 321 :
2014-11-12 00:55:07 +00:00
// decrement calls left before triggering
if ( triggerline - > callcount > 0 )
{
if ( - - triggerline - > callcount > 0 )
2023-03-16 10:47:21 +00:00
return ;
2014-11-12 00:55:07 +00:00
}
break ;
2021-09-23 08:21:53 +00:00
case 323 : // nightserize
case 325 : // denightserize
case 327 : // nights lap
case 329 : // nights egg capsule touch
2018-04-02 12:11:28 +00:00
if ( ! P_CheckNightsTriggerLine ( triggerline , actor ) )
2023-03-16 10:47:21 +00:00
return ;
2018-03-31 14:11:27 +00:00
break ;
2021-12-09 18:33:02 +00:00
case 331 :
if ( ! ( actor & & actor - > player ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 18:33:02 +00:00
if ( ! triggerline - > stringargs [ 0 ] )
2023-03-16 10:47:21 +00:00
return ;
2023-10-28 22:09:42 +00:00
if ( ! ( stricmp ( triggerline - > stringargs [ 0 ] , skins [ actor - > player - > skin ] - > name ) = = 0 ) ^ ! ! ( triggerline - > args [ 1 ] ) )
2023-03-16 10:47:21 +00:00
return ;
2019-07-03 07:19:29 +00:00
break ;
2021-12-09 18:49:17 +00:00
case 334 : // object dye
2020-01-11 23:40:57 +00:00
{
2021-12-09 18:49:17 +00:00
INT32 triggercolor = triggerline - > stringargs [ 0 ] ? get_number ( triggerline - > stringargs [ 0 ] ) : SKINCOLOR_NONE ;
2020-08-25 02:53:21 +00:00
UINT16 color = ( actor - > player ? actor - > player - > powers [ pw_dye ] : actor - > color ) ;
2020-04-14 08:31:07 +00:00
2021-12-09 18:49:17 +00:00
if ( ! ! ( triggerline - > args [ 1 ] ) ^ ( triggercolor ! = color ) )
2023-03-16 10:47:21 +00:00
return ;
2020-01-11 23:40:57 +00:00
}
2021-12-09 19:37:39 +00:00
break ;
case 337 : // emerald check
if ( ! P_CheckEmeralds ( triggerline - > args [ 2 ] , ( UINT16 ) triggerline - > args [ 1 ] ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 19:37:39 +00:00
break ;
2021-12-09 19:57:44 +00:00
case 340 : // NiGHTS mare
if ( ! P_CheckPlayerMare ( triggerline ) )
2023-03-16 10:47:21 +00:00
return ;
2021-12-09 19:57:44 +00:00
break ;
2022-09-17 00:14:25 +00:00
case 343 : // gravity check
2022-10-04 08:38:57 +00:00
if ( triggerline - > args [ 1 ] = = TMG_TEMPREVERSE & & ( ! ( actor - > flags2 & MF2_OBJECTFLIP ) ! = ! ( actor - > player - > powers [ pw_gravityboots ] ) ) )
2023-03-16 10:47:21 +00:00
return ;
2022-10-04 08:38:57 +00:00
if ( ( triggerline - > args [ 1 ] = = TMG_NORMAL ) ! = ! ( actor - > eflags & MFE_VERTICALFLIP ) )
2023-03-16 10:47:21 +00:00
return ;
2022-09-17 00:14:25 +00:00
break ;
2014-08-04 03:49:33 +00:00
default :
break ;
}
2014-03-15 16:59:03 +00:00
2014-08-04 03:49:33 +00:00
/////////////////////////////////
// Processing linedef specials //
/////////////////////////////////
2014-03-15 16:59:03 +00:00
2021-12-09 06:10:16 +00:00
if ( ! P_ActivateLinedefExecutorsInSector ( triggerline , actor , caller ) )
2023-03-16 10:47:21 +00:00
return ;
2014-08-04 03:49:33 +00:00
2021-12-09 17:15:27 +00:00
// "Trigger on X calls" linedefs reset if args[2] is set
if ( specialtype = = 321 & & triggerline - > args [ 2 ] )
2022-12-04 08:14:55 +00:00
triggerline - > callcount = triggerline - > args [ 1 ] ;
2014-11-12 00:55:07 +00:00
else
2021-12-09 19:02:03 +00:00
{
// These special types work only once
if ( specialtype = = 313 // No more enemies
| | specialtype = = 321 // Trigger on X calls
| | specialtype = = 399 ) // Level Load
triggerline - > special = 0 ;
else if ( ( specialtype = = 323 // Nightserize
| | specialtype = = 325 // DeNightserize
| | specialtype = = 327 // Nights lap
| | specialtype = = 329 ) // Nights bonus time
& & triggerline - > args [ 0 ] )
triggerline - > special = 0 ;
else if ( ( specialtype = = 300 // Basic
| | specialtype = = 303 // Ring count
| | specialtype = = 305 // Character ability
| | specialtype = = 308 // Gametype
| | specialtype = = 309 // CTF team
| | specialtype = = 314 // No of pushables
| | specialtype = = 317 // Unlockable trigger
| | specialtype = = 319 // Unlockable
| | specialtype = = 331 // Player skin
2021-12-09 19:37:39 +00:00
| | specialtype = = 334 // Object dye
2022-09-17 00:14:25 +00:00
| | specialtype = = 337 // Emerald check
| | specialtype = = 343 ) // Gravity check
2021-12-09 19:02:03 +00:00
& & triggerline - > args [ 0 ] = = TMT_ONCE )
triggerline - > special = 0 ;
}
2014-08-04 03:49:33 +00:00
}
/** Runs a linedef executor.
* Can be called by :
* - a player moving into a special sector or FOF .
* - a pushable object moving into a special sector or FOF .
* - a ceiling or floor movement from a previous linedef executor finishing .
* - any object in a state with the A_LinedefExecute ( ) action .
*
* \ param tag Tag of the linedef executor to run .
* \ param actor Object initiating the action ; should not be NULL .
* \ param caller Sector in which the action was started . May be NULL .
* \ sa P_ProcessLineSpecial , P_RunTriggerLinedef
* \ author Graue < graue @ oceanbase . org >
*/
void P_LinedefExecute ( INT16 tag , mobj_t * actor , sector_t * caller )
{
2021-09-22 08:38:08 +00:00
INT32 masterline ;
2014-08-04 03:49:33 +00:00
CONS_Debug ( DBG_GAMELOGIC , " P_LinedefExecute: Executing trigger linedefs of tag %d \n " , tag ) ;
I_Assert ( ! actor | | ! P_MobjWasRemoved ( actor ) ) ; // If actor is there, it must be valid.
2021-09-22 08:38:08 +00:00
TAG_ITER_LINES ( tag , masterline )
2014-08-04 03:49:33 +00:00
{
2021-09-22 08:38:08 +00:00
if ( lines [ masterline ] . special < 300
| | lines [ masterline ] . special > 399 )
2014-08-04 03:49:33 +00:00
continue ;
2014-08-27 03:56:30 +00:00
// "No More Enemies" and "Level Load" take care of themselves.
2021-12-09 17:56:50 +00:00
if ( lines [ masterline ] . special = = 313 | | lines [ masterline ] . special = = 399 )
continue ;
// Each-time executors handle themselves, too
if ( ( lines [ masterline ] . special = = 300 // Basic
| | lines [ masterline ] . special = = 303 // Ring count
2021-12-09 18:17:16 +00:00
| | lines [ masterline ] . special = = 305 // Character ability
2021-12-09 17:56:50 +00:00
| | lines [ masterline ] . special = = 308 // Gametype
| | lines [ masterline ] . special = = 309 // CTF team
| | lines [ masterline ] . special = = 314 // Number of pushables
| | lines [ masterline ] . special = = 317 // Condition set trigger
2021-12-09 18:33:02 +00:00
| | lines [ masterline ] . special = = 319 // Unlockable trigger
2021-12-09 18:49:17 +00:00
| | lines [ masterline ] . special = = 331 // Player skin
2021-12-09 19:37:39 +00:00
| | lines [ masterline ] . special = = 334 // Object dye
2022-09-17 00:14:25 +00:00
| | lines [ masterline ] . special = = 337 // Emerald check
| | lines [ masterline ] . special = = 343 ) // Gravity check
2021-12-09 17:56:50 +00:00
& & lines [ masterline ] . args [ 0 ] > TMT_EACHTIMEMASK )
continue ;
if ( lines [ masterline ] . special = = 321 & & lines [ masterline ] . args [ 0 ] > TMXT_EACHTIMEMASK ) // Trigger after X calls
continue ;
2023-03-16 10:47:21 +00:00
P_RunTriggerLinedef ( & lines [ masterline ] , actor , caller ) ; // Even if it fails, there might be more linedefs to trigger
2014-08-04 03:49:33 +00:00
}
2014-03-15 16:59:03 +00:00
}
2021-09-21 08:14:55 +00:00
static void P_PlaySFX ( INT32 sfxnum , mobj_t * mo , sector_t * callsec , INT16 tag , textmapsoundsource_t source , textmapsoundlistener_t listener )
{
if ( sfxnum = = sfx_None )
return ; // Do nothing!
if ( sfxnum < sfx_None | | sfxnum > = NUMSFX )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 414 Executor: sfx number %d is invalid! \n " , sfxnum ) ;
return ;
}
// Check if you can hear the sound
switch ( listener )
{
case TMSL_TRIGGERER : // only play sound if displayplayer
if ( ! mo )
return ;
if ( ! mo - > player )
return ;
if ( mo - > player ! = & players [ displayplayer ] & & mo - > player ! = & players [ secondarydisplayplayer ] )
return ;
break ;
case TMSL_TAGGEDSECTOR : // only play if touching tagged sectors
{
UINT8 i = 0 ;
mobj_t * camobj = players [ displayplayer ] . mo ;
ffloor_t * rover ;
boolean foundit = false ;
for ( i = 0 ; i < 2 ; camobj = players [ secondarydisplayplayer ] . mo , i + + )
{
if ( ! camobj )
continue ;
if ( foundit | | Tag_Find ( & camobj - > subsector - > sector - > tags , tag ) )
{
foundit = true ;
break ;
}
// Only trigger if mobj is touching the tag
for ( rover = camobj - > subsector - > sector - > ffloors ; rover ; rover = rover - > next )
{
if ( ! Tag_Find ( & rover - > master - > frontsector - > tags , tag ) )
continue ;
if ( camobj - > z > P_GetSpecialTopZ ( camobj , sectors + rover - > secnum , camobj - > subsector - > sector ) )
continue ;
if ( camobj - > z + camobj - > height < P_GetSpecialBottomZ ( camobj , sectors + rover - > secnum , camobj - > subsector - > sector ) )
continue ;
foundit = true ;
break ;
}
}
if ( ! foundit )
return ;
break ;
}
case TMSL_EVERYONE : // no additional check
default :
break ;
}
// Play the sound from the specified source
switch ( source )
{
case TMSS_TRIGGERMOBJ : // play the sound from mobj that triggered it
if ( mo )
S_StartSound ( mo , sfxnum ) ;
break ;
case TMSS_TRIGGERSECTOR : // play the sound from calling sector's soundorg
if ( callsec )
S_StartSound ( & callsec - > soundorg , sfxnum ) ;
else if ( mo )
S_StartSound ( & mo - > subsector - > sector - > soundorg , sfxnum ) ;
break ;
case TMSS_NOWHERE : // play the sound from nowhere
S_StartSound ( NULL , sfxnum ) ;
break ;
case TMSS_TAGGEDSECTOR : // play the sound from tagged sectors' soundorgs
{
INT32 secnum ;
TAG_ITER_SECTORS ( tag , secnum )
S_StartSound ( & sectors [ secnum ] . soundorg , sfxnum ) ;
break ;
}
default :
break ;
}
}
2021-09-13 02:01:30 +00:00
static boolean is_rain_type ( INT32 weathernum )
{
switch ( weathernum )
{
case PRECIP_SNOW :
case PRECIP_RAIN :
case PRECIP_STORM :
case PRECIP_STORM_NOSTRIKES :
case PRECIP_BLANK :
return true ;
default :
return false ;
}
}
2014-03-15 16:59:03 +00:00
//
// P_SwitchWeather
//
// Switches the weather!
//
void P_SwitchWeather ( INT32 weathernum )
{
2021-09-10 17:00:43 +00:00
boolean purge = true ;
2014-03-15 16:59:03 +00:00
2021-09-10 17:00:43 +00:00
if ( weathernum = = curWeather )
return ;
2021-09-13 02:01:30 +00:00
if ( is_rain_type ( weathernum ) & &
is_rain_type ( curWeather ) )
2021-09-10 17:00:43 +00:00
purge = false ;
2014-03-15 16:59:03 +00:00
if ( purge )
{
thinker_t * think ;
precipmobj_t * precipmobj ;
2019-07-12 23:42:03 +00:00
for ( think = thlist [ THINK_PRECIP ] . next ; think ! = & thlist [ THINK_PRECIP ] ; think = think - > next )
2014-03-15 16:59:03 +00:00
{
2018-10-07 14:00:58 +00:00
if ( think - > function . acp1 ! = ( actionf_p1 ) P_NullPrecipThinker )
2014-03-15 16:59:03 +00:00
continue ; // not a precipmobj thinker
precipmobj = ( precipmobj_t * ) think ;
P_RemovePrecipMobj ( precipmobj ) ;
}
}
2021-09-10 17:00:43 +00:00
else // Rather than respawn all that crap, reuse it!
2014-03-15 16:59:03 +00:00
{
thinker_t * think ;
precipmobj_t * precipmobj ;
state_t * st ;
2019-07-12 23:42:03 +00:00
for ( think = thlist [ THINK_PRECIP ] . next ; think ! = & thlist [ THINK_PRECIP ] ; think = think - > next )
2014-03-15 16:59:03 +00:00
{
2018-10-07 14:00:58 +00:00
if ( think - > function . acp1 ! = ( actionf_p1 ) P_NullPrecipThinker )
continue ; // not a precipmobj thinker
precipmobj = ( precipmobj_t * ) think ;
2021-10-27 17:52:29 +00:00
if ( weathernum = = PRECIP_RAIN | | weathernum = = PRECIP_STORM | | weathernum = = PRECIP_STORM_NOSTRIKES ) // Snow To Rain
2014-03-15 16:59:03 +00:00
{
precipmobj - > flags = mobjinfo [ MT_RAIN ] . flags ;
st = & states [ mobjinfo [ MT_RAIN ] . spawnstate ] ;
precipmobj - > state = st ;
precipmobj - > tics = st - > tics ;
precipmobj - > sprite = st - > sprite ;
precipmobj - > frame = st - > frame ;
precipmobj - > momz = mobjinfo [ MT_RAIN ] . speed ;
precipmobj - > precipflags & = ~ PCF_INVISIBLE ;
2018-10-07 14:00:58 +00:00
precipmobj - > precipflags | = PCF_RAIN ;
//think->function.acp1 = (actionf_p1)P_RainThinker;
2014-03-15 16:59:03 +00:00
}
2021-09-10 17:00:43 +00:00
else if ( weathernum = = PRECIP_SNOW ) // Rain To Snow
2014-03-15 16:59:03 +00:00
{
INT32 z ;
precipmobj - > flags = mobjinfo [ MT_SNOWFLAKE ] . flags ;
2016-03-27 14:33:15 +00:00
z = M_RandomByte ( ) ;
2014-03-15 16:59:03 +00:00
if ( z < 64 )
z = 2 ;
else if ( z < 144 )
z = 1 ;
else
z = 0 ;
st = & states [ mobjinfo [ MT_SNOWFLAKE ] . spawnstate + z ] ;
precipmobj - > state = st ;
precipmobj - > tics = st - > tics ;
precipmobj - > sprite = st - > sprite ;
precipmobj - > frame = st - > frame ;
precipmobj - > momz = mobjinfo [ MT_SNOWFLAKE ] . speed ;
2018-10-07 14:00:58 +00:00
precipmobj - > precipflags & = ~ ( PCF_INVISIBLE | PCF_RAIN ) ;
2014-03-15 16:59:03 +00:00
2018-10-07 14:00:58 +00:00
//think->function.acp1 = (actionf_p1)P_SnowThinker;
2014-03-15 16:59:03 +00:00
}
2021-09-10 17:00:43 +00:00
else // Remove precip, but keep it around for reuse.
2014-03-15 16:59:03 +00:00
{
2018-10-07 14:00:58 +00:00
//think->function.acp1 = (actionf_p1)P_NullPrecipThinker;
2014-03-15 16:59:03 +00:00
precipmobj - > precipflags | = PCF_INVISIBLE ;
}
}
}
switch ( weathernum )
{
case PRECIP_SNOW : // snow
curWeather = PRECIP_SNOW ;
2022-03-03 19:24:46 +00:00
2021-09-10 17:00:43 +00:00
if ( purge )
2014-03-15 16:59:03 +00:00
P_SpawnPrecipitation ( ) ;
break ;
case PRECIP_RAIN : // rain
{
curWeather = PRECIP_RAIN ;
2021-09-10 17:00:43 +00:00
if ( purge )
2014-03-15 16:59:03 +00:00
P_SpawnPrecipitation ( ) ;
break ;
}
case PRECIP_STORM : // storm
{
curWeather = PRECIP_STORM ;
2021-09-10 17:00:43 +00:00
if ( purge )
2014-03-15 16:59:03 +00:00
P_SpawnPrecipitation ( ) ;
break ;
}
case PRECIP_STORM_NOSTRIKES : // storm w/o lightning
{
curWeather = PRECIP_STORM_NOSTRIKES ;
2021-09-10 17:00:43 +00:00
if ( purge )
2014-03-15 16:59:03 +00:00
P_SpawnPrecipitation ( ) ;
break ;
}
case PRECIP_STORM_NORAIN : // storm w/o rain
curWeather = PRECIP_STORM_NORAIN ;
break ;
2021-09-10 17:00:43 +00:00
case PRECIP_BLANK : //preloaded
2014-03-15 16:59:03 +00:00
curWeather = PRECIP_BLANK ;
2021-09-10 17:00:43 +00:00
if ( purge )
2014-03-15 16:59:03 +00:00
P_SpawnPrecipitation ( ) ;
break ;
default :
curWeather = PRECIP_NONE ;
break ;
}
}
/** Gets an object.
*
* \ param type Object type to look for .
* \ param s Sector number to look in .
* \ return Pointer to the first : : type found in the sector .
* \ sa P_GetPushThing
*/
static mobj_t * P_GetObjectTypeInSectorNum ( mobjtype_t type , size_t s )
{
sector_t * sec = sectors + s ;
mobj_t * thing = sec - > thinglist ;
while ( thing )
{
if ( thing - > type = = type )
return thing ;
thing = thing - > snext ;
}
return NULL ;
}
2021-09-21 12:59:58 +00:00
static mobj_t * P_FindObjectTypeFromTag ( mobjtype_t type , mtag_t tag )
{
if ( udmf )
{
INT32 mtnum ;
mobj_t * mo ;
TAG_ITER_THINGS ( tag , mtnum )
{
mo = mapthings [ mtnum ] . mobj ;
if ( ! mo )
continue ;
if ( mo - > type ! = type )
continue ;
return mo ;
}
return NULL ;
}
else
{
INT32 secnum ;
if ( ( secnum = Tag_Iterate_Sectors ( tag , 0 ) ) < 0 )
return NULL ;
return P_GetObjectTypeInSectorNum ( type , secnum ) ;
}
}
2014-03-15 16:59:03 +00:00
/** Processes the line special triggered by an object.
*
* \ param line Line with the special command on it .
* \ param mo mobj that triggered the line . Must be valid and non - NULL .
2014-11-12 00:55:07 +00:00
* \ param callsec sector in which action was initiated ; this can be NULL .
* Because of the A_LinedefExecute ( ) action , even if non - NULL ,
* this sector might not have the same tag as the linedef executor ,
* and it might not have the linedef executor sector type .
2014-03-15 16:59:03 +00:00
* \ todo Handle mo being NULL gracefully . T_MoveFloor ( ) and T_MoveCeiling ( )
* don ' t have an object to pass .
* \ todo Split up into multiple functions .
* \ sa P_LinedefExecute
* \ author Graue < graue @ oceanbase . org >
*/
2014-11-12 00:55:07 +00:00
static void P_ProcessLineSpecial ( line_t * line , mobj_t * mo , sector_t * callsec )
2014-03-15 16:59:03 +00:00
{
INT32 secnum = - 1 ;
mobj_t * bot = NULL ;
I_Assert ( ! mo | | ! P_MobjWasRemoved ( mo ) ) ; // If mo is there, mo must be valid!
if ( mo & & mo - > player & & botingame )
bot = players [ secondarydisplayplayer ] . mo ;
// note: only commands with linedef types >= 400 && < 500 can be used
switch ( line - > special )
{
2021-06-25 16:34:56 +00:00
case 400 : // Set tagged sector's heights/flats
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_CEILING )
2021-06-25 16:34:56 +00:00
EV_DoFloor ( line - > args [ 0 ] , line , instantMoveFloorByFrontSector ) ;
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_FLOOR )
2021-06-25 16:34:56 +00:00
EV_DoCeiling ( line - > args [ 0 ] , line , instantMoveCeilingByFrontSector ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-09-19 12:23:04 +00:00
case 402 : // Copy light level to tagged sectors
2014-03-15 16:59:03 +00:00
{
INT16 newlightlevel ;
2021-09-19 06:58:21 +00:00
INT16 newfloorlightlevel , newceilinglightlevel ;
boolean newfloorlightabsolute , newceilinglightabsolute ;
2014-03-15 16:59:03 +00:00
INT32 newfloorlightsec , newceilinglightsec ;
newlightlevel = line - > frontsector - > lightlevel ;
2021-09-19 06:58:21 +00:00
newfloorlightlevel = line - > frontsector - > floorlightlevel ;
newfloorlightabsolute = line - > frontsector - > floorlightabsolute ;
newceilinglightlevel = line - > frontsector - > ceilinglightlevel ;
newceilinglightabsolute = line - > frontsector - > ceilinglightabsolute ;
2014-03-15 16:59:03 +00:00
newfloorlightsec = line - > frontsector - > floorlightsec ;
newceilinglightsec = line - > frontsector - > ceilinglightsec ;
2021-09-19 12:23:04 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2014-03-15 16:59:03 +00:00
{
if ( sectors [ secnum ] . lightingdata )
{
// Stop the lighting madness going on in this sector!
2021-09-20 06:18:59 +00:00
P_RemoveThinker ( & ( ( thinkerdata_t * ) sectors [ secnum ] . lightingdata ) - > thinker ) ;
2014-03-15 16:59:03 +00:00
sectors [ secnum ] . lightingdata = NULL ;
}
2021-09-19 12:23:04 +00:00
if ( ! ( line - > args [ 1 ] & TMLC_NOSECTOR ) )
sectors [ secnum ] . lightlevel = newlightlevel ;
if ( ! ( line - > args [ 1 ] & TMLC_NOFLOOR ) )
{
sectors [ secnum ] . floorlightlevel = newfloorlightlevel ;
sectors [ secnum ] . floorlightabsolute = newfloorlightabsolute ;
sectors [ secnum ] . floorlightsec = newfloorlightsec ;
}
if ( ! ( line - > args [ 1 ] & TMLC_NOCEILING ) )
{
sectors [ secnum ] . ceilinglightlevel = newceilinglightlevel ;
sectors [ secnum ] . ceilinglightabsolute = newceilinglightabsolute ;
sectors [ secnum ] . ceilinglightsec = newceilinglightsec ;
}
2014-03-15 16:59:03 +00:00
}
}
break ;
2021-06-26 06:31:59 +00:00
case 403 : // Move planes by front sector
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_CEILING )
2021-06-26 06:31:59 +00:00
EV_DoFloor ( line - > args [ 0 ] , line , moveFloorByFrontSector ) ;
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_FLOOR )
2021-06-26 06:31:59 +00:00
EV_DoCeiling ( line - > args [ 0 ] , line , moveCeilingByFrontSector ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-06-26 06:43:35 +00:00
case 405 : // Move planes by distance
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_CEILING )
2021-06-26 06:43:35 +00:00
EV_DoFloor ( line - > args [ 0 ] , line , moveFloorByDistance ) ;
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_FLOOR )
2021-06-26 06:43:35 +00:00
EV_DoCeiling ( line - > args [ 0 ] , line , moveCeilingByDistance ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-06-27 09:56:06 +00:00
case 408 : // Set flats
{
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
{
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_CEILING )
2021-06-27 09:56:06 +00:00
sectors [ secnum ] . floorpic = line - > frontsector - > floorpic ;
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] ! = TMP_FLOOR )
2021-06-27 09:56:06 +00:00
sectors [ secnum ] . ceilingpic = line - > frontsector - > ceilingpic ;
}
break ;
}
2014-03-15 16:59:03 +00:00
case 409 : // Change tagged sectors' tag
2014-08-04 03:49:33 +00:00
// (formerly "Change calling sectors' tag", but behavior was changed)
2014-03-15 16:59:03 +00:00
{
2021-09-21 09:30:34 +00:00
mtag_t newtag = line - > args [ 1 ] ;
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
{
switch ( line - > args [ 2 ] )
{
case TMT_ADD :
Tag_SectorAdd ( secnum , newtag ) ;
break ;
case TMT_REMOVE :
Tag_SectorRemove ( secnum , newtag ) ;
break ;
case TMT_REPLACEFIRST :
default :
Tag_SectorFSet ( secnum , newtag ) ;
break ;
2022-05-29 10:24:14 +00:00
case TMT_TRIGGERTAG :
sectors [ secnum ] . triggertag = newtag ;
break ;
2021-09-21 09:30:34 +00:00
}
}
2014-03-15 16:59:03 +00:00
break ;
}
case 410 : // Change front sector's tag
2021-09-21 09:30:34 +00:00
{
mtag_t newtag = line - > args [ 1 ] ;
secnum = ( UINT32 ) ( line - > frontsector - sectors ) ;
switch ( line - > args [ 2 ] )
{
case TMT_ADD :
Tag_SectorAdd ( secnum , newtag ) ;
break ;
case TMT_REMOVE :
Tag_SectorRemove ( secnum , newtag ) ;
break ;
case TMT_REPLACEFIRST :
default :
Tag_SectorFSet ( secnum , newtag ) ;
break ;
2022-05-29 10:24:14 +00:00
case TMT_TRIGGERTAG :
sectors [ secnum ] . triggertag = newtag ;
break ;
2021-09-21 09:30:34 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
2021-09-21 09:30:34 +00:00
}
2014-03-15 16:59:03 +00:00
case 411 : // Stop floor/ceiling movement in tagged sector(s)
2021-06-27 07:53:57 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2014-03-15 16:59:03 +00:00
{
if ( sectors [ secnum ] . floordata )
{
if ( sectors [ secnum ] . floordata = = sectors [ secnum ] . ceilingdata ) // elevator
{
P_RemoveThinker ( & ( ( elevator_t * ) sectors [ secnum ] . floordata ) - > thinker ) ;
sectors [ secnum ] . floordata = sectors [ secnum ] . ceilingdata = NULL ;
sectors [ secnum ] . floorspeed = sectors [ secnum ] . ceilspeed = 0 ;
}
else // floormove
{
P_RemoveThinker ( & ( ( floormove_t * ) sectors [ secnum ] . floordata ) - > thinker ) ;
sectors [ secnum ] . floordata = NULL ;
sectors [ secnum ] . floorspeed = 0 ;
}
}
if ( sectors [ secnum ] . ceilingdata ) // ceiling
{
P_RemoveThinker ( & ( ( ceiling_t * ) sectors [ secnum ] . ceilingdata ) - > thinker ) ;
sectors [ secnum ] . ceilingdata = NULL ;
sectors [ secnum ] . ceilspeed = 0 ;
}
}
break ;
case 412 : // Teleport the player or thing
{
mobj_t * dest ;
if ( ! mo ) // nothing to teleport
return ;
2021-09-21 13:31:53 +00:00
if ( line - > args [ 1 ] & TMT_RELATIVE ) // Relative silent teleport
2014-03-15 16:59:03 +00:00
{
fixed_t x , y , z ;
2021-09-21 13:31:53 +00:00
x = line - > args [ 2 ] < < FRACBITS ;
y = line - > args [ 3 ] < < FRACBITS ;
z = line - > args [ 4 ] < < FRACBITS ;
2014-03-15 16:59:03 +00:00
2024-07-01 10:46:51 +00:00
P_SetOrigin ( mo , mo - > x + x , mo - > y + y , mo - > z + z ) ;
2014-03-15 16:59:03 +00:00
if ( mo - > player )
{
if ( bot ) // This might put poor Tails in a wall if he's too far behind! D: But okay, whatever! >:3
2022-04-25 19:36:54 +00:00
P_SetOrigin ( bot , bot - > x + x , bot - > y + y , bot - > z + z ) ;
2014-03-15 16:59:03 +00:00
if ( splitscreen & & mo - > player = = & players [ secondarydisplayplayer ] & & camera2 . chase )
{
2024-07-01 10:46:51 +00:00
camera2 . reset = true ;
2014-03-15 16:59:03 +00:00
camera2 . x + = x ;
camera2 . y + = y ;
camera2 . z + = z ;
camera2 . subsector = R_PointInSubsector ( camera2 . x , camera2 . y ) ;
}
else if ( camera . chase & & mo - > player = = & players [ displayplayer ] )
{
2024-07-01 10:46:51 +00:00
camera . reset = true ;
2014-03-15 16:59:03 +00:00
camera . x + = x ;
camera . y + = y ;
camera . z + = z ;
camera . subsector = R_PointInSubsector ( camera . x , camera . y ) ;
}
}
}
else
{
2021-09-21 13:31:53 +00:00
angle_t angle ;
boolean silent , keepmomentum ;
2014-03-15 16:59:03 +00:00
2021-09-21 13:31:53 +00:00
dest = P_FindObjectTypeFromTag ( MT_TELEPORTMAN , line - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
if ( ! dest )
return ;
2021-09-21 13:31:53 +00:00
angle = ( line - > args [ 1 ] & TMT_KEEPANGLE ) ? mo - > angle : dest - > angle ;
silent = ! ! ( line - > args [ 1 ] & TMT_SILENT ) ;
keepmomentum = ! ! ( line - > args [ 1 ] & TMT_KEEPMOMENTUM ) ;
2014-03-15 16:59:03 +00:00
if ( bot )
2021-09-21 13:31:53 +00:00
P_Teleport ( bot , dest - > x , dest - > y , dest - > z , angle , ! silent , keepmomentum ) ;
P_Teleport ( mo , dest - > x , dest - > y , dest - > z , angle , ! silent , keepmomentum ) ;
if ( ! silent )
S_StartSound ( dest , sfx_mixup ) ; // Play the 'bowrwoosh!' sound
2014-03-15 16:59:03 +00:00
}
}
break ;
case 413 : // Change music
2021-09-22 06:57:48 +00:00
// console player only unless TMM_ALLPLAYERS is set
if ( ( line - > args [ 0 ] & TMM_ALLPLAYERS ) | | ( mo & & mo - > player & & P_IsLocalPlayer ( mo - > player ) ) | | titlemapinaction )
2014-03-15 16:59:03 +00:00
{
2021-09-22 06:57:48 +00:00
boolean musicsame = ( ! line - > stringargs [ 0 ] | | ! line - > stringargs [ 0 ] [ 0 ] | | ! strnicmp ( line - > stringargs [ 0 ] , S_MusicName ( ) , 7 ) ) ;
UINT16 tracknum = ( UINT16 ) max ( line - > args [ 6 ] , 0 ) ;
INT32 position = ( INT32 ) max ( line - > args [ 1 ] , 0 ) ;
UINT32 prefadems = ( UINT32 ) max ( line - > args [ 2 ] , 0 ) ;
UINT32 postfadems = ( UINT32 ) max ( line - > args [ 3 ] , 0 ) ;
UINT8 fadetarget = ( UINT8 ) max ( line - > args [ 4 ] , 0 ) ;
INT16 fadesource = ( INT16 ) max ( line - > args [ 5 ] , - 1 ) ;
2018-09-18 19:28:57 +00:00
2019-03-13 23:44:24 +00:00
// Seek offset from current song position
2021-09-22 06:57:48 +00:00
if ( line - > args [ 0 ] & TMM_OFFSET )
2018-09-18 19:28:57 +00:00
{
// adjust for loop point if subtracting
if ( position < 0 & & S_GetMusicLength ( ) & &
S_GetMusicPosition ( ) > S_GetMusicLoopPoint ( ) & &
S_GetMusicPosition ( ) + position < S_GetMusicLoopPoint ( ) )
position = max ( S_GetMusicLength ( ) - ( S_GetMusicLoopPoint ( ) - ( S_GetMusicPosition ( ) + position ) ) , 0 ) ;
else
position = max ( S_GetMusicPosition ( ) + position , 0 ) ;
}
2014-03-15 16:59:03 +00:00
2019-03-13 23:44:24 +00:00
// Fade current music to target volume (if music won't be changed)
2021-09-22 06:57:48 +00:00
if ( ( line - > args [ 0 ] & TMM_FADE ) & & fadetarget & & musicsame )
2018-09-18 19:28:57 +00:00
{
2019-03-13 23:44:24 +00:00
// 0 fadesource means fade from current volume.
// meaning that we can't specify volume 0 as the source volume -- this starts at 1.
if ( ! fadesource )
fadesource = - 1 ;
2016-01-08 03:48:20 +00:00
2018-09-18 19:28:57 +00:00
if ( ! postfadems )
S_SetInternalMusicVolume ( fadetarget ) ;
else
S_FadeMusicFromVolume ( fadetarget , fadesource , postfadems ) ;
2014-03-15 16:59:03 +00:00
2018-09-18 19:28:57 +00:00
if ( position )
S_SetMusicPosition ( position ) ;
}
2019-03-13 23:44:24 +00:00
// Change the music and apply position/fade operations
2018-09-18 19:28:57 +00:00
else
{
2023-08-02 15:05:38 +00:00
if ( ! line - > stringargs [ 0 ] | | ! strcmp ( line - > stringargs [ 0 ] , " - " ) )
strcpy ( mapmusname , " " ) ;
else
{
strncpy ( mapmusname , line - > stringargs [ 0 ] , 7 ) ;
mapmusname [ 6 ] = 0 ;
}
2018-09-18 19:28:57 +00:00
mapmusflags = tracknum & MUSIC_TRACKMASK ;
2021-09-22 06:57:48 +00:00
if ( ! ( line - > args [ 0 ] & TMM_NORELOAD ) )
2018-09-18 19:28:57 +00:00
mapmusflags | = MUSIC_RELOADRESET ;
2021-09-22 06:57:48 +00:00
if ( line - > args [ 0 ] & TMM_FORCERESET )
2018-09-18 19:28:57 +00:00
mapmusflags | = MUSIC_FORCERESET ;
mapmusposition = position ;
2014-03-15 16:59:03 +00:00
2021-09-22 06:57:48 +00:00
S_ChangeMusicEx ( mapmusname , mapmusflags , ! ( line - > args [ 0 ] & TMM_NOLOOP ) , position ,
! ( line - > args [ 0 ] & TMM_FADE ) ? prefadems : 0 ,
! ( line - > args [ 0 ] & TMM_FADE ) ? postfadems : 0 ) ;
2018-09-18 14:22:17 +00:00
2021-09-22 06:57:48 +00:00
if ( ( line - > args [ 0 ] & TMM_FADE ) & & fadetarget )
2018-09-18 19:28:57 +00:00
{
if ( ! postfadems )
S_SetInternalMusicVolume ( fadetarget ) ;
else
S_FadeMusicFromVolume ( fadetarget , fadesource , postfadems ) ;
}
}
2014-03-15 16:59:03 +00:00
2021-09-22 06:57:48 +00:00
// Except, you can use the TMM_NORELOAD flag to change this behavior.
2016-01-08 03:48:20 +00:00
// if (mapmusflags & MUSIC_RELOADRESET) then it will reset the music in G_PlayerReborn.
2014-03-15 16:59:03 +00:00
}
break ;
case 414 : // Play SFX
2021-09-21 08:14:55 +00:00
P_PlaySFX ( line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : sfx_None , mo , callsec , line - > args [ 2 ] , line - > args [ 0 ] , line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 415 : // Run a script
if ( cv_runscripts . value )
{
2021-09-20 10:32:01 +00:00
lumpnum_t lumpnum = W_CheckNumForName ( line - > stringargs [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
if ( lumpnum = = LUMPERROR | | W_LumpLength ( lumpnum ) = = 0 )
2021-09-20 10:32:01 +00:00
CONS_Debug ( DBG_SETUP , " Line type 415 Executor: script lump %s not found/not valid. \n " , line - > stringargs [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
else
2023-06-02 20:19:30 +00:00
{
void * lump = W_CacheLumpNum ( lumpnum , PU_CACHE ) ;
size_t len = W_LumpLength ( lumpnum ) ;
char * text = Z_Malloc ( len + 1 , PU_CACHE , NULL ) ;
memcpy ( text , lump , len ) ;
text [ len ] = ' \0 ' ;
COM_BufInsertText ( text ) ;
Z_Free ( text ) ;
}
2014-03-15 16:59:03 +00:00
}
break ;
case 416 : // Spawn adjustable fire flicker
2021-09-19 15:07:08 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
P_SpawnAdjustableFireFlicker ( & sectors [ secnum ] , line - > args [ 2 ] ,
line - > args [ 3 ] ? sectors [ secnum ] . lightlevel : line - > args [ 4 ] , line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 417 : // Spawn adjustable glowing light
2021-09-19 15:07:08 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
P_SpawnAdjustableGlowingLight ( & sectors [ secnum ] , line - > args [ 2 ] ,
line - > args [ 3 ] ? sectors [ secnum ] . lightlevel : line - > args [ 4 ] , line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-09-19 15:07:08 +00:00
case 418 : // Spawn adjustable strobe flash
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
P_SpawnAdjustableStrobeFlash ( & sectors [ secnum ] , line - > args [ 3 ] ,
( line - > args [ 4 ] & TMB_USETARGET ) ? sectors [ secnum ] . lightlevel : line - > args [ 5 ] ,
line - > args [ 1 ] , line - > args [ 2 ] , line - > args [ 4 ] & TMB_SYNC ) ;
2014-03-15 16:59:03 +00:00
break ;
case 420 : // Fade light levels in tagged sectors to new value
2021-09-19 21:12:30 +00:00
P_FadeLight ( line - > args [ 0 ] , line - > args [ 1 ] , line - > args [ 2 ] , line - > args [ 3 ] & TMF_TICBASED , line - > args [ 3 ] & TMF_OVERRIDE , line - > args [ 3 ] & TMF_RELATIVE ) ;
2014-03-15 16:59:03 +00:00
break ;
case 421 : // Stop lighting effect in tagged sectors
2021-09-19 12:23:04 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2014-03-15 16:59:03 +00:00
if ( sectors [ secnum ] . lightingdata )
{
2021-09-20 06:18:59 +00:00
P_RemoveThinker ( & ( ( thinkerdata_t * ) sectors [ secnum ] . lightingdata ) - > thinker ) ;
2014-03-15 16:59:03 +00:00
sectors [ secnum ] . lightingdata = NULL ;
}
break ;
case 422 : // Cut away to another view
{
mobj_t * altview ;
2021-09-21 12:59:58 +00:00
INT32 aim ;
2014-03-15 16:59:03 +00:00
2018-11-18 06:53:18 +00:00
if ( ( ! mo | | ! mo - > player ) & & ! titlemapinaction ) // only players have views, and title screens
2014-03-15 16:59:03 +00:00
return ;
2021-09-21 12:59:58 +00:00
altview = P_FindObjectTypeFromTag ( MT_ALTVIEWMAN , line - > args [ 0 ] ) ;
if ( ! altview | | ! altview - > spawnpoint )
2014-03-15 16:59:03 +00:00
return ;
2018-11-18 06:53:18 +00:00
// If titlemap, set the camera ref for title's thinker
// This is not revoked until overwritten; awayviewtics is ignored
if ( titlemapinaction )
titlemapcameraref = altview ;
2023-07-23 01:44:22 +00:00
else if ( ! mo - > player - > awayviewtics | | mo - > player - > awayviewmobj ! = altview ) {
2020-06-21 08:06:42 +00:00
P_SetTarget ( & mo - > player - > awayviewmobj , altview ) ;
2023-10-28 13:19:35 +00:00
2023-06-03 20:43:58 +00:00
if ( mo - > player = = & players [ displayplayer ] )
P_ResetCamera ( mo - > player , & camera ) ; // reset p1 camera on p1 getting an awayviewmobj
else if ( splitscreen & & mo - > player = = & players [ secondarydisplayplayer ] )
P_ResetCamera ( mo - > player , & camera2 ) ; // reset p2 camera on p2 getting an awayviewmobj
2018-11-18 06:53:18 +00:00
}
2021-09-21 12:59:58 +00:00
aim = udmf ? altview - > spawnpoint - > pitch : line - > args [ 2 ] ;
aim = ( aim + 360 ) % 360 ;
aim * = ( ANGLE_90 > > 8 ) ;
aim / = 90 ;
aim < < = 8 ;
if ( titlemapinaction )
titlemapcameraref - > cusval = ( angle_t ) aim ;
2023-07-23 01:44:22 +00:00
else {
2021-09-21 12:59:58 +00:00
mo - > player - > awayviewaiming = ( angle_t ) aim ;
2023-07-23 01:44:22 +00:00
mo - > player - > awayviewtics = line - > args [ 1 ] ;
}
2014-03-15 16:59:03 +00:00
}
break ;
case 423 : // Change Sky
2021-09-20 10:42:31 +00:00
if ( ( mo & & mo - > player & & P_IsLocalPlayer ( mo - > player ) ) | | line - > args [ 1 ] )
P_SetupLevelSky ( line - > args [ 0 ] , line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 424 : // Change Weather
2021-09-20 10:42:31 +00:00
if ( line - > args [ 1 ] )
2014-03-15 16:59:03 +00:00
{
2021-09-20 10:42:31 +00:00
globalweather = ( UINT8 ) ( line - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
P_SwitchWeather ( globalweather ) ;
}
else if ( mo & & mo - > player & & P_IsLocalPlayer ( mo - > player ) )
2021-09-20 10:42:31 +00:00
P_SwitchWeather ( line - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 425 : // Calls P_SetMobjState on calling mobj
if ( mo & & ! mo - > player )
2021-09-21 05:36:54 +00:00
{
statenum_t state = line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : S_NULL ;
if ( state > = 0 & & state < NUMSTATES )
P_SetMobjState ( mo , state ) ;
}
2014-03-15 16:59:03 +00:00
break ;
case 426 : // Moves the mobj to its sector's soundorg and on the floor, and stops it
if ( ! mo )
return ;
2021-09-20 11:31:32 +00:00
if ( line - > args [ 0 ] )
2014-03-15 16:59:03 +00:00
{
P_UnsetThingPosition ( mo ) ;
mo - > x = mo - > subsector - > sector - > soundorg . x ;
mo - > y = mo - > subsector - > sector - > soundorg . y ;
mo - > z = mo - > floorz ;
P_SetThingPosition ( mo ) ;
}
mo - > momx = mo - > momy = mo - > momz = 1 ;
mo - > pmomz = 0 ;
if ( mo - > player )
{
mo - > player - > rmomx = mo - > player - > rmomy = 1 ;
mo - > player - > cmomx = mo - > player - > cmomy = 0 ;
P_ResetPlayer ( mo - > player ) ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( mo , S_PLAY_STND ) ;
2014-03-15 16:59:03 +00:00
// Reset bot too.
if ( bot ) {
2021-12-29 22:12:46 +00:00
if ( line - > args [ 0 ] )
2022-04-25 19:36:54 +00:00
P_SetOrigin ( bot , mo - > x , mo - > y , mo - > z ) ;
2014-03-15 16:59:03 +00:00
bot - > momx = bot - > momy = bot - > momz = 1 ;
bot - > pmomz = 0 ;
bot - > player - > rmomx = bot - > player - > rmomy = 1 ;
bot - > player - > cmomx = bot - > player - > cmomy = 0 ;
P_ResetPlayer ( bot - > player ) ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( bot , S_PLAY_STND ) ;
2014-03-15 16:59:03 +00:00
}
}
break ;
case 427 : // Awards points if the mobj is a player
if ( mo & & mo - > player )
2021-09-20 11:31:32 +00:00
P_AddPlayerScore ( mo - > player , line - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 428 : // Start floating platform movement
2021-06-25 09:11:16 +00:00
EV_DoElevator ( line - > args [ 0 ] , line , elevateContinuous ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-06-26 10:53:14 +00:00
case 429 : // Crush planes once
2021-06-27 10:36:46 +00:00
if ( line - > args [ 1 ] = = TMP_FLOOR )
2021-06-26 10:53:14 +00:00
EV_DoFloor ( line - > args [ 0 ] , line , crushFloorOnce ) ;
2021-06-27 10:36:46 +00:00
else if ( line - > args [ 1 ] = = TMP_CEILING )
2021-06-26 10:53:14 +00:00
EV_DoCrush ( line - > args [ 0 ] , line , crushCeilOnce ) ;
else
EV_DoCrush ( line - > args [ 0 ] , line , crushBothOnce ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-09-20 11:31:32 +00:00
case 432 : // Enable/Disable 2D Mode
2019-08-20 19:31:28 +00:00
if ( mo & & mo - > player )
2014-03-15 16:59:03 +00:00
{
2021-09-20 11:31:32 +00:00
if ( line - > args [ 0 ] )
2014-03-15 16:59:03 +00:00
mo - > flags2 & = ~ MF2_TWOD ;
else
mo - > flags2 | = MF2_TWOD ;
// Copy effect to bot if necessary
// (Teleport them to you so they don't break it.)
if ( bot & & ( bot - > flags2 & MF2_TWOD ) ! = ( mo - > flags2 & MF2_TWOD ) ) {
bot - > flags2 = ( bot - > flags2 & ~ MF2_TWOD ) | ( mo - > flags2 & MF2_TWOD ) ;
2022-04-25 19:36:54 +00:00
P_SetOrigin ( bot , mo - > x , mo - > y , mo - > z ) ;
2014-03-15 16:59:03 +00:00
}
}
break ;
2021-09-20 11:31:32 +00:00
case 433 : // Flip/flop gravity. Works on pushables, too!
2022-09-07 21:11:02 +00:00
if ( line - > args [ 1 ] )
mo - > flags2 ^ = MF2_OBJECTFLIP ;
else if ( line - > args [ 0 ] )
2014-03-15 16:59:03 +00:00
mo - > flags2 & = ~ MF2_OBJECTFLIP ;
else
mo - > flags2 | = MF2_OBJECTFLIP ;
2023-03-20 14:46:15 +00:00
if ( line - > args [ 2 ] )
mo - > eflags | = MFE_VERTICALFLIP ;
2014-03-15 16:59:03 +00:00
if ( bot )
bot - > flags2 = ( bot - > flags2 & ~ MF2_OBJECTFLIP ) | ( mo - > flags2 & MF2_OBJECTFLIP ) ;
break ;
case 434 : // Custom Power
2019-08-20 19:31:28 +00:00
if ( mo & & mo - > player )
2014-03-15 16:59:03 +00:00
{
2022-04-20 18:41:46 +00:00
powertype_t power = line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : 0 ;
INT32 value = line - > stringargs [ 1 ] ? get_number ( line - > stringargs [ 1 ] ) : 0 ;
if ( value = = - 1 ) // 'Infinite'
2022-02-01 11:51:01 +00:00
value = UINT16_MAX ;
2014-03-15 16:59:03 +00:00
2022-02-01 11:51:01 +00:00
P_SetPower ( mo - > player , power , value ) ;
2014-03-15 16:59:03 +00:00
2022-02-01 11:51:01 +00:00
if ( bot )
P_SetPower ( bot - > player , power , value ) ;
2014-03-15 16:59:03 +00:00
}
break ;
case 435 : // Change scroller direction
{
scroll_t * scroller ;
thinker_t * th ;
2021-07-02 10:51:39 +00:00
fixed_t length = R_PointToDist2 ( line - > v2 - > x , line - > v2 - > y , line - > v1 - > x , line - > v1 - > y ) ;
fixed_t speed = line - > args [ 1 ] < < FRACBITS ;
fixed_t dx = FixedMul ( FixedMul ( FixedDiv ( line - > dx , length ) , speed ) > > SCROLL_SHIFT , CARRYFACTOR ) ;
fixed_t dy = FixedMul ( FixedMul ( FixedDiv ( line - > dy , length ) , speed ) > > SCROLL_SHIFT , CARRYFACTOR ) ;
2019-04-20 20:37:19 +00:00
for ( th = thlist [ THINK_MAIN ] . next ; th ! = & thlist [ THINK_MAIN ] ; th = th - > next )
2014-03-15 16:59:03 +00:00
{
if ( th - > function . acp1 ! = ( actionf_p1 ) T_Scroll )
continue ;
scroller = ( scroll_t * ) th ;
2021-07-02 10:51:39 +00:00
if ( ! Tag_Find ( & sectors [ scroller - > affectee ] . tags , line - > args [ 0 ] ) )
2014-03-15 16:59:03 +00:00
continue ;
2021-07-02 10:51:39 +00:00
scroller - > dx = dx ;
scroller - > dy = dy ;
2014-03-15 16:59:03 +00:00
}
}
break ;
case 436 : // Shatter block remotely
{
2021-09-22 05:32:39 +00:00
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
sector_t * sec ; // Sector that the FOF is visible in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF that we are going to crumble
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2014-03-15 16:59:03 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2014-03-15 16:59:03 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 436 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
EV_CrumbleChain ( sec , rover ) ;
}
2014-03-15 16:59:03 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2014-03-15 16:59:03 +00:00
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 436 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
return ;
}
}
}
break ;
case 437 : // Disable Player Controls
2019-08-20 19:31:28 +00:00
if ( mo & & mo - > player )
2014-03-15 16:59:03 +00:00
{
2021-09-20 11:31:32 +00:00
UINT16 fractime = ( UINT16 ) ( line - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
if ( fractime < 1 )
fractime = 1 ; //instantly wears off upon leaving
2021-09-20 11:31:32 +00:00
if ( line - > args [ 1 ] )
2014-03-15 16:59:03 +00:00
fractime | = 1 < < 15 ; //more crazy &ing, as if music stuff wasn't enough
mo - > player - > powers [ pw_nocontrol ] = fractime ;
if ( bot )
bot - > player - > powers [ pw_nocontrol ] = fractime ;
}
break ;
case 438 : // Set player scale
if ( mo )
{
2021-09-20 12:50:18 +00:00
mo - > destscale = FixedDiv ( line - > args [ 0 ] < < FRACBITS , 100 < < FRACBITS ) ;
2014-03-15 16:59:03 +00:00
if ( mo - > destscale < FRACUNIT / 100 )
mo - > destscale = FRACUNIT / 100 ;
if ( mo - > player & & bot )
bot - > destscale = mo - > destscale ;
}
break ;
case 439 : // Set texture
{
size_t linenum ;
2022-10-10 05:39:34 +00:00
side_t * setfront = & sides [ line - > sidenum [ 0 ] ] ;
2023-09-21 05:06:06 +00:00
side_t * setback = ( line - > args [ 3 ] & & line - > sidenum [ 1 ] ! = NO_SIDEDEF ) ? & sides [ line - > sidenum [ 1 ] ] : setfront ;
2022-10-10 05:39:34 +00:00
side_t * this ;
2021-09-20 12:42:21 +00:00
boolean always = ! ( line - > args [ 2 ] ) ; // If args[2] is set: Only change mid texture if mid texture already exists on tagged lines, etc.
2020-04-17 16:15:25 +00:00
2014-03-15 16:59:03 +00:00
for ( linenum = 0 ; linenum < numlines ; linenum + + )
{
if ( lines [ linenum ] . special = = 439 )
continue ; // Don't override other set texture lines!
2021-09-20 12:42:21 +00:00
if ( ! Tag_Find ( & lines [ linenum ] . tags , line - > args [ 0 ] ) )
2014-03-15 16:59:03 +00:00
continue ; // Find tagged lines
// Front side
2021-09-20 12:42:21 +00:00
if ( line - > args [ 1 ] ! = TMSD_BACK )
{
this = & sides [ lines [ linenum ] . sidenum [ 0 ] ] ;
2022-10-10 05:39:34 +00:00
if ( always | | this - > toptexture ) this - > toptexture = setfront - > toptexture ;
if ( always | | this - > midtexture ) this - > midtexture = setfront - > midtexture ;
if ( always | | this - > bottomtexture ) this - > bottomtexture = setfront - > bottomtexture ;
2021-09-20 12:42:21 +00:00
}
2014-03-15 16:59:03 +00:00
// Back side
2023-09-21 05:06:06 +00:00
if ( line - > args [ 1 ] ! = TMSD_FRONT & & lines [ linenum ] . sidenum [ 1 ] ! = NO_SIDEDEF )
2021-09-20 12:42:21 +00:00
{
this = & sides [ lines [ linenum ] . sidenum [ 1 ] ] ;
2022-10-10 05:39:34 +00:00
if ( always | | this - > toptexture ) this - > toptexture = setback - > toptexture ;
if ( always | | this - > midtexture ) this - > midtexture = setback - > midtexture ;
if ( always | | this - > bottomtexture ) this - > bottomtexture = setback - > bottomtexture ;
2021-09-20 12:42:21 +00:00
}
2014-03-15 16:59:03 +00:00
}
}
break ;
case 440 : // Play race countdown and start Metal Sonic
if ( ! metalrecording & & ! metalplayback )
G_DoPlayMetal ( ) ;
break ;
case 441 : // Trigger unlockable
{
2021-09-20 12:25:07 +00:00
INT32 trigid = line - > args [ 0 ] ;
2014-03-15 16:59:03 +00:00
if ( trigid < 0 | | trigid > 31 ) // limited by 32 bit variable
2023-10-25 16:16:10 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Unlockable trigger (sidedef %u): bad trigger ID %d \n " , line - > sidenum [ 0 ] , trigid ) ;
2014-03-15 16:59:03 +00:00
else
{
unlocktriggers | = 1 < < trigid ;
// Unlocked something?
2022-02-27 21:52:05 +00:00
M_SilentUpdateUnlockablesAndEmblems ( serverGamedata ) ;
if ( M_UpdateUnlockablesAndExtraEmblems ( clientGamedata ) )
2014-03-15 16:59:03 +00:00
{
2018-06-03 21:41:54 +00:00
S_StartSound ( NULL , sfx_s3k68 ) ;
2022-02-27 21:52:05 +00:00
G_SaveGameData ( clientGamedata ) ; // only save if unlocked something
2014-03-15 16:59:03 +00:00
}
}
}
// Execute one time only
line - > special = 0 ;
break ;
case 442 : // Calls P_SetMobjState on mobjs of a given type in the tagged sectors
{
2021-09-21 05:36:54 +00:00
const mobjtype_t type = line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : MT_NULL ;
2014-03-15 16:59:03 +00:00
statenum_t state = NUMSTATES ;
mobj_t * thing ;
2021-09-21 05:36:54 +00:00
if ( type < 0 | | type > = NUMMOBJTYPES )
break ;
2014-03-15 16:59:03 +00:00
2021-09-21 05:36:54 +00:00
if ( ! line - > args [ 1 ] )
{
state = line - > stringargs [ 1 ] ? get_number ( line - > stringargs [ 1 ] ) : S_NULL ;
if ( state < 0 | | state > = NUMSTATES )
break ;
}
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2014-03-15 16:59:03 +00:00
{
2014-03-18 17:56:54 +00:00
boolean tryagain ;
do {
tryagain = false ;
2021-09-21 05:36:54 +00:00
for ( thing = sectors [ secnum ] . thinglist ; thing ; thing = thing - > snext )
{
if ( thing - > type ! = type )
continue ;
if ( ! P_SetMobjState ( thing , line - > args [ 1 ] ? thing - > state - > nextstate : state ) )
{ // mobj was removed
tryagain = true ; // snext is corrupt, we'll have to start over.
break ;
2014-03-18 17:56:54 +00:00
}
2021-09-21 05:36:54 +00:00
}
2014-03-18 17:56:54 +00:00
} while ( tryagain ) ;
2014-03-15 16:59:03 +00:00
}
break ;
}
case 443 : // Calls a named Lua function
2020-01-11 14:38:50 +00:00
if ( line - > stringargs [ 0 ] )
2020-11-17 12:14:45 +00:00
LUA_HookLinedefExecute ( line , mo , callsec ) ;
2017-04-01 19:16:48 +00:00
else
2021-12-29 20:03:47 +00:00
CONS_Alert ( CONS_WARNING , " Linedef %s is missing the hook name of the Lua function to call! (This should be given in stringarg0) \n " , sizeu1 ( line - lines ) ) ;
2014-03-15 16:59:03 +00:00
break ;
case 444 : // Earthquake camera
{
2021-09-20 12:25:07 +00:00
quake . intensity = line - > args [ 1 ] < < FRACBITS ;
quake . radius = line - > args [ 2 ] < < FRACBITS ;
quake . time = line - > args [ 0 ] ;
2014-03-15 16:59:03 +00:00
quake . epicenter = NULL ; /// \todo
// reasonable defaults.
if ( ! quake . intensity )
quake . intensity = 8 < < FRACBITS ;
if ( ! quake . radius )
quake . radius = 512 < < FRACBITS ;
break ;
}
2021-09-22 05:32:39 +00:00
case 445 : // Force block disappear remotely (reappear if args[2] is set)
2014-03-15 16:59:03 +00:00
{
2021-09-22 05:32:39 +00:00
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
sector_t * sec ; // Sector that the FOF is visible (or not visible) in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF to vanish/un-vanish
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2016-02-26 18:11:12 +00:00
ffloortype_e oldflags ; // store FOF's old flags
2014-03-15 16:59:03 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2014-03-15 16:59:03 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 445 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
2022-07-31 10:04:42 +00:00
oldflags = rover - > fofflags ;
2018-09-19 21:49:03 +00:00
// Abracadabra!
2021-09-22 05:32:39 +00:00
if ( line - > args [ 2 ] )
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_EXISTS ;
2018-09-19 21:49:03 +00:00
else
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_EXISTS ;
2018-09-19 21:49:03 +00:00
// if flags changed, reset sector's light list
2022-07-31 10:04:42 +00:00
if ( rover - > fofflags ! = oldflags )
2019-07-04 13:43:18 +00:00
{
2018-09-19 21:49:03 +00:00
sec - > moved = true ;
2019-07-04 13:43:18 +00:00
P_RecalcPrecipInSector ( sec ) ;
}
2018-09-19 21:49:03 +00:00
}
2014-03-15 16:59:03 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2014-03-15 16:59:03 +00:00
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 445 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
return ;
}
}
}
break ;
2022-07-31 10:04:42 +00:00
case 446 : // Make block fall remotely (acts like FOF_CRUMBLE)
2016-06-21 21:04:58 +00:00
{
2021-09-22 05:32:39 +00:00
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2016-06-21 21:04:58 +00:00
sector_t * sec ; // Sector that the FOF is visible in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF that we are going to make fall down
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2016-06-21 21:04:58 +00:00
player_t * player = NULL ; // player that caused FOF to fall
boolean respawn = true ; // should the fallen FOF respawn?
if ( mo ) // NULL check
player = mo - > player ;
2021-09-22 05:32:39 +00:00
if ( line - > args [ 2 ] & TMFR_NORETURN ) // don't respawn!
2016-06-21 21:04:58 +00:00
respawn = false ;
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2016-06-21 21:04:58 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 446 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
2021-09-22 05:32:39 +00:00
if ( line - > args [ 2 ] & TMFR_CHECKFLAG ) // FOF flags determine respawn ability instead?
2022-07-31 10:04:42 +00:00
respawn = ! ( rover - > fofflags & FOF_NORETURN ) ^ ! ! ( line - > args [ 2 ] & TMFR_NORETURN ) ; // TMFR_NORETURN inverts
2018-09-19 21:49:03 +00:00
2022-07-31 10:04:42 +00:00
EV_StartCrumble ( rover - > master - > frontsector , rover , ( rover - > fofflags & FOF_FLOATBOB ) , player , rover - > alpha , respawn ) ;
2018-09-19 21:49:03 +00:00
}
2016-06-21 21:04:58 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2016-06-21 21:04:58 +00:00
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 446 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
return ;
}
}
}
break ;
2018-06-14 20:51:21 +00:00
case 447 : // Change colormap of tagged sectors!
// Basically this special applies a colormap to the tagged sectors, just like 606 (the colormap linedef)
// Except it is activated by linedef executor, not level load
// This could even override existing colormaps I believe
// -- Monster Iestyn 14/06/18
2020-03-20 10:19:30 +00:00
{
extracolormap_t * source ;
if ( ! udmf )
source = sides [ line - > sidenum [ 0 ] ] . colormap_data ;
else
2018-09-11 20:50:35 +00:00
{
2020-03-20 10:19:30 +00:00
if ( ! line - > args [ 1 ] )
source = line - > frontsector - > extra_colormap ;
else
2018-09-12 11:31:10 +00:00
{
2020-04-18 12:04:15 +00:00
INT32 sourcesec = Tag_Iterate_Sectors ( line - > args [ 1 ] , 0 ) ;
2020-03-20 10:19:30 +00:00
if ( sourcesec = = - 1 )
2018-09-12 11:31:10 +00:00
{
2020-03-20 10:19:30 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 447 Executor: Can't find sector with source colormap (tag %d)! \n " , line - > args [ 1 ] ) ;
return ;
2018-09-12 11:31:10 +00:00
}
2020-03-20 10:19:30 +00:00
source = sectors [ sourcesec ] . extra_colormap ;
2018-09-12 11:31:10 +00:00
}
2020-03-20 10:19:30 +00:00
}
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2020-03-20 10:19:30 +00:00
{
2020-03-20 11:19:02 +00:00
if ( sectors [ secnum ] . colormap_protected )
continue ;
2020-03-20 10:19:30 +00:00
P_ResetColormapFader ( & sectors [ secnum ] ) ;
2020-04-18 15:15:25 +00:00
if ( line - > args [ 2 ] & TMCF_RELATIVE )
2018-09-12 11:38:51 +00:00
{
2023-09-21 05:06:06 +00:00
extracolormap_t * target = ( ! udmf & & ( line - > flags & ML_TFERLINE ) & & line - > sidenum [ 1 ] ! = NO_SIDEDEF ) ?
2020-03-20 10:19:30 +00:00
sides [ line - > sidenum [ 1 ] ] . colormap_data : sectors [ secnum ] . extra_colormap ; // use back colormap instead of target sector
extracolormap_t * exc = R_AddColormaps (
target ,
source ,
2020-04-18 15:15:25 +00:00
line - > args [ 2 ] & TMCF_SUBLIGHTR ,
line - > args [ 2 ] & TMCF_SUBLIGHTG ,
line - > args [ 2 ] & TMCF_SUBLIGHTB ,
line - > args [ 2 ] & TMCF_SUBLIGHTA ,
line - > args [ 2 ] & TMCF_SUBFADER ,
line - > args [ 2 ] & TMCF_SUBFADEG ,
line - > args [ 2 ] & TMCF_SUBFADEB ,
line - > args [ 2 ] & TMCF_SUBFADEA ,
line - > args [ 2 ] & TMCF_SUBFADESTART ,
line - > args [ 2 ] & TMCF_SUBFADEEND ,
line - > args [ 2 ] & TMCF_IGNOREFLAGS ,
2020-03-20 10:19:30 +00:00
false ) ;
2018-09-12 11:38:51 +00:00
2018-09-12 12:06:44 +00:00
if ( ! ( sectors [ secnum ] . extra_colormap = R_GetColormapFromList ( exc ) ) )
{
exc - > colormap = R_CreateLightTable ( exc ) ;
R_AddColormapToList ( exc ) ;
sectors [ secnum ] . extra_colormap = exc ;
2018-09-12 11:38:51 +00:00
}
else
2018-09-12 12:06:44 +00:00
Z_Free ( exc ) ;
2018-09-12 11:38:51 +00:00
}
2018-09-12 12:06:44 +00:00
else
2020-03-20 10:19:30 +00:00
sectors [ secnum ] . extra_colormap = source ;
2018-09-11 20:50:35 +00:00
}
2018-06-14 20:51:21 +00:00
break ;
2020-03-20 10:19:30 +00:00
}
2017-04-26 17:16:01 +00:00
case 448 : // Change skybox viewpoint/centerpoint
2021-09-20 12:25:07 +00:00
if ( ( mo & & mo - > player & & P_IsLocalPlayer ( mo - > player ) ) | | line - > args [ 3 ] )
2017-04-26 17:16:01 +00:00
{
2021-09-20 12:25:07 +00:00
INT32 viewid = line - > args [ 0 ] ;
INT32 centerid = line - > args [ 1 ] ;
2017-04-26 17:16:01 +00:00
2021-09-20 12:25:07 +00:00
// set viewpoint mobj
if ( line - > args [ 2 ] ! = TMS_CENTERPOINT )
2017-04-26 17:16:01 +00:00
{
2021-09-20 12:25:07 +00:00
if ( viewid > = 0 & & viewid < 16 )
skyboxmo [ 0 ] = skyboxviewpnts [ viewid ] ;
else
skyboxmo [ 0 ] = NULL ;
2017-04-26 17:16:01 +00:00
}
2017-05-01 18:34:53 +00:00
2021-09-20 12:25:07 +00:00
// set centerpoint mobj
if ( line - > args [ 2 ] ! = TMS_VIEWPOINT )
{
if ( centerid > = 0 & & centerid < 16 )
skyboxmo [ 1 ] = skyboxcenterpnts [ centerid ] ;
else
skyboxmo [ 1 ] = NULL ;
2017-04-26 17:16:01 +00:00
}
CONS_Debug ( DBG_GAMELOGIC , " Line type 448 Executor: viewid = %d, centerid = %d, viewpoint? = %s, centerpoint? = %s \n " ,
viewid ,
centerid ,
2021-09-20 12:25:07 +00:00
( ( line - > args [ 2 ] = = TMS_CENTERPOINT ) ? " no " : " yes " ) ,
( ( line - > args [ 2 ] = = TMS_VIEWPOINT ) ? " no " : " yes " ) ) ;
2017-04-26 17:16:01 +00:00
}
break ;
2019-07-31 22:17:17 +00:00
case 449 : // Enable bosses with parameter
{
2021-09-20 12:25:07 +00:00
INT32 bossid = line - > args [ 0 ] ;
2019-07-31 22:17:17 +00:00
if ( bossid & ~ 15 ) // if any bits other than first 16 are set
{
CONS_Alert ( CONS_WARNING ,
2021-09-20 12:25:07 +00:00
M_GetText ( " Boss enable linedef has an invalid boss ID (%d). \n Consider changing it or removing it entirely. \n " ) ,
bossid ) ;
2019-07-31 22:17:17 +00:00
break ;
}
2021-09-20 12:25:07 +00:00
if ( line - > args [ 1 ] )
2019-07-31 22:17:17 +00:00
{
bossdisabled | = ( 1 < < bossid ) ;
CONS_Debug ( DBG_GAMELOGIC , " Line type 449 Executor: bossid disabled = %d " , bossid ) ;
}
else
{
bossdisabled & = ~ ( 1 < < bossid ) ;
CONS_Debug ( DBG_GAMELOGIC , " Line type 449 Executor: bossid enabled = %d " , bossid ) ;
}
break ;
}
2014-03-15 16:59:03 +00:00
case 450 : // Execute Linedef Executor - for recursion
2021-09-20 12:25:07 +00:00
P_LinedefExecute ( line - > args [ 0 ] , mo , NULL ) ;
2014-03-15 16:59:03 +00:00
break ;
case 451 : // Execute Random Linedef Executor
{
2021-09-20 12:25:07 +00:00
INT32 rvalue1 = line - > args [ 0 ] ;
INT32 rvalue2 = line - > args [ 1 ] ;
2014-03-15 16:59:03 +00:00
INT32 result ;
if ( rvalue1 < = rvalue2 )
result = P_RandomRange ( rvalue1 , rvalue2 ) ;
else
result = P_RandomRange ( rvalue2 , rvalue1 ) ;
P_LinedefExecute ( ( INT16 ) result , mo , NULL ) ;
break ;
}
2018-09-10 14:09:02 +00:00
case 452 : // Set FOF alpha
{
2021-09-22 05:32:39 +00:00
INT16 destvalue = ( INT16 ) ( line - > args [ 2 ] ) ;
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2018-09-10 14:09:02 +00:00
sector_t * sec ; // Sector that the FOF is visible in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF that we are going to operate
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2018-09-10 14:09:02 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2018-09-10 14:09:02 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 452 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
// If fading an invisible FOF whose render flags we did not yet set,
2023-08-20 15:42:27 +00:00
// initialize its alpha to 0 for relative alpha calculation
2021-09-22 05:32:39 +00:00
if ( ! ( line - > args [ 3 ] & TMST_DONTDOTRANSLUCENT ) & & // do translucent
2022-07-31 10:04:42 +00:00
( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) & &
! ( rover - > fofflags & FOF_RENDERALL ) )
2023-08-20 15:42:27 +00:00
rover - > alpha = 0 ;
2018-09-19 21:49:03 +00:00
P_RemoveFakeFloorFader ( rover ) ;
P_FadeFakeFloor ( rover ,
rover - > alpha ,
2023-08-20 15:42:27 +00:00
max ( 0 , min ( 255 , ( line - > args [ 3 ] & TMST_RELATIVE ) ? rover - > alpha + destvalue : destvalue ) ) ,
2021-09-22 05:32:39 +00:00
0 , // set alpha immediately
false , NULL , // tic-based logic
2022-07-31 10:04:42 +00:00
false , // do not handle FOF_EXISTS
! ( line - > args [ 3 ] & TMST_DONTDOTRANSLUCENT ) , // handle FOF_TRANSLUCENT
2021-09-22 05:32:39 +00:00
false , // do not handle lighting
false , // do not handle colormap
false , // do not handle collision
false , // do not do ghost fade (no collision during fade)
true ) ; // use exact alpha values (for opengl)
2018-09-19 21:49:03 +00:00
}
2018-09-10 14:09:02 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2018-09-10 14:09:02 +00:00
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 452 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
return ;
}
}
break ;
}
case 453 : // Fade FOF
2018-03-30 20:17:35 +00:00
{
2021-09-22 05:32:39 +00:00
INT16 destvalue = ( INT16 ) ( line - > args [ 2 ] ) ;
INT16 speed = ( INT16 ) ( line - > args [ 3 ] ) ;
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2018-08-17 04:32:20 +00:00
sector_t * sec ; // Sector that the FOF is visible in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF that we are going to operate
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2018-08-17 06:28:52 +00:00
size_t j = 0 ; // sec->ffloors is saved as ffloor #0, ss->ffloors->next is #1, etc
2018-08-17 04:32:20 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2018-08-17 04:32:20 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
2018-09-10 14:09:02 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 453 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
2018-08-17 04:32:20 +00:00
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
// Prevent continuous execs from interfering on an existing fade
2021-09-22 05:32:39 +00:00
if ( ! ( line - > args [ 4 ] & TMFT_OVERRIDE )
2018-09-19 21:49:03 +00:00
& & rover - > fadingdata )
//&& ((fade_t*)rover->fadingdata)->timer > (ticbased ? 2 : speed*2))
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 453 Executor: Fade FOF thinker already exists, timer: %d \n " , ( ( fade_t * ) rover - > fadingdata ) - > timer ) ;
continue ;
}
if ( speed > 0 )
P_AddFakeFloorFader ( rover , secnum , j ,
destvalue ,
speed ,
2021-09-22 05:32:39 +00:00
( line - > args [ 4 ] & TMFT_TICBASED ) , // tic-based logic
( line - > args [ 4 ] & TMFT_RELATIVE ) , // Relative destvalue
2022-07-31 10:04:42 +00:00
! ( line - > args [ 4 ] & TMFT_DONTDOEXISTS ) , // do not handle FOF_EXISTS
! ( line - > args [ 4 ] & TMFT_DONTDOTRANSLUCENT ) , // do not handle FOF_TRANSLUCENT
2021-09-22 05:32:39 +00:00
! ( line - > args [ 4 ] & TMFT_DONTDOLIGHTING ) , // do not handle lighting
! ( line - > args [ 4 ] & TMFT_DONTDOCOLORMAP ) , // do not handle colormap
! ( line - > args [ 4 ] & TMFT_IGNORECOLLISION ) , // do not handle collision
( line - > args [ 4 ] & TMFT_GHOSTFADE ) , // do ghost fade (no collision during fade)
( line - > args [ 4 ] & TMFT_USEEXACTALPHA ) ) ; // use exact alpha values (for opengl)
2018-09-19 21:49:03 +00:00
else
{
// If fading an invisible FOF whose render flags we did not yet set,
2023-08-20 15:42:27 +00:00
// initialize its alpha to 1 for relative alpha calculation
2021-09-22 05:32:39 +00:00
if ( ! ( line - > args [ 4 ] & TMFT_DONTDOTRANSLUCENT ) & & // do translucent
2022-07-31 10:04:42 +00:00
( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) & &
! ( rover - > fofflags & FOF_RENDERALL ) )
2023-08-20 15:42:27 +00:00
rover - > alpha = 0 ;
2018-09-19 21:49:03 +00:00
P_RemoveFakeFloorFader ( rover ) ;
P_FadeFakeFloor ( rover ,
rover - > alpha ,
2023-08-20 15:42:27 +00:00
max ( 0 , min ( 255 , ( line - > args [ 4 ] & TMFT_RELATIVE ) ? rover - > alpha + destvalue : destvalue ) ) ,
2021-09-22 05:32:39 +00:00
0 , // set alpha immediately
false , NULL , // tic-based logic
2022-07-31 10:04:42 +00:00
! ( line - > args [ 4 ] & TMFT_DONTDOEXISTS ) , // do not handle FOF_EXISTS
! ( line - > args [ 4 ] & TMFT_DONTDOTRANSLUCENT ) , // do not handle FOF_TRANSLUCENT
2021-09-22 05:32:39 +00:00
! ( line - > args [ 4 ] & TMFT_DONTDOLIGHTING ) , // do not handle lighting
! ( line - > args [ 4 ] & TMFT_DONTDOCOLORMAP ) , // do not handle colormap
! ( line - > args [ 4 ] & TMFT_IGNORECOLLISION ) , // do not handle collision
( line - > args [ 4 ] & TMFT_GHOSTFADE ) , // do ghost fade (no collision during fade)
( line - > args [ 4 ] & TMFT_USEEXACTALPHA ) ) ; // use exact alpha values (for opengl)
2018-09-19 21:49:03 +00:00
}
}
2018-08-17 06:28:52 +00:00
j + + ;
2018-08-17 04:32:20 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2018-08-17 04:32:20 +00:00
{
2018-09-10 14:09:02 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 453 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
2018-08-17 04:32:20 +00:00
return ;
}
}
2018-03-30 20:17:35 +00:00
break ;
}
2018-08-14 02:10:16 +00:00
2018-09-10 14:09:02 +00:00
case 454 : // Stop fading FOF
2018-03-31 05:15:24 +00:00
{
2021-09-22 05:32:39 +00:00
INT16 sectag = ( INT16 ) ( line - > args [ 0 ] ) ;
INT16 foftag = ( INT16 ) ( line - > args [ 1 ] ) ;
2018-08-17 04:32:20 +00:00
sector_t * sec ; // Sector that the FOF is visible in
2018-09-20 00:11:30 +00:00
ffloor_t * rover ; // FOF that we are going to operate
boolean foundrover = false ; // for debug, "Can't find a FOF" message
2018-08-17 04:32:20 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( sectag , secnum )
2018-08-17 04:32:20 +00:00
{
sec = sectors + secnum ;
if ( ! sec - > ffloors )
{
2018-09-10 14:09:02 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 454 Executor: Target sector #%d has no FOFs. \n " , secnum ) ;
2018-08-17 04:32:20 +00:00
return ;
}
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
{
2020-04-17 20:26:04 +00:00
if ( Tag_Find ( & rover - > master - > frontsector - > tags , foftag ) )
2018-09-19 21:49:03 +00:00
{
2018-09-20 00:11:30 +00:00
foundrover = true ;
2018-09-19 21:49:03 +00:00
P_ResetFakeFloorFader ( rover , NULL ,
2021-09-22 05:32:39 +00:00
! ( line - > args [ 2 ] ) ) ; // do not finalize collision flags
2018-09-19 21:49:03 +00:00
}
2018-08-17 04:32:20 +00:00
}
2018-09-19 21:49:03 +00:00
if ( ! foundrover )
2018-08-17 04:32:20 +00:00
{
2018-09-10 14:09:02 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 454 Executor: Can't find a FOF control sector with tag %d \n " , foftag ) ;
2018-08-17 04:32:20 +00:00
return ;
}
}
2018-03-30 20:17:35 +00:00
break ;
2018-03-31 05:15:24 +00:00
}
2018-03-30 20:17:35 +00:00
2018-09-12 13:06:38 +00:00
case 455 : // Fade colormap
2020-03-20 10:19:30 +00:00
{
extracolormap_t * dest ;
if ( ! udmf )
dest = sides [ line - > sidenum [ 0 ] ] . colormap_data ;
else
{
if ( ! line - > args [ 1 ] )
dest = line - > frontsector - > extra_colormap ;
else
{
2020-04-18 12:04:15 +00:00
INT32 destsec = Tag_Iterate_Sectors ( line - > args [ 1 ] , 0 ) ;
2020-03-20 10:19:30 +00:00
if ( destsec = = - 1 )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 455 Executor: Can't find sector with destination colormap (tag %d)! \n " , line - > args [ 1 ] ) ;
return ;
}
dest = sectors [ destsec ] . extra_colormap ;
}
}
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2018-09-12 13:06:38 +00:00
{
extracolormap_t * source_exc , * dest_exc , * exc ;
2020-03-20 11:19:02 +00:00
if ( sectors [ secnum ] . colormap_protected )
continue ;
2020-03-20 10:19:30 +00:00
// Don't interrupt ongoing fade
2020-04-18 15:15:25 +00:00
if ( ! ( line - > args [ 3 ] & TMCF_OVERRIDE )
2018-09-18 01:16:27 +00:00
& & sectors [ secnum ] . fadecolormapdata )
2018-09-18 09:45:28 +00:00
//&& ((fadecolormap_t*)sectors[secnum].fadecolormapdata)->timer > (ticbased ? 2 : speed*2))
{
2018-09-18 11:07:30 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Line type 455 Executor: Fade color thinker already exists, timer: %d \n " , ( ( fadecolormap_t * ) sectors [ secnum ] . fadecolormapdata ) - > timer ) ;
2018-09-18 01:16:27 +00:00
continue ;
2018-09-18 09:45:28 +00:00
}
2018-09-18 01:16:27 +00:00
2020-03-20 10:19:30 +00:00
if ( ! udmf & & ( line - > flags & ML_TFERLINE ) ) // use back colormap instead of target sector
2023-09-21 05:06:06 +00:00
sectors [ secnum ] . extra_colormap = ( line - > sidenum [ 1 ] ! = NO_SIDEDEF ) ?
2020-03-20 10:19:30 +00:00
sides [ line - > sidenum [ 1 ] ] . colormap_data : NULL ;
2018-09-12 15:21:49 +00:00
exc = sectors [ secnum ] . extra_colormap ;
2018-09-12 13:06:38 +00:00
2020-04-18 15:15:25 +00:00
if ( ! ( line - > args [ 3 ] & TMCF_FROMBLACK ) // Override fade from default rgba
2020-03-20 10:19:30 +00:00
& & ! R_CheckDefaultColormap ( dest , true , false , false )
2018-09-12 13:06:38 +00:00
& & R_CheckDefaultColormap ( exc , true , false , false ) )
{
exc = R_CopyColormap ( exc , false ) ;
2020-03-20 10:19:30 +00:00
exc - > rgba = R_GetRgbaRGB ( dest - > rgba ) + R_PutRgbaA ( R_GetRgbaA ( exc - > rgba ) ) ;
//exc->fadergba = R_GetRgbaRGB(dest->rgba) + R_PutRgbaA(R_GetRgbaA(exc->fadergba));
2018-09-12 13:06:38 +00:00
if ( ! ( source_exc = R_GetColormapFromList ( exc ) ) )
{
exc - > colormap = R_CreateLightTable ( exc ) ;
R_AddColormapToList ( exc ) ;
source_exc = exc ;
}
else
Z_Free ( exc ) ;
2018-09-12 15:11:22 +00:00
sectors [ secnum ] . extra_colormap = source_exc ;
2018-09-12 13:06:38 +00:00
}
else
2018-09-12 15:11:22 +00:00
source_exc = exc ? exc : R_GetDefaultColormap ( ) ;
2018-09-12 13:06:38 +00:00
2020-04-18 15:15:25 +00:00
if ( line - > args [ 3 ] & TMCF_RELATIVE )
2018-09-12 13:06:38 +00:00
{
exc = R_AddColormaps (
source_exc ,
2020-03-20 10:19:30 +00:00
dest ,
2020-04-18 15:15:25 +00:00
line - > args [ 3 ] & TMCF_SUBLIGHTR ,
line - > args [ 3 ] & TMCF_SUBLIGHTG ,
line - > args [ 3 ] & TMCF_SUBLIGHTB ,
line - > args [ 3 ] & TMCF_SUBLIGHTA ,
line - > args [ 3 ] & TMCF_SUBFADER ,
line - > args [ 3 ] & TMCF_SUBFADEG ,
line - > args [ 3 ] & TMCF_SUBFADEB ,
line - > args [ 3 ] & TMCF_SUBFADEA ,
line - > args [ 3 ] & TMCF_SUBFADESTART ,
line - > args [ 3 ] & TMCF_SUBFADEEND ,
line - > args [ 3 ] & TMCF_IGNOREFLAGS ,
2018-09-12 13:06:38 +00:00
false ) ;
}
else
2020-03-20 10:19:30 +00:00
exc = R_CopyColormap ( dest , false ) ;
2018-09-12 13:06:38 +00:00
if ( ! ( dest_exc = R_GetColormapFromList ( exc ) ) )
{
exc - > colormap = R_CreateLightTable ( exc ) ;
R_AddColormapToList ( exc ) ;
dest_exc = exc ;
}
else
Z_Free ( exc ) ;
2020-03-20 10:19:30 +00:00
Add_ColormapFader ( & sectors [ secnum ] , source_exc , dest_exc , true , // tic-based timing
line - > args [ 2 ] ) ;
2018-09-12 13:06:38 +00:00
}
break ;
2020-03-20 10:19:30 +00:00
}
2018-09-12 13:06:38 +00:00
case 456 : // Stop fade colormap
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
2018-09-12 13:06:38 +00:00
P_ResetColormapFader ( & sectors [ secnum ] ) ;
2018-09-12 13:42:51 +00:00
break ;
2018-09-12 13:06:38 +00:00
2018-11-04 21:28:25 +00:00
case 457 : // Track mobj angle to point
if ( mo )
{
2021-09-21 13:10:50 +00:00
INT32 failureangle = FixedAngle ( ( min ( max ( abs ( line - > args [ 1 ] ) , 0 ) , 360 ) ) * FRACUNIT ) ;
INT32 failuredelay = abs ( line - > args [ 2 ] ) ;
INT32 failureexectag = line - > args [ 3 ] ;
boolean persist = ! ! ( line - > args [ 4 ] ) ;
2018-11-04 21:28:25 +00:00
mobj_t * anchormo ;
2021-09-21 13:10:50 +00:00
anchormo = P_FindObjectTypeFromTag ( MT_ANGLEMAN , line - > args [ 0 ] ) ;
2018-11-04 21:28:25 +00:00
if ( ! anchormo )
return ;
mo - > eflags | = MFE_TRACERANGLE ;
P_SetTarget ( & mo - > tracer , anchormo ) ;
mo - > lastlook = persist ; // don't disable behavior after first failure
mo - > extravalue1 = failureangle ; // angle to exceed for failure state
mo - > extravalue2 = failureexectag ; // exec tag for failure state (angle is not within range)
mo - > cusval = mo - > cvmem = failuredelay ; // cusval = tics to allow failure before line trigger; cvmem = decrement timer
}
break ;
case 458 : // Stop tracking mobj angle to point
2018-11-04 21:48:18 +00:00
if ( mo & & ( mo - > eflags & MFE_TRACERANGLE ) )
2018-11-04 21:28:25 +00:00
{
mo - > eflags & = ~ MFE_TRACERANGLE ;
P_SetTarget ( & mo - > tracer , NULL ) ;
mo - > lastlook = mo - > cvmem = mo - > cusval = mo - > extravalue1 = mo - > extravalue2 = 0 ;
}
break ;
2018-11-04 21:45:29 +00:00
case 459 : // Control Text Prompt
2021-09-21 07:05:40 +00:00
// console player only
2018-11-10 20:34:56 +00:00
if ( mo & & mo - > player & & P_IsLocalPlayer ( mo - > player ) & & ( ! bot | | bot ! = mo ) )
2018-11-04 21:45:29 +00:00
{
2021-09-21 07:05:40 +00:00
INT32 promptnum = max ( 0 , line - > args [ 0 ] - 1 ) ;
INT32 pagenum = max ( 0 , line - > args [ 1 ] - 1 ) ;
INT32 postexectag = abs ( line - > args [ 3 ] ) ;
boolean closetextprompt = ( line - > args [ 2 ] & TMP_CLOSE ) ;
//boolean allplayers = (line->args[2] & TMP_ALLPLAYERS);
boolean runpostexec = ( line - > args [ 2 ] & TMP_RUNPOSTEXEC ) ;
boolean blockcontrols = ! ( line - > args [ 2 ] & TMP_KEEPCONTROLS ) ;
boolean freezerealtime = ! ( line - > args [ 2 ] & TMP_KEEPREALTIME ) ;
//boolean freezethinkers = (line->args[2] & TMP_FREEZETHINKERS);
boolean callbynamedtag = ( line - > args [ 2 ] & TMP_CALLBYNAME ) ;
2018-11-04 21:45:29 +00:00
if ( closetextprompt )
2018-11-05 03:22:47 +00:00
F_EndTextPrompt ( false , false ) ;
2018-11-04 21:45:29 +00:00
else
2018-11-13 05:13:36 +00:00
{
2021-09-21 07:05:40 +00:00
if ( callbynamedtag & & line - > stringargs [ 0 ] & & line - > stringargs [ 0 ] [ 0 ] )
F_GetPromptPageByNamedTag ( line - > stringargs [ 0 ] , & promptnum , & pagenum ) ;
2018-11-04 21:45:29 +00:00
F_StartTextPrompt ( promptnum , pagenum , mo , runpostexec ? postexectag : 0 , blockcontrols , freezerealtime ) ;
2018-11-13 05:13:36 +00:00
}
2018-11-04 21:45:29 +00:00
}
break ;
2019-07-19 02:42:46 +00:00
case 460 : // Award rings
{
2021-09-20 12:25:07 +00:00
INT16 rings = line - > args [ 0 ] ;
INT32 delay = line - > args [ 1 ] ;
2019-07-25 01:18:07 +00:00
if ( mo & & mo - > player )
2019-07-19 02:42:46 +00:00
{
2019-08-03 12:04:02 +00:00
if ( delay < = 0 | | ! ( leveltime % delay ) )
2019-07-19 02:42:46 +00:00
P_GivePlayerRings ( mo - > player , rings ) ;
}
}
2018-11-04 21:45:29 +00:00
break ;
2019-08-29 05:57:58 +00:00
case 461 : // Spawns an object on the map based on texture offsets
{
2021-09-22 06:57:48 +00:00
const mobjtype_t type = line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : MT_NULL ;
2019-09-08 21:14:47 +00:00
mobj_t * mobj ;
2019-08-29 05:57:58 +00:00
fixed_t x , y , z ;
2021-09-22 06:57:48 +00:00
if ( line - > args [ 4 ] ) // If args[4] is set, spawn randomly within a range
2019-08-29 05:57:58 +00:00
{
2021-09-22 06:57:48 +00:00
x = P_RandomRange ( line - > args [ 0 ] , line - > args [ 5 ] ) < < FRACBITS ;
y = P_RandomRange ( line - > args [ 1 ] , line - > args [ 6 ] ) < < FRACBITS ;
z = P_RandomRange ( line - > args [ 2 ] , line - > args [ 7 ] ) < < FRACBITS ;
}
else
{
x = line - > args [ 0 ] < < FRACBITS ;
y = line - > args [ 1 ] < < FRACBITS ;
z = line - > args [ 2 ] < < FRACBITS ;
2019-08-29 05:57:58 +00:00
}
2019-09-08 21:14:47 +00:00
mobj = P_SpawnMobj ( x , y , z , type ) ;
2019-08-30 03:56:15 +00:00
if ( mobj )
2020-04-13 15:10:25 +00:00
{
2021-09-22 06:57:48 +00:00
mobj - > angle = FixedAngle ( line - > args [ 3 ] < < FRACBITS ) ;
2019-08-30 03:56:15 +00:00
CONS_Debug ( DBG_GAMELOGIC , " Linedef Type %d - Spawn Object: %d spawned at (%d, %d, %d) \n " , line - > special , mobj - > type , mobj - > x > > FRACBITS , mobj - > y > > FRACBITS , mobj - > z > > FRACBITS ) ; //TODO: Convert mobj->type to a string somehow.
2020-04-13 15:10:25 +00:00
}
2019-08-30 03:56:15 +00:00
else
CONS_Alert ( CONS_ERROR , " Linedef Type %d - Spawn Object: Object did not spawn! \n " , line - > special ) ;
2019-08-29 05:57:58 +00:00
}
break ;
2019-11-08 15:47:12 +00:00
case 462 : // Stop clock (and end level in record attack)
if ( G_PlatformGametype ( ) )
{
stoppedclock = true ;
CONS_Debug ( DBG_GAMELOGIC , " Clock stopped! \n " ) ;
if ( modeattacking )
{
UINT8 i ;
for ( i = 0 ; i < MAXPLAYERS ; i + + )
{
if ( ! playeringame [ i ] )
continue ;
2023-09-20 02:48:01 +00:00
P_DoPlayerExit ( & players [ i ] , true ) ;
2019-11-08 15:47:12 +00:00
}
}
}
break ;
2020-01-11 23:40:57 +00:00
case 463 : // Dye object
{
if ( mo )
{
2021-09-21 05:44:55 +00:00
INT32 color = line - > stringargs [ 0 ] ? get_number ( line - > stringargs [ 0 ] ) : SKINCOLOR_NONE ;
2020-05-24 00:29:07 +00:00
if ( color < 0 | | color > = numskincolors )
2020-01-11 23:40:57 +00:00
return ;
2020-04-14 08:31:07 +00:00
2020-01-11 23:40:57 +00:00
var1 = 0 ;
var2 = color ;
A_Dye ( mo ) ;
}
}
break ;
2020-04-14 08:31:07 +00:00
2020-05-03 13:22:13 +00:00
case 464 : // Trigger Egg Capsule
{
2021-09-21 10:55:56 +00:00
INT32 mtnum ;
2020-05-03 13:22:13 +00:00
mobj_t * mo2 ;
// Find the center of the Eggtrap and release all the pretty animals!
// The chimps are my friends.. heeheeheheehehee..... - LouisJM
2021-09-21 10:55:56 +00:00
TAG_ITER_THINGS ( line - > args [ 0 ] , mtnum )
2020-05-03 13:22:13 +00:00
{
2021-09-21 10:55:56 +00:00
mo2 = mapthings [ mtnum ] . mobj ;
2020-05-03 13:22:13 +00:00
2021-09-21 10:55:56 +00:00
if ( ! mo2 )
2020-05-03 13:22:13 +00:00
continue ;
2021-09-21 10:59:20 +00:00
if ( mo2 - > type ! = MT_EGGTRAP )
continue ;
2021-09-21 10:55:56 +00:00
if ( mo2 - > thinker . function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
2020-05-03 13:22:13 +00:00
continue ;
P_KillMobj ( mo2 , NULL , mo , 0 ) ;
}
2021-09-21 08:35:01 +00:00
if ( ! ( line - > args [ 1 ] ) )
2020-05-03 13:22:13 +00:00
{
INT32 i ;
// Mark all players with the time to exit thingy!
for ( i = 0 ; i < MAXPLAYERS ; i + + )
{
if ( ! playeringame [ i ] )
continue ;
2023-09-20 02:48:01 +00:00
P_DoPlayerExit ( & players [ i ] , true ) ;
2020-05-03 13:22:13 +00:00
}
}
}
break ;
2020-05-03 19:03:15 +00:00
case 465 : // Set linedef executor delay
{
INT32 linenum ;
if ( ! udmf )
break ;
2021-02-11 12:24:20 +00:00
TAG_ITER_LINES ( line - > args [ 0 ] , linenum )
2020-05-03 19:03:15 +00:00
{
if ( line - > args [ 2 ] )
lines [ linenum ] . executordelay + = line - > args [ 1 ] ;
else
lines [ linenum ] . executordelay = line - > args [ 1 ] ;
}
}
break ;
2021-04-19 21:20:34 +00:00
case 466 : // Set level failure state
{
2024-05-19 19:24:03 +00:00
if ( line - > args [ 0 ] )
2021-04-19 21:20:34 +00:00
{
stagefailed = false ;
CONS_Debug ( DBG_GAMELOGIC , " Stage can be completed successfully! \n " ) ;
}
else
{
stagefailed = true ;
CONS_Debug ( DBG_GAMELOGIC , " Stage will end in failure... \n " ) ;
}
}
break ;
2021-09-20 05:54:10 +00:00
case 467 : // Set light level
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
{
if ( sectors [ secnum ] . lightingdata )
{
// Stop any lighting effects going on in the sector
2021-09-20 06:18:59 +00:00
P_RemoveThinker ( & ( ( thinkerdata_t * ) sectors [ secnum ] . lightingdata ) - > thinker ) ;
2021-09-20 05:54:10 +00:00
sectors [ secnum ] . lightingdata = NULL ;
}
if ( line - > args [ 2 ] = = TML_FLOOR )
{
if ( line - > args [ 3 ] )
sectors [ secnum ] . floorlightlevel + = line - > args [ 1 ] ;
else
sectors [ secnum ] . floorlightlevel = line - > args [ 1 ] ;
}
else if ( line - > args [ 2 ] = = TML_CEILING )
{
if ( line - > args [ 3 ] )
sectors [ secnum ] . ceilinglightlevel + = line - > args [ 1 ] ;
else
sectors [ secnum ] . ceilinglightlevel = line - > args [ 1 ] ;
}
else
{
if ( line - > args [ 3 ] )
sectors [ secnum ] . lightlevel + = line - > args [ 1 ] ;
else
sectors [ secnum ] . lightlevel = line - > args [ 1 ] ;
sectors [ secnum ] . lightlevel = max ( 0 , min ( 255 , sectors [ secnum ] . lightlevel ) ) ;
}
}
break ;
2021-12-29 09:42:20 +00:00
case 468 : // Change linedef executor argument
{
INT32 linenum ;
if ( ! udmf )
break ;
if ( line - > args [ 1 ] < 0 | | line - > args [ 1 ] > = NUMLINEARGS )
{
CONS_Debug ( DBG_GAMELOGIC , " Linedef type 468: Invalid linedef arg %d \n " , line - > args [ 1 ] ) ;
break ;
}
TAG_ITER_LINES ( line - > args [ 0 ] , linenum )
{
if ( line - > args [ 3 ] )
lines [ linenum ] . args [ line - > args [ 1 ] ] + = line - > args [ 2 ] ;
else
lines [ linenum ] . args [ line - > args [ 1 ] ] = line - > args [ 2 ] ;
}
}
break ;
2021-12-30 18:28:01 +00:00
case 469 : // Change sector gravity
{
fixed_t gravityvalue ;
if ( ! udmf )
break ;
if ( ! line - > stringargs [ 0 ] )
break ;
gravityvalue = FloatToFixed ( atof ( line - > stringargs [ 0 ] ) ) ;
TAG_ITER_SECTORS ( line - > args [ 0 ] , secnum )
{
if ( line - > args [ 1 ] )
sectors [ secnum ] . gravity = FixedMul ( sectors [ secnum ] . gravity , gravityvalue ) ;
else
sectors [ secnum ] . gravity = gravityvalue ;
if ( line - > args [ 2 ] = = TMF_ADD )
sectors [ secnum ] . flags | = MSF_GRAVITYFLIP ;
else if ( line - > args [ 2 ] = = TMF_REMOVE )
sectors [ secnum ] . flags & = ~ MSF_GRAVITYFLIP ;
2022-09-11 17:46:57 +00:00
if ( line - > args [ 3 ] )
sectors [ secnum ] . specialflags | = SSF_GRAVITYOVERRIDE ;
2021-12-30 18:28:01 +00:00
}
}
break ;
2014-03-15 16:59:03 +00:00
case 480 : // Polyobj_DoorSlide
case 481 : // Polyobj_DoorSwing
PolyDoor ( line ) ;
break ;
case 482 : // Polyobj_Move
PolyMove ( line ) ;
break ;
case 484 : // Polyobj_RotateRight
PolyRotate ( line ) ;
break ;
case 488 : // Polyobj_Waypoint
PolyWaypoint ( line ) ;
break ;
case 489 :
2021-07-04 11:45:41 +00:00
PolySetVisibilityTangibility ( line ) ;
2014-03-15 16:59:03 +00:00
break ;
case 491 :
PolyTranslucency ( line ) ;
break ;
2018-09-07 19:27:18 +00:00
case 492 :
PolyFade ( line ) ;
break ;
2014-03-15 16:59:03 +00:00
default :
break ;
}
}
//
// P_SetupSignExit
//
// Finds the exit sign in the current sector and
// sets its target to the player who passed the map.
//
void P_SetupSignExit ( player_t * player )
{
mobj_t * thing ;
msecnode_t * node = player - > mo - > subsector - > sector - > touching_thinglist ; // things touching this sector
thinker_t * think ;
INT32 numfound = 0 ;
2016-06-09 13:16:02 +00:00
for ( ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
if ( thing - > type ! = MT_SIGN )
continue ;
2019-12-05 04:36:25 +00:00
if ( ! numfound
& & ! ( player - > mo - > target & & player - > mo - > target - > type = = MT_SIGN )
2020-02-29 04:51:45 +00:00
& & ! ( ( gametyperules & GTR_FRIENDLY ) & & ( netgame | | multiplayer ) & & cv_exitmove . value ) )
2019-12-05 04:36:25 +00:00
P_SetTarget ( & player - > mo - > target , thing ) ;
2019-11-08 16:47:51 +00:00
2014-03-15 16:59:03 +00:00
if ( thing - > state ! = & states [ thing - > info - > spawnstate ] )
continue ;
P_SetTarget ( & thing - > target , player - > mo ) ;
2019-11-04 08:28:44 +00:00
P_SetObjectMomZ ( thing , 12 * FRACUNIT , false ) ;
2019-11-04 17:33:09 +00:00
P_SetMobjState ( thing , S_SIGNSPIN1 ) ;
2014-03-15 16:59:03 +00:00
if ( thing - > info - > seesound )
S_StartSound ( thing , thing - > info - > seesound ) ;
+ + numfound ;
}
if ( numfound )
return ;
// didn't find any signposts in the exit sector.
// spin all signposts in the level then.
2019-04-20 21:29:20 +00:00
for ( think = thlist [ THINK_MOBJ ] . next ; think ! = & thlist [ THINK_MOBJ ] ; think = think - > next )
2014-03-15 16:59:03 +00:00
{
2019-07-12 23:42:03 +00:00
if ( think - > function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
continue ;
2014-03-15 16:59:03 +00:00
thing = ( mobj_t * ) think ;
if ( thing - > type ! = MT_SIGN )
continue ;
2019-12-05 04:36:25 +00:00
if ( ! numfound
& & ! ( player - > mo - > target & & player - > mo - > target - > type = = MT_SIGN )
2020-02-29 04:51:45 +00:00
& & ! ( ( gametyperules & GTR_FRIENDLY ) & & ( netgame | | multiplayer ) & & cv_exitmove . value ) )
2019-12-05 04:36:25 +00:00
P_SetTarget ( & player - > mo - > target , thing ) ;
2019-11-08 16:47:51 +00:00
2014-03-15 16:59:03 +00:00
if ( thing - > state ! = & states [ thing - > info - > spawnstate ] )
continue ;
P_SetTarget ( & thing - > target , player - > mo ) ;
2019-11-04 08:28:44 +00:00
P_SetObjectMomZ ( thing , 12 * FRACUNIT , false ) ;
2019-11-04 17:33:09 +00:00
P_SetMobjState ( thing , S_SIGNSPIN1 ) ;
2014-03-15 16:59:03 +00:00
if ( thing - > info - > seesound )
S_StartSound ( thing , thing - > info - > seesound ) ;
+ + numfound ;
}
}
//
// P_IsFlagAtBase
//
// Checks to see if a flag is at its base.
//
2014-05-26 02:41:05 +00:00
boolean P_IsFlagAtBase ( mobjtype_t flag )
2014-03-15 16:59:03 +00:00
{
thinker_t * think ;
mobj_t * mo ;
2021-12-30 23:03:24 +00:00
sectorspecialflags_t specialflag = ( flag = = MT_REDFLAG ) ? SSF_REDTEAMBASE : SSF_BLUETEAMBASE ;
2014-03-15 16:59:03 +00:00
2019-04-20 21:29:20 +00:00
for ( think = thlist [ THINK_MOBJ ] . next ; think ! = & thlist [ THINK_MOBJ ] ; think = think - > next )
2014-03-15 16:59:03 +00:00
{
2019-07-12 23:42:03 +00:00
if ( think - > function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
continue ;
2014-03-15 16:59:03 +00:00
mo = ( mobj_t * ) think ;
if ( mo - > type ! = flag )
continue ;
2021-12-30 23:03:24 +00:00
if ( mo - > subsector - > sector - > specialflags & specialflag )
2014-03-15 16:59:03 +00:00
return true ;
else if ( mo - > subsector - > sector - > ffloors ) // Check the 3D floors
{
ffloor_t * rover ;
for ( rover = mo - > subsector - > sector - > ffloors ; rover ; rover = rover - > next )
{
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2014-03-15 16:59:03 +00:00
continue ;
2021-12-30 23:03:24 +00:00
if ( ! ( rover - > master - > frontsector - > specialflags & specialflag ) )
2014-03-15 16:59:03 +00:00
continue ;
2019-07-12 23:42:03 +00:00
if ( ! ( mo - > z < = P_GetSpecialTopZ ( mo , sectors + rover - > secnum , mo - > subsector - > sector )
& & mo - > z > = P_GetSpecialBottomZ ( mo , sectors + rover - > secnum , mo - > subsector - > sector ) ) )
continue ;
return true ;
2014-03-15 16:59:03 +00:00
}
}
}
return false ;
}
2021-12-06 18:33:36 +00:00
static boolean P_IsMobjTouchingPlane ( mobj_t * mo , sector_t * sec , fixed_t floorz , fixed_t ceilingz )
{
2021-12-30 13:16:00 +00:00
boolean floorallowed = ( ( sec - > flags & MSF_FLIPSPECIAL_FLOOR ) & & ( ( sec - > flags & MSF_TRIGGERSPECIAL_HEADBUMP ) | | ! ( mo - > eflags & MFE_VERTICALFLIP ) ) & & ( mo - > z = = floorz ) ) ;
boolean ceilingallowed = ( ( sec - > flags & MSF_FLIPSPECIAL_CEILING ) & & ( ( sec - > flags & MSF_TRIGGERSPECIAL_HEADBUMP ) | | ( mo - > eflags & MFE_VERTICALFLIP ) ) & & ( mo - > z + mo - > height = = ceilingz ) ) ;
2021-12-06 18:33:36 +00:00
return ( floorallowed | | ceilingallowed ) ;
}
2021-12-07 06:17:58 +00:00
boolean P_IsMobjTouchingSectorPlane ( mobj_t * mo , sector_t * sec )
2021-12-06 18:33:36 +00:00
{
return P_IsMobjTouchingPlane ( mo , sec , P_GetSpecialBottomZ ( mo , sec , sec ) , P_GetSpecialTopZ ( mo , sec , sec ) ) ;
}
2021-12-07 06:17:58 +00:00
boolean P_IsMobjTouching3DFloor ( mobj_t * mo , ffloor_t * ffloor , sector_t * sec )
2021-12-06 18:33:36 +00:00
{
fixed_t topheight = P_GetSpecialTopZ ( mo , sectors + ffloor - > secnum , sec ) ;
fixed_t bottomheight = P_GetSpecialBottomZ ( mo , sectors + ffloor - > secnum , sec ) ;
2022-07-31 10:04:42 +00:00
if ( ( ( ffloor - > fofflags & FOF_BLOCKPLAYER ) & & mo - > player )
| | ( ( ffloor - > fofflags & FOF_BLOCKOTHERS ) & & ! mo - > player ) )
2021-12-06 18:33:36 +00:00
{
// Solid 3D floor: Mobj must touch the top or bottom
return P_IsMobjTouchingPlane ( mo , ffloor - > master - > frontsector , topheight , bottomheight ) ;
}
else
{
// Water or intangible 3D floor: Mobj must be inside
return mo - > z < = topheight & & ( mo - > z + mo - > height ) > = bottomheight ;
}
}
2021-12-07 06:17:58 +00:00
boolean P_IsMobjTouchingPolyobj ( mobj_t * mo , polyobj_t * po , sector_t * polysec )
2021-12-06 18:33:36 +00:00
{
if ( ! ( po - > flags & POF_TESTHEIGHT ) ) // Don't do height checking
return true ;
if ( po - > flags & POF_SOLID )
{
// Solid polyobject: Player must touch the top or bottom
return P_IsMobjTouchingPlane ( mo , polysec , polysec - > ceilingheight , polysec - > floorheight ) ;
}
else
{
// Water or intangible polyobject: Player must be inside
return mo - > z < = polysec - > ceilingheight & & ( mo - > z + mo - > height ) > = polysec - > floorheight ;
}
}
2021-12-08 20:07:26 +00:00
static sector_t * P_MobjTouching3DFloorSpecial ( mobj_t * mo , sector_t * sector , INT32 section , INT32 number )
2014-03-15 16:59:03 +00:00
{
ffloor_t * rover ;
2021-12-08 20:07:26 +00:00
for ( rover = sector - > ffloors ; rover ; rover = rover - > next )
2014-03-15 16:59:03 +00:00
{
if ( GETSECSPECIAL ( rover - > master - > frontsector - > special , section ) ! = number )
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2014-03-15 16:59:03 +00:00
continue ;
2021-12-08 20:07:26 +00:00
if ( ! P_IsMobjTouching3DFloor ( mo , rover , sector ) )
2021-12-06 18:33:36 +00:00
continue ;
2014-03-15 16:59:03 +00:00
2021-12-08 20:07:26 +00:00
// This FOF has the special we're looking for, but are we allowed to touch it?
if ( sector = = mo - > subsector - > sector
2021-12-30 13:16:00 +00:00
| | ( rover - > master - > frontsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2021-12-08 20:07:26 +00:00
return rover - > master - > frontsector ;
2014-03-15 16:59:03 +00:00
}
2021-12-08 20:07:26 +00:00
return NULL ;
}
2021-12-30 23:03:24 +00:00
static sector_t * P_MobjTouching3DFloorSpecialFlag ( mobj_t * mo , sector_t * sector , sectorspecialflags_t flag )
{
ffloor_t * rover ;
for ( rover = sector - > ffloors ; rover ; rover = rover - > next )
{
if ( ! ( rover - > master - > frontsector - > specialflags & flag ) )
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2021-12-30 23:03:24 +00:00
continue ;
if ( ! P_IsMobjTouching3DFloor ( mo , rover , sector ) )
continue ;
// This FOF has the special we're looking for, but are we allowed to touch it?
if ( sector = = mo - > subsector - > sector
| | ( rover - > master - > frontsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
return rover - > master - > frontsector ;
}
return NULL ;
}
2021-12-08 20:07:26 +00:00
static sector_t * P_MobjTouchingPolyobjSpecial ( mobj_t * mo , INT32 section , INT32 number )
{
polyobj_t * po ;
sector_t * polysec ;
boolean touching = false ;
boolean inside = false ;
for ( po = mo - > subsector - > polyList ; po ; po = ( polyobj_t * ) ( po - > link . next ) )
2014-03-15 16:59:03 +00:00
{
2021-12-08 20:07:26 +00:00
if ( po - > flags & POF_NOSPECIALS )
continue ;
2014-03-15 16:59:03 +00:00
2021-12-08 20:07:26 +00:00
polysec = po - > lines [ 0 ] - > backsector ;
2014-03-15 16:59:03 +00:00
2021-12-08 20:07:26 +00:00
if ( GETSECSPECIAL ( polysec - > special , section ) ! = number )
continue ;
2014-03-15 16:59:03 +00:00
2021-12-30 13:16:00 +00:00
touching = ( polysec - > flags & MSF_TRIGGERSPECIAL_TOUCH ) & & P_MobjTouchingPolyobj ( po , mo ) ;
2021-12-08 20:07:26 +00:00
inside = P_MobjInsidePolyobj ( po , mo ) ;
2014-03-15 16:59:03 +00:00
2021-12-08 20:07:26 +00:00
if ( ! ( inside | | touching ) )
continue ;
if ( ! P_IsMobjTouchingPolyobj ( mo , po , polysec ) )
continue ;
return polysec ;
}
return NULL ;
}
2021-12-30 23:03:24 +00:00
static sector_t * P_MobjTouchingPolyobjSpecialFlag ( mobj_t * mo , sectorspecialflags_t flag )
{
polyobj_t * po ;
sector_t * polysec ;
boolean touching = false ;
boolean inside = false ;
for ( po = mo - > subsector - > polyList ; po ; po = ( polyobj_t * ) ( po - > link . next ) )
{
if ( po - > flags & POF_NOSPECIALS )
continue ;
polysec = po - > lines [ 0 ] - > backsector ;
if ( ! ( polysec - > specialflags & flag ) )
continue ;
touching = ( polysec - > flags & MSF_TRIGGERSPECIAL_TOUCH ) & & P_MobjTouchingPolyobj ( po , mo ) ;
inside = P_MobjInsidePolyobj ( po , mo ) ;
if ( ! ( inside | | touching ) )
continue ;
if ( ! P_IsMobjTouchingPolyobj ( mo , po , polysec ) )
continue ;
return polysec ;
}
return NULL ;
}
2021-12-08 20:07:26 +00:00
sector_t * P_MobjTouchingSectorSpecial ( mobj_t * mo , INT32 section , INT32 number )
{
msecnode_t * node ;
sector_t * result ;
result = P_MobjTouching3DFloorSpecial ( mo , mo - > subsector - > sector , section , number ) ;
if ( result )
return result ;
result = P_MobjTouchingPolyobjSpecial ( mo , section , number ) ;
if ( result )
return result ;
if ( GETSECSPECIAL ( mo - > subsector - > sector - > special , section ) = = number )
return mo - > subsector - > sector ;
for ( node = mo - > touching_sectorlist ; node ; node = node - > m_sectorlist_next )
{
if ( node - > m_sector = = mo - > subsector - > sector ) // Don't duplicate
continue ;
result = P_MobjTouching3DFloorSpecial ( mo , node - > m_sector , section , number ) ;
if ( result )
return result ;
2021-12-30 13:16:00 +00:00
if ( ! ( node - > m_sector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2021-12-08 20:07:26 +00:00
continue ;
2021-12-30 23:03:24 +00:00
if ( GETSECSPECIAL ( node - > m_sector - > special , section ) = = number )
return node - > m_sector ;
}
return NULL ;
}
2023-10-28 13:19:35 +00:00
// TODO: 2.3: Delete
2023-01-01 10:31:44 +00:00
// Deprecated in favor of P_MobjTouchingSectorSpecial
// Kept for Lua backwards compatibility only
sector_t * P_ThingOnSpecial3DFloor ( mobj_t * mo )
{
ffloor_t * rover ;
for ( rover = mo - > subsector - > sector - > ffloors ; rover ; rover = rover - > next )
{
if ( ! rover - > master - > frontsector - > special )
continue ;
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
continue ;
if ( ! P_IsMobjTouching3DFloor ( mo , rover , mo - > subsector - > sector ) )
continue ;
return rover - > master - > frontsector ;
}
return NULL ;
}
2021-12-30 23:03:24 +00:00
sector_t * P_MobjTouchingSectorSpecialFlag ( mobj_t * mo , sectorspecialflags_t flag )
{
msecnode_t * node ;
sector_t * result ;
result = P_MobjTouching3DFloorSpecialFlag ( mo , mo - > subsector - > sector , flag ) ;
if ( result )
return result ;
result = P_MobjTouchingPolyobjSpecialFlag ( mo , flag ) ;
if ( result )
return result ;
if ( mo - > subsector - > sector - > specialflags & flag )
return mo - > subsector - > sector ;
for ( node = mo - > touching_sectorlist ; node ; node = node - > m_sectorlist_next )
{
if ( node - > m_sector = = mo - > subsector - > sector ) // Don't duplicate
continue ;
result = P_MobjTouching3DFloorSpecialFlag ( mo , node - > m_sector , flag ) ;
if ( result )
return result ;
if ( ! ( node - > m_sector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
continue ;
if ( node - > m_sector - > specialflags & flag )
2021-12-08 20:07:26 +00:00
return node - > m_sector ;
2014-03-15 16:59:03 +00:00
}
return NULL ;
}
2021-12-08 20:07:26 +00:00
//
// P_PlayerTouchingSectorSpecial
//
// Replaces the old player->specialsector.
// This allows a player to touch more than
// one sector at a time, if necessary.
//
// Returns a pointer to the first sector of
// the particular type that it finds.
// Returns NULL if it doesn't find it.
//
sector_t * P_PlayerTouchingSectorSpecial ( player_t * player , INT32 section , INT32 number )
{
if ( ! player - > mo )
return NULL ;
return P_MobjTouchingSectorSpecial ( player - > mo , section , number ) ;
}
2021-12-30 23:03:24 +00:00
sector_t * P_PlayerTouchingSectorSpecialFlag ( player_t * player , sectorspecialflags_t flag )
{
if ( ! player - > mo )
return NULL ;
return P_MobjTouchingSectorSpecialFlag ( player - > mo , flag ) ;
}
2021-12-31 10:39:34 +00:00
static sector_t * P_CheckPlayer3DFloorTrigger ( player_t * player , sector_t * sector , line_t * sourceline )
2014-08-27 03:56:30 +00:00
{
ffloor_t * rover ;
2021-12-08 19:29:10 +00:00
for ( rover = sector - > ffloors ; rover ; rover = rover - > next )
{
2021-12-31 10:39:34 +00:00
if ( ! rover - > master - > frontsector - > triggertag )
continue ;
if ( rover - > master - > frontsector - > triggerer = = TO_MOBJ )
2021-12-08 19:29:10 +00:00
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2021-12-08 19:29:10 +00:00
continue ;
2021-12-31 10:39:34 +00:00
if ( ! Tag_Find ( & sourceline - > tags , rover - > master - > frontsector - > triggertag ) )
continue ;
2021-12-08 19:29:10 +00:00
if ( ! P_IsMobjTouching3DFloor ( player - > mo , rover , sector ) )
continue ;
// This FOF has the special we're looking for, but are we allowed to touch it?
if ( sector = = player - > mo - > subsector - > sector
2021-12-30 13:16:00 +00:00
| | ( rover - > master - > frontsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2021-12-08 19:29:10 +00:00
return rover - > master - > frontsector ;
}
return NULL ;
}
2021-12-31 10:39:34 +00:00
static sector_t * P_CheckPlayerPolyobjTrigger ( player_t * player , line_t * sourceline )
2021-12-08 19:29:10 +00:00
{
polyobj_t * po ;
sector_t * polysec ;
boolean touching = false ;
boolean inside = false ;
2021-12-31 10:39:34 +00:00
for ( po = player - > mo - > subsector - > polyList ; po ; po = ( polyobj_t * ) ( po - > link . next ) )
2021-12-08 19:29:10 +00:00
{
if ( po - > flags & POF_NOSPECIALS )
continue ;
polysec = po - > lines [ 0 ] - > backsector ;
2021-12-31 10:39:34 +00:00
if ( ! polysec - > triggertag )
2021-12-08 19:29:10 +00:00
continue ;
2021-12-31 10:39:34 +00:00
if ( polysec - > triggerer = = TO_MOBJ )
continue ;
if ( ! Tag_Find ( & sourceline - > tags , polysec - > triggertag ) )
continue ;
2021-12-08 19:29:10 +00:00
2021-12-30 13:16:00 +00:00
touching = ( polysec - > flags & MSF_TRIGGERSPECIAL_TOUCH ) & & P_MobjTouchingPolyobj ( po , player - > mo ) ;
2021-12-08 19:29:10 +00:00
inside = P_MobjInsidePolyobj ( po , player - > mo ) ;
if ( ! ( inside | | touching ) )
continue ;
if ( ! P_IsMobjTouchingPolyobj ( player - > mo , po , polysec ) )
continue ;
return polysec ;
}
return NULL ;
}
2021-12-31 10:39:34 +00:00
static boolean P_CheckPlayerSectorTrigger ( player_t * player , sector_t * sector , line_t * sourceline )
2021-12-08 19:29:10 +00:00
{
2021-12-31 10:39:34 +00:00
if ( ! sector - > triggertag )
2021-12-08 19:29:10 +00:00
return false ;
2021-12-31 10:39:34 +00:00
if ( sector - > triggerer = = TO_MOBJ )
2014-08-27 03:56:30 +00:00
return false ;
2021-12-31 10:39:34 +00:00
if ( ! Tag_Find ( & sourceline - > tags , sector - > triggertag ) )
return false ;
if ( ! ( sector - > flags & MSF_TRIGGERLINE_PLANE ) )
return true ; // Don't require plane touch
2021-12-08 19:29:10 +00:00
return P_IsMobjTouchingSectorPlane ( player - > mo , sector ) ;
}
sector_t * P_FindPlayerTrigger ( player_t * player , line_t * sourceline )
{
sector_t * originalsector ;
sector_t * loopsector ;
msecnode_t * node ;
sector_t * caller ;
if ( ! player - > mo )
return NULL ;
originalsector = player - > mo - > subsector - > sector ;
2021-12-31 10:39:34 +00:00
caller = P_CheckPlayer3DFloorTrigger ( player , originalsector , sourceline ) ; // Handle FOFs first.
2021-12-08 19:29:10 +00:00
if ( caller )
return caller ;
// Allow sector specials to be applied to polyobjects!
2021-12-31 10:39:34 +00:00
caller = P_CheckPlayerPolyobjTrigger ( player , sourceline ) ;
2021-12-08 19:29:10 +00:00
if ( caller )
return caller ;
2021-12-31 10:39:34 +00:00
if ( P_CheckPlayerSectorTrigger ( player , originalsector , sourceline ) )
2021-12-08 19:29:10 +00:00
return originalsector ;
// Iterate through touching_sectorlist for SF_TRIGGERSPECIAL_TOUCH
for ( node = player - > mo - > touching_sectorlist ; node ; node = node - > m_sectorlist_next )
2014-08-27 03:56:30 +00:00
{
2021-12-08 19:29:10 +00:00
loopsector = node - > m_sector ;
if ( loopsector = = originalsector ) // Don't duplicate
2014-08-27 03:56:30 +00:00
continue ;
2021-12-08 19:29:10 +00:00
// Check 3D floors...
2021-12-31 10:39:34 +00:00
caller = P_CheckPlayer3DFloorTrigger ( player , loopsector , sourceline ) ; // Handle FOFs first.
2021-12-08 19:29:10 +00:00
if ( caller )
return caller ;
2014-08-27 03:56:30 +00:00
2021-12-30 13:16:00 +00:00
if ( ! ( loopsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2021-12-08 19:29:10 +00:00
continue ;
2021-12-31 10:39:34 +00:00
if ( P_CheckPlayerSectorTrigger ( player , loopsector , sourceline ) )
2021-12-08 19:29:10 +00:00
return loopsector ;
2014-08-27 03:56:30 +00:00
}
2023-01-28 03:12:29 +00:00
return NULL ;
2014-08-27 03:56:30 +00:00
}
2021-12-09 20:36:46 +00:00
boolean P_IsPlayerValid ( size_t playernum )
{
if ( ! playeringame [ playernum ] )
return false ;
if ( ! players [ playernum ] . mo )
return false ;
if ( players [ playernum ] . mo - > health < = 0 )
return false ;
if ( players [ playernum ] . spectator )
return false ;
return true ;
}
boolean P_CanPlayerTrigger ( size_t playernum )
{
return P_IsPlayerValid ( playernum ) & & ! players [ playernum ] . bot ;
}
2021-12-06 17:50:06 +00:00
/// \todo check continues for proper splitscreen support?
2021-12-31 10:39:34 +00:00
static boolean P_DoAllPlayersTrigger ( mtag_t triggertag )
2021-12-06 17:50:06 +00:00
{
INT32 i ;
2021-12-08 19:29:10 +00:00
line_t dummyline ;
dummyline . tags . count = 1 ;
2021-12-31 10:39:34 +00:00
dummyline . tags . tags = & triggertag ;
2021-12-06 17:50:06 +00:00
for ( i = 0 ; i < MAXPLAYERS ; i + + )
{
2021-12-09 20:36:46 +00:00
if ( ! P_CanPlayerTrigger ( i ) )
2021-12-06 17:50:06 +00:00
continue ;
2021-12-08 19:29:10 +00:00
if ( ! P_FindPlayerTrigger ( & players [ i ] , & dummyline ) )
return false ;
2021-12-06 17:50:06 +00:00
}
return true ;
}
static void P_ProcessEggCapsule ( player_t * player , sector_t * sector )
{
thinker_t * th ;
mobj_t * mo2 ;
INT32 i ;
if ( player - > bot | | sector - > ceilingdata | | sector - > floordata )
return ;
// Find the center of the Eggtrap and release all the pretty animals!
// The chimps are my friends.. heeheeheheehehee..... - LouisJM
for ( th = thlist [ THINK_MOBJ ] . next ; th ! = & thlist [ THINK_MOBJ ] ; th = th - > next )
{
if ( th - > function . acp1 = = ( actionf_p1 ) P_RemoveThinkerDelayed )
continue ;
mo2 = ( mobj_t * ) th ;
if ( mo2 - > type ! = MT_EGGTRAP )
continue ;
P_KillMobj ( mo2 , NULL , player - > mo , 0 ) ;
}
// clear the special so you can't push the button twice.
sector - > special = 0 ;
// Move the button down
EV_DoElevator ( LE_CAPSULE0 , NULL , elevateDown ) ;
// Open the top FOF
EV_DoFloor ( LE_CAPSULE1 , NULL , raiseFloorToNearestFast ) ;
// Open the bottom FOF
EV_DoCeiling ( LE_CAPSULE2 , NULL , lowerToLowestFast ) ;
// Mark all players with the time to exit thingy!
for ( i = 0 ; i < MAXPLAYERS ; i + + )
{
if ( ! playeringame [ i ] )
continue ;
2023-09-20 02:48:01 +00:00
P_DoPlayerExit ( & players [ i ] , true ) ;
2021-12-06 17:50:06 +00:00
}
}
static void P_ProcessSpeedPad ( player_t * player , sector_t * sector , sector_t * roversector , mtag_t sectag )
{
2021-12-29 22:03:37 +00:00
INT32 lineindex = - 1 ;
2021-12-06 17:50:06 +00:00
angle_t lineangle ;
fixed_t linespeed ;
fixed_t sfxnum ;
2021-12-29 22:03:37 +00:00
size_t i ;
2021-12-06 17:50:06 +00:00
if ( player - > powers [ pw_flashing ] ! = 0 & & player - > powers [ pw_flashing ] < TICRATE / 2 )
return ;
2021-12-29 22:03:37 +00:00
// Try for lines facing the sector itself, with tag 0.
for ( i = 0 ; i < sector - > linecount ; i + + )
{
line_t * li = sector - > lines [ i ] ;
if ( li - > frontsector ! = sector )
continue ;
if ( li - > special ! = 4 )
continue ;
if ( ! Tag_Find ( & li - > tags , 0 ) )
continue ;
lineindex = li - lines ;
break ;
}
// Nothing found? Look via tag.
if ( lineindex = = - 1 )
lineindex = Tag_FindLineSpecial ( 4 , sectag ) ;
2021-12-06 17:50:06 +00:00
if ( lineindex = = - 1 )
{
2021-12-30 23:03:24 +00:00
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Speed pad missing line special #4. \n " ) ;
2021-12-06 17:50:06 +00:00
return ;
}
lineangle = R_PointToAngle2 ( lines [ lineindex ] . v1 - > x , lines [ lineindex ] . v1 - > y , lines [ lineindex ] . v2 - > x , lines [ lineindex ] . v2 - > y ) ;
2021-12-29 22:03:37 +00:00
linespeed = lines [ lineindex ] . args [ 0 ] < < FRACBITS ;
2021-12-06 17:50:06 +00:00
if ( linespeed = = 0 )
{
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Speed pad (tag %d) at zero speed. \n " , sectag ) ;
return ;
}
player - > mo - > angle = player - > drawangle = lineangle ;
if ( ! demoplayback | | P_ControlStyle ( player ) = = CS_LMAOGALOG )
P_SetPlayerAngle ( player , player - > mo - > angle ) ;
2021-12-29 22:03:37 +00:00
if ( ! ( lines [ lineindex ] . args [ 1 ] & TMSP_NOTELEPORT ) )
2021-12-06 17:50:06 +00:00
{
P_UnsetThingPosition ( player - > mo ) ;
if ( roversector ) // make FOF speed pads work
{
player - > mo - > x = roversector - > soundorg . x ;
player - > mo - > y = roversector - > soundorg . y ;
}
else
{
player - > mo - > x = sector - > soundorg . x ;
player - > mo - > y = sector - > soundorg . y ;
}
P_SetThingPosition ( player - > mo ) ;
}
P_InstaThrust ( player - > mo , player - > mo - > angle , linespeed ) ;
2021-12-29 22:03:37 +00:00
if ( lines [ lineindex ] . args [ 1 ] & TMSP_FORCESPIN ) // Roll!
2021-12-06 17:50:06 +00:00
{
if ( ! ( player - > pflags & PF_SPINNING ) )
player - > pflags | = PF_SPINNING ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( player - > mo , S_PLAY_ROLL ) ;
2021-12-06 17:50:06 +00:00
}
player - > powers [ pw_flashing ] = TICRATE / 3 ;
2021-12-29 22:03:37 +00:00
sfxnum = lines [ lineindex ] . stringargs [ 0 ] ? get_number ( lines [ lineindex ] . stringargs [ 0 ] ) : sfx_spdpad ;
2021-12-06 17:50:06 +00:00
if ( ! sfxnum )
sfxnum = sfx_spdpad ;
S_StartSound ( player - > mo , sfxnum ) ;
}
2021-12-30 23:03:24 +00:00
static void P_ProcessSpecialStagePit ( player_t * player )
2021-12-06 17:50:06 +00:00
{
if ( ! ( gametyperules & GTR_ALLOWEXIT ) )
return ;
if ( player - > bot )
return ;
2021-12-30 23:03:24 +00:00
if ( ! G_IsSpecialStage ( gamemap ) )
return ;
if ( maptol & TOL_NIGHTS )
return ;
if ( player - > nightstime < = 6 )
return ;
player - > nightstime = 6 ; // Just let P_Ticker take care of the rest.
}
static void P_ProcessExitSector ( player_t * player , mtag_t sectag )
{
INT32 lineindex ;
if ( ! ( gametyperules & GTR_ALLOWEXIT ) )
return ;
if ( player - > bot )
2021-12-06 17:50:06 +00:00
return ;
2022-12-27 15:49:52 +00:00
if ( G_IsSpecialStage ( gamemap ) & & ! ( maptol & TOL_NIGHTS ) )
return ;
2021-12-06 17:50:06 +00:00
// Exit (for FOF exits; others are handled in P_PlayerThink in p_user.c)
P_DoPlayerFinish ( player ) ;
P_SetupSignExit ( player ) ;
if ( ! G_CoopGametype ( ) )
return ;
// Custom exit!
// important: use sectag on next line instead of player->mo->subsector->tag
// this part is different from in P_PlayerThink, this is what was causing
// FOF custom exits not to work.
lineindex = Tag_FindLineSpecial ( 2 , sectag ) ;
if ( lineindex = = - 1 )
{
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Exit sector missing line special #2. \n " ) ;
return ;
}
2021-12-29 22:03:37 +00:00
// Special goodies depending on emeralds collected
if ( ( lines [ lineindex ] . args [ 1 ] & TMEF_EMERALDCHECK ) & & ALL7EMERALDS ( emeralds ) )
nextmapoverride = ( INT16 ) ( udmf ? lines [ lineindex ] . args [ 2 ] : lines [ lineindex ] . frontsector - > ceilingheight > > FRACBITS ) ;
2021-12-06 17:50:06 +00:00
else
2021-12-29 22:03:37 +00:00
nextmapoverride = ( INT16 ) ( udmf ? lines [ lineindex ] . args [ 0 ] : lines [ lineindex ] . frontsector - > floorheight > > FRACBITS ) ;
2021-12-06 17:50:06 +00:00
2021-12-29 22:03:37 +00:00
if ( lines [ lineindex ] . args [ 1 ] & TMEF_SKIPTALLY )
2021-12-06 17:50:06 +00:00
skipstats = 1 ;
}
static void P_ProcessTeamBase ( player_t * player , boolean redteam )
{
mobj_t * mo ;
if ( ! ( gametyperules & GTR_TEAMFLAGS ) )
return ;
if ( ! P_IsObjectOnGround ( player - > mo ) )
return ;
if ( player - > ctfteam ! = ( redteam ? 1 : 2 ) )
return ;
if ( ! ( player - > gotflag & ( redteam ? GF_BLUEFLAG : GF_REDFLAG ) ) )
return ;
// Make sure the team still has their own
// flag at their base so they can score.
2023-02-25 14:43:51 +00:00
if ( ! P_IsFlagAtBase ( redteam ? MT_REDFLAG : MT_BLUEFLAG ) )
2021-12-06 17:50:06 +00:00
return ;
HU_SetCEchoFlags ( V_AUTOFADEOUT | V_ALLOWLOWERCASE ) ;
HU_SetCEchoDuration ( 5 ) ;
HU_DoCEcho ( va ( M_GetText ( " %s%s \200 \\ CAPTURED THE %s%s FLAG \200 . \\ \\ \\ \\ " ) , redteam ? " \205 " : " \204 " , player_names [ player - players ] , redteam ? " \204 " : " \205 " , redteam ? " BLUE " : " RED " ) ) ;
if ( splitscreen | | players [ consoleplayer ] . ctfteam = = ( redteam ? 1 : 2 ) )
S_StartSound ( NULL , sfx_flgcap ) ;
else if ( players [ consoleplayer ] . ctfteam = = ( redteam ? 2 : 1 ) )
S_StartSound ( NULL , sfx_lose ) ;
mo = P_SpawnMobj ( player - > mo - > x , player - > mo - > y , player - > mo - > z , redteam ? MT_BLUEFLAG : MT_REDFLAG ) ;
player - > gotflag & = ~ ( redteam ? GF_BLUEFLAG : GF_REDFLAG ) ;
mo - > flags & = ~ MF_SPECIAL ;
mo - > fuse = TICRATE ;
mo - > spawnpoint = redteam ? bflagpoint : rflagpoint ;
mo - > flags2 | = MF2_JUSTATTACKED ;
if ( redteam )
redscore + = 1 ;
else
bluescore + = 1 ;
P_AddPlayerScore ( player , 250 ) ;
}
2021-12-30 23:03:24 +00:00
static void P_ProcessZoomTube ( player_t * player , mtag_t sectag , boolean end )
2021-12-06 17:50:06 +00:00
{
INT32 sequence ;
fixed_t speed ;
INT32 lineindex ;
mobj_t * waypoint = NULL ;
angle_t an ;
if ( player - > mo - > tracer & & player - > mo - > tracer - > type = = MT_TUBEWAYPOINT & & player - > powers [ pw_carry ] = = CR_ZOOMTUBE )
return ;
if ( player - > powers [ pw_ignorelatch ] & ( 1 < < 15 ) )
return ;
// Find line #3 tagged to this sector
lineindex = Tag_FindLineSpecial ( 3 , sectag ) ;
if ( lineindex = = - 1 )
{
2021-12-30 23:03:24 +00:00
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Zoom tube missing line special #3. \n " ) ;
2021-12-06 17:50:06 +00:00
return ;
}
// Grab speed and sequence values
2021-12-29 22:03:37 +00:00
speed = abs ( lines [ lineindex ] . args [ 0 ] ) < < ( FRACBITS - 3 ) ;
2021-12-06 17:50:06 +00:00
if ( end )
speed * = - 1 ;
2021-12-29 22:03:37 +00:00
sequence = abs ( lines [ lineindex ] . args [ 1 ] ) ;
2021-12-06 17:50:06 +00:00
if ( speed = = 0 )
{
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Waypoint sequence %d at zero speed. \n " , sequence ) ;
return ;
}
waypoint = end ? P_GetLastWaypoint ( sequence ) : P_GetFirstWaypoint ( sequence ) ;
if ( ! waypoint )
{
CONS_Debug ( DBG_GAMELOGIC , " ERROR: %s WAYPOINT IN SEQUENCE %d NOT FOUND. \n " , end ? " LAST " : " FIRST " , sequence ) ;
return ;
}
CONS_Debug ( DBG_GAMELOGIC , " Waypoint %d found in sequence %d - speed = %d \n " , waypoint - > health , sequence , speed ) ;
an = R_PointToAngle2 ( player - > mo - > x , player - > mo - > y , waypoint - > x , waypoint - > y ) - player - > mo - > angle ;
2021-12-29 22:03:37 +00:00
if ( an > ANGLE_90 & & an < ANGLE_270 & & ! ( lines [ lineindex ] . args [ 2 ] ) )
2021-12-06 17:50:06 +00:00
return ; // behind back
P_SetTarget ( & player - > mo - > tracer , waypoint ) ;
player - > powers [ pw_carry ] = CR_ZOOMTUBE ;
player - > speed = speed ;
player - > pflags | = PF_SPINNING ;
player - > pflags & = ~ ( PF_JUMPED | PF_NOJUMPDAMAGE | PF_GLIDING | PF_BOUNCING | PF_SLIDING | PF_CANCARRY ) ;
player - > climbing = 0 ;
if ( player - > mo - > state - states ! = S_PLAY_ROLL )
{
2023-12-23 12:45:16 +00:00
P_SetMobjState ( player - > mo , S_PLAY_ROLL ) ;
2021-12-06 17:50:06 +00:00
S_StartSound ( player - > mo , sfx_spin ) ;
}
}
static void P_ProcessFinishLine ( player_t * player )
{
if ( ( gametyperules & ( GTR_RACE | GTR_LIVES ) ) ! = GTR_RACE )
return ;
if ( player - > exiting )
return ;
if ( player - > starpostnum = = numstarposts ) // Must have touched all the starposts
{
player - > laps + + ;
if ( player - > powers [ pw_carry ] = = CR_NIGHTSMODE )
player - > drillmeter + = 48 * 20 ;
if ( player - > laps > = ( UINT8 ) cv_numlaps . value )
CONS_Printf ( M_GetText ( " %s has finished the race. \n " ) , player_names [ player - players ] ) ;
else if ( player - > laps = = ( UINT8 ) cv_numlaps . value - 1 )
CONS_Printf ( M_GetText ( " %s started the \205 final lap \200 ! \n " ) , player_names [ player - players ] ) ;
else
CONS_Printf ( M_GetText ( " %s started lap %u \n " ) , player_names [ player - players ] , ( UINT32 ) player - > laps + 1 ) ;
// Reset starposts (checkpoints) info
player - > starpostscale = player - > starpostangle = player - > starposttime = player - > starpostnum = 0 ;
player - > starpostx = player - > starposty = player - > starpostz = 0 ;
P_ResetStarposts ( ) ;
// Play the starpost sound for 'consistency'
S_StartSound ( player - > mo , sfx_strpst ) ;
}
else if ( player - > starpostnum )
{
// blatant reuse of a variable that's normally unused in circuit
if ( ! player - > tossdelay )
S_StartSound ( player - > mo , sfx_lose ) ;
player - > tossdelay = 3 ;
}
if ( player - > laps > = ( unsigned ) cv_numlaps . value )
{
if ( P_IsLocalPlayer ( player ) )
{
HU_SetCEchoFlags ( 0 ) ;
HU_SetCEchoDuration ( 5 ) ;
HU_DoCEcho ( " FINISHED! " ) ;
}
2023-09-20 02:48:01 +00:00
P_DoPlayerExit ( player , true ) ;
2021-12-06 17:50:06 +00:00
}
}
2021-12-30 23:03:24 +00:00
static void P_ProcessRopeHang ( player_t * player , mtag_t sectag )
2021-12-06 17:50:06 +00:00
{
INT32 sequence ;
fixed_t speed ;
INT32 lineindex ;
mobj_t * waypointmid = NULL ;
mobj_t * waypointhigh = NULL ;
mobj_t * waypointlow = NULL ;
mobj_t * closest = NULL ;
vector3_t p , line [ 2 ] , resulthigh , resultlow ;
if ( player - > mo - > tracer & & player - > mo - > tracer - > type = = MT_TUBEWAYPOINT & & player - > powers [ pw_carry ] = = CR_ROPEHANG )
return ;
if ( player - > powers [ pw_ignorelatch ] & ( 1 < < 15 ) )
return ;
if ( player - > mo - > momz > 0 )
return ;
if ( player - > cmd . buttons & BT_SPIN )
return ;
if ( ! ( player - > pflags & PF_SLIDING ) & & player - > mo - > state = = & states [ player - > mo - > info - > painstate ] )
return ;
if ( player - > exiting )
return ;
//initialize resulthigh and resultlow with 0
memset ( & resultlow , 0x00 , sizeof ( resultlow ) ) ;
memset ( & resulthigh , 0x00 , sizeof ( resulthigh ) ) ;
// Find line #11 tagged to this sector
lineindex = Tag_FindLineSpecial ( 11 , sectag ) ;
if ( lineindex = = - 1 )
{
2021-12-30 23:03:24 +00:00
CONS_Debug ( DBG_GAMELOGIC , " ERROR: Rope hang missing line special #11. \n " ) ;
2021-12-06 17:50:06 +00:00
return ;
}
// Grab speed and sequence values
2021-12-29 22:03:37 +00:00
speed = abs ( lines [ lineindex ] . args [ 0 ] ) < < ( FRACBITS - 3 ) ;
sequence = abs ( lines [ lineindex ] . args [ 1 ] ) ;
2021-12-06 17:50:06 +00:00
// Find the closest waypoint
// Find the preceding waypoint
// Find the proceeding waypoint
// Determine the closest spot on the line between the three waypoints
// Put player at that location.
waypointmid = P_GetClosestWaypoint ( sequence , player - > mo ) ;
if ( ! waypointmid )
{
CONS_Debug ( DBG_GAMELOGIC , " ERROR: WAYPOINT(S) IN SEQUENCE %d NOT FOUND. \n " , sequence ) ;
return ;
}
waypointlow = P_GetPreviousWaypoint ( waypointmid , true ) ;
waypointhigh = P_GetNextWaypoint ( waypointmid , true ) ;
CONS_Debug ( DBG_GAMELOGIC , " WaypointMid: %d; WaypointLow: %d; WaypointHigh: %d \n " ,
waypointmid - > health , waypointlow ? waypointlow - > health : - 1 , waypointhigh ? waypointhigh - > health : - 1 ) ;
// Now we have three waypoints... the closest one we're near, and the one that comes before, and after.
// Next, we need to find the closest point on the line between each set, and determine which one we're
// closest to.
p . x = player - > mo - > x ;
p . y = player - > mo - > y ;
p . z = player - > mo - > z ;
// Waypointmid and Waypointlow:
if ( waypointlow )
{
line [ 0 ] . x = waypointmid - > x ;
line [ 0 ] . y = waypointmid - > y ;
line [ 0 ] . z = waypointmid - > z ;
line [ 1 ] . x = waypointlow - > x ;
line [ 1 ] . y = waypointlow - > y ;
line [ 1 ] . z = waypointlow - > z ;
P_ClosestPointOnLine3D ( & p , line , & resultlow ) ;
}
// Waypointmid and Waypointhigh:
if ( waypointhigh )
{
line [ 0 ] . x = waypointmid - > x ;
line [ 0 ] . y = waypointmid - > y ;
line [ 0 ] . z = waypointmid - > z ;
line [ 1 ] . x = waypointhigh - > x ;
line [ 1 ] . y = waypointhigh - > y ;
line [ 1 ] . z = waypointhigh - > z ;
P_ClosestPointOnLine3D ( & p , line , & resulthigh ) ;
}
// 3D support now available. Disregard the previous notice here. -Red
P_UnsetThingPosition ( player - > mo ) ;
P_ResetPlayer ( player ) ;
player - > mo - > momx = player - > mo - > momy = player - > mo - > momz = 0 ;
2021-12-29 22:03:37 +00:00
if ( lines [ lineindex ] . args [ 2 ] ) // Don't wrap
2021-12-06 17:50:06 +00:00
{
mobj_t * highest = P_GetLastWaypoint ( sequence ) ;
highest - > flags | = MF_SLIDEME ;
}
// Changing the conditions on these ifs to fix issues with snapping to the wrong spot -Red
2021-12-29 22:03:37 +00:00
if ( ( lines [ lineindex ] . args [ 2 ] ) & & waypointmid - > health = = 0 )
2021-12-06 17:50:06 +00:00
{
closest = waypointhigh ;
player - > mo - > x = resulthigh . x ;
player - > mo - > y = resulthigh . y ;
player - > mo - > z = resulthigh . z - P_GetPlayerHeight ( player ) ;
}
2021-12-29 22:03:37 +00:00
else if ( ( lines [ lineindex ] . args [ 2 ] ) & & waypointmid - > health = = numwaypoints [ sequence ] - 1 )
2021-12-06 17:50:06 +00:00
{
closest = waypointmid ;
player - > mo - > x = resultlow . x ;
player - > mo - > y = resultlow . y ;
player - > mo - > z = resultlow . z - P_GetPlayerHeight ( player ) ;
}
else
{
if ( P_AproxDistance ( P_AproxDistance ( player - > mo - > x - resultlow . x , player - > mo - > y - resultlow . y ) ,
player - > mo - > z - resultlow . z ) < P_AproxDistance ( P_AproxDistance ( player - > mo - > x - resulthigh . x ,
player - > mo - > y - resulthigh . y ) , player - > mo - > z - resulthigh . z ) )
{
// Line between Mid and Low is closer
closest = waypointmid ;
player - > mo - > x = resultlow . x ;
player - > mo - > y = resultlow . y ;
player - > mo - > z = resultlow . z - P_GetPlayerHeight ( player ) ;
}
else
{
// Line between Mid and High is closer
closest = waypointhigh ;
player - > mo - > x = resulthigh . x ;
player - > mo - > y = resulthigh . y ;
player - > mo - > z = resulthigh . z - P_GetPlayerHeight ( player ) ;
}
}
P_SetTarget ( & player - > mo - > tracer , closest ) ;
player - > powers [ pw_carry ] = CR_ROPEHANG ;
2021-12-29 22:03:37 +00:00
player - > speed = speed ;
2021-12-06 17:50:06 +00:00
S_StartSound ( player - > mo , sfx_s3k4a ) ;
player - > pflags & = ~ ( PF_JUMPED | PF_NOJUMPDAMAGE | PF_GLIDING | PF_BOUNCING | PF_SLIDING | PF_CANCARRY ) ;
player - > climbing = 0 ;
P_SetThingPosition ( player - > mo ) ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( player - > mo , S_PLAY_RIDE ) ;
2021-12-06 17:50:06 +00:00
}
2021-12-31 10:39:34 +00:00
static boolean P_SectorHasSpecial ( sector_t * sec )
2014-03-15 16:59:03 +00:00
{
2021-12-31 10:39:34 +00:00
if ( sec - > specialflags )
return true ;
2021-12-07 20:39:45 +00:00
2021-12-31 10:39:34 +00:00
if ( sec - > damagetype ! = SD_NONE )
return true ;
2014-03-15 16:59:03 +00:00
2021-12-31 10:39:34 +00:00
if ( sec - > triggertag )
return true ;
2014-03-15 16:59:03 +00:00
2021-12-31 10:39:34 +00:00
if ( sec - > special )
return true ;
2014-03-15 16:59:03 +00:00
2021-12-31 10:39:34 +00:00
return false ;
}
static void P_EvaluateSpecialFlags ( player_t * player , sector_t * sector , sector_t * roversector , boolean isTouching )
{
mtag_t sectag = Tag_FGet ( & sector - > tags ) ;
2021-12-07 20:39:45 +00:00
2021-12-30 23:03:24 +00:00
if ( sector - > specialflags & SSF_OUTERSPACE )
{
if ( ! ( player - > powers [ pw_shield ] & SH_PROTECTWATER ) & & ! player - > powers [ pw_spacetime ] )
player - > powers [ pw_spacetime ] = spacetimetics + 1 ;
}
if ( sector - > specialflags & SSF_WINDCURRENT )
player - > onconveyor = 2 ;
if ( sector - > specialflags & SSF_CONVEYOR )
player - > onconveyor = 4 ;
if ( ( sector - > specialflags & SSF_SPEEDPAD ) & & isTouching )
P_ProcessSpeedPad ( player , sector , roversector , sectag ) ;
if ( sector - > specialflags & SSF_STARPOSTACTIVATOR )
{
mobj_t * post = P_GetObjectTypeInSectorNum ( MT_STARPOST , sector - sectors ) ;
if ( post )
P_TouchStarPost ( post , player , false ) ;
}
if ( sector - > specialflags & SSF_EXIT )
P_ProcessExitSector ( player , sectag ) ;
if ( ( sector - > specialflags & SSF_SPECIALSTAGEPIT ) & & isTouching )
P_ProcessSpecialStagePit ( player ) ;
if ( ( sector - > specialflags & SSF_REDTEAMBASE ) & & isTouching )
P_ProcessTeamBase ( player , true ) ;
if ( ( sector - > specialflags & SSF_BLUETEAMBASE ) & & isTouching )
P_ProcessTeamBase ( player , false ) ;
if ( sector - > specialflags & SSF_FAN )
{
player - > mo - > momz + = mobjinfo [ MT_FAN ] . mass / 4 ;
if ( player - > mo - > momz > mobjinfo [ MT_FAN ] . mass )
player - > mo - > momz = mobjinfo [ MT_FAN ] . mass ;
2022-09-16 20:56:26 +00:00
if ( ! player - > powers [ pw_carry ] )
{
P_ResetPlayer ( player ) ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( player - > mo , S_PLAY_FALL ) ;
2022-09-16 20:56:26 +00:00
P_SetTarget ( & player - > mo - > tracer , player - > mo ) ;
player - > powers [ pw_carry ] = CR_FAN ;
}
2021-12-30 23:03:24 +00:00
}
if ( sector - > specialflags & SSF_SUPERTRANSFORM )
{
if ( player - > mo - > health > 0 & & ! player - > bot & & ( player - > charflags & SF_SUPER ) & & ! player - > powers [ pw_super ] & & ALL7EMERALDS ( emeralds ) )
P_DoSuperTransformation ( player , true ) ;
}
if ( ( sector - > specialflags & SSF_FORCESPIN ) & & isTouching )
{
if ( ! ( player - > pflags & PF_SPINNING ) )
{
player - > pflags | = PF_SPINNING ;
2023-12-23 12:45:16 +00:00
P_SetMobjState ( player - > mo , S_PLAY_ROLL ) ;
2021-12-30 23:03:24 +00:00
S_StartAttackSound ( player - > mo , sfx_spin ) ;
if ( abs ( player - > rmomx ) < FixedMul ( 5 * FRACUNIT , player - > mo - > scale )
& & abs ( player - > rmomy ) < FixedMul ( 5 * FRACUNIT , player - > mo - > scale ) )
P_InstaThrust ( player - > mo , player - > mo - > angle , FixedMul ( 10 * FRACUNIT , player - > mo - > scale ) ) ;
}
}
if ( sector - > specialflags & SSF_ZOOMTUBESTART )
P_ProcessZoomTube ( player , sectag , false ) ;
if ( sector - > specialflags & SSF_ZOOMTUBEEND )
P_ProcessZoomTube ( player , sectag , true ) ;
if ( sector - > specialflags & SSF_FINISHLINE )
P_ProcessFinishLine ( player ) ;
if ( ( sector - > specialflags & SSF_ROPEHANG ) & & isTouching )
P_ProcessRopeHang ( player , sectag ) ;
2021-12-31 10:39:34 +00:00
}
2021-12-30 23:03:24 +00:00
2021-12-31 10:39:34 +00:00
static void P_EvaluateDamageType ( player_t * player , sector_t * sector , boolean isTouching )
{
2021-12-31 07:53:00 +00:00
switch ( sector - > damagetype )
2014-03-15 16:59:03 +00:00
{
2021-12-31 07:53:00 +00:00
case SD_GENERIC :
2021-12-07 20:39:45 +00:00
if ( isTouching )
2015-02-13 16:15:58 +00:00
P_DamageMobj ( player - > mo , NULL , NULL , 1 , 0 ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_WATER :
2021-12-07 20:39:45 +00:00
if ( isTouching & & ( player - > powers [ pw_underwater ] | | player - > powers [ pw_carry ] = = CR_NIGHTSMODE ) )
2015-02-13 16:15:58 +00:00
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_WATER ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_FIRE :
2021-12-31 11:15:01 +00:00
case SD_LAVA :
2021-12-07 20:39:45 +00:00
if ( isTouching )
2015-02-13 16:15:58 +00:00
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_FIRE ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_ELECTRIC :
2021-12-07 20:39:45 +00:00
if ( isTouching )
2015-02-13 16:15:58 +00:00
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_ELECTRIC ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_SPIKE :
2021-12-07 20:39:45 +00:00
if ( isTouching )
2020-04-17 20:54:35 +00:00
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_SPIKE ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_DEATHPITTILT :
case SD_DEATHPITNOTILT :
2021-12-07 20:39:45 +00:00
if ( ! isTouching )
break ;
if ( player - > quittime )
G_MovePlayerToSpawnOrStarpost ( player - players ) ;
else
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_DEATHPIT ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_INSTAKILL :
2020-01-22 02:20:27 +00:00
if ( player - > quittime )
G_MovePlayerToSpawnOrStarpost ( player - players ) ;
else
P_DamageMobj ( player - > mo , NULL , NULL , 1 , DMG_INSTAKILL ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-12-31 07:53:00 +00:00
case SD_SPECIALSTAGE :
if ( ! isTouching )
break ;
if ( player - > exiting | | player - > bot ) // Don't do anything for bots or players who have just finished
break ;
if ( ! ( player - > powers [ pw_shield ] | | player - > spheres > 0 ) ) // Don't do anything if no shield or spheres anyway
break ;
P_SpecialStageDamage ( player , NULL , NULL ) ;
break ;
default :
break ;
}
2021-12-31 10:39:34 +00:00
}
static void P_EvaluateLinedefExecutorTrigger ( player_t * player , sector_t * sector , boolean isTouching )
{
if ( player - > bot )
return ;
if ( ! sector - > triggertag )
return ;
if ( sector - > triggerer = = TO_MOBJ )
return ;
else if ( sector - > triggerer = = TO_ALLPLAYERS & & ! P_DoAllPlayersTrigger ( sector - > triggertag ) )
return ;
if ( ( sector - > flags & MSF_TRIGGERLINE_PLANE ) & & ! isTouching )
return ;
2021-12-31 07:53:00 +00:00
2021-12-31 10:39:34 +00:00
P_LinedefExecute ( sector - > triggertag , player - > mo , sector ) ;
}
2021-12-31 07:53:00 +00:00
2021-12-31 10:39:34 +00:00
static void P_EvaluateOldSectorSpecial ( player_t * player , sector_t * sector , sector_t * roversector , boolean isTouching )
{
switch ( GETSECSPECIAL ( sector - > special , 1 ) )
2021-12-31 07:53:00 +00:00
{
2014-03-15 16:59:03 +00:00
case 9 : // Ring Drainer (Floor Touch)
2021-12-07 20:39:45 +00:00
if ( ! isTouching )
break ;
/* FALLTHRU */
2014-03-15 16:59:03 +00:00
case 10 : // Ring Drainer (No Floor Touch)
2015-08-15 20:07:16 +00:00
if ( leveltime % ( TICRATE / 2 ) = = 0 & & player - > rings > 0 )
2014-03-15 16:59:03 +00:00
{
2015-08-15 20:07:16 +00:00
player - > rings - - ;
2020-12-11 18:34:30 +00:00
S_StartSound ( player - > mo , sfx_antiri ) ;
2014-03-15 16:59:03 +00:00
}
break ;
}
2021-12-31 10:39:34 +00:00
switch ( GETSECSPECIAL ( sector - > special , 2 ) )
2014-03-15 16:59:03 +00:00
{
case 9 : // Egg trap capsule
2021-12-31 10:39:34 +00:00
if ( roversector )
2021-12-07 20:39:45 +00:00
P_ProcessEggCapsule ( player , sector ) ;
2014-03-15 16:59:03 +00:00
break ;
}
}
2021-12-31 10:39:34 +00:00
/** Applies a sector special to a player.
*
* \ param player Player in the sector .
* \ param sector Sector with the special .
* \ param roversector If ! NULL , sector is actually an FOF ; otherwise , sector
* is being physically contacted by the player .
* \ sa P_PlayerInSpecialSector , P_PlayerOnSpecial3DFloor
*/
void P_ProcessSpecialSector ( player_t * player , sector_t * sector , sector_t * roversector )
{
boolean isTouching ;
if ( ! P_SectorHasSpecial ( sector ) )
return ;
// Ignore spectators
if ( player - > spectator )
return ;
// Ignore dead players.
// If this strange phenomenon could be potentially used in levels,
// TODO: modify this to accommodate for it.
if ( player - > playerstate ! = PST_LIVE )
return ;
isTouching = roversector | | P_IsMobjTouchingSectorPlane ( player - > mo , sector ) ;
P_EvaluateSpecialFlags ( player , sector , roversector , isTouching ) ;
P_EvaluateDamageType ( player , sector , isTouching ) ;
P_EvaluateLinedefExecutorTrigger ( player , sector , isTouching ) ;
if ( ! udmf )
P_EvaluateOldSectorSpecial ( player , sector , roversector , isTouching ) ;
}
2021-12-07 18:20:26 +00:00
# define TELEPORTED(mo) (mo->subsector->sector != originalsector)
2016-10-31 00:02:38 +00:00
2014-03-15 16:59:03 +00:00
/** Checks if a player is standing on or is inside a 3D floor (e.g. water) and
* applies any specials .
*
* \ param player Player to check .
* \ sa P_ThingOnSpecial3DFloor , P_PlayerInSpecialSector
*/
static void P_PlayerOnSpecial3DFloor ( player_t * player , sector_t * sector )
{
2016-10-31 14:03:53 +00:00
sector_t * originalsector = player - > mo - > subsector - > sector ;
2014-03-15 16:59:03 +00:00
ffloor_t * rover ;
for ( rover = sector - > ffloors ; rover ; rover = rover - > next )
{
2021-12-31 10:39:34 +00:00
if ( ! P_SectorHasSpecial ( rover - > master - > frontsector ) )
2014-03-15 16:59:03 +00:00
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2014-03-15 16:59:03 +00:00
continue ;
2021-12-06 18:33:36 +00:00
if ( ! P_IsMobjTouching3DFloor ( player - > mo , rover , sector ) )
continue ;
2014-03-15 16:59:03 +00:00
// This FOF has the special we're looking for, but are we allowed to touch it?
if ( sector = = player - > mo - > subsector - > sector
2021-12-30 13:16:00 +00:00
| | ( rover - > master - > frontsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2016-10-31 00:02:38 +00:00
{
2014-03-15 16:59:03 +00:00
P_ProcessSpecialSector ( player , rover - > master - > frontsector , sector ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2016-10-31 00:02:38 +00:00
}
2014-03-15 16:59:03 +00:00
}
2021-12-06 18:33:36 +00:00
}
2014-03-15 16:59:03 +00:00
2021-12-09 20:15:11 +00:00
static void P_PlayerOnSpecialPolyobj ( player_t * player )
2021-12-06 18:33:36 +00:00
{
sector_t * originalsector = player - > mo - > subsector - > sector ;
polyobj_t * po ;
sector_t * polysec ;
boolean touching = false ;
boolean inside = false ;
2014-03-15 16:59:03 +00:00
2021-12-06 18:33:36 +00:00
for ( po = player - > mo - > subsector - > polyList ; po ; po = ( polyobj_t * ) ( po - > link . next ) )
{
if ( po - > flags & POF_NOSPECIALS )
continue ;
2014-03-15 16:59:03 +00:00
2021-12-06 18:33:36 +00:00
polysec = po - > lines [ 0 ] - > backsector ;
2014-03-15 16:59:03 +00:00
2021-12-31 10:39:34 +00:00
if ( ! P_SectorHasSpecial ( polysec ) )
2021-12-06 18:33:36 +00:00
continue ;
2014-03-15 16:59:03 +00:00
2021-12-30 13:16:00 +00:00
touching = ( polysec - > flags & MSF_TRIGGERSPECIAL_TOUCH ) & & P_MobjTouchingPolyobj ( po , player - > mo ) ;
2021-12-06 18:33:36 +00:00
inside = P_MobjInsidePolyobj ( po , player - > mo ) ;
2014-03-15 16:59:03 +00:00
2021-12-06 18:33:36 +00:00
if ( ! ( inside | | touching ) )
continue ;
2014-03-15 16:59:03 +00:00
2021-12-06 18:33:36 +00:00
if ( ! P_IsMobjTouchingPolyobj ( player - > mo , po , polysec ) )
continue ;
2014-03-15 16:59:03 +00:00
2021-12-09 20:15:11 +00:00
P_ProcessSpecialSector ( player , polysec , originalsector ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2014-03-15 16:59:03 +00:00
}
}
/** Checks if the player is in a special sector or FOF and apply any specials.
*
* \ param player Player to check .
* \ sa P_PlayerOnSpecial3DFloor , P_ProcessSpecialSector
*/
void P_PlayerInSpecialSector ( player_t * player )
{
2016-10-31 00:02:38 +00:00
sector_t * originalsector ;
sector_t * loopsector ;
2014-03-15 16:59:03 +00:00
msecnode_t * node ;
if ( ! player - > mo )
return ;
2016-10-31 00:02:38 +00:00
originalsector = player - > mo - > subsector - > sector ;
2014-03-15 16:59:03 +00:00
2016-10-31 14:03:53 +00:00
P_PlayerOnSpecial3DFloor ( player , originalsector ) ; // Handle FOFs first.
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2014-03-15 16:59:03 +00:00
2021-12-06 18:33:36 +00:00
// Allow sector specials to be applied to polyobjects!
2021-12-09 20:15:11 +00:00
P_PlayerOnSpecialPolyobj ( player ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2021-12-06 18:33:36 +00:00
2021-12-07 20:39:45 +00:00
P_ProcessSpecialSector ( player , originalsector , NULL ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2014-03-15 16:59:03 +00:00
2016-10-31 00:02:38 +00:00
// Iterate through touching_sectorlist for SF_TRIGGERSPECIAL_TOUCH
2016-06-09 13:56:24 +00:00
for ( node = player - > mo - > touching_sectorlist ; node ; node = node - > m_sectorlist_next )
2014-03-15 16:59:03 +00:00
{
2016-10-31 00:02:38 +00:00
loopsector = node - > m_sector ;
2014-03-15 16:59:03 +00:00
2016-10-31 00:02:38 +00:00
if ( loopsector = = originalsector ) // Don't duplicate
2014-03-15 16:59:03 +00:00
continue ;
// Check 3D floors...
2016-10-31 14:03:53 +00:00
P_PlayerOnSpecial3DFloor ( player , loopsector ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2014-03-15 16:59:03 +00:00
2021-12-30 13:16:00 +00:00
if ( ! ( loopsector - > flags & MSF_TRIGGERSPECIAL_TOUCH ) )
2016-10-31 00:02:38 +00:00
continue ;
2021-12-07 20:39:45 +00:00
P_ProcessSpecialSector ( player , loopsector , NULL ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( player - > mo ) return ;
2014-03-15 16:59:03 +00:00
}
}
2021-12-31 10:39:34 +00:00
static void P_CheckMobj3DFloorTrigger ( mobj_t * mo , sector_t * sec )
2021-12-07 18:20:26 +00:00
{
sector_t * originalsector = mo - > subsector - > sector ;
ffloor_t * rover ;
2021-12-31 10:39:34 +00:00
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
2021-12-07 18:20:26 +00:00
{
2021-12-31 10:39:34 +00:00
if ( ! rover - > master - > frontsector - > triggertag )
continue ;
if ( rover - > master - > frontsector - > triggerer ! = TO_MOBJ )
2021-12-07 18:20:26 +00:00
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) )
2021-12-07 18:20:26 +00:00
continue ;
2021-12-31 10:39:34 +00:00
if ( ! P_IsMobjTouching3DFloor ( mo , rover , sec ) )
2021-12-07 18:20:26 +00:00
continue ;
2021-12-31 10:39:34 +00:00
P_LinedefExecute ( rover - > master - > frontsector - > triggertag , mo , rover - > master - > frontsector ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( mo ) return ;
}
}
static void P_CheckMobjPolyobjTrigger ( mobj_t * mo )
{
sector_t * originalsector = mo - > subsector - > sector ;
polyobj_t * po ;
sector_t * polysec ;
boolean touching = false ;
boolean inside = false ;
for ( po = mo - > subsector - > polyList ; po ; po = ( polyobj_t * ) ( po - > link . next ) )
{
if ( po - > flags & POF_NOSPECIALS )
continue ;
polysec = po - > lines [ 0 ] - > backsector ;
2021-12-31 10:39:34 +00:00
if ( ! polysec - > triggertag )
continue ;
if ( polysec - > triggerer ! = TO_MOBJ )
2021-12-07 18:20:26 +00:00
continue ;
2021-12-30 13:16:00 +00:00
touching = ( polysec - > flags & MSF_TRIGGERSPECIAL_TOUCH ) & & P_MobjTouchingPolyobj ( po , mo ) ;
2021-12-07 18:20:26 +00:00
inside = P_MobjInsidePolyobj ( po , mo ) ;
if ( ! ( inside | | touching ) )
continue ;
if ( ! P_IsMobjTouchingPolyobj ( mo , po , polysec ) )
continue ;
2021-12-31 10:39:34 +00:00
P_LinedefExecute ( polysec - > triggertag , mo , polysec ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( mo ) return ;
}
}
2021-12-31 10:39:34 +00:00
static void P_CheckMobjSectorTrigger ( mobj_t * mo , sector_t * sec )
2021-12-07 18:20:26 +00:00
{
2021-12-31 10:39:34 +00:00
if ( ! sec - > triggertag )
return ;
2021-12-07 18:20:26 +00:00
2021-12-31 10:39:34 +00:00
if ( sec - > triggerer ! = TO_MOBJ )
return ;
2021-12-07 18:20:26 +00:00
2021-12-31 10:39:34 +00:00
if ( ( sec - > flags & MSF_TRIGGERLINE_PLANE ) & & ! P_IsMobjTouchingSectorPlane ( mo , sec ) )
return ;
2021-12-07 18:20:26 +00:00
2021-12-31 10:39:34 +00:00
P_LinedefExecute ( sec - > triggertag , mo , sec ) ;
2021-12-07 18:20:26 +00:00
}
2021-12-31 10:39:34 +00:00
void P_CheckMobjTrigger ( mobj_t * mobj , boolean pushable )
2021-12-07 18:20:26 +00:00
{
sector_t * originalsector ;
if ( ! mobj - > subsector )
return ;
originalsector = mobj - > subsector - > sector ;
2021-12-31 10:39:34 +00:00
if ( ! pushable & & ! ( originalsector - > flags & MSF_TRIGGERLINE_MOBJ ) )
2021-12-07 18:20:26 +00:00
return ;
2021-12-31 10:39:34 +00:00
P_CheckMobj3DFloorTrigger ( mobj , originalsector ) ;
2021-12-07 18:20:26 +00:00
if TELEPORTED ( mobj ) return ;
P_CheckMobjPolyobjTrigger ( mobj ) ;
2021-12-31 10:39:34 +00:00
if TELEPORTED ( mobj ) return ;
P_CheckMobjSectorTrigger ( mobj , originalsector ) ;
2021-12-07 18:20:26 +00:00
}
2016-10-31 00:02:38 +00:00
# undef TELEPORTED
2014-03-15 16:59:03 +00:00
/** Animate planes, scroll walls, etc. and keeps track of level timelimit and exits if time is up.
*
2015-10-11 20:01:04 +00:00
* \ sa P_CheckTimeLimit , P_CheckPointLimit
2014-03-15 16:59:03 +00:00
*/
void P_UpdateSpecials ( void )
{
// LEVEL TIMER
2015-10-11 20:01:04 +00:00
P_CheckTimeLimit ( ) ;
2014-03-15 16:59:03 +00:00
// POINT LIMIT
P_CheckPointLimit ( ) ;
// ANIMATE TEXTURES
2023-08-01 20:19:58 +00:00
for ( anim_t * anim = anims ; anim < lastanim ; anim + + )
2014-03-15 16:59:03 +00:00
{
2023-08-01 20:19:58 +00:00
for ( INT32 i = 0 ; i < anim - > numpics ; i + + )
2014-03-15 16:59:03 +00:00
{
2023-08-01 20:19:58 +00:00
INT32 pic = anim - > basepic + ( ( leveltime / anim - > speed + i ) % anim - > numpics ) ;
2024-01-15 05:40:43 +00:00
texturetranslation [ anim - > basepic + i ] = pic ;
2014-03-15 16:59:03 +00:00
}
}
}
2020-04-24 21:29:41 +00:00
//
2023-08-01 20:19:58 +00:00
// 3D floors
2020-04-24 21:29:41 +00:00
//
/** Gets the ID number for a 3Dfloor in its target sector.
*
* \ param fflr The 3 Dfloor we want an ID for .
* \ return ID of 3 Dfloor in target sector . Note that the first FOF ' s ID is 0. UINT16_MAX is given if invalid .
* \ sa P_GetFFloorByID
*/
UINT16 P_GetFFloorID ( ffloor_t * fflr )
{
ffloor_t * rover ;
sector_t * sec ;
UINT16 i = 0 ;
2020-04-25 15:52:11 +00:00
if ( ! fflr )
2020-04-24 21:29:41 +00:00
return UINT16_MAX ;
2020-04-25 15:53:52 +00:00
sec = fflr - > target ;
2020-04-24 21:29:41 +00:00
if ( ! sec - > ffloors )
return UINT16_MAX ;
for ( rover = sec - > ffloors ; rover ; rover = rover - > next , i + + )
if ( rover = = fflr )
return i ;
return UINT16_MAX ;
}
2016-07-10 16:58:54 +00:00
/** Gets a 3Dfloor by control sector.
*
* \ param sec Target sector .
* \ param sec2 Control sector .
* \ return Pointer to found 3 Dfloor , or NULL .
* \ sa P_GetFFloorByID
*/
2015-01-01 19:50:31 +00:00
static inline ffloor_t * P_GetFFloorBySec ( sector_t * sec , sector_t * sec2 )
{
ffloor_t * rover ;
if ( ! sec - > ffloors )
return NULL ;
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
if ( rover - > secnum = = ( size_t ) ( sec2 - sectors ) )
return rover ;
return NULL ;
}
2016-07-10 16:58:54 +00:00
/** Gets a 3Dfloor by ID number.
*
* \ param sec Target sector .
2016-10-25 15:15:17 +00:00
* \ param id ID of 3 Dfloor in target sector . Note that the first FOF ' s ID is 0.
2016-07-10 16:58:54 +00:00
* \ return Pointer to found 3 Dfloor , or NULL .
2020-04-24 21:29:41 +00:00
* \ sa P_GetFFloorBySec , P_GetFFloorID
2016-07-10 16:58:54 +00:00
*/
ffloor_t * P_GetFFloorByID ( sector_t * sec , UINT16 id )
{
ffloor_t * rover ;
UINT16 i = 0 ;
if ( ! sec - > ffloors )
return NULL ;
for ( rover = sec - > ffloors ; rover ; rover = rover - > next )
if ( i + + = = id )
return rover ;
return NULL ;
}
2014-03-15 16:59:03 +00:00
/** Adds a newly formed 3Dfloor structure to a sector's ffloors list.
*
* \ param sec Target sector .
2019-07-09 20:15:12 +00:00
* \ param fflr Newly formed 3 Dfloor structure .
2014-03-15 16:59:03 +00:00
* \ sa P_AddFakeFloor
*/
2019-07-09 20:15:12 +00:00
static inline void P_AddFFloorToList ( sector_t * sec , ffloor_t * fflr )
2014-03-15 16:59:03 +00:00
{
ffloor_t * rover ;
if ( ! sec - > ffloors )
{
2019-07-09 20:15:12 +00:00
sec - > ffloors = fflr ;
fflr - > next = 0 ;
fflr - > prev = 0 ;
2014-03-15 16:59:03 +00:00
return ;
}
for ( rover = sec - > ffloors ; rover - > next ; rover = rover - > next ) ;
2019-07-09 20:15:12 +00:00
rover - > next = fflr ;
fflr - > prev = rover ;
fflr - > next = 0 ;
2014-03-15 16:59:03 +00:00
}
/** Adds a 3Dfloor.
*
* \ param sec Target sector .
* \ param sec2 Control sector .
* \ param master Control linedef .
2020-06-12 06:16:50 +00:00
* \ param alpha Alpha value ( 0 - 255 ) .
2022-01-04 18:33:17 +00:00
* \ param blendmode Blending mode .
2014-03-15 16:59:03 +00:00
* \ param flags Options affecting this 3 Dfloor .
* \ param secthinkers List of relevant thinkers sorted by sector . May be NULL .
* \ return Pointer to the new 3 Dfloor .
* \ sa P_AddFFloor , P_AddFakeFloorsByLine , P_SpawnSpecials
*/
2022-01-04 18:33:17 +00:00
static ffloor_t * P_AddFakeFloor ( sector_t * sec , sector_t * sec2 , line_t * master , INT32 alpha , UINT8 blendmode , ffloortype_e flags , thinkerlist_t * secthinkers )
2014-03-15 16:59:03 +00:00
{
2019-07-09 20:15:12 +00:00
ffloor_t * fflr ;
2014-03-15 16:59:03 +00:00
thinker_t * th ;
friction_t * f ;
pusher_t * p ;
size_t sec2num ;
size_t i ;
if ( sec = = sec2 )
2015-01-01 19:50:31 +00:00
return NULL ; //Don't need a fake floor on a control sector.
2019-07-09 20:15:12 +00:00
if ( ( fflr = ( P_GetFFloorBySec ( sec , sec2 ) ) ) )
return fflr ; // If this ffloor already exists, return it
2014-03-15 16:59:03 +00:00
if ( sec2 - > ceilingheight < sec2 - > floorheight )
{
fixed_t tempceiling = sec2 - > ceilingheight ;
//flip the sector around and print an error instead of crashing 12.1.08 -Inuyasha
2020-03-20 15:03:48 +00:00
CONS_Alert ( CONS_ERROR , M_GetText ( " A FOF tagged %d has a top height below its bottom. \n " ) , master - > args [ 0 ] ) ;
2014-03-15 16:59:03 +00:00
sec2 - > ceilingheight = sec2 - > floorheight ;
sec2 - > floorheight = tempceiling ;
}
if ( sec2 - > numattached = = 0 )
{
sec2 - > attached = Z_Malloc ( sizeof ( * sec2 - > attached ) * sec2 - > maxattached , PU_STATIC , NULL ) ;
sec2 - > attachedsolid = Z_Malloc ( sizeof ( * sec2 - > attachedsolid ) * sec2 - > maxattached , PU_STATIC , NULL ) ;
sec2 - > attached [ 0 ] = sec - sectors ;
sec2 - > numattached = 1 ;
2022-07-31 10:04:42 +00:00
sec2 - > attachedsolid [ 0 ] = ( flags & FOF_SOLID ) ;
2014-03-15 16:59:03 +00:00
}
else
{
for ( i = 0 ; i < sec2 - > numattached ; i + + )
if ( sec2 - > attached [ i ] = = ( size_t ) ( sec - sectors ) )
return NULL ;
if ( sec2 - > numattached > = sec2 - > maxattached )
{
sec2 - > maxattached * = 2 ;
sec2 - > attached = Z_Realloc ( sec2 - > attached , sizeof ( * sec2 - > attached ) * sec2 - > maxattached , PU_STATIC , NULL ) ;
sec2 - > attachedsolid = Z_Realloc ( sec2 - > attachedsolid , sizeof ( * sec2 - > attachedsolid ) * sec2 - > maxattached , PU_STATIC , NULL ) ;
}
sec2 - > attached [ sec2 - > numattached ] = sec - sectors ;
2022-07-31 10:04:42 +00:00
sec2 - > attachedsolid [ sec2 - > numattached ] = ( flags & FOF_SOLID ) ;
2014-03-15 16:59:03 +00:00
sec2 - > numattached + + ;
}
// Add the floor
2019-07-09 20:15:12 +00:00
fflr = Z_Calloc ( sizeof ( * fflr ) , PU_LEVEL , NULL ) ;
fflr - > secnum = sec2 - sectors ;
fflr - > target = sec ;
fflr - > bottomheight = & sec2 - > floorheight ;
fflr - > bottompic = & sec2 - > floorpic ;
2022-11-25 23:01:27 +00:00
fflr - > bottomxoffs = & sec2 - > floorxoffset ;
fflr - > bottomyoffs = & sec2 - > flooryoffset ;
2023-11-24 04:52:57 +00:00
fflr - > bottomxscale = & sec2 - > floorxscale ;
fflr - > bottomyscale = & sec2 - > flooryscale ;
2022-11-25 23:01:27 +00:00
fflr - > bottomangle = & sec2 - > floorangle ;
2014-03-15 16:59:03 +00:00
// Add the ceiling
2019-07-09 20:15:12 +00:00
fflr - > topheight = & sec2 - > ceilingheight ;
fflr - > toppic = & sec2 - > ceilingpic ;
fflr - > toplightlevel = & sec2 - > lightlevel ;
2022-11-25 23:01:27 +00:00
fflr - > topxoffs = & sec2 - > ceilingxoffset ;
fflr - > topyoffs = & sec2 - > ceilingyoffset ;
2023-11-24 04:52:57 +00:00
fflr - > topxscale = & sec2 - > ceilingxscale ;
fflr - > topyscale = & sec2 - > ceilingyscale ;
2022-11-25 23:01:27 +00:00
fflr - > topangle = & sec2 - > ceilingangle ;
2014-03-15 16:59:03 +00:00
2015-05-17 03:32:12 +00:00
// Add slopes
2019-07-09 20:15:12 +00:00
fflr - > t_slope = & sec2 - > c_slope ;
fflr - > b_slope = & sec2 - > f_slope ;
2019-06-17 17:46:51 +00:00
// mark the target sector as having slopes, if the FOF has any of its own
// (this fixes FOF slopes glitching initially at level load in software mode)
if ( sec2 - > hasslope )
sec - > hasslope = true ;
2015-05-17 03:32:12 +00:00
2022-07-31 10:04:42 +00:00
fflr - > spawnflags = fflr - > fofflags = flags ;
2019-07-09 20:15:12 +00:00
fflr - > master = master ;
fflr - > norender = INFTICS ;
fflr - > fadingdata = NULL ;
2014-03-15 16:59:03 +00:00
// Scan the thinkers to check for special conditions applying to this FOF.
// If we have thinkers sorted by sector, just check the relevant ones;
// otherwise, check them all. Apologies for the ugly loop...
sec2num = sec2 - sectors ;
// Just initialise both of these to placate the compiler.
i = 0 ;
2019-04-20 20:37:19 +00:00
th = thlist [ THINK_MAIN ] . next ;
2014-03-15 16:59:03 +00:00
for ( ; ; )
{
if ( secthinkers )
{
if ( i < secthinkers [ sec2num ] . count )
th = secthinkers [ sec2num ] . thinkers [ i ] ;
else break ;
}
2019-04-20 20:37:19 +00:00
else if ( th = = & thlist [ THINK_MAIN ] )
2014-03-15 16:59:03 +00:00
break ;
// Should this FOF have friction?
2020-04-17 20:54:35 +00:00
if ( th - > function . acp1 = = ( actionf_p1 ) T_Friction )
2014-03-15 16:59:03 +00:00
{
f = ( friction_t * ) th ;
if ( f - > affectee = = ( INT32 ) sec2num )
Add_Friction ( f - > friction , f - > movefactor , ( INT32 ) ( sec - sectors ) , f - > affectee ) ;
}
// Should this FOF have wind/current/pusher?
else if ( th - > function . acp1 = = ( actionf_p1 ) T_Pusher )
{
p = ( pusher_t * ) th ;
if ( p - > affectee = = ( INT32 ) sec2num )
2021-07-01 18:05:23 +00:00
Add_Pusher ( p - > type , p - > x_mag , p - > y_mag , p - > z_mag , ( INT32 ) ( sec - sectors ) , p - > affectee , p - > exclusive , p - > slider ) ;
2014-03-15 16:59:03 +00:00
}
if ( secthinkers ) i + + ;
else th = th - > next ;
}
2020-06-12 06:16:50 +00:00
fflr - > alpha = max ( 0 , min ( 0xff , alpha ) ) ;
2022-07-31 10:04:42 +00:00
if ( fflr - > alpha < 0xff | | flags & FOF_SPLAT )
2020-06-12 06:16:50 +00:00
{
2022-07-31 10:04:42 +00:00
fflr - > fofflags | = FOF_TRANSLUCENT ;
fflr - > spawnflags = fflr - > fofflags ;
2020-06-12 06:16:50 +00:00
}
2019-07-09 20:15:12 +00:00
fflr - > spawnalpha = fflr - > alpha ; // save for netgames
2014-11-12 00:55:07 +00:00
2022-01-04 18:33:17 +00:00
switch ( blendmode )
{
case TMB_TRANSLUCENT :
default :
2022-01-10 19:59:57 +00:00
fflr - > blend = AST_COPY ;
2022-01-04 18:33:17 +00:00
break ;
case TMB_ADD :
fflr - > blend = AST_ADD ;
break ;
case TMB_SUBTRACT :
fflr - > blend = AST_SUBTRACT ;
break ;
case TMB_REVERSESUBTRACT :
fflr - > blend = AST_REVERSESUBTRACT ;
break ;
case TMB_MODULATE :
fflr - > blend = AST_MODULATE ;
break ;
}
2022-07-31 10:04:42 +00:00
if ( flags & FOF_QUICKSAND )
2014-03-15 16:59:03 +00:00
CheckForQuicksand = true ;
2022-07-31 10:04:42 +00:00
if ( flags & FOF_BUSTUP )
2014-03-15 16:59:03 +00:00
CheckForBustableBlocks = true ;
2022-07-31 10:04:42 +00:00
if ( ( flags & FOF_MARIO ) )
2014-03-15 16:59:03 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ! ( flags & FOF_GOOWATER ) ) // Don't change the textures of a brick block, just a question block
2016-06-04 19:02:11 +00:00
P_AddBlockThinker ( sec2 , master ) ;
2014-03-15 16:59:03 +00:00
CheckForMarioBlocks = true ;
}
2022-07-31 10:04:42 +00:00
if ( ( flags & FOF_CRUMBLE ) )
2020-04-17 21:54:37 +00:00
sec2 - > crumblestate = CRUMBLE_WAIT ;
2014-03-15 16:59:03 +00:00
2022-07-31 10:04:42 +00:00
if ( ( flags & FOF_FLOATBOB ) )
2014-03-15 16:59:03 +00:00
{
2021-09-26 04:48:26 +00:00
P_AddFloatThinker ( sec2 , master - > args [ 0 ] , master ) ;
2014-03-15 16:59:03 +00:00
CheckForFloatBob = true ;
}
2019-07-09 20:15:12 +00:00
P_AddFFloorToList ( sec , fflr ) ;
2014-03-15 16:59:03 +00:00
2019-07-09 20:15:12 +00:00
return fflr ;
2014-03-15 16:59:03 +00:00
}
//
// SPECIAL SPAWNING
//
/** Adds a float thinker.
* Float thinkers cause solid 3 Dfloors to float on water .
*
* \ param sec Control sector .
* \ param actionsector Target sector .
* \ sa P_SpawnSpecials , T_FloatSector
* \ author SSNTails < http : //www.ssntails.org>
*/
2020-04-17 22:26:49 +00:00
static void P_AddFloatThinker ( sector_t * sec , UINT16 tag , line_t * sourceline )
2014-03-15 16:59:03 +00:00
{
2020-04-17 22:26:49 +00:00
floatthink_t * floater ;
2014-03-15 16:59:03 +00:00
// create and initialize new thinker
floater = Z_Calloc ( sizeof ( * floater ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & floater - > thinker ) ;
2014-03-15 16:59:03 +00:00
floater - > thinker . function . acp1 = ( actionf_p1 ) T_FloatSector ;
floater - > sector = sec ;
2020-04-17 22:26:49 +00:00
floater - > tag = ( INT16 ) tag ;
2014-03-15 16:59:03 +00:00
floater - > sourceline = sourceline ;
2022-04-25 20:49:07 +00:00
// interpolation
R_CreateInterpolator_SectorPlane ( & floater - > thinker , sec , false ) ;
R_CreateInterpolator_SectorPlane ( & floater - > thinker , sec , true ) ;
2014-03-15 16:59:03 +00:00
}
2017-01-05 20:29:48 +00:00
/**
* Adds a plane displacement thinker .
* Whenever the " control " sector moves ,
* the " affectee " sector ' s floor or ceiling plane moves too !
*
* \ param speed Rate of movement relative to control sector
* \ param control Control sector .
* \ param affectee Target sector .
2017-04-11 21:33:04 +00:00
* \ param reverse Reverse direction ?
2017-01-05 20:29:48 +00:00
* \ sa P_SpawnSpecials , T_PlaneDisplace
* \ author Monster Iestyn
*/
2017-04-11 21:33:04 +00:00
static void P_AddPlaneDisplaceThinker ( INT32 type , fixed_t speed , INT32 control , INT32 affectee , UINT8 reverse )
2017-01-05 20:29:48 +00:00
{
planedisplace_t * displace ;
// create and initialize new displacement thinker
displace = Z_Calloc ( sizeof ( * displace ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & displace - > thinker ) ;
2017-01-05 20:29:48 +00:00
displace - > thinker . function . acp1 = ( actionf_p1 ) T_PlaneDisplace ;
displace - > affectee = affectee ;
displace - > control = control ;
displace - > last_height = sectors [ control ] . floorheight ;
displace - > speed = speed ;
displace - > type = type ;
2017-04-11 21:33:04 +00:00
displace - > reverse = reverse ;
2022-04-13 01:45:49 +00:00
// interpolation
R_CreateInterpolator_SectorPlane ( & displace - > thinker , & sectors [ affectee ] , false ) ;
2017-01-05 20:29:48 +00:00
}
2014-03-15 16:59:03 +00:00
/** Adds a Mario block thinker, which changes the block's texture between blank
* and ? depending on whether it has contents .
* Needed in case objects respawn inside .
*
* \ param sec Control sector .
* \ param actionsector Target sector .
* \ param sourceline Control linedef .
* \ sa P_SpawnSpecials , T_MarioBlockChecker
* \ author SSNTails < http : //www.ssntails.org>
*/
static void P_AddBlockThinker ( sector_t * sec , line_t * sourceline )
{
2020-04-18 09:05:58 +00:00
mariocheck_t * block ;
2014-03-15 16:59:03 +00:00
// create and initialize new elevator thinker
block = Z_Calloc ( sizeof ( * block ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & block - > thinker ) ;
2014-03-15 16:59:03 +00:00
block - > thinker . function . acp1 = ( actionf_p1 ) T_MarioBlockChecker ;
block - > sourceline = sourceline ;
block - > sector = sec ;
}
/** Adds a raise thinker.
* A raise thinker checks to see if the
* player is standing on its 3 D Floor ,
* and if so , raises the platform towards
* it ' s destination . Otherwise , it lowers
* to the lowest nearby height if not
* there already .
*
* \ param sec Control sector .
* \ sa P_SpawnSpecials , T_RaiseSector
* \ author SSNTails < http : //www.ssntails.org>
*/
2020-04-27 10:54:08 +00:00
static void P_AddRaiseThinker ( sector_t * sec , INT16 tag , fixed_t speed , fixed_t ceilingtop , fixed_t ceilingbottom , boolean lower , boolean spindash )
2014-03-15 16:59:03 +00:00
{
2020-04-26 08:33:13 +00:00
raise_t * raise ;
2014-03-15 16:59:03 +00:00
raise = Z_Calloc ( sizeof ( * raise ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & raise - > thinker ) ;
2014-03-15 16:59:03 +00:00
raise - > thinker . function . acp1 = ( actionf_p1 ) T_RaiseSector ;
2020-04-27 10:54:08 +00:00
raise - > tag = tag ;
2014-03-15 16:59:03 +00:00
raise - > sector = sec ;
2020-04-27 09:19:07 +00:00
raise - > ceilingtop = ceilingtop ;
raise - > ceilingbottom = ceilingbottom ;
2014-03-15 16:59:03 +00:00
2020-04-27 09:56:29 +00:00
raise - > basespeed = speed > > 2 ;
2014-03-15 16:59:03 +00:00
2020-04-26 08:33:13 +00:00
if ( lower )
raise - > flags | = RF_REVERSE ;
if ( spindash )
raise - > flags | = RF_SPINDASH ;
2022-04-13 01:45:49 +00:00
// interpolation
R_CreateInterpolator_SectorPlane ( & raise - > thinker , sec , false ) ;
R_CreateInterpolator_SectorPlane ( & raise - > thinker , sec , true ) ;
2014-03-15 16:59:03 +00:00
}
2020-04-27 10:54:08 +00:00
static void P_AddAirbob ( sector_t * sec , INT16 tag , fixed_t dist , boolean raise , boolean spindash , boolean dynamic )
2014-03-15 16:59:03 +00:00
{
2020-04-17 07:58:53 +00:00
raise_t * airbob ;
2014-03-15 16:59:03 +00:00
airbob = Z_Calloc ( sizeof ( * airbob ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & airbob - > thinker ) ;
2014-03-15 16:59:03 +00:00
airbob - > thinker . function . acp1 = ( actionf_p1 ) T_RaiseSector ;
2020-04-27 10:54:08 +00:00
airbob - > tag = tag ;
2014-03-15 16:59:03 +00:00
airbob - > sector = sec ;
2020-04-17 07:58:53 +00:00
airbob - > ceilingtop = sec - > ceilingheight ;
2020-04-17 09:13:13 +00:00
airbob - > ceilingbottom = sec - > ceilingheight - dist ;
2019-11-07 22:56:42 +00:00
2020-04-17 07:58:53 +00:00
airbob - > basespeed = FRACUNIT ;
2014-03-15 16:59:03 +00:00
2020-04-17 09:11:36 +00:00
if ( ! raise )
2020-04-17 07:58:53 +00:00
airbob - > flags | = RF_REVERSE ;
2020-04-17 09:11:36 +00:00
if ( spindash )
2020-04-17 07:58:53 +00:00
airbob - > flags | = RF_SPINDASH ;
if ( dynamic )
airbob - > flags | = RF_DYNAMIC ;
2022-04-13 01:45:49 +00:00
// interpolation
R_CreateInterpolator_SectorPlane ( & airbob - > thinker , sec , false ) ;
R_CreateInterpolator_SectorPlane ( & airbob - > thinker , sec , true ) ;
2014-03-15 16:59:03 +00:00
}
/** Adds a thwomp thinker.
* Even thwomps need to think !
*
* \ param sec Control sector .
* \ sa P_SpawnSpecials , T_ThwompSector
* \ author SSNTails < http : //www.ssntails.org>
*/
2020-11-10 12:10:01 +00:00
static inline void P_AddThwompThinker ( sector_t * sec , line_t * sourceline , fixed_t crushspeed , fixed_t retractspeed , UINT16 sound )
2014-03-15 16:59:03 +00:00
{
2020-04-25 07:20:53 +00:00
thwomp_t * thwomp ;
2014-03-15 16:59:03 +00:00
// You *probably* already have a thwomp in this sector. If you've combined it with something
// else that uses the floordata/ceilingdata, you must be weird.
if ( sec - > floordata | | sec - > ceilingdata )
return ;
// create and initialize new elevator thinker
thwomp = Z_Calloc ( sizeof ( * thwomp ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & thwomp - > thinker ) ;
2014-03-15 16:59:03 +00:00
thwomp - > thinker . function . acp1 = ( actionf_p1 ) T_ThwompSector ;
// set up the fields according to the type of elevator action
2020-04-25 07:20:53 +00:00
thwomp - > sourceline = sourceline ;
2014-03-15 16:59:03 +00:00
thwomp - > sector = sec ;
2020-04-27 11:01:31 +00:00
thwomp - > crushspeed = crushspeed ;
thwomp - > retractspeed = retractspeed ;
2014-03-15 16:59:03 +00:00
thwomp - > direction = 0 ;
2020-04-25 07:20:53 +00:00
thwomp - > floorstartheight = sec - > floorheight ;
thwomp - > ceilingstartheight = sec - > ceilingheight ;
thwomp - > delay = 1 ;
2021-09-26 04:48:26 +00:00
thwomp - > tag = sourceline - > args [ 0 ] ;
2020-04-27 11:01:31 +00:00
thwomp - > sound = sound ;
2020-04-25 07:20:53 +00:00
sec - > floordata = thwomp ;
sec - > ceilingdata = thwomp ;
// Start with 'resting' texture
sides [ sourceline - > sidenum [ 0 ] ] . midtexture = sides [ sourceline - > sidenum [ 0 ] ] . bottomtexture ;
2022-04-13 01:45:49 +00:00
// interpolation
R_CreateInterpolator_SectorPlane ( & thwomp - > thinker , sec , false ) ;
R_CreateInterpolator_SectorPlane ( & thwomp - > thinker , sec , true ) ;
2014-03-15 16:59:03 +00:00
}
/** Adds a thinker which checks if any MF_ENEMY objects with health are in the defined area.
* If not , a linedef executor is run once .
*
* \ param sourceline Control linedef .
* \ sa P_SpawnSpecials , T_NoEnemiesSector
* \ author SSNTails < http : //www.ssntails.org>
*/
2020-04-18 07:21:04 +00:00
static inline void P_AddNoEnemiesThinker ( line_t * sourceline )
2014-03-15 16:59:03 +00:00
{
2020-04-18 07:21:04 +00:00
noenemies_t * nobaddies ;
2014-03-15 16:59:03 +00:00
// create and initialize new thinker
nobaddies = Z_Calloc ( sizeof ( * nobaddies ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & nobaddies - > thinker ) ;
2014-03-15 16:59:03 +00:00
nobaddies - > thinker . function . acp1 = ( actionf_p1 ) T_NoEnemiesSector ;
nobaddies - > sourceline = sourceline ;
}
/** Adds a thinker for Each-Time linedef executors. A linedef executor is run
* only when a player enters the area and doesn ' t run again until they re - enter .
*
* \ param sourceline Control linedef .
* \ sa P_SpawnSpecials , T_EachTimeThinker
* \ author SSNTails < http : //www.ssntails.org>
*/
2021-12-09 06:30:55 +00:00
static void P_AddEachTimeThinker ( line_t * sourceline , boolean triggerOnExit )
2014-03-15 16:59:03 +00:00
{
2020-04-17 12:00:48 +00:00
eachtime_t * eachtime ;
2014-03-15 16:59:03 +00:00
// create and initialize new thinker
eachtime = Z_Calloc ( sizeof ( * eachtime ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & eachtime - > thinker ) ;
2014-03-15 16:59:03 +00:00
eachtime - > thinker . function . acp1 = ( actionf_p1 ) T_EachTimeThinker ;
eachtime - > sourceline = sourceline ;
2021-12-09 06:30:55 +00:00
eachtime - > triggerOnExit = triggerOnExit ;
2014-03-15 16:59:03 +00:00
}
/** Adds a camera scanner.
*
* \ param sourcesec Control sector .
* \ param actionsector Target sector .
* \ param angle Angle of the source line .
* \ sa P_SpawnSpecials , T_CameraScanner
* \ author SSNTails < http : //www.ssntails.org>
*/
static inline void P_AddCameraScanner ( sector_t * sourcesec , sector_t * actionsector , angle_t angle )
{
elevator_t * elevator ; // Why not? LOL
// create and initialize new elevator thinker
elevator = Z_Calloc ( sizeof ( * elevator ) , PU_LEVSPEC , NULL ) ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & elevator - > thinker ) ;
2014-03-15 16:59:03 +00:00
elevator - > thinker . function . acp1 = ( actionf_p1 ) T_CameraScanner ;
elevator - > type = elevateBounce ;
// set up the fields according to the type of elevator action
elevator - > sector = sourcesec ;
elevator - > actionsector = actionsector ;
elevator - > distance = FixedInt ( AngleFixed ( angle ) ) ;
}
/** Flashes a laser block.
*
* \ param flash Thinker structure for this laser .
2020-05-02 08:03:16 +00:00
* \ sa P_AddLaserThinker
2014-03-15 16:59:03 +00:00
* \ author SSNTails < http : //www.ssntails.org>
*/
void T_LaserFlash ( laserthink_t * flash )
{
msecnode_t * node ;
mobj_t * thing ;
2020-05-02 08:03:16 +00:00
INT32 s ;
ffloor_t * fflr ;
sector_t * sector ;
sector_t * sourcesec = flash - > sourceline - > frontsector ;
2015-05-24 17:53:30 +00:00
fixed_t top , bottom ;
2014-03-15 16:59:03 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( flash - > tag , s )
2020-05-02 08:03:16 +00:00
{
sector = & sectors [ s ] ;
for ( fflr = sector - > ffloors ; fflr ; fflr = fflr - > next )
{
if ( fflr - > master ! = flash - > sourceline )
continue ;
2014-03-15 16:59:03 +00:00
2022-07-31 10:04:42 +00:00
if ( ! ( fflr - > fofflags & FOF_EXISTS ) )
2020-05-02 08:03:16 +00:00
break ;
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
if ( leveltime & 2 )
2022-07-31 10:04:42 +00:00
//fflr->flags |= FOF_RENDERALL;
2020-05-02 08:03:16 +00:00
fflr - > alpha = 0xB0 ;
else
2022-07-31 10:04:42 +00:00
//fflr->flags &= ~FOF_RENDERALL;
2020-05-02 08:03:16 +00:00
fflr - > alpha = 0x90 ;
2014-03-15 16:59:03 +00:00
2020-05-18 14:14:05 +00:00
top = P_GetFFloorTopZAt ( fflr , sector - > soundorg . x , sector - > soundorg . y ) ;
bottom = P_GetFFloorBottomZAt ( fflr , sector - > soundorg . x , sector - > soundorg . y ) ;
2020-05-02 08:03:16 +00:00
sector - > soundorg . z = ( top + bottom ) / 2 ;
S_StartSound ( & sector - > soundorg , sfx_laser ) ;
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
// Seek out objects to DESTROY! MUAHAHHAHAHAA!!!*cough*
for ( node = sector - > touching_thinglist ; node & & node - > m_thing ; node = node - > m_thinglist_next )
{
thing = node - > m_thing ;
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
if ( flash - > nobosses & & thing - > flags & MF_BOSS )
continue ; // Don't hurt bosses
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
// Don't endlessly kill egg guard shields (or anything else for that matter)
if ( thing - > health < = 0 )
continue ;
2017-01-07 18:02:55 +00:00
2020-05-02 08:03:16 +00:00
top = P_GetSpecialTopZ ( thing , sourcesec , sector ) ;
bottom = P_GetSpecialBottomZ ( thing , sourcesec , sector ) ;
2015-05-24 17:53:30 +00:00
2020-05-02 08:03:16 +00:00
if ( thing - > z > = top
| | thing - > z + thing - > height < = bottom )
continue ;
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
if ( thing - > flags & MF_SHOOTABLE )
P_DamageMobj ( thing , NULL , NULL , 1 , 0 ) ;
else if ( thing - > type = = MT_EGGSHIELD )
P_KillMobj ( thing , NULL , NULL , 0 ) ;
}
2014-03-15 16:59:03 +00:00
2020-05-02 08:03:16 +00:00
break ;
}
2014-03-15 16:59:03 +00:00
}
}
2020-05-02 08:03:16 +00:00
static inline void P_AddLaserThinker ( INT16 tag , line_t * line , boolean nobosses )
2014-03-15 16:59:03 +00:00
{
2020-05-02 08:03:16 +00:00
laserthink_t * flash = Z_Calloc ( sizeof ( * flash ) , PU_LEVSPEC , NULL ) ;
2014-03-15 16:59:03 +00:00
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & flash - > thinker ) ;
2014-03-15 16:59:03 +00:00
flash - > thinker . function . acp1 = ( actionf_p1 ) T_LaserFlash ;
2020-05-02 08:03:16 +00:00
flash - > tag = tag ;
2014-03-15 16:59:03 +00:00
flash - > sourceline = line ;
2020-04-27 11:09:57 +00:00
flash - > nobosses = nobosses ;
2014-03-15 16:59:03 +00:00
}
//
// P_RunLevelLoadExecutors
//
// After loading/spawning all other specials
// and items, execute these.
//
static void P_RunLevelLoadExecutors ( void )
{
size_t i ;
for ( i = 0 ; i < numlines ; i + + )
{
if ( lines [ i ] . special = = 399 )
2014-08-27 03:56:30 +00:00
P_RunTriggerLinedef ( & lines [ i ] , NULL , NULL ) ;
2014-03-15 16:59:03 +00:00
}
}
2017-04-24 19:33:39 +00:00
/** Before things are loaded, initialises certain stuff in case they're needed
2020-01-10 18:56:29 +00:00
* by P_SpawnSlopes or P_LoadThings . This was split off from
2017-04-24 19:33:39 +00:00
* P_SpawnSpecials , in case you couldn ' t tell .
*
2020-05-04 20:40:49 +00:00
* \ sa P_SpawnSpecials
2017-04-24 19:33:39 +00:00
* \ author Monster Iestyn
*/
void P_InitSpecials ( void )
{
// Set the default gravity. Custom gravity overrides this setting.
2020-05-03 16:33:18 +00:00
gravity = mapheaderinfo [ gamemap - 1 ] - > gravity ;
2017-04-24 19:33:39 +00:00
// Defaults in case levels don't have them set.
2020-05-03 15:56:49 +00:00
sstimer = mapheaderinfo [ gamemap - 1 ] - > sstimer * TICRATE + 6 ;
ssspheres = mapheaderinfo [ gamemap - 1 ] - > ssspheres ;
2017-04-24 19:33:39 +00:00
CheckForBustableBlocks = CheckForBouncySector = CheckForQuicksand = CheckForMarioBlocks = CheckForFloatBob = CheckForReverseGravity = false ;
// Set curWeather
switch ( mapheaderinfo [ gamemap - 1 ] - > weather )
{
case PRECIP_SNOW : // snow
case PRECIP_RAIN : // rain
case PRECIP_STORM : // storm
case PRECIP_STORM_NORAIN : // storm w/o rain
case PRECIP_STORM_NOSTRIKES : // storm w/o lightning
curWeather = mapheaderinfo [ gamemap - 1 ] - > weather ;
break ;
default : // blank/none
2017-04-24 19:43:58 +00:00
curWeather = PRECIP_NONE ;
2017-04-24 19:33:39 +00:00
break ;
}
// Set globalweather
globalweather = mapheaderinfo [ gamemap - 1 ] - > weather ;
}
2021-12-29 07:12:28 +00:00
void P_ApplyFlatAlignment ( sector_t * sector , angle_t flatangle , fixed_t xoffs , fixed_t yoffs , boolean floor , boolean ceiling )
2018-05-14 17:34:22 +00:00
{
2021-12-29 07:12:28 +00:00
if ( floor )
2018-05-14 17:34:22 +00:00
{
2022-11-25 23:01:27 +00:00
sector - > floorangle = flatangle ;
sector - > floorxoffset + = xoffs ;
sector - > flooryoffset + = yoffs ;
2018-05-14 17:34:22 +00:00
}
2021-12-29 07:12:28 +00:00
if ( ceiling )
2018-05-14 17:34:22 +00:00
{
2022-11-25 23:01:27 +00:00
sector - > ceilingangle = flatangle ;
sector - > ceilingxoffset + = xoffs ;
sector - > ceilingyoffset + = yoffs ;
2018-05-14 17:34:22 +00:00
}
}
2020-05-03 10:44:30 +00:00
static void P_MakeFOFBouncy ( line_t * paramline , line_t * masterline )
{
INT32 s ;
if ( masterline - > special < 100 | | masterline - > special > = 300 )
return ;
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( masterline - > args [ 0 ] , s )
2020-05-03 10:44:30 +00:00
{
ffloor_t * rover ;
for ( rover = sectors [ s ] . ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master ! = masterline )
continue ;
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_BOUNCY ;
rover - > spawnflags | = FOF_BOUNCY ;
2020-05-03 10:44:30 +00:00
rover - > bouncestrength = ( paramline - > args [ 1 ] < < FRACBITS ) / 100 ;
CheckForBouncySector = true ;
break ;
}
}
}
2021-12-09 17:56:50 +00:00
static boolean P_CheckGametypeRules ( INT32 checktype , UINT32 target )
{
switch ( checktype )
{
2021-12-09 19:37:39 +00:00
case TMF_HASALL :
2021-12-09 17:56:50 +00:00
default :
return ( gametyperules & target ) = = target ;
2021-12-09 19:37:39 +00:00
case TMF_HASANY :
2021-12-09 17:56:50 +00:00
return ! ! ( gametyperules & target ) ;
2021-12-09 19:37:39 +00:00
case TMF_HASEXACTLY :
2021-12-09 17:56:50 +00:00
return gametyperules = = target ;
2021-12-09 19:37:39 +00:00
case TMF_DOESNTHAVEALL :
2021-12-09 17:56:50 +00:00
return ( gametyperules & target ) ! = target ;
2021-12-09 19:37:39 +00:00
case TMF_DOESNTHAVEANY :
2021-12-09 17:56:50 +00:00
return ! ( gametyperules & target ) ;
}
}
2021-12-30 17:19:42 +00:00
fixed_t P_GetSectorGravityFactor ( sector_t * sec )
{
if ( sec - > gravityptr )
return FixedDiv ( * sec - > gravityptr > > FRACBITS , 1000 ) ;
else
return sec - > gravity ;
}
2023-08-23 20:49:29 +00:00
void P_InitSectorPortals ( void )
{
secportalcount = 0 ;
secportalcapacity = 0 ;
secportals = NULL ;
}
UINT32 P_NewSectorPortal ( void )
{
size_t i = secportalcount + + ;
if ( i = = UINT32_MAX )
I_Error ( " Too many sector portals " ) ;
if ( secportalcapacity = = 0 | | secportalcount = = secportalcapacity )
{
secportalcapacity = secportalcapacity ? ( secportalcapacity * 2 ) : 16 ;
secportals = Z_Realloc ( secportals , secportalcapacity * sizeof ( sectorportal_t ) , PU_LEVEL , NULL ) ;
}
2023-08-24 19:04:31 +00:00
secportals [ i ] . type = SECPORTAL_NONE ;
2023-08-23 20:49:29 +00:00
return ( UINT32 ) i ;
}
2023-08-24 20:36:38 +00:00
boolean P_IsSectorPortalValid ( sectorportal_t * secportal )
2023-08-23 16:43:02 +00:00
{
2023-08-23 20:49:29 +00:00
if ( secportal = = NULL )
return false ;
2023-08-23 18:04:50 +00:00
switch ( secportal - > type )
{
case SECPORTAL_LINE :
case SECPORTAL_FLOOR :
case SECPORTAL_CEILING :
return true ;
case SECPORTAL_OBJECT :
return secportal - > mobj & & ! P_MobjWasRemoved ( secportal - > mobj ) ;
case SECPORTAL_SKYBOX :
return skyboxmo [ 0 ] & & ! P_MobjWasRemoved ( skyboxmo [ 0 ] ) ;
2023-08-24 03:51:52 +00:00
case SECPORTAL_PLANE :
case SECPORTAL_HORIZON :
return true ;
2023-08-23 18:04:50 +00:00
default :
return false ;
}
2023-08-23 16:43:02 +00:00
}
2023-08-23 18:46:48 +00:00
boolean P_SectorHasPortal ( sector_t * sector )
{
return P_SectorHasFloorPortal ( sector ) | | P_SectorHasCeilingPortal ( sector ) ;
}
2023-08-23 20:49:29 +00:00
sectorportal_t * P_SectorGetFloorPortal ( sector_t * sector )
{
UINT32 num = sector - > portal_floor ;
if ( num > = secportalcount )
return NULL ;
return & secportals [ num ] ;
}
sectorportal_t * P_SectorGetCeilingPortal ( sector_t * sector )
{
UINT32 num = sector - > portal_ceiling ;
if ( num > = secportalcount )
return NULL ;
return & secportals [ num ] ;
}
2023-08-23 18:04:50 +00:00
boolean P_SectorHasFloorPortal ( sector_t * sector )
2023-08-23 07:24:06 +00:00
{
2023-08-23 20:49:29 +00:00
return P_IsSectorPortalValid ( P_SectorGetFloorPortal ( sector ) ) ;
2023-08-23 18:04:50 +00:00
}
2023-08-23 07:24:06 +00:00
2023-08-23 18:04:50 +00:00
boolean P_SectorHasCeilingPortal ( sector_t * sector )
{
2023-08-23 20:49:29 +00:00
return P_IsSectorPortalValid ( P_SectorGetCeilingPortal ( sector ) ) ;
2023-08-23 18:04:50 +00:00
}
boolean P_CompareSectorPortals ( sectorportal_t * a , sectorportal_t * b )
{
2023-08-23 18:46:48 +00:00
if ( a = = NULL & & b = = NULL )
return true ;
else if ( ! a | | ! b )
return false ;
else if ( a - > type ! = b - > type )
2023-08-23 18:04:50 +00:00
return false ;
switch ( a - > type )
{
case SECPORTAL_LINE :
return a - > line . start = = b - > line . start & & a - > line . dest = = b - > line . dest ;
case SECPORTAL_FLOOR :
case SECPORTAL_CEILING :
return a - > sector = = b - > sector ;
case SECPORTAL_OBJECT :
return a - > mobj = = b - > mobj ;
default :
return true ;
}
}
2023-08-23 07:24:06 +00:00
2023-08-23 18:29:16 +00:00
static mobj_t * P_GetMobjByTag ( INT32 tag )
2023-08-23 18:04:50 +00:00
{
2023-08-25 08:02:23 +00:00
INT32 mtnum = - 1 ;
2023-08-23 07:24:06 +00:00
2023-08-25 08:02:23 +00:00
TAG_ITER_THINGS ( tag , mtnum )
{
mobj_t * mo = mapthings [ mtnum ] . mobj ;
if ( mo )
2023-08-23 18:04:50 +00:00
return mo ;
2023-08-23 07:24:06 +00:00
}
2023-08-23 18:04:50 +00:00
return NULL ;
2023-08-23 07:24:06 +00:00
}
2023-08-23 20:49:29 +00:00
static void P_DoPortalCopyFromLine ( sector_t * dest_sector , int plane_type , int tag )
2023-08-23 18:29:16 +00:00
{
2023-08-23 20:49:29 +00:00
INT32 secnum = - 1 ;
TAG_ITER_SECTORS ( tag , secnum )
{
sector_t * src_sector = & sectors [ secnum ] ;
if ( plane_type = = TMP_FLOOR | | plane_type = = TMP_BOTH )
dest_sector - > portal_floor = src_sector - > portal_floor ;
if ( plane_type = = TMP_CEILING | | plane_type = = TMP_BOTH )
dest_sector - > portal_ceiling = src_sector - > portal_ceiling ;
}
}
2023-08-25 03:18:20 +00:00
static sectorportal_t * P_SectorGetPortalOrCreate ( sector_t * sector , UINT32 * num , UINT32 * result )
2023-08-23 20:49:29 +00:00
{
2023-08-25 03:18:20 +00:00
sectorportal_t * secportal = NULL ;
2023-08-24 20:36:38 +00:00
2023-08-25 03:18:20 +00:00
if ( * num > = secportalcount )
{
* num = P_NewSectorPortal ( ) ;
secportal = & secportals [ * num ] ;
secportal - > origin . x = sector - > soundorg . x ;
secportal - > origin . y = sector - > soundorg . y ;
* result = * num ;
}
else
2023-08-23 20:49:29 +00:00
{
2023-08-25 03:18:20 +00:00
* result = * num ;
secportal = & secportals [ * num ] ;
2023-08-23 20:49:29 +00:00
}
2023-08-23 18:29:16 +00:00
2023-08-25 03:18:20 +00:00
return secportal ;
2023-08-23 18:29:16 +00:00
}
2023-08-25 03:18:20 +00:00
static sectorportal_t * P_SectorGetFloorPortalOrCreate ( sector_t * sector , UINT32 * result )
2023-08-23 18:29:16 +00:00
{
2023-08-25 03:18:20 +00:00
return P_SectorGetPortalOrCreate ( sector , & sector - > portal_floor , result ) ;
}
2023-08-23 20:49:29 +00:00
2023-08-25 03:18:20 +00:00
static sectorportal_t * P_SectorGetCeilingPortalOrCreate ( sector_t * sector , UINT32 * result )
{
return P_SectorGetPortalOrCreate ( sector , & sector - > portal_ceiling , result ) ;
2023-08-24 20:36:38 +00:00
}
static void P_CopySectorPortalToLines ( UINT32 portal_num , int sector_tag )
{
for ( size_t i = 0 ; i < numlines ; i + + )
{
if ( lines [ i ] . special ! = SPECIAL_SECTOR_SETPORTAL )
continue ;
if ( lines [ i ] . args [ 1 ] ! = TMSECPORTAL_COPY_PORTAL_TO_LINE )
continue ;
if ( lines [ i ] . args [ 3 ] ! = sector_tag )
continue ;
if ( lines [ i ] . args [ 0 ] ! = 0 )
{
INT32 linenum = - 1 ;
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , linenum )
{
lines [ linenum ] . secportal = portal_num ;
}
}
else
{
// Just transfer it to this line
lines [ i ] . secportal = portal_num ;
}
}
2023-08-23 18:29:16 +00:00
}
2014-03-15 16:59:03 +00:00
/** After the map has loaded, scans for specials that spawn 3Dfloors and
* thinkers .
*
* \ todo Split up into multiple functions .
* \ todo Get rid of all the magic numbers .
* \ todo Potentially use ' fromnetsave ' to stop any new thinkers from being created
* as they ' ll just be erased by UnArchiveThinkers .
* \ sa P_SpawnPrecipitation , P_SpawnFriction , P_SpawnPushers , P_SpawnScrollers
*/
2019-12-28 11:48:32 +00:00
void P_SpawnSpecials ( boolean fromnetsave )
2014-03-15 16:59:03 +00:00
{
sector_t * sector ;
size_t i ;
INT32 j ;
thinkerlist_t * secthinkers ;
thinker_t * th ;
2015-01-01 19:50:31 +00:00
// This used to be used, and *should* be used in the future,
// but currently isn't.
( void ) fromnetsave ;
2019-08-03 11:09:18 +00:00
// yep, we do this here - "bossdisabled" is considered an apparatus of specials.
bossdisabled = 0 ;
2019-11-08 15:47:12 +00:00
stoppedclock = false ;
2015-01-01 19:50:31 +00:00
2014-03-15 16:59:03 +00:00
// Init special SECTORs.
sector = sectors ;
for ( i = 0 ; i < numsectors ; i + + , sector + + )
{
2021-12-30 17:50:02 +00:00
CheckForReverseGravity | = ( sector - > flags & MSF_GRAVITYFLIP ) ;
2021-12-30 23:03:24 +00:00
if ( sector - > specialflags & SSF_FINISHLINE )
{
if ( ( gametyperules & ( GTR_RACE | GTR_LIVES ) ) = = GTR_RACE )
circuitmap = true ;
}
2023-02-19 06:27:16 +00:00
if ( sector - > damagetype = = SD_SPIKE ) {
//Terrible hack to replace an even worse hack:
//Spike damage automatically sets MSF_TRIGGERSPECIAL_TOUCH.
//Yes, this also affects other specials on the same sector. Sorry.
sector - > flags | = MSF_TRIGGERSPECIAL_TOUCH ;
}
2023-10-28 13:19:35 +00:00
// TODO: 2.3: Delete everything below
2023-02-19 06:27:16 +00:00
// Process deprecated binary sector specials
if ( udmf | | ! sector - > special )
2014-03-15 16:59:03 +00:00
continue ;
// Process Section 1
switch ( GETSECSPECIAL ( sector - > special , 1 ) )
{
2021-06-23 20:21:19 +00:00
case 15 : // Bouncy FOF
CheckForBouncySector = true ;
break ;
2014-03-15 16:59:03 +00:00
}
// Process Section 2
switch ( GETSECSPECIAL ( sector - > special , 2 ) )
{
case 10 : // Time for special stage
sstimer = ( sector - > floorheight > > FRACBITS ) * TICRATE + 6 ; // Time to finish
2018-06-03 21:41:54 +00:00
ssspheres = sector - > ceilingheight > > FRACBITS ; // Ring count for special stage
2014-03-15 16:59:03 +00:00
break ;
case 11 : // Custom global gravity!
gravity = sector - > floorheight / 1000 ;
break ;
}
}
P_SpawnScrollers ( ) ; // Add generalized scrollers
P_SpawnFriction ( ) ; // Friction model using linedefs
P_SpawnPushers ( ) ; // Pusher model using linedefs
// Look for thinkers that affect FOFs, and sort them by sector
secthinkers = Z_Calloc ( numsectors * sizeof ( thinkerlist_t ) , PU_STATIC , NULL ) ;
// Firstly, find out how many there are in each sector
2019-04-20 20:37:19 +00:00
for ( th = thlist [ THINK_MAIN ] . next ; th ! = & thlist [ THINK_MAIN ] ; th = th - > next )
2014-03-15 16:59:03 +00:00
{
2020-04-17 20:54:35 +00:00
if ( th - > function . acp1 = = ( actionf_p1 ) T_Friction )
2014-03-15 16:59:03 +00:00
secthinkers [ ( ( friction_t * ) th ) - > affectee ] . count + + ;
else if ( th - > function . acp1 = = ( actionf_p1 ) T_Pusher )
secthinkers [ ( ( pusher_t * ) th ) - > affectee ] . count + + ;
}
// Allocate each list, and then zero the count so we can use it to track
// the end of the list as we add the thinkers
for ( i = 0 ; i < numsectors ; i + + )
if ( secthinkers [ i ] . count > 0 )
{
secthinkers [ i ] . thinkers = Z_Malloc ( secthinkers [ i ] . count * sizeof ( thinker_t * ) , PU_STATIC , NULL ) ;
secthinkers [ i ] . count = 0 ;
}
// Finally, populate the lists.
2019-04-20 20:37:19 +00:00
for ( th = thlist [ THINK_MAIN ] . next ; th ! = & thlist [ THINK_MAIN ] ; th = th - > next )
2014-03-15 16:59:03 +00:00
{
size_t secnum = ( size_t ) - 1 ;
2020-04-17 20:54:35 +00:00
if ( th - > function . acp1 = = ( actionf_p1 ) T_Friction )
2014-03-15 16:59:03 +00:00
secnum = ( ( friction_t * ) th ) - > affectee ;
else if ( th - > function . acp1 = = ( actionf_p1 ) T_Pusher )
secnum = ( ( pusher_t * ) th ) - > affectee ;
if ( secnum ! = ( size_t ) - 1 )
secthinkers [ secnum ] . thinkers [ secthinkers [ secnum ] . count + + ] = th ;
}
// Init line EFFECTs
for ( i = 0 ; i < numlines ; i + + )
{
2021-12-29 07:12:28 +00:00
// set line specials to 0 here too, same reason as above
if ( netgame | | multiplayer )
2014-03-15 16:59:03 +00:00
{
2021-12-29 07:12:28 +00:00
if ( lines [ i ] . flags & ML_NONET )
2014-03-15 16:59:03 +00:00
{
lines [ i ] . special = 0 ;
continue ;
}
}
2021-12-29 07:12:28 +00:00
else if ( lines [ i ] . flags & ML_NETONLY )
{
lines [ i ] . special = 0 ;
continue ;
}
2014-03-15 16:59:03 +00:00
switch ( lines [ i ] . special )
{
INT32 s ;
2020-05-01 17:07:29 +00:00
INT32 l ;
2014-03-15 16:59:03 +00:00
size_t sec ;
ffloortype_e ffloorflags ;
case 1 : // Definable gravity per sector
2021-12-30 18:28:01 +00:00
if ( udmf )
break ;
2014-03-15 16:59:03 +00:00
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-12-31 13:17:17 +00:00
TAG_ITER_SECTORS ( Tag_FGet ( & lines [ i ] . tags ) , s )
2014-03-15 16:59:03 +00:00
{
2021-12-30 17:19:42 +00:00
sectors [ s ] . gravityptr = & sectors [ sec ] . floorheight ; // This allows it to change in realtime!
2014-03-15 16:59:03 +00:00
if ( lines [ i ] . flags & ML_NOCLIMB )
2021-12-30 17:50:02 +00:00
sectors [ s ] . flags | = MSF_GRAVITYFLIP ;
2014-03-15 16:59:03 +00:00
else
2021-12-30 17:50:02 +00:00
sectors [ s ] . flags & = ~ MSF_GRAVITYFLIP ;
2014-03-15 16:59:03 +00:00
2022-09-11 14:15:50 +00:00
if ( lines [ i ] . flags & ML_EFFECT6 )
sectors [ s ] . specialflags | = SSF_GRAVITYOVERRIDE ;
2021-12-30 17:50:02 +00:00
CheckForReverseGravity | = ( sectors [ s ] . flags & MSF_GRAVITYFLIP ) ;
2014-03-15 16:59:03 +00:00
}
break ;
case 5 : // Change camera info
2021-12-29 09:55:39 +00:00
if ( udmf )
break ;
2014-03-15 16:59:03 +00:00
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-12-31 13:17:17 +00:00
TAG_ITER_SECTORS ( Tag_FGet ( & lines [ i ] . tags ) , s )
2014-03-15 16:59:03 +00:00
P_AddCameraScanner ( & sectors [ sec ] , & sectors [ s ] , R_PointToAngle2 ( lines [ i ] . v2 - > x , lines [ i ] . v2 - > y , lines [ i ] . v1 - > x , lines [ i ] . v1 - > y ) ) ;
break ;
2023-08-24 20:36:38 +00:00
case SPECIAL_SECTOR_SETPORTAL : // Sector portal
2023-08-23 07:24:06 +00:00
{
2023-08-23 18:04:50 +00:00
int target_sector_tag = lines [ i ] . args [ 0 ] ;
int portal_type = lines [ i ] . args [ 1 ] ;
int plane_type = lines [ i ] . args [ 2 ] ;
int misc = lines [ i ] . args [ 3 ] ;
boolean floor , ceiling ;
if ( plane_type = = TMP_BOTH )
floor = ceiling = true ;
else
2023-08-23 07:24:06 +00:00
{
2023-08-23 18:04:50 +00:00
floor = plane_type = = TMP_FLOOR ;
ceiling = plane_type = = TMP_CEILING ;
}
2023-08-23 07:24:06 +00:00
2023-08-24 20:36:38 +00:00
UINT32 portal_num = UINT32_MAX ;
2023-08-24 03:51:52 +00:00
// Eternity's floor and horizon portal types
if ( portal_type = = TMSECPORTAL_PLANE | | portal_type = = TMSECPORTAL_HORIZON )
{
secportaltype_e type = portal_type = = TMSECPORTAL_HORIZON ? SECPORTAL_HORIZON : SECPORTAL_PLANE ;
if ( floor )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * floorportal = P_SectorGetFloorPortalOrCreate ( lines [ i ] . frontsector , & portal_num ) ;
2023-08-24 03:51:52 +00:00
floorportal - > type = type ;
2023-08-24 19:04:31 +00:00
floorportal - > sector = lines [ i ] . frontsector ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-24 03:51:52 +00:00
}
if ( ceiling )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * ceilportal = P_SectorGetCeilingPortalOrCreate ( lines [ i ] . frontsector , & portal_num ) ;
2023-08-24 03:51:52 +00:00
ceilportal - > type = type ;
2023-08-24 19:04:31 +00:00
ceilportal - > sector = lines [ i ] . frontsector ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-24 03:51:52 +00:00
}
break ;
}
2023-08-23 18:04:50 +00:00
INT32 s1 = - 1 ;
2023-08-24 20:36:38 +00:00
TAG_ITER_SECTORS ( target_sector_tag , s1 )
2023-08-23 18:04:50 +00:00
{
2023-08-23 20:49:29 +00:00
sector_t * target_sector = & sectors [ s1 ] ;
2023-08-23 18:04:50 +00:00
// Line portal
2023-08-24 03:51:52 +00:00
if ( portal_type = = TMSECPORTAL_NORMAL )
2023-08-23 18:04:50 +00:00
{
INT32 linenum = - 1 ;
TAG_ITER_LINES ( misc , linenum )
2023-08-23 07:24:06 +00:00
{
2023-08-24 20:36:38 +00:00
if ( lines [ linenum ] . special = = SPECIAL_SECTOR_SETPORTAL
2023-08-23 18:04:50 +00:00
& & lines [ linenum ] . args [ 0 ] = = target_sector_tag
& & lines [ linenum ] . args [ 1 ] = = portal_type
& & lines [ linenum ] . args [ 2 ] = = plane_type
& & lines [ linenum ] . args [ 3 ] = = 1 )
{
if ( floor )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * floorportal = P_SectorGetFloorPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
floorportal - > type = SECPORTAL_LINE ;
floorportal - > line . start = & lines [ i ] ;
floorportal - > line . dest = & lines [ linenum ] ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
if ( ceiling )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * ceilportal = P_SectorGetCeilingPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
ceilportal - > type = SECPORTAL_LINE ;
ceilportal - > line . start = & lines [ i ] ;
ceilportal - > line . dest = & lines [ linenum ] ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
}
2023-08-23 07:24:06 +00:00
}
2023-08-23 18:04:50 +00:00
}
// Skybox portal
2023-08-24 03:51:52 +00:00
else if ( portal_type = = TMSECPORTAL_SKYBOX )
2023-08-23 18:04:50 +00:00
{
2023-08-23 07:24:06 +00:00
if ( floor )
2023-08-23 20:49:29 +00:00
{
2023-08-24 20:36:38 +00:00
sectorportal_t * floorportal = P_SectorGetFloorPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
floorportal - > type = SECPORTAL_SKYBOX ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 20:49:29 +00:00
}
2023-08-23 07:24:06 +00:00
if ( ceiling )
2023-08-23 20:49:29 +00:00
{
2023-08-24 20:36:38 +00:00
sectorportal_t * ceilportal = P_SectorGetCeilingPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
ceilportal - > type = SECPORTAL_SKYBOX ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 20:49:29 +00:00
}
2023-08-23 18:04:50 +00:00
}
// Plane portal
2023-08-24 03:51:52 +00:00
else if ( portal_type = = TMSECPORTAL_SECTOR )
2023-08-23 18:04:50 +00:00
{
INT32 s2 = - 1 ;
TAG_ITER_SECTORS ( misc , s2 ) // Sector tag to make a portal to
{
2023-08-23 20:49:29 +00:00
sector_t * view_sector = & sectors [ s2 ] ;
2023-08-23 18:04:50 +00:00
if ( floor )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * floorportal = P_SectorGetFloorPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
floorportal - > type = SECPORTAL_CEILING ;
2023-08-23 20:49:29 +00:00
floorportal - > sector = view_sector ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
if ( ceiling )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * ceilportal = P_SectorGetCeilingPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
ceilportal - > type = SECPORTAL_FLOOR ;
2023-08-23 20:49:29 +00:00
ceilportal - > sector = view_sector ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
}
}
// Use mobj as viewpoint
2023-08-24 03:51:52 +00:00
else if ( portal_type = = TMSECPORTAL_OBJECT )
2023-08-23 18:04:50 +00:00
{
2023-08-23 18:29:16 +00:00
mobj_t * mobj = P_GetMobjByTag ( misc ) ;
2023-08-23 18:04:50 +00:00
if ( ! mobj )
break ;
if ( floor )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * floorportal = P_SectorGetFloorPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
floorportal - > type = SECPORTAL_OBJECT ;
2023-08-24 04:19:58 +00:00
P_SetTarget ( & floorportal - > mobj , mobj ) ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
if ( ceiling )
{
2023-08-24 20:36:38 +00:00
sectorportal_t * ceilportal = P_SectorGetCeilingPortalOrCreate ( target_sector , & portal_num ) ;
2023-08-23 18:04:50 +00:00
ceilportal - > type = SECPORTAL_OBJECT ;
2023-08-24 04:19:58 +00:00
P_SetTarget ( & ceilportal - > mobj , mobj ) ;
2023-08-24 20:36:38 +00:00
P_CopySectorPortalToLines ( portal_num , target_sector_tag ) ;
2023-08-23 18:04:50 +00:00
}
2023-08-23 07:24:06 +00:00
}
}
break ;
}
2016-05-30 21:44:23 +00:00
case 7 : // Flat alignment - redone by toast
2021-12-29 07:12:28 +00:00
{
// Set calculated offsets such that line's v1 is the apparent origin
angle_t flatangle = InvAngle ( R_PointToAngle2 ( lines [ i ] . v1 - > x , lines [ i ] . v1 - > y , lines [ i ] . v2 - > x , lines [ i ] . v2 - > y ) ) ;
fixed_t xoffs = - lines [ i ] . v1 - > x ;
fixed_t yoffs = lines [ i ] . v1 - > y ;
//If no tag is given, apply to front sector
if ( lines [ i ] . args [ 0 ] = = 0 )
P_ApplyFlatAlignment ( lines [ i ] . frontsector , flatangle , xoffs , yoffs , lines [ i ] . args [ 1 ] ! = TMP_CEILING , lines [ i ] . args [ 1 ] ! = TMP_FLOOR ) ;
else
2014-03-15 16:59:03 +00:00
{
2021-12-29 07:12:28 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
P_ApplyFlatAlignment ( sectors + s , flatangle , xoffs , yoffs , lines [ i ] . args [ 1 ] ! = TMP_CEILING , lines [ i ] . args [ 1 ] ! = TMP_FLOOR ) ;
2014-03-15 16:59:03 +00:00
}
break ;
2021-12-29 07:12:28 +00:00
}
2014-03-15 16:59:03 +00:00
2021-12-30 13:26:51 +00:00
case 8 : // Set camera collision planes
if ( lines [ i ] . frontsector )
2021-12-31 13:06:06 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2021-12-30 13:26:51 +00:00
sectors [ s ] . camsec = lines [ i ] . frontsector - sectors ;
2014-03-15 16:59:03 +00:00
break ;
case 10 : // Vertical culling plane for sprites and FOFs
2021-12-29 07:12:28 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2014-03-15 16:59:03 +00:00
sectors [ s ] . cullheight = & lines [ i ] ; // This allows it to change in realtime!
break ;
case 50 : // Insta-Lower Sector
2021-06-25 09:11:16 +00:00
if ( ! udmf )
EV_DoFloor ( lines [ i ] . args [ 0 ] , & lines [ i ] , instantLower ) ;
2014-03-15 16:59:03 +00:00
break ;
case 51 : // Instant raise for ceilings
2021-06-25 09:11:16 +00:00
if ( ! udmf )
2021-06-27 07:54:48 +00:00
EV_DoCeiling ( lines [ i ] . args [ 0 ] , & lines [ i ] , instantRaise ) ;
2014-03-15 16:59:03 +00:00
break ;
case 52 : // Continuously Falling sector
2021-06-27 07:53:57 +00:00
EV_DoContinuousFall ( lines [ i ] . frontsector , lines [ i ] . backsector , lines [ i ] . args [ 0 ] < < FRACBITS , lines [ i ] . args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-06-25 09:11:16 +00:00
case 53 : // Continuous plane movement (slowdown)
2014-03-15 16:59:03 +00:00
if ( lines [ i ] . backsector )
2021-06-25 09:11:16 +00:00
{
2021-06-27 10:36:46 +00:00
if ( lines [ i ] . args [ 1 ] ! = TMP_CEILING )
2021-06-25 09:11:16 +00:00
EV_DoFloor ( lines [ i ] . args [ 0 ] , & lines [ i ] , bounceFloor ) ;
2021-06-27 10:36:46 +00:00
if ( lines [ i ] . args [ 1 ] ! = TMP_FLOOR )
2021-06-25 09:11:16 +00:00
EV_DoCeiling ( lines [ i ] . args [ 0 ] , & lines [ i ] , bounceCeiling ) ;
}
2014-03-15 16:59:03 +00:00
break ;
2021-06-25 09:11:16 +00:00
case 56 : // Continuous plane movement (constant)
2014-03-15 16:59:03 +00:00
if ( lines [ i ] . backsector )
2021-06-25 09:11:16 +00:00
{
2021-06-27 10:36:46 +00:00
if ( lines [ i ] . args [ 1 ] ! = TMP_CEILING )
2021-06-25 09:11:16 +00:00
EV_DoFloor ( lines [ i ] . args [ 0 ] , & lines [ i ] , bounceFloorCrush ) ;
2021-06-27 10:36:46 +00:00
if ( lines [ i ] . args [ 1 ] ! = TMP_FLOOR )
2021-06-25 09:11:16 +00:00
EV_DoCeiling ( lines [ i ] . args [ 0 ] , & lines [ i ] , bounceCeilingCrush ) ;
}
2014-03-15 16:59:03 +00:00
break ;
2021-06-25 09:11:16 +00:00
case 60 : // Moving platform
EV_DoElevator ( lines [ i ] . args [ 0 ] , & lines [ i ] , elevateContinuous ) ;
2014-03-15 16:59:03 +00:00
break ;
case 61 : // Crusher!
2021-06-26 10:53:14 +00:00
EV_DoCrush ( lines [ i ] . args [ 0 ] , & lines [ i ] , lines [ i ] . args [ 1 ] ? raiseAndCrush : crushAndRaise ) ;
2014-03-15 16:59:03 +00:00
break ;
case 63 : // support for drawn heights coming from different sector
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-12-29 10:03:00 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2014-03-15 16:59:03 +00:00
sectors [ s ] . heightsec = ( INT32 ) sec ;
break ;
case 64 : // Appearing/Disappearing FOF option
2021-12-29 20:03:47 +00:00
if ( lines [ i ] . args [ 0 ] = = 0 ) // Find FOFs by control sector tag
{
TAG_ITER_SECTORS ( lines [ i ] . args [ 1 ] , s )
{
2014-03-15 16:59:03 +00:00
for ( j = 0 ; ( unsigned ) j < sectors [ s ] . linecount ; j + + )
2021-12-29 20:03:47 +00:00
{
if ( sectors [ s ] . lines [ j ] - > special < 100 | | sectors [ s ] . lines [ j ] - > special > = 300 )
continue ;
Add_MasterDisappearer ( abs ( lines [ i ] . args [ 2 ] ) , abs ( lines [ i ] . args [ 3 ] ) , abs ( lines [ i ] . args [ 4 ] ) , ( INT32 ) ( sectors [ s ] . lines [ j ] - lines ) , ( INT32 ) i ) ;
}
}
}
else // Find FOFs by effect sector tag
2020-04-13 11:10:38 +00:00
{
2021-12-29 20:03:47 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , s )
2014-03-15 16:59:03 +00:00
{
2021-12-29 20:03:47 +00:00
if ( lines [ s ] . special < 100 | | lines [ s ] . special > = 300 )
2014-03-15 16:59:03 +00:00
continue ;
2021-12-29 20:03:47 +00:00
if ( lines [ i ] . args [ 1 ] ! = 0 & & ! Tag_Find ( & lines [ s ] . frontsector - > tags , lines [ i ] . args [ 1 ] ) )
continue ;
Add_MasterDisappearer ( abs ( lines [ i ] . args [ 2 ] ) , abs ( lines [ i ] . args [ 3 ] ) , abs ( lines [ i ] . args [ 4 ] ) , s , ( INT32 ) i ) ;
2014-03-15 16:59:03 +00:00
}
2020-04-13 11:10:38 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
2021-06-27 07:53:57 +00:00
case 66 : // Displace planes by front sector
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
P_AddPlaneDisplaceThinker ( lines [ i ] . args [ 1 ] , abs ( lines [ i ] . args [ 2 ] ) < < 8 , sides [ lines [ i ] . sidenum [ 0 ] ] . sector - sectors , s , lines [ i ] . args [ 2 ] < 0 ) ;
2016-12-13 22:42:47 +00:00
break ;
2020-05-03 06:43:16 +00:00
case 70 : // Add raise thinker to FOF
if ( udmf )
{
fixed_t destheight = lines [ i ] . args [ 2 ] < < FRACBITS ;
fixed_t startheight , topheight , bottomheight ;
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-03 06:43:16 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
startheight = lines [ l ] . frontsector - > ceilingheight ;
topheight = max ( startheight , destheight ) ;
bottomheight = min ( startheight , destheight ) ;
P_AddRaiseThinker ( lines [ l ] . frontsector , lines [ l ] . args [ 0 ] , lines [ i ] . args [ 1 ] < < FRACBITS , topheight , bottomheight , ( destheight < startheight ) , ! ! ( lines [ i ] . args [ 3 ] ) ) ;
}
}
break ;
case 71 : // Add air bob thinker to FOF
if ( udmf )
{
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-03 06:43:16 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
P_AddAirbob ( lines [ l ] . frontsector , lines [ l ] . args [ 0 ] , lines [ i ] . args [ 1 ] < < FRACBITS , ! ! ( lines [ i ] . args [ 2 ] & TMFB_REVERSE ) , ! ! ( lines [ i ] . args [ 2 ] & TMFB_SPINDASH ) , ! ! ( lines [ i ] . args [ 2 ] & TMFB_DYNAMIC ) ) ;
}
}
break ;
case 72 : // Add thwomp thinker to FOF
if ( udmf )
{
UINT16 sound = ( lines [ i ] . stringargs [ 0 ] ) ? get_number ( lines [ i ] . stringargs [ 0 ] ) : sfx_thwomp ;
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-03 06:43:16 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
2021-06-25 16:41:14 +00:00
P_AddThwompThinker ( lines [ l ] . frontsector , & lines [ l ] , lines [ i ] . args [ 1 ] < < ( FRACBITS - 3 ) , lines [ i ] . args [ 2 ] < < ( FRACBITS - 3 ) , sound ) ;
2020-05-03 06:43:16 +00:00
}
}
break ;
case 73 : // Add laser thinker to FOF
if ( udmf )
{
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-03 06:43:16 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
P_AddLaserThinker ( lines [ l ] . args [ 0 ] , lines + l , ! ! ( lines [ i ] . args [ 1 ] ) ) ;
}
}
break ;
2020-03-20 15:03:48 +00:00
case 100 : // FOF (solid)
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_SOLID | FOF_RENDERALL ;
2014-03-15 16:59:03 +00:00
2020-06-12 14:45:18 +00:00
//Appearance settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOPLANES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERPLANES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSIDES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERSIDES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_INSIDES )
2014-03-15 16:59:03 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_BOTHPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_ALLSIDES ;
2014-03-15 16:59:03 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_ONLYINSIDES )
2020-06-12 14:45:18 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_INVERTPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_INVERTSIDES ;
2020-06-12 14:45:18 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSHADE )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_NOSHADE ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2014-03-15 16:59:03 +00:00
2020-04-13 23:34:11 +00:00
//Tangibility settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_INTANGIBLETOP )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_REVERSEPLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_INTANGIBLEBOTTOM )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_PLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_DONTBLOCKPLAYER )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKPLAYER ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_DONTBLOCKOTHERS )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKOTHERS ;
2020-04-13 23:34:11 +00:00
//Cutting options
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERALL )
2020-04-13 23:34:11 +00:00
{
2023-08-17 17:31:55 +00:00
//If inside is visible from the outside, cut inner walls
if ( lines [ i ] . args [ 1 ] < 255 | | ( lines [ i ] . args [ 3 ] & TMFA_SPLAT ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_CUTEXTRA | FOF_EXTRA ;
2023-08-20 15:55:06 +00:00
else if ( ! ( lines [ i ] . args [ 4 ] & TMFT_VISIBLEFROMINSIDE ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_CUTLEVEL ;
2020-04-13 23:34:11 +00:00
}
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-03-20 15:15:39 +00:00
case 120 : // FOF (water)
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_RENDERPLANES | FOF_SWIMMABLE | FOF_BOTHPLANES | FOF_CUTEXTRA | FOF_EXTRA | FOF_CUTSPRITES ;
2022-01-04 18:33:17 +00:00
if ( ! ( lines [ i ] . args [ 3 ] & TMFW_NOSIDES ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_RENDERSIDES | FOF_ALLSIDES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFW_DOUBLESHADOW )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_DOUBLESHADOW ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFW_COLORMAPONLY )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_COLORMAPONLY ;
2022-01-04 18:33:17 +00:00
if ( ! ( lines [ i ] . args [ 3 ] & TMFW_NORIPPLE ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_RIPPLE ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFW_GOOWATER )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_GOOWATER ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFW_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 09:05:27 +00:00
case 150 : // FOF (Air bobbing)
2022-07-31 10:04:42 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , FOF_EXISTS | FOF_SOLID | FOF_RENDERALL , secthinkers ) ;
2020-06-12 15:48:43 +00:00
P_AddAirbob ( lines [ i ] . frontsector , lines [ i ] . args [ 0 ] , lines [ i ] . args [ 1 ] < < FRACBITS , ! ! ( lines [ i ] . args [ 2 ] & TMFB_REVERSE ) , ! ! ( lines [ i ] . args [ 2 ] & TMFB_SPINDASH ) , ! ! ( lines [ i ] . args [ 2 ] & TMFB_DYNAMIC ) ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 09:05:27 +00:00
case 160 : // FOF (Water bobbing)
2022-07-31 10:04:42 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , FOF_EXISTS | FOF_SOLID | FOF_RENDERALL | FOF_FLOATBOB , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 09:05:27 +00:00
case 170 : // FOF (Crumbling)
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_SOLID | FOF_RENDERALL | FOF_CRUMBLE ;
2014-03-15 16:59:03 +00:00
2020-04-15 07:44:33 +00:00
//Tangibility settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFT_INTANGIBLETOP )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_REVERSEPLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFT_INTANGIBLEBOTTOM )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_PLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFT_DONTBLOCKPLAYER )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKPLAYER ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFT_DONTBLOCKOTHERS )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKOTHERS ;
2014-03-15 16:59:03 +00:00
2020-04-15 07:44:33 +00:00
//Flags
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFC_NOSHADE )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_NOSHADE ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFC_NORETURN )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_NORETURN ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFC_FLOATBOB )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_FLOATBOB ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFC_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2020-04-15 07:44:33 +00:00
2023-08-17 17:31:55 +00:00
//If inside is visible from the outside, cut inner walls
if ( lines [ i ] . args [ 1 ] < 255 | | ( lines [ i ] . args [ 4 ] & TMFC_SPLAT ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_CUTEXTRA | FOF_EXTRA ;
2023-08-17 17:31:55 +00:00
//If player can view it from the inside, render insides
else if ( lines [ i ] . args [ 3 ] & TMFT_VISIBLEFROMINSIDE )
2020-04-15 07:44:33 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_BOTHPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_ALLSIDES ;
2020-04-15 07:44:33 +00:00
}
2023-08-17 17:31:55 +00:00
else
ffloorflags | = FOF_CUTLEVEL ;
2014-03-15 16:59:03 +00:00
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
if ( lines [ i ] . args [ 4 ] & TMFC_AIRBOB )
2020-05-01 12:02:48 +00:00
P_AddAirbob ( lines [ i ] . frontsector , lines [ i ] . args [ 0 ] , 16 * FRACUNIT , false , false , false ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 09:52:31 +00:00
case 190 : // FOF (Rising)
2020-04-27 09:19:07 +00:00
{
fixed_t ceilingtop = P_FindHighestCeilingSurrounding ( lines [ i ] . frontsector ) ;
fixed_t ceilingbottom = P_FindLowestCeilingSurrounding ( lines [ i ] . frontsector ) ;
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_SOLID | FOF_RENDERALL ;
2020-04-27 09:52:31 +00:00
2020-06-12 14:45:18 +00:00
//Appearance settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOPLANES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERPLANES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSIDES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERSIDES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_INSIDES )
2020-04-27 09:52:31 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_BOTHPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_ALLSIDES ;
2020-04-27 09:52:31 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_ONLYINSIDES )
2020-06-12 14:45:18 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_INVERTPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_INVERTSIDES ;
2020-06-12 14:45:18 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSHADE )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_NOSHADE ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2020-04-27 09:52:31 +00:00
//Tangibility settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_INTANGIBLETOP )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_REVERSEPLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_INTANGIBLEBOTTOM )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_PLATFORM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_DONTBLOCKPLAYER )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKPLAYER ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFT_DONTBLOCKOTHERS )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_BLOCKOTHERS ;
2020-04-27 09:52:31 +00:00
//Cutting options
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERALL )
2020-04-27 09:52:31 +00:00
{
2023-08-17 17:31:55 +00:00
//If inside is visible from the outside, cut inner walls
if ( lines [ i ] . args [ 1 ] < 255 | | ( lines [ i ] . args [ 3 ] & TMFA_SPLAT ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_CUTEXTRA | FOF_EXTRA ;
2023-08-20 15:55:06 +00:00
else if ( ! ( lines [ i ] . args [ 4 ] & TMFT_VISIBLEFROMINSIDE ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_CUTLEVEL ;
2020-04-27 09:52:31 +00:00
}
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2022-01-13 22:11:10 +00:00
P_AddRaiseThinker ( lines [ i ] . frontsector , lines [ i ] . args [ 0 ] , lines [ i ] . args [ 5 ] < < FRACBITS , ceilingtop , ceilingbottom , ! ! ( lines [ i ] . args [ 6 ] & TMFR_REVERSE ) , ! ! ( lines [ i ] . args [ 6 ] & TMFR_SPINDASH ) ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 09:19:07 +00:00
}
2020-04-14 07:03:00 +00:00
case 200 : // Light block
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_CUTSPRITES ;
2020-04-14 07:03:00 +00:00
if ( ! lines [ i ] . args [ 1 ] )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_DOUBLESHADOW ;
2022-01-10 19:59:57 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
case 202 : // Fog
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_RENDERALL | FOF_FOG | FOF_INVERTPLANES | FOF_INVERTSIDES | FOF_CUTEXTRA | FOF_EXTRA | FOF_DOUBLESHADOW | FOF_CUTSPRITES ;
2014-03-15 16:59:03 +00:00
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2020-02-16 19:19:24 +00:00
// SoM: Because it's fog, check for an extra colormap and set the fog flag...
2014-03-15 16:59:03 +00:00
if ( sectors [ sec ] . extra_colormap )
2020-02-16 19:19:24 +00:00
sectors [ sec ] . extra_colormap - > flags = CMF_FOG ;
2022-01-10 19:59:57 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-14 08:13:38 +00:00
case 220 : //Intangible
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_RENDERALL | FOF_CUTEXTRA | FOF_EXTRA | FOF_CUTSPRITES ;
2014-03-15 16:59:03 +00:00
2020-06-12 14:45:18 +00:00
//Appearance settings
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOPLANES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERPLANES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSIDES )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ FOF_RENDERSIDES ;
2022-01-04 18:33:17 +00:00
if ( ! ( lines [ i ] . args [ 3 ] & TMFA_INSIDES ) )
2020-04-14 08:13:38 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_BOTHPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_ALLSIDES ;
2020-04-14 08:13:38 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_ONLYINSIDES )
2020-06-12 14:45:18 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ffloorflags & FOF_RENDERPLANES )
ffloorflags | = FOF_INVERTPLANES ;
if ( ffloorflags & FOF_RENDERSIDES )
ffloorflags | = FOF_INVERTSIDES ;
2020-06-12 14:45:18 +00:00
}
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_NOSHADE )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_NOSHADE ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 3 ] & TMFA_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2014-03-15 16:59:03 +00:00
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
case 223 : // FOF (intangible, invisible) - for combining specials in a sector
2022-07-31 10:04:42 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , FOF_EXISTS | FOF_NOSHADE , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
case 250 : // Mario Block
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_SOLID | FOF_RENDERALL | FOF_CUTLEVEL | FOF_MARIO ;
2020-04-18 16:02:43 +00:00
if ( lines [ i ] . args [ 1 ] & TMFM_BRICK )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_GOOWATER ;
2020-04-18 16:02:43 +00:00
if ( lines [ i ] . args [ 1 ] & TMFM_INVISIBLE )
2022-07-31 10:04:42 +00:00
ffloorflags & = ~ ( FOF_SOLID | FOF_RENDERALL | FOF_CUTLEVEL ) ;
2016-06-06 17:30:58 +00:00
2022-01-10 19:59:57 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
case 251 : // A THWOMP!
2020-04-27 11:01:31 +00:00
{
2020-05-01 12:02:48 +00:00
UINT16 sound = ( lines [ i ] . stringargs [ 0 ] ) ? get_number ( lines [ i ] . stringargs [ 0 ] ) : sfx_thwomp ;
2021-06-25 16:41:14 +00:00
P_AddThwompThinker ( lines [ i ] . frontsector , & lines [ i ] , lines [ i ] . args [ 1 ] < < ( FRACBITS - 3 ) , lines [ i ] . args [ 2 ] < < ( FRACBITS - 3 ) , sound ) ;
2022-07-31 10:04:42 +00:00
P_AddFakeFloorsByLine ( i , 0xff , TMB_TRANSLUCENT , FOF_EXISTS | FOF_SOLID | FOF_RENDERALL | FOF_CUTLEVEL , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2020-04-27 11:01:31 +00:00
}
2014-03-15 16:59:03 +00:00
2020-04-14 09:45:37 +00:00
case 254 : // Bustable block
2020-05-02 19:19:55 +00:00
{
UINT8 busttype = BT_REGULAR ;
2021-12-03 17:48:16 +00:00
ffloorbustflags_e bustflags = 0 ;
2020-05-02 19:19:55 +00:00
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_BLOCKOTHERS | FOF_RENDERALL | FOF_BUSTUP ;
2014-03-15 16:59:03 +00:00
2020-04-14 09:45:37 +00:00
//Bustable type
2022-01-04 18:33:17 +00:00
switch ( lines [ i ] . args [ 3 ] )
2020-04-14 09:45:37 +00:00
{
2020-04-18 16:02:43 +00:00
case TMFB_TOUCH :
2020-05-02 19:19:55 +00:00
busttype = BT_TOUCH ;
2020-04-14 09:45:37 +00:00
break ;
2020-04-18 16:02:43 +00:00
case TMFB_SPIN :
2020-10-27 21:13:10 +00:00
busttype = BT_SPINBUST ;
2020-04-14 09:45:37 +00:00
break ;
2020-05-02 19:19:55 +00:00
case TMFB_REGULAR :
busttype = BT_REGULAR ;
2020-04-14 09:45:37 +00:00
break ;
2020-05-02 19:19:55 +00:00
case TMFB_STRONG :
busttype = BT_STRONG ;
2020-04-14 09:45:37 +00:00
break ;
}
2014-03-15 16:59:03 +00:00
2020-05-02 19:19:55 +00:00
//Flags
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFB_PUSHABLES )
2021-12-03 17:48:16 +00:00
bustflags | = FB_PUSHABLES ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFB_EXECUTOR )
2021-12-03 17:48:16 +00:00
bustflags | = FB_EXECUTOR ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFB_ONLYBOTTOM )
2021-12-03 17:48:16 +00:00
bustflags | = FB_ONLYBOTTOM ;
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFB_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2014-03-15 16:59:03 +00:00
2021-12-03 17:48:16 +00:00
if ( busttype ! = BT_TOUCH | | bustflags & FB_ONLYBOTTOM )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_BLOCKPLAYER ;
2014-03-15 16:59:03 +00:00
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2020-05-02 19:19:55 +00:00
{
2022-01-04 18:33:17 +00:00
ffloor_t * fflr = P_AddFakeFloor ( & sectors [ s ] , lines [ i ] . frontsector , lines + i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2021-06-27 15:49:00 +00:00
if ( ! fflr )
continue ;
2021-12-03 17:48:16 +00:00
fflr - > bustflags = bustflags ;
2020-05-02 19:19:55 +00:00
fflr - > busttype = busttype ;
2022-01-04 18:33:17 +00:00
fflr - > busttag = lines [ i ] . args [ 5 ] ;
2020-05-02 19:19:55 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
2020-05-02 19:19:55 +00:00
}
2014-03-15 16:59:03 +00:00
case 257 : // Quicksand
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_QUICKSAND | FOF_RENDERALL | FOF_ALLSIDES | FOF_CUTSPRITES ;
2020-06-12 16:22:16 +00:00
if ( ! ( lines [ i ] . args [ 1 ] ) )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_RIPPLE ;
2016-06-06 17:30:58 +00:00
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2020-05-02 20:07:42 +00:00
{
2022-01-10 19:59:57 +00:00
ffloor_t * fflr = P_AddFakeFloor ( & sectors [ s ] , lines [ i ] . frontsector , lines + i , 0xff , TMB_TRANSLUCENT , ffloorflags , secthinkers ) ;
2021-06-27 15:49:00 +00:00
if ( ! fflr )
continue ;
2020-06-12 16:22:16 +00:00
fflr - > sinkspeed = abs ( lines [ i ] . args [ 2 ] ) < < ( FRACBITS - 1 ) ;
fflr - > friction = abs ( lines [ i ] . args [ 3 ] ) < < ( FRACBITS - 6 ) ;
2020-05-02 20:07:42 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
case 258 : // Laser block
2022-07-31 10:04:42 +00:00
ffloorflags = FOF_EXISTS | FOF_RENDERALL | FOF_NOSHADE | FOF_EXTRA | FOF_CUTEXTRA | FOF_TRANSLUCENT ;
2022-01-04 18:33:17 +00:00
P_AddLaserThinker ( lines [ i ] . args [ 0 ] , lines + i , ! ! ( lines [ i ] . args [ 3 ] & TMFL_NOBOSSES ) ) ;
if ( lines [ i ] . args [ 3 ] & TMFL_SPLAT )
2022-07-31 10:04:42 +00:00
ffloorflags | = FOF_SPLAT ;
2022-01-04 18:33:17 +00:00
P_AddFakeFloorsByLine ( i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
break ;
2019-12-29 11:01:41 +00:00
case 259 : // Custom FOF
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2014-03-15 16:59:03 +00:00
{
2022-01-04 18:33:17 +00:00
ffloor_t * fflr = P_AddFakeFloor ( & sectors [ s ] , lines [ i ] . frontsector , lines + i , lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , lines [ i ] . args [ 3 ] , secthinkers ) ;
2021-06-27 15:49:00 +00:00
if ( ! fflr )
continue ;
2020-05-02 21:00:22 +00:00
if ( ! udmf ) // Ugly backwards compatibility stuff
{
2022-07-31 10:04:42 +00:00
if ( lines [ i ] . args [ 3 ] & FOF_QUICKSAND )
2020-05-02 21:00:22 +00:00
{
fflr - > sinkspeed = abs ( lines [ i ] . dx ) > > 1 ;
fflr - > friction = abs ( lines [ i ] . dy ) > > 6 ;
}
2022-07-31 10:04:42 +00:00
if ( lines [ i ] . args [ 3 ] & FOF_BUSTUP )
2020-05-02 21:00:22 +00:00
{
2022-01-04 18:33:17 +00:00
switch ( lines [ i ] . args [ 4 ] % TMFB_ONLYBOTTOM )
2020-05-03 07:13:14 +00:00
{
case TMFB_TOUCH :
fflr - > busttype = BT_TOUCH ;
break ;
case TMFB_SPIN :
2020-10-27 21:13:10 +00:00
fflr - > busttype = BT_SPINBUST ;
2020-05-03 07:13:14 +00:00
break ;
case TMFB_REGULAR :
fflr - > busttype = BT_REGULAR ;
break ;
case TMFB_STRONG :
fflr - > busttype = BT_STRONG ;
break ;
}
2020-05-02 21:00:22 +00:00
2022-01-04 18:33:17 +00:00
if ( lines [ i ] . args [ 4 ] & TMFB_ONLYBOTTOM )
2021-12-03 17:48:16 +00:00
fflr - > bustflags | = FB_ONLYBOTTOM ;
2022-01-05 18:43:40 +00:00
if ( lines [ i ] . flags & ML_MIDSOLID )
2021-12-03 17:48:16 +00:00
fflr - > bustflags | = FB_PUSHABLES ;
2022-01-05 18:43:40 +00:00
if ( lines [ i ] . flags & ML_WRAPMIDTEX )
2020-05-02 21:00:22 +00:00
{
2021-12-03 17:48:16 +00:00
fflr - > bustflags | = FB_EXECUTOR ;
2020-05-02 21:00:22 +00:00
fflr - > busttag = P_AproxDistance ( lines [ i ] . dx , lines [ i ] . dy ) > > FRACBITS ;
}
}
}
2014-03-15 16:59:03 +00:00
}
break ;
2021-12-09 06:30:55 +00:00
case 300 : // Trigger linedef executor
2021-12-09 06:47:01 +00:00
case 303 : // Count rings
2021-12-09 18:17:16 +00:00
case 305 : // Character ability
2021-12-09 06:56:15 +00:00
case 314 : // Pushable linedef executors (count # of pushables)
2021-12-09 07:08:04 +00:00
case 317 : // Condition set trigger
case 319 : // Unlockable trigger
2021-12-09 18:33:02 +00:00
case 331 : // Player skin
2021-12-09 18:49:17 +00:00
case 334 : // Object dye
2021-12-09 19:37:39 +00:00
case 337 : // Emerald check
2022-09-17 00:14:25 +00:00
case 343 : // Gravity check
2021-12-09 06:30:55 +00:00
if ( lines [ i ] . args [ 0 ] > TMT_EACHTIMEMASK )
P_AddEachTimeThinker ( & lines [ i ] , lines [ i ] . args [ 0 ] = = TMT_EACHTIMEENTERANDEXIT ) ;
break ;
2014-03-15 16:59:03 +00:00
case 308 : // Race-only linedef executor. Triggers once.
2021-12-09 17:56:50 +00:00
if ( ! P_CheckGametypeRules ( lines [ i ] . args [ 2 ] , ( UINT32 ) lines [ i ] . args [ 1 ] ) )
{
lines [ i ] . special = 0 ;
break ;
}
if ( lines [ i ] . args [ 0 ] > TMT_EACHTIMEMASK )
P_AddEachTimeThinker ( & lines [ i ] , lines [ i ] . args [ 0 ] = = TMT_EACHTIMEENTERANDEXIT ) ;
break ;
// Linedef executor triggers for CTF teams.
case 309 :
if ( ! ( gametyperules & GTR_TEAMFLAGS ) )
{
2014-03-15 16:59:03 +00:00
lines [ i ] . special = 0 ;
2021-12-09 17:56:50 +00:00
break ;
}
if ( lines [ i ] . args [ 0 ] > TMT_EACHTIMEMASK )
P_AddEachTimeThinker ( & lines [ i ] , lines [ i ] . args [ 0 ] = = TMT_EACHTIMEENTERANDEXIT ) ;
2014-03-15 16:59:03 +00:00
break ;
// No More Enemies Linedef Exec
case 313 :
2020-04-18 07:21:04 +00:00
P_AddNoEnemiesThinker ( & lines [ i ] ) ;
2014-03-15 16:59:03 +00:00
break ;
2014-11-12 00:55:07 +00:00
// Trigger on X calls
case 321 :
2021-12-09 17:15:27 +00:00
lines [ i ] . callcount = ( lines [ i ] . args [ 2 ] & & lines [ i ] . args [ 3 ] > 0 ) ? lines [ i ] . args [ 3 ] : lines [ i ] . args [ 1 ] ; // optional "starting" count
if ( lines [ i ] . args [ 0 ] > TMXT_EACHTIMEMASK ) // Each time
P_AddEachTimeThinker ( & lines [ i ] , lines [ i ] . args [ 0 ] = = TMXT_EACHTIMEENTERANDEXIT ) ;
2014-11-12 00:55:07 +00:00
break ;
2019-07-31 22:17:17 +00:00
case 449 : // Enable bosses with parameter
{
2021-09-20 13:07:08 +00:00
INT32 bossid = lines [ i ] . args [ 0 ] ;
2019-07-31 22:17:17 +00:00
if ( bossid & ~ 15 ) // if any bits other than first 16 are set
{
CONS_Alert ( CONS_WARNING ,
2021-09-20 13:07:08 +00:00
M_GetText ( " Boss enable linedef has an invalid boss ID (%d). \n Consider changing it or removing it entirely. \n " ) ,
bossid ) ;
2019-07-31 22:17:17 +00:00
break ;
}
2021-09-20 13:07:08 +00:00
if ( ! ( lines [ i ] . args [ 1 ] ) )
2019-07-31 22:17:17 +00:00
{
bossdisabled | = ( 1 < < bossid ) ; // gotta disable in the first place to enable
CONS_Debug ( DBG_GAMELOGIC , " Line type 449 spawn effect: bossid disabled = %d " , bossid ) ;
}
break ;
}
2021-09-20 06:36:55 +00:00
case 600 : // Copy light level to tagged sector's planes
2014-03-15 16:59:03 +00:00
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-09-20 06:36:55 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
{
if ( lines [ i ] . args [ 1 ] ! = TMP_CEILING )
sectors [ s ] . floorlightsec = ( INT32 ) sec ;
if ( lines [ i ] . args [ 1 ] ! = TMP_FLOOR )
sectors [ s ] . ceilinglightsec = ( INT32 ) sec ;
}
2014-03-15 16:59:03 +00:00
break ;
case 602 : // Adjustable pulsating light
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-09-19 15:07:08 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
P_SpawnAdjustableGlowingLight ( & sectors [ s ] , lines [ i ] . args [ 2 ] ,
lines [ i ] . args [ 3 ] ? sectors [ s ] . lightlevel : lines [ i ] . args [ 4 ] , lines [ i ] . args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
case 603 : // Adjustable flickering light
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-09-19 15:07:08 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
P_SpawnAdjustableFireFlicker ( & sectors [ s ] , lines [ i ] . args [ 2 ] ,
lines [ i ] . args [ 3 ] ? sectors [ s ] . lightlevel : lines [ i ] . args [ 4 ] , lines [ i ] . args [ 1 ] ) ;
2014-03-15 16:59:03 +00:00
break ;
2021-09-19 15:07:08 +00:00
case 604 : // Adjustable Blinking Light
2014-03-15 16:59:03 +00:00
sec = sides [ * lines [ i ] . sidenum ] . sector - sectors ;
2021-12-31 13:09:22 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2021-09-19 15:07:08 +00:00
P_SpawnAdjustableStrobeFlash ( & sectors [ s ] , lines [ i ] . args [ 3 ] ,
( lines [ i ] . args [ 4 ] & TMB_USETARGET ) ? sectors [ s ] . lightlevel : lines [ i ] . args [ 5 ] ,
lines [ i ] . args [ 1 ] , lines [ i ] . args [ 2 ] , lines [ i ] . args [ 4 ] & TMB_SYNC ) ;
2014-03-15 16:59:03 +00:00
break ;
case 606 : // HACK! Copy colormaps. Just plain colormaps.
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2020-03-20 10:19:30 +00:00
{
extracolormap_t * exc ;
2020-03-20 11:19:02 +00:00
if ( sectors [ s ] . colormap_protected )
continue ;
2020-03-20 10:19:30 +00:00
if ( ! udmf )
exc = sides [ lines [ i ] . sidenum [ 0 ] ] . colormap_data ;
else
{
if ( ! lines [ i ] . args [ 1 ] )
exc = lines [ i ] . frontsector - > extra_colormap ;
else
{
2020-04-18 12:04:15 +00:00
INT32 sourcesec = Tag_Iterate_Sectors ( lines [ i ] . args [ 1 ] , 0 ) ;
2020-03-20 10:19:30 +00:00
if ( sourcesec = = - 1 )
{
CONS_Debug ( DBG_GAMELOGIC , " Line type 606: Can't find sector with source colormap (tag %d)! \n " , lines [ i ] . args [ 1 ] ) ;
return ;
}
exc = sectors [ sourcesec ] . extra_colormap ;
}
}
sectors [ s ] . extra_colormap = sectors [ s ] . spawn_extra_colormap = exc ;
}
2014-03-15 16:59:03 +00:00
break ;
default :
break ;
}
}
2020-05-02 19:50:49 +00:00
// And another round, this time with all FOFs already created
for ( i = 0 ; i < numlines ; i + + )
{
switch ( lines [ i ] . special )
{
INT32 s ;
INT32 l ;
2020-05-03 06:43:16 +00:00
case 74 : // Make FOF bustable
2020-05-02 19:50:49 +00:00
{
UINT8 busttype = BT_REGULAR ;
2021-12-03 17:48:16 +00:00
ffloorbustflags_e bustflags = 0 ;
2020-05-02 19:50:49 +00:00
if ( ! udmf )
break ;
switch ( lines [ i ] . args [ 1 ] )
{
case TMFB_TOUCH :
busttype = BT_TOUCH ;
break ;
case TMFB_SPIN :
2020-10-27 21:13:10 +00:00
busttype = BT_SPINBUST ;
2020-05-02 19:50:49 +00:00
break ;
case TMFB_REGULAR :
busttype = BT_REGULAR ;
break ;
case TMFB_STRONG :
busttype = BT_STRONG ;
break ;
}
if ( lines [ i ] . args [ 2 ] & TMFB_PUSHABLES )
2021-12-03 17:48:16 +00:00
bustflags | = FB_PUSHABLES ;
2020-05-02 19:50:49 +00:00
if ( lines [ i ] . args [ 2 ] & TMFB_EXECUTOR )
2021-12-03 17:48:16 +00:00
bustflags | = FB_EXECUTOR ;
2020-05-02 19:50:49 +00:00
if ( lines [ i ] . args [ 2 ] & TMFB_ONLYBOTTOM )
2021-12-03 17:48:16 +00:00
bustflags | = FB_ONLYBOTTOM ;
2020-05-02 19:50:49 +00:00
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-02 19:50:49 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ l ] . args [ 0 ] , s )
2020-05-02 19:50:49 +00:00
{
ffloor_t * rover ;
for ( rover = sectors [ s ] . ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master ! = lines + l )
continue ;
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_BUSTUP ;
rover - > spawnflags | = FOF_BUSTUP ;
2021-12-03 17:48:16 +00:00
rover - > bustflags = bustflags ;
2020-05-02 19:50:49 +00:00
rover - > busttype = busttype ;
rover - > busttag = lines [ i ] . args [ 3 ] ;
CheckForBustableBlocks = true ;
break ;
}
}
}
break ;
}
2020-05-02 21:00:22 +00:00
2020-05-03 06:43:16 +00:00
case 75 : // Make FOF quicksand
2020-05-02 21:00:22 +00:00
{
if ( ! udmf )
break ;
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-02 21:00:22 +00:00
{
if ( lines [ l ] . special < 100 | | lines [ l ] . special > = 300 )
continue ;
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ l ] . args [ 0 ] , s )
2020-05-02 21:00:22 +00:00
{
ffloor_t * rover ;
for ( rover = sectors [ s ] . ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master ! = lines + l )
continue ;
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_QUICKSAND ;
rover - > spawnflags | = FOF_QUICKSAND ;
2020-05-02 21:00:22 +00:00
rover - > sinkspeed = abs ( lines [ i ] . args [ 1 ] ) < < ( FRACBITS - 1 ) ;
rover - > friction = abs ( lines [ i ] . args [ 2 ] ) < < ( FRACBITS - 6 ) ;
CheckForQuicksand = true ;
break ;
}
}
}
break ;
}
2020-05-03 10:44:30 +00:00
case 76 : // Make FOF bouncy
{
if ( udmf )
{
2021-06-23 15:55:30 +00:00
TAG_ITER_LINES ( lines [ i ] . args [ 0 ] , l )
2020-05-03 10:44:30 +00:00
P_MakeFOFBouncy ( lines + i , lines + l ) ;
}
else
{
2021-06-23 15:55:30 +00:00
TAG_ITER_SECTORS ( lines [ i ] . args [ 0 ] , s )
2020-05-03 10:44:30 +00:00
for ( j = 0 ; ( unsigned ) j < sectors [ s ] . linecount ; j + + )
P_MakeFOFBouncy ( lines + i , sectors [ s ] . lines [ j ] ) ;
}
break ;
}
2020-05-02 19:50:49 +00:00
}
}
2014-03-15 16:59:03 +00:00
// Allocate each list
for ( i = 0 ; i < numsectors ; i + + )
if ( secthinkers [ i ] . thinkers )
Z_Free ( secthinkers [ i ] . thinkers ) ;
Z_Free ( secthinkers ) ;
// haleyjd 02/20/06: spawn polyobjects
Polyobj_InitLevel ( ) ;
for ( i = 0 ; i < numlines ; i + + )
{
switch ( lines [ i ] . special )
{
case 30 : // Polyobj_Flag
2020-05-04 07:58:27 +00:00
PolyFlag ( & lines [ i ] ) ;
2014-03-15 16:59:03 +00:00
break ;
2014-11-12 00:55:07 +00:00
case 31 : // Polyobj_Displace
PolyDisplace ( & lines [ i ] ) ;
break ;
2019-04-05 09:50:59 +00:00
case 32 : // Polyobj_RotDisplace
PolyRotDisplace ( & lines [ i ] ) ;
break ;
2014-03-15 16:59:03 +00:00
}
}
2023-08-23 18:29:16 +00:00
// Copy portals
for ( i = 0 ; i < numlines ; i + + )
{
2023-08-24 20:36:38 +00:00
if ( lines [ i ] . special ! = SPECIAL_SECTOR_SETPORTAL )
2023-08-23 18:29:16 +00:00
continue ;
int portal_type = lines [ i ] . args [ 1 ] ;
2023-08-24 03:51:52 +00:00
if ( portal_type ! = TMSECPORTAL_COPIED )
2023-08-23 18:29:16 +00:00
continue ;
int target_sector_tag = lines [ i ] . args [ 0 ] ;
int plane_type = lines [ i ] . args [ 2 ] ;
int tag_to_copy = lines [ i ] . args [ 3 ] ;
2023-08-23 20:49:29 +00:00
if ( plane_type = = 3 )
plane_type = TMP_BOTH ;
2023-08-23 18:29:16 +00:00
if ( target_sector_tag = = 0 )
2023-08-23 20:49:29 +00:00
P_DoPortalCopyFromLine ( lines [ i ] . frontsector , plane_type , tag_to_copy ) ;
2023-08-23 18:29:16 +00:00
else
{
INT32 s1 = - 1 ;
TAG_ITER_SECTORS ( target_sector_tag , s1 )
2023-08-23 20:49:29 +00:00
P_DoPortalCopyFromLine ( & sectors [ s1 ] , plane_type , tag_to_copy ) ;
2023-08-23 18:29:16 +00:00
}
}
2021-08-08 17:43:42 +00:00
if ( ! fromnetsave )
P_RunLevelLoadExecutors ( ) ;
2014-03-15 16:59:03 +00:00
}
/** Adds 3Dfloors as appropriate based on a common control linedef.
*
* \ param line Control linedef to use .
2020-06-12 06:16:50 +00:00
* \ param alpha Alpha value ( 0 - 255 ) .
2022-01-04 18:33:17 +00:00
* \ param blendmode Blending mode .
2014-03-15 16:59:03 +00:00
* \ param ffloorflags 3 Dfloor flags to use .
* \ param secthkiners Lists of thinkers sorted by sector . May be NULL .
* \ sa P_SpawnSpecials , P_AddFakeFloor
* \ author Graue < graue @ oceanbase . org >
*/
2022-01-04 18:33:17 +00:00
static void P_AddFakeFloorsByLine ( size_t line , INT32 alpha , UINT8 blendmode , ffloortype_e ffloorflags , thinkerlist_t * secthinkers )
2014-03-15 16:59:03 +00:00
{
INT32 s ;
2020-11-13 12:23:14 +00:00
mtag_t tag = lines [ line ] . args [ 0 ] ;
2014-03-15 16:59:03 +00:00
size_t sec = sides [ * lines [ line ] . sidenum ] . sector - sectors ;
2020-04-12 16:52:47 +00:00
line_t * li = lines + line ;
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( tag , s )
2022-01-04 18:33:17 +00:00
P_AddFakeFloor ( & sectors [ s ] , & sectors [ sec ] , li , alpha , blendmode , ffloorflags , secthinkers ) ;
2014-03-15 16:59:03 +00:00
}
/*
SoM : 3 / 8 / 2000 : General scrolling functions .
T_Scroll ,
Add_Scroller ,
Add_WallScroller ,
P_SpawnScrollers
*/
2014-11-12 00:55:07 +00:00
// helper function for T_Scroll
static void P_DoScrollMove ( mobj_t * thing , fixed_t dx , fixed_t dy , INT32 exclusive )
{
fixed_t fuckaj = 0 ; // Nov 05 14:12:08 <+MonsterIestyn> I've heard of explicitly defined variables but this is ridiculous
if ( thing - > player )
{
if ( ! ( dx | dy ) )
{
thing - > player - > cmomx = 0 ;
thing - > player - > cmomy = 0 ;
}
else
{
thing - > player - > cmomx + = dx ;
thing - > player - > cmomy + = dy ;
thing - > player - > cmomx = FixedMul ( thing - > player - > cmomx , 0xe800 ) ;
thing - > player - > cmomy = FixedMul ( thing - > player - > cmomy , 0xe800 ) ;
}
}
if ( thing - > player & & ( thing - > player - > pflags & PF_SPINNING ) & & ( thing - > player - > rmomx | | thing - > player - > rmomy ) & & ! ( thing - > player - > pflags & PF_STARTDASH ) )
fuckaj = FixedDiv ( 549 * ORIG_FRICTION , 500 * FRACUNIT ) ;
else if ( thing - > friction ! = ORIG_FRICTION )
fuckaj = thing - > friction ;
if ( fuckaj ) {
// refactor thrust for new friction
dx = FixedDiv ( dx , CARRYFACTOR ) ;
dy = FixedDiv ( dy , CARRYFACTOR ) ;
dx = FixedMul ( dx , FRACUNIT - fuckaj ) ;
dy = FixedMul ( dy , FRACUNIT - fuckaj ) ;
}
thing - > momx + = dx ;
thing - > momy + = dy ;
if ( exclusive )
2016-03-09 09:30:52 +00:00
thing - > eflags | = MFE_PUSHED ;
2014-11-12 00:55:07 +00:00
}
2014-03-15 16:59:03 +00:00
/** Processes an active scroller.
* 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 an
* edge around a sector can be used to control the direction of the sector ' s
* scrolling , which is usually what is desired .
*
* \ param s Thinker for the scroller to process .
* \ todo Split up into multiple functions .
* \ todo Use attached lists to make : : sc_carry_ceiling case faster and
* cleaner .
* \ sa Add_Scroller , Add_WallScroller , P_SpawnScrollers
* \ author Steven McGranahan
* \ author Graue < graue @ oceanbase . org >
*/
void T_Scroll ( scroll_t * s )
{
fixed_t dx = s - > dx , dy = s - > dy ;
boolean is3dblock = false ;
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 ) ;
}
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 ;
msecnode_t * node ;
mobj_t * thing ;
line_t * line ;
size_t i ;
INT32 sect ;
2018-05-21 19:02:30 +00:00
ffloor_t * rover ;
2014-03-15 16:59:03 +00:00
case sc_side : // scroll wall texture
side = sides + s - > affectee ;
side - > textureoffset + = dx ;
side - > rowoffset + = dy ;
break ;
case sc_floor : // scroll floor texture
sec = sectors + s - > affectee ;
2022-11-25 23:01:27 +00:00
sec - > floorxoffset + = dx ;
sec - > flooryoffset + = dy ;
2014-03-15 16:59:03 +00:00
break ;
case sc_ceiling : // scroll ceiling texture
sec = sectors + s - > affectee ;
2022-11-25 23:01:27 +00:00
sec - > ceilingxoffset + = dx ;
sec - > ceilingyoffset + = dy ;
2014-03-15 16:59:03 +00:00
break ;
case sc_carry :
sec = sectors + s - > affectee ;
height = sec - > floorheight ;
// sec is the control sector, find the real sector(s) to use
for ( i = 0 ; i < sec - > linecount ; i + + )
{
line = sec - > lines [ i ] ;
if ( line - > special < 100 | | line - > special > = 300 )
is3dblock = false ;
else
is3dblock = true ;
if ( ! is3dblock )
continue ;
2021-09-26 04:48:26 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , sect )
2014-03-15 16:59:03 +00:00
{
sector_t * psec ;
psec = sectors + sect ;
2018-05-21 19:02:30 +00:00
// Find the FOF corresponding to the control linedef
for ( rover = psec - > ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master = = sec - > lines [ i ] )
break ;
}
if ( ! rover ) // This should be impossible, but don't complain if it is the case somehow
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) ) // If the FOF does not "exist", we pretend that nobody's there
2018-05-21 19:02:30 +00:00
continue ;
2016-06-09 13:16:02 +00:00
for ( node = psec - > touching_thinglist ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
2016-03-09 09:30:52 +00:00
if ( thing - > eflags & MFE_PUSHED ) // Already pushed this tic by an exclusive pusher.
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
height = P_GetSpecialBottomZ ( thing , sec , psec ) ;
2014-03-15 16:59:03 +00:00
if ( ! ( thing - > flags & MF_NOCLIP ) ) // Thing must be clipped
if ( ! ( thing - > flags & MF_NOGRAVITY | | thing - > z + thing - > height ! = height ) ) // Thing must a) be non-floating and have z+height == height
{
// Move objects only if on floor
// non-floating, and clipped.
2014-11-12 00:55:07 +00:00
P_DoScrollMove ( thing , dx , dy , s - > exclusive ) ;
2014-03-15 16:59:03 +00:00
}
} // end of for loop through touching_thinglist
} // end of loop through sectors
}
if ( ! is3dblock )
{
2016-06-09 13:16:02 +00:00
for ( node = sec - > touching_thinglist ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
2016-03-09 09:30:52 +00:00
if ( thing - > eflags & MFE_PUSHED )
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
height = P_GetSpecialBottomZ ( thing , sec , sec ) ;
2014-11-12 00:55:07 +00:00
if ( ! ( thing - > flags & MF_NOCLIP ) & &
2014-03-15 16:59:03 +00:00
( ! ( thing - > flags & MF_NOGRAVITY | | thing - > z > height ) ) )
{
// Move objects only if on floor or underwater,
// non-floating, and clipped.
2014-11-12 00:55:07 +00:00
P_DoScrollMove ( thing , dx , dy , s - > exclusive ) ;
2014-03-15 16:59:03 +00:00
}
}
}
break ;
case sc_carry_ceiling : // carry on ceiling (FOF scrolling)
sec = sectors + s - > affectee ;
height = sec - > ceilingheight ;
// sec is the control sector, find the real sector(s) to use
for ( i = 0 ; i < sec - > linecount ; i + + )
{
line = sec - > lines [ i ] ;
if ( line - > special < 100 | | line - > special > = 300 )
is3dblock = false ;
else
is3dblock = true ;
if ( ! is3dblock )
continue ;
2021-09-26 04:48:26 +00:00
TAG_ITER_SECTORS ( line - > args [ 0 ] , sect )
2014-03-15 16:59:03 +00:00
{
sector_t * psec ;
psec = sectors + sect ;
2018-05-21 19:02:30 +00:00
// Find the FOF corresponding to the control linedef
for ( rover = psec - > ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master = = sec - > lines [ i ] )
break ;
}
if ( ! rover ) // This should be impossible, but don't complain if it is the case somehow
continue ;
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_EXISTS ) ) // If the FOF does not "exist", we pretend that nobody's there
2018-05-21 19:02:30 +00:00
continue ;
2016-06-09 13:16:02 +00:00
for ( node = psec - > touching_thinglist ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
2016-03-09 09:30:52 +00:00
if ( thing - > eflags & MFE_PUSHED )
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
height = P_GetSpecialTopZ ( thing , sec , psec ) ;
2014-03-15 16:59:03 +00:00
if ( ! ( thing - > flags & MF_NOCLIP ) ) // Thing must be clipped
if ( ! ( thing - > flags & MF_NOGRAVITY | | thing - > z ! = height ) ) // Thing must a) be non-floating and have z == height
{
// Move objects only if on floor or underwater,
// non-floating, and clipped.
2014-11-12 00:55:07 +00:00
P_DoScrollMove ( thing , dx , dy , s - > exclusive ) ;
2014-03-15 16:59:03 +00:00
}
} // end of for loop through touching_thinglist
} // end of loop through sectors
}
if ( ! is3dblock )
{
2016-06-09 13:16:02 +00:00
for ( node = sec - > touching_thinglist ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
2016-03-09 09:30:52 +00:00
if ( thing - > eflags & MFE_PUSHED )
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
height = P_GetSpecialTopZ ( thing , sec , sec ) ;
2014-11-12 00:55:07 +00:00
if ( ! ( thing - > flags & MF_NOCLIP ) & &
2014-08-04 03:49:33 +00:00
( ! ( thing - > flags & MF_NOGRAVITY | | thing - > z + thing - > height < height ) ) )
2014-03-15 16:59:03 +00:00
{
// Move objects only if on floor or underwater,
// non-floating, and clipped.
2014-11-12 00:55:07 +00:00
P_DoScrollMove ( thing , dx , dy , s - > exclusive ) ;
2014-03-15 16:59:03 +00:00
}
}
}
break ; // end of sc_carry_ceiling
} // end of switch
}
2022-10-08 09:06:14 +00:00
static boolean IsSector3DBlock ( sector_t * sec )
{
size_t i ;
for ( i = 0 ; i < sec - > linecount ; i + + )
{
if ( sec - > lines [ i ] - > special > = 100 & & sec - > lines [ i ] - > special < 300 )
return true ;
}
return false ;
}
2014-03-15 16:59:03 +00:00
/** Adds a generalized scroller to the thinker list.
*
* \ param type The enumerated type of scrolling .
* \ param dx x speed of scrolling or its acceleration .
* \ param dy y speed of scrolling or its acceleration .
* \ param control Sector whose heights control this scroller ' s effect
* remotely , or - 1 if there is no control sector .
* \ param affectee Index of the affected object , sector or sidedef .
* \ param accel Nonzero for an accelerative effect .
* \ sa Add_WallScroller , P_SpawnScrollers , T_Scroll
*/
static void Add_Scroller ( INT32 type , fixed_t dx , fixed_t dy , INT32 control , INT32 affectee , INT32 accel , INT32 exclusive )
{
scroll_t * s = Z_Calloc ( sizeof * s , PU_LEVSPEC , NULL ) ;
s - > thinker . function . acp1 = ( actionf_p1 ) T_Scroll ;
s - > type = type ;
s - > dx = dx ;
s - > dy = dy ;
s - > accel = accel ;
s - > exclusive = exclusive ;
s - > vdx = s - > vdy = 0 ;
2021-12-31 12:58:41 +00:00
s - > control = control ;
if ( s - > control ! = - 1 )
2014-03-15 16:59:03 +00:00
s - > last_height = sectors [ control ] . floorheight + sectors [ control ] . ceilingheight ;
s - > affectee = affectee ;
2021-12-31 12:58:41 +00:00
if ( type = = sc_carry | | type = = sc_carry_ceiling )
2022-10-08 09:06:14 +00:00
{
2021-12-31 12:58:41 +00:00
sectors [ affectee ] . specialflags | = SSF_CONVEYOR ;
2022-11-06 06:48:05 +00:00
if ( IsSector3DBlock ( & sectors [ affectee ] ) )
2022-10-09 13:27:07 +00:00
{
if ( type = = sc_carry )
sectors [ affectee ] . flags | = MSF_FLIPSPECIAL_CEILING ;
else
sectors [ affectee ] . flags | = MSF_FLIPSPECIAL_FLOOR ;
}
2022-10-08 09:06:14 +00:00
}
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & s - > thinker ) ;
2022-04-13 02:22:39 +00:00
// interpolation
switch ( type )
{
case sc_side :
2022-04-13 02:35:52 +00:00
R_CreateInterpolator_SideScroll ( & s - > thinker , & sides [ affectee ] ) ;
2022-04-13 02:22:39 +00:00
break ;
case sc_floor :
R_CreateInterpolator_SectorScroll ( & s - > thinker , & sectors [ affectee ] , false ) ;
break ;
case sc_ceiling :
R_CreateInterpolator_SectorScroll ( & s - > thinker , & sectors [ affectee ] , true ) ;
break ;
default :
break ;
}
2014-03-15 16:59:03 +00:00
}
2021-06-30 21:50:54 +00:00
static void P_SpawnPlaneScroller ( line_t * l , fixed_t dx , fixed_t dy , INT32 control , INT32 affectee , INT32 accel , INT32 exclusive )
{
if ( l - > args [ 1 ] ! = TMP_CEILING )
{
if ( l - > args [ 2 ] ! = TMS_SCROLLONLY )
Add_Scroller ( sc_carry , FixedMul ( dx , CARRYFACTOR ) , FixedMul ( dy , CARRYFACTOR ) , control , affectee , accel , exclusive ) ;
if ( l - > args [ 2 ] ! = TMS_CARRYONLY )
Add_Scroller ( sc_floor , - dx , dy , control , affectee , accel , exclusive ) ;
}
if ( l - > args [ 1 ] ! = TMP_FLOOR )
{
if ( l - > args [ 2 ] ! = TMS_SCROLLONLY )
Add_Scroller ( sc_carry_ceiling , FixedMul ( dx , CARRYFACTOR ) , FixedMul ( dy , CARRYFACTOR ) , control , affectee , accel , exclusive ) ;
if ( l - > args [ 2 ] ! = TMS_CARRYONLY )
Add_Scroller ( sc_ceiling , - dx , dy , control , affectee , accel , exclusive ) ;
}
}
2014-03-15 16:59:03 +00:00
/** Initializes the scrollers.
*
* \ todo Get rid of all the magic numbers .
* \ sa P_SpawnSpecials , Add_Scroller , Add_WallScroller
*/
static void P_SpawnScrollers ( void )
{
size_t i ;
line_t * l = lines ;
for ( i = 0 ; i < numlines ; i + + , l + + )
{
INT32 control = - 1 , accel = 0 ; // no control sector or acceleration
2020-04-17 16:15:25 +00:00
2021-06-30 21:50:54 +00:00
if ( l - > special = = 502 | | l - > special = = 510 )
2014-03-15 16:59:03 +00:00
{
2021-06-30 21:50:54 +00:00
if ( ( l - > args [ 4 ] & TMST_TYPEMASK ) ! = TMST_REGULAR )
control = ( INT32 ) ( sides [ * l - > sidenum ] . sector - sectors ) ;
if ( ( l - > args [ 4 ] & TMST_TYPEMASK ) = = TMST_ACCELERATIVE )
accel = 1 ;
2014-03-15 16:59:03 +00:00
}
2021-06-30 21:50:54 +00:00
switch ( l - > special )
2014-03-15 16:59:03 +00:00
{
register INT32 s ;
2021-06-30 21:50:54 +00:00
case 510 : // plane scroller
{
fixed_t length = R_PointToDist2 ( l - > v2 - > x , l - > v2 - > y , l - > v1 - > x , l - > v1 - > y ) ;
fixed_t speed = l - > args [ 3 ] < < FRACBITS ;
fixed_t dx = FixedMul ( FixedDiv ( l - > dx , length ) , speed ) > > SCROLL_SHIFT ;
fixed_t dy = FixedMul ( FixedDiv ( l - > dy , length ) , speed ) > > SCROLL_SHIFT ;
2014-03-15 16:59:03 +00:00
2021-06-30 21:50:54 +00:00
if ( l - > args [ 0 ] = = 0 )
2022-09-18 09:21:00 +00:00
P_SpawnPlaneScroller ( l , dx , dy , control , ( INT32 ) ( l - > frontsector - sectors ) , accel , ! ( l - > args [ 4 ] & TMST_NONEXCLUSIVE ) ) ;
2021-06-30 21:50:54 +00:00
else
{
TAG_ITER_SECTORS ( l - > args [ 0 ] , s )
2022-09-18 09:21:00 +00:00
P_SpawnPlaneScroller ( l , dx , dy , control , s , accel , ! ( l - > args [ 4 ] & TMST_NONEXCLUSIVE ) ) ;
2021-06-30 21:50:54 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
2021-06-30 21:50:54 +00:00
}
2014-03-15 16:59:03 +00:00
// scroll wall according to linedef
// (same direction and speed as scrolling floors)
case 502 :
2020-04-13 11:10:38 +00:00
{
2021-06-30 21:50:54 +00:00
TAG_ITER_LINES ( l - > args [ 0 ] , s )
2014-03-15 16:59:03 +00:00
if ( s ! = ( INT32 ) i )
2020-05-19 21:39:35 +00:00
{
2021-06-30 21:50:54 +00:00
if ( l - > args [ 1 ] ! = TMSD_BACK )
2022-05-27 04:38:50 +00:00
Add_Scroller ( sc_side , l - > args [ 2 ] < < ( FRACBITS - SCROLL_SHIFT ) , l - > args [ 3 ] < < ( FRACBITS - SCROLL_SHIFT ) , control , lines [ s ] . sidenum [ 0 ] , accel , 0 ) ;
2023-09-21 05:06:06 +00:00
if ( l - > args [ 1 ] ! = TMSD_FRONT & & lines [ s ] . sidenum [ 1 ] ! = NO_SIDEDEF )
2022-05-27 04:38:50 +00:00
Add_Scroller ( sc_side , l - > args [ 2 ] < < ( FRACBITS - SCROLL_SHIFT ) , l - > args [ 3 ] < < ( FRACBITS - SCROLL_SHIFT ) , control , lines [ s ] . sidenum [ 1 ] , accel , 0 ) ;
2020-05-19 21:39:35 +00:00
}
2014-03-15 16:59:03 +00:00
break ;
2020-04-13 11:10:38 +00:00
}
2014-03-15 16:59:03 +00:00
2021-06-30 21:50:54 +00:00
case 500 :
if ( l - > args [ 0 ] ! = TMSD_BACK )
Add_Scroller ( sc_side , - l - > args [ 1 ] < < FRACBITS , l - > args [ 2 ] < < FRACBITS , - 1 , l - > sidenum [ 0 ] , accel , 0 ) ;
if ( l - > args [ 0 ] ! = TMSD_FRONT )
{
2023-09-21 05:06:06 +00:00
if ( l - > sidenum [ 1 ] ! = NO_SIDEDEF )
2021-06-30 21:50:54 +00:00
Add_Scroller ( sc_side , - l - > args [ 1 ] < < FRACBITS , l - > args [ 2 ] < < FRACBITS , - 1 , l - > sidenum [ 1 ] , accel , 0 ) ;
else
CONS_Debug ( DBG_GAMELOGIC , " Line special 500 (line #%s) missing back side! \n " , sizeu1 ( i ) ) ;
}
2014-03-15 16:59:03 +00:00
break ;
}
}
}
/** Adds master appear/disappear thinker.
*
* \ param appeartime tics to be existent
* \ param disappeartime tics to be nonexistent
* \ param sector pointer to control sector
*/
static void Add_MasterDisappearer ( tic_t appeartime , tic_t disappeartime , tic_t offset , INT32 line , INT32 sourceline )
{
disappear_t * d = Z_Malloc ( sizeof * d , PU_LEVSPEC , NULL ) ;
d - > thinker . function . acp1 = ( actionf_p1 ) T_Disappear ;
d - > appeartime = appeartime ;
d - > disappeartime = disappeartime ;
d - > offset = offset ;
d - > affectee = line ;
d - > sourceline = sourceline ;
d - > exists = true ;
d - > timer = 1 ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & d - > thinker ) ;
2014-03-15 16:59:03 +00:00
}
/** Makes a FOF appear/disappear
*
* \ param d Disappear thinker .
* \ sa Add_MasterDisappearer
*/
void T_Disappear ( disappear_t * d )
{
if ( d - > offset & & ! d - > exists )
{
d - > offset - - ;
return ;
}
if ( - - d - > timer < = 0 )
{
ffloor_t * rover ;
register INT32 s ;
2021-09-26 04:48:26 +00:00
mtag_t afftag = lines [ d - > affectee ] . args [ 0 ] ;
2014-03-15 16:59:03 +00:00
2021-02-11 12:24:20 +00:00
TAG_ITER_SECTORS ( afftag , s )
2014-03-15 16:59:03 +00:00
{
for ( rover = sectors [ s ] . ffloors ; rover ; rover = rover - > next )
{
if ( rover - > master ! = & lines [ d - > affectee ] )
continue ;
if ( d - > exists )
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_EXISTS ;
2014-03-15 16:59:03 +00:00
else
{
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_EXISTS ;
2014-03-15 16:59:03 +00:00
2021-12-29 20:03:47 +00:00
if ( ! ( lines [ d - > sourceline ] . args [ 5 ] ) )
2014-03-15 16:59:03 +00:00
{
2020-03-22 14:17:16 +00:00
sectors [ s ] . soundorg . z = P_GetFFloorTopZAt ( rover , sectors [ s ] . soundorg . x , sectors [ s ] . soundorg . y ) ;
2014-03-15 16:59:03 +00:00
S_StartSound ( & sectors [ s ] . soundorg , sfx_appear ) ;
}
}
}
sectors [ s ] . moved = true ;
2019-07-04 13:43:18 +00:00
P_RecalcPrecipInSector ( & sectors [ s ] ) ;
2014-03-15 16:59:03 +00:00
}
if ( d - > exists )
{
d - > timer = d - > disappeartime ;
d - > exists = false ;
}
else
{
d - > timer = d - > appeartime ;
d - > exists = true ;
}
}
}
2018-03-31 05:15:24 +00:00
/** Removes fadingdata from FOF control sector
2018-08-14 02:10:16 +00:00
*
2018-03-31 05:15:24 +00:00
* \ param line line to search for target faders
* \ param data pointer to set new fadingdata to . Can be NULL to erase .
*/
2018-08-18 09:55:49 +00:00
static void P_ResetFakeFloorFader ( ffloor_t * rover , fade_t * data , boolean finalize )
2018-03-31 05:15:24 +00:00
{
2018-08-18 09:55:49 +00:00
fade_t * fadingdata = ( fade_t * ) rover - > fadingdata ;
2018-03-31 05:15:24 +00:00
// find any existing thinkers and remove them, then replace with new data
2018-08-18 09:55:49 +00:00
if ( fadingdata ! = data )
2018-03-31 05:15:24 +00:00
{
2018-09-15 06:24:44 +00:00
if ( fadingdata )
2018-08-18 09:31:41 +00:00
{
2018-08-18 09:55:49 +00:00
if ( finalize )
P_FadeFakeFloor ( rover ,
2018-09-11 14:28:24 +00:00
fadingdata - > sourcevalue ,
2018-08-18 09:55:49 +00:00
fadingdata - > alpha > = fadingdata - > destvalue ?
fadingdata - > alpha - 1 : // trigger fade-out finish
fadingdata - > alpha + 1 , // trigger fade-in finish
0 ,
2018-09-10 00:45:12 +00:00
fadingdata - > ticbased ,
& fadingdata - > timer ,
2018-08-18 09:55:49 +00:00
fadingdata - > doexists ,
fadingdata - > dotranslucent ,
2018-09-09 00:41:45 +00:00
fadingdata - > dolighting ,
2018-09-12 20:57:35 +00:00
fadingdata - > docolormap ,
2018-09-12 16:40:56 +00:00
fadingdata - > docollision ,
2018-08-18 09:55:49 +00:00
fadingdata - > doghostfade ,
fadingdata - > exactalpha ) ;
rover - > alpha = fadingdata - > alpha ;
2018-09-09 03:44:29 +00:00
if ( fadingdata - > dolighting )
P_RemoveLighting ( & sectors [ rover - > secnum ] ) ;
2018-09-12 20:57:35 +00:00
if ( fadingdata - > docolormap )
P_ResetColormapFader ( & sectors [ rover - > secnum ] ) ;
2018-08-18 09:55:49 +00:00
P_RemoveThinker ( & fadingdata - > thinker ) ;
2018-08-18 09:31:41 +00:00
}
2018-03-31 05:15:24 +00:00
2018-08-17 06:28:52 +00:00
rover - > fadingdata = data ;
2018-03-31 05:15:24 +00:00
}
}
2018-09-11 14:28:24 +00:00
static boolean P_FadeFakeFloor ( ffloor_t * rover , INT16 sourcevalue , INT16 destvalue , INT16 speed , boolean ticbased , INT32 * timer ,
2018-09-12 20:57:35 +00:00
boolean doexists , boolean dotranslucent , boolean dolighting , boolean docolormap ,
boolean docollision , boolean doghostfade , boolean exactalpha )
2018-04-07 11:09:04 +00:00
{
2018-08-17 18:49:33 +00:00
boolean stillfading = false ;
2018-08-17 20:27:32 +00:00
INT32 alpha ;
fade_t * fadingdata = ( fade_t * ) rover - > fadingdata ;
2018-09-15 05:04:15 +00:00
( void ) docolormap ; // *shrug* maybe we can use this in the future. For now, let's be consistent with our other function params
2018-08-17 20:27:32 +00:00
2018-09-08 05:33:12 +00:00
if ( rover - > master - > special = = 258 ) // Laser block
return false ;
2023-08-20 15:42:27 +00:00
// If fading an invisible FOF whose render flags we did not yet set, initialize its alpha to 1
2018-09-08 13:31:23 +00:00
if ( dotranslucent & &
2022-07-31 10:04:42 +00:00
( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > fofflags & FOF_FOG ) & & // do not include fog
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) & &
! ( rover - > fofflags & FOF_RENDERALL ) )
2023-08-20 15:42:27 +00:00
rover - > alpha = 0 ;
2018-09-08 13:31:23 +00:00
2018-08-17 20:27:32 +00:00
if ( fadingdata )
alpha = fadingdata - > alpha ;
else
alpha = rover - > alpha ;
2018-04-07 11:09:04 +00:00
2018-08-17 18:49:33 +00:00
// routines specific to fade in and fade out
2018-09-09 23:57:41 +00:00
if ( ! ticbased & & alpha = = destvalue )
2018-08-17 18:49:33 +00:00
return stillfading ;
2018-08-17 20:27:32 +00:00
else if ( alpha > destvalue ) // fade out
2018-04-07 11:09:04 +00:00
{
// finish fading out
2018-09-09 23:31:32 +00:00
if ( speed < 1 | | ( ! ticbased & & alpha - speed < = destvalue + speed ) | |
2018-09-10 00:45:12 +00:00
( ticbased & & ( - - ( * timer ) < = 0 | | alpha < = destvalue ) ) )
2018-04-07 11:09:04 +00:00
{
2018-08-17 20:27:32 +00:00
alpha = destvalue ;
2018-04-07 11:09:04 +00:00
2018-08-17 19:13:05 +00:00
if ( docollision )
2018-04-07 11:09:04 +00:00
{
2022-07-31 10:04:42 +00:00
if ( rover - > spawnflags & FOF_SOLID )
rover - > fofflags & = ~ FOF_SOLID ;
if ( rover - > spawnflags & FOF_SWIMMABLE )
rover - > fofflags & = ~ FOF_SWIMMABLE ;
if ( rover - > spawnflags & FOF_QUICKSAND )
rover - > fofflags & = ~ FOF_QUICKSAND ;
if ( rover - > spawnflags & FOF_BUSTUP )
rover - > fofflags & = ~ FOF_BUSTUP ;
if ( rover - > spawnflags & FOF_MARIO )
rover - > fofflags & = ~ FOF_MARIO ;
2018-04-07 11:09:04 +00:00
}
}
else // continue fading out
{
2018-09-09 23:31:32 +00:00
if ( ! ticbased )
alpha - = speed ;
2018-09-11 14:28:24 +00:00
else
{
INT16 delta = abs ( destvalue - sourcevalue ) ;
fixed_t factor = min ( FixedDiv ( speed - ( * timer ) , speed ) , 1 * FRACUNIT ) ;
alpha = max ( min ( alpha , sourcevalue - ( INT16 ) FixedMul ( delta , factor ) ) , destvalue ) ;
}
2018-08-17 18:49:33 +00:00
stillfading = true ;
2018-04-07 11:09:04 +00:00
}
}
else // fade in
{
// finish fading in
2018-09-09 23:31:32 +00:00
if ( speed < 1 | | ( ! ticbased & & alpha + speed > = destvalue - speed ) | |
2018-09-11 14:28:24 +00:00
( ticbased & & ( - - ( * timer ) < = 0 | | alpha > = destvalue ) ) )
2018-04-07 11:09:04 +00:00
{
2018-08-17 20:27:32 +00:00
alpha = destvalue ;
2018-08-14 02:10:16 +00:00
2018-08-17 19:13:05 +00:00
if ( docollision )
2018-04-07 11:09:04 +00:00
{
2022-07-31 10:04:42 +00:00
if ( rover - > spawnflags & FOF_SOLID )
rover - > fofflags | = FOF_SOLID ;
if ( rover - > spawnflags & FOF_SWIMMABLE )
rover - > fofflags | = FOF_SWIMMABLE ;
if ( rover - > spawnflags & FOF_QUICKSAND )
rover - > fofflags | = FOF_QUICKSAND ;
if ( rover - > spawnflags & FOF_BUSTUP )
rover - > fofflags | = FOF_BUSTUP ;
if ( rover - > spawnflags & FOF_MARIO )
rover - > fofflags | = FOF_MARIO ;
2018-04-07 11:09:04 +00:00
}
}
else // continue fading in
{
2018-09-09 23:31:32 +00:00
if ( ! ticbased )
alpha + = speed ;
2018-09-11 14:28:24 +00:00
else
{
INT16 delta = abs ( destvalue - sourcevalue ) ;
fixed_t factor = min ( FixedDiv ( speed - ( * timer ) , speed ) , 1 * FRACUNIT ) ;
alpha = min ( max ( alpha , sourcevalue + ( INT16 ) FixedMul ( delta , factor ) ) , destvalue ) ;
}
2018-08-17 18:49:33 +00:00
stillfading = true ;
}
}
2018-08-14 02:10:16 +00:00
2018-08-17 18:49:33 +00:00
// routines common to both fade in and fade out
if ( ! stillfading )
{
2022-07-31 10:04:42 +00:00
if ( doexists & & ! ( rover - > spawnflags & FOF_BUSTUP ) )
2018-08-17 18:49:33 +00:00
{
2023-08-20 15:42:27 +00:00
if ( alpha < = 0 )
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_EXISTS ;
2018-08-17 18:49:33 +00:00
else
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_EXISTS ;
2018-09-09 00:49:32 +00:00
// Re-render lighting at end of fade
2022-07-31 10:04:42 +00:00
if ( dolighting & & ! ( rover - > spawnflags & FOF_NOSHADE ) & & ! ( rover - > fofflags & FOF_EXISTS ) )
2018-09-09 00:49:32 +00:00
rover - > target - > moved = true ;
2018-08-17 18:49:33 +00:00
}
2018-04-07 11:09:04 +00:00
2022-07-31 10:04:42 +00:00
if ( dotranslucent & & ! ( rover - > fofflags & FOF_FOG ) )
2018-08-17 18:49:33 +00:00
{
2023-08-20 15:42:27 +00:00
if ( alpha > = 255 )
2018-04-07 11:09:04 +00:00
{
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_CUTSOLIDS ) & &
( rover - > spawnflags & FOF_CUTSOLIDS ) )
2018-09-08 12:46:56 +00:00
{
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_CUTSOLIDS ;
2018-09-08 12:46:56 +00:00
rover - > target - > moved = true ;
}
2018-09-08 13:08:32 +00:00
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_TRANSLUCENT ;
2018-08-17 18:49:33 +00:00
}
else
{
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_TRANSLUCENT ;
2018-09-08 13:08:32 +00:00
2022-07-31 10:04:42 +00:00
if ( ( rover - > fofflags & FOF_CUTSOLIDS ) & &
( rover - > spawnflags & FOF_CUTSOLIDS ) )
2018-09-08 12:46:56 +00:00
{
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_CUTSOLIDS ;
2018-09-08 12:46:56 +00:00
rover - > target - > moved = true ;
}
2018-04-07 11:09:04 +00:00
}
2018-09-08 13:08:32 +00:00
2022-07-31 10:04:42 +00:00
if ( ( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) )
2018-09-08 13:08:32 +00:00
{
if ( rover - > alpha > 1 )
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_RENDERALL ;
2018-09-08 13:08:32 +00:00
else
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_RENDERALL ;
2018-09-08 13:08:32 +00:00
}
2018-08-17 18:49:33 +00:00
}
}
else
{
2022-07-31 10:04:42 +00:00
if ( doexists & & ! ( rover - > spawnflags & FOF_BUSTUP ) )
2018-09-09 00:49:32 +00:00
{
2022-07-31 10:04:42 +00:00
// Re-render lighting if we haven't yet set FOF_EXISTS (beginning of fade)
if ( dolighting & & ! ( rover - > spawnflags & FOF_NOSHADE ) & & ! ( rover - > fofflags & FOF_EXISTS ) )
2018-09-09 00:49:32 +00:00
rover - > target - > moved = true ;
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_EXISTS ;
2018-09-09 00:49:32 +00:00
}
2018-08-17 18:49:33 +00:00
2022-07-31 10:04:42 +00:00
if ( dotranslucent & & ! ( rover - > fofflags & FOF_FOG ) )
2018-08-17 18:49:33 +00:00
{
2022-07-31 10:04:42 +00:00
rover - > fofflags | = FOF_TRANSLUCENT ;
2018-09-08 13:08:32 +00:00
2022-07-31 10:04:42 +00:00
if ( ( rover - > fofflags & FOF_CUTSOLIDS ) & &
( rover - > spawnflags & FOF_CUTSOLIDS ) )
2018-09-08 12:46:56 +00:00
{
2022-07-31 10:04:42 +00:00
rover - > fofflags & = ~ FOF_CUTSOLIDS ;
2018-09-08 12:46:56 +00:00
rover - > target - > moved = true ;
}
2018-09-08 13:08:32 +00:00
2022-07-31 10:04:42 +00:00
if ( ( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) )
rover - > fofflags | = FOF_RENDERALL ;
2018-08-17 18:49:33 +00:00
}
2018-08-17 19:13:05 +00:00
if ( docollision )
2018-08-17 18:49:33 +00:00
{
2018-08-17 19:59:34 +00:00
if ( doghostfade ) // remove collision flags during fade
2018-08-17 18:49:33 +00:00
{
2022-07-31 10:04:42 +00:00
if ( rover - > spawnflags & FOF_SOLID )
rover - > fofflags & = ~ FOF_SOLID ;
if ( rover - > spawnflags & FOF_SWIMMABLE )
rover - > fofflags & = ~ FOF_SWIMMABLE ;
if ( rover - > spawnflags & FOF_QUICKSAND )
rover - > fofflags & = ~ FOF_QUICKSAND ;
if ( rover - > spawnflags & FOF_BUSTUP )
rover - > fofflags & = ~ FOF_BUSTUP ;
if ( rover - > spawnflags & FOF_MARIO )
rover - > fofflags & = ~ FOF_MARIO ;
2018-08-17 18:49:33 +00:00
}
2018-08-17 19:59:34 +00:00
else // keep collision during fade
2018-08-17 18:49:33 +00:00
{
2022-07-31 10:04:42 +00:00
if ( rover - > spawnflags & FOF_SOLID )
rover - > fofflags | = FOF_SOLID ;
if ( rover - > spawnflags & FOF_SWIMMABLE )
rover - > fofflags | = FOF_SWIMMABLE ;
if ( rover - > spawnflags & FOF_QUICKSAND )
rover - > fofflags | = FOF_QUICKSAND ;
if ( rover - > spawnflags & FOF_BUSTUP )
rover - > fofflags | = FOF_BUSTUP ;
if ( rover - > spawnflags & FOF_MARIO )
rover - > fofflags | = FOF_MARIO ;
2018-08-17 18:49:33 +00:00
}
2018-04-07 11:09:04 +00:00
}
}
2022-07-31 10:04:42 +00:00
if ( ! ( rover - > fofflags & FOF_FOG ) ) // don't set FOG alpha
2018-08-17 20:27:32 +00:00
{
2018-09-10 02:14:24 +00:00
if ( ! stillfading | | exactalpha )
rover - > alpha = alpha ;
else // clamp fadingdata->alpha to software's alpha levels
{
if ( alpha < 12 )
2023-08-20 15:42:27 +00:00
rover - > alpha = destvalue < 12 ? destvalue : 0 ; // Don't even draw it
2018-09-10 02:14:24 +00:00
else if ( alpha < 38 )
rover - > alpha = destvalue > = 12 & & destvalue < 38 ? destvalue : 25 ;
else if ( alpha < 64 )
2023-08-20 15:42:27 +00:00
rover - > alpha = destvalue > = 38 & & destvalue < 64 ? destvalue : 51 ;
2018-09-10 02:14:24 +00:00
else if ( alpha < 89 )
rover - > alpha = destvalue > = 64 & & destvalue < 89 ? destvalue : 76 ;
else if ( alpha < 115 )
rover - > alpha = destvalue > = 89 & & destvalue < 115 ? destvalue : 102 ;
else if ( alpha < 140 )
rover - > alpha = destvalue > = 115 & & destvalue < 140 ? destvalue : 128 ;
else if ( alpha < 166 )
rover - > alpha = destvalue > = 140 & & destvalue < 166 ? destvalue : 154 ;
else if ( alpha < 192 )
rover - > alpha = destvalue > = 166 & & destvalue < 192 ? destvalue : 179 ;
else if ( alpha < 217 )
rover - > alpha = destvalue > = 192 & & destvalue < 217 ? destvalue : 204 ;
else if ( alpha < 243 )
rover - > alpha = destvalue > = 217 & & destvalue < 243 ? destvalue : 230 ;
else // Opaque
2023-08-20 15:42:27 +00:00
rover - > alpha = destvalue > = 243 ? destvalue : 255 ;
2018-09-10 02:14:24 +00:00
}
2018-08-17 20:27:32 +00:00
}
if ( fadingdata )
fadingdata - > alpha = alpha ;
2018-08-17 18:49:33 +00:00
return stillfading ;
2018-04-07 11:09:04 +00:00
}
2018-08-17 19:13:05 +00:00
/** Adds fake floor fader thinker.
2018-03-30 19:27:37 +00:00
*
2018-08-17 19:13:05 +00:00
* \ param destvalue transparency value to fade to
* \ param speed speed to fade by
2018-09-09 23:31:32 +00:00
* \ param ticbased tic - based logic , speed = duration
2018-09-10 13:48:24 +00:00
* \ param relative Destvalue is relative to rover - > alpha
2022-07-31 10:04:42 +00:00
* \ param doexists handle FOF_EXISTS
* \ param dotranslucent handle FOF_TRANSLUCENT
2018-09-09 19:41:30 +00:00
* \ param dolighting fade FOF light
2018-08-17 19:13:05 +00:00
* \ param docollision handle interactive flags
* \ param doghostfade no interactive flags during fading
* \ param exactalpha use exact alpha values ( opengl )
2018-03-30 19:27:37 +00:00
*/
2018-08-17 19:13:05 +00:00
static void P_AddFakeFloorFader ( ffloor_t * rover , size_t sectornum , size_t ffloornum ,
2018-09-10 13:48:24 +00:00
INT16 destvalue , INT16 speed , boolean ticbased , boolean relative ,
2018-09-12 20:57:35 +00:00
boolean doexists , boolean dotranslucent , boolean dolighting , boolean docolormap ,
boolean docollision , boolean doghostfade , boolean exactalpha )
2018-03-30 19:27:37 +00:00
{
2018-09-15 05:04:15 +00:00
fade_t * d ;
2023-08-20 15:42:27 +00:00
// If fading an invisible FOF whose render flags we did not yet set, initialize its alpha to 1
2018-09-10 13:48:24 +00:00
if ( dotranslucent & &
2022-07-31 10:04:42 +00:00
( rover - > spawnflags & FOF_NOSHADE ) & & // do not include light blocks, which don't set FOF_NOSHADE
! ( rover - > spawnflags & FOF_RENDERSIDES ) & &
! ( rover - > spawnflags & FOF_RENDERPLANES ) & &
! ( rover - > fofflags & FOF_RENDERALL ) )
2023-08-20 15:42:27 +00:00
rover - > alpha = 0 ;
2018-09-10 13:48:24 +00:00
2018-09-09 12:10:45 +00:00
// already equal, nothing to do
2023-08-20 15:42:27 +00:00
if ( rover - > alpha = = max ( 0 , min ( 255 , relative ? rover - > alpha + destvalue : destvalue ) ) )
2018-09-09 12:10:45 +00:00
return ;
2018-09-15 05:04:15 +00:00
d = Z_Malloc ( sizeof * d , PU_LEVSPEC , NULL ) ;
2018-03-30 19:27:37 +00:00
2018-03-30 20:17:35 +00:00
d - > thinker . function . acp1 = ( actionf_p1 ) T_Fade ;
2018-08-17 06:28:52 +00:00
d - > rover = rover ;
2018-09-10 03:52:36 +00:00
d - > sectornum = ( UINT32 ) sectornum ;
d - > ffloornum = ( UINT32 ) ffloornum ;
2018-09-08 13:31:23 +00:00
2018-09-11 14:28:24 +00:00
d - > alpha = d - > sourcevalue = rover - > alpha ;
2023-08-20 15:42:27 +00:00
d - > destvalue = max ( 0 , min ( 255 , relative ? rover - > alpha + destvalue : destvalue ) ) ; // rover->alpha is 0-255
2018-09-09 23:31:32 +00:00
if ( ticbased )
{
2018-09-10 00:45:12 +00:00
d - > ticbased = true ;
2018-09-11 14:28:24 +00:00
d - > timer = d - > speed = abs ( speed ) ; // use d->speed as total duration
2018-09-09 23:31:32 +00:00
}
else
{
2018-09-10 00:45:12 +00:00
d - > ticbased = false ;
2018-09-11 14:28:24 +00:00
d - > speed = max ( 1 , speed ) ; // minimum speed 1/tic // if speed < 1, alpha is set immediately in thinker
2018-09-10 00:45:12 +00:00
d - > timer = - 1 ;
2018-09-09 23:31:32 +00:00
}
2018-04-07 03:14:48 +00:00
d - > doexists = doexists ;
d - > dotranslucent = dotranslucent ;
2018-09-09 00:41:45 +00:00
d - > dolighting = dolighting ;
2018-09-12 20:57:35 +00:00
d - > docolormap = docolormap ;
2018-09-09 21:19:49 +00:00
d - > docollision = docollision ;
2018-08-14 14:18:08 +00:00
d - > doghostfade = doghostfade ;
2018-08-17 19:13:05 +00:00
d - > exactalpha = exactalpha ;
2018-03-31 03:29:41 +00:00
2018-03-31 05:15:24 +00:00
// find any existing thinkers and remove them, then replace with new data
2018-08-18 09:55:49 +00:00
P_ResetFakeFloorFader ( rover , d , false ) ;
2018-03-31 05:15:24 +00:00
2018-09-09 03:44:29 +00:00
// Set a separate thinker for shadow fading
2022-07-31 10:04:42 +00:00
if ( dolighting & & ! ( rover - > fofflags & FOF_NOSHADE ) )
2018-09-09 19:41:30 +00:00
{
UINT16 lightdelta = abs ( sectors [ rover - > secnum ] . spawn_lightlevel - rover - > target - > lightlevel ) ;
2018-09-09 23:44:35 +00:00
fixed_t alphapercent = min ( FixedDiv ( d - > destvalue , rover - > spawnalpha ) , 1 * FRACUNIT ) ; // don't make darker than spawn_lightlevel
2018-09-09 19:41:30 +00:00
fixed_t adjustedlightdelta = FixedMul ( lightdelta , alphapercent ) ;
if ( rover - > target - > lightlevel > = sectors [ rover - > secnum ] . spawn_lightlevel ) // fading out, get lighter
2018-09-09 21:19:49 +00:00
d - > destlightlevel = rover - > target - > lightlevel - adjustedlightdelta ;
2018-09-09 19:41:30 +00:00
else // fading in, get darker
2018-09-09 21:19:49 +00:00
d - > destlightlevel = rover - > target - > lightlevel + adjustedlightdelta ;
2018-09-09 19:41:30 +00:00
2018-09-09 03:44:29 +00:00
P_FadeLightBySector ( & sectors [ rover - > secnum ] ,
2018-09-09 21:19:49 +00:00
d - > destlightlevel ,
2018-09-10 00:45:12 +00:00
ticbased ? d - > timer :
2018-09-09 23:31:32 +00:00
FixedFloor ( FixedDiv ( abs ( d - > destvalue - d - > alpha ) , d - > speed ) ) / FRACUNIT ,
2018-09-09 19:41:30 +00:00
true ) ;
}
2018-09-09 21:19:49 +00:00
else
d - > destlightlevel = - 1 ;
2018-09-09 03:44:29 +00:00
2018-09-12 20:57:35 +00:00
// Set a separate thinker for colormap fading
2022-07-31 10:04:42 +00:00
if ( docolormap & & ! ( rover - > fofflags & FOF_NOSHADE ) & & sectors [ rover - > secnum ] . spawn_extra_colormap & & ! sectors [ rover - > secnum ] . colormap_protected )
2018-09-12 20:57:35 +00:00
{
extracolormap_t * dest_exc ,
* source_exc = sectors [ rover - > secnum ] . extra_colormap ? sectors [ rover - > secnum ] . extra_colormap : R_GetDefaultColormap ( ) ;
INT32 colordelta = R_GetRgbaA ( sectors [ rover - > secnum ] . spawn_extra_colormap - > rgba ) ; // alpha is from 0
fixed_t alphapercent = min ( FixedDiv ( d - > destvalue , rover - > spawnalpha ) , 1 * FRACUNIT ) ; // don't make darker than spawn_lightlevel
fixed_t adjustedcolordelta = FixedMul ( colordelta , alphapercent ) ;
INT32 coloralpha ;
coloralpha = adjustedcolordelta ;
dest_exc = R_CopyColormap ( sectors [ rover - > secnum ] . spawn_extra_colormap , false ) ;
dest_exc - > rgba = R_GetRgbaRGB ( dest_exc - > rgba ) + R_PutRgbaA ( coloralpha ) ;
if ( ! ( d - > dest_exc = R_GetColormapFromList ( dest_exc ) ) )
{
dest_exc - > colormap = R_CreateLightTable ( dest_exc ) ;
R_AddColormapToList ( dest_exc ) ;
d - > dest_exc = dest_exc ;
}
else
Z_Free ( dest_exc ) ;
// If fading from 0, set source_exc rgb same to dest_exc
if ( ! R_CheckDefaultColormap ( d - > dest_exc , true , false , false )
& & R_CheckDefaultColormap ( source_exc , true , false , false ) )
{
extracolormap_t * exc = R_CopyColormap ( source_exc , false ) ;
exc - > rgba = R_GetRgbaRGB ( d - > dest_exc - > rgba ) + R_PutRgbaA ( R_GetRgbaA ( source_exc - > rgba ) ) ;
exc - > fadergba = R_GetRgbaRGB ( d - > dest_exc - > rgba ) + R_PutRgbaA ( R_GetRgbaA ( source_exc - > fadergba ) ) ;
if ( ! ( source_exc = R_GetColormapFromList ( exc ) ) )
{
exc - > colormap = R_CreateLightTable ( exc ) ;
R_AddColormapToList ( exc ) ;
source_exc = exc ;
}
else
Z_Free ( exc ) ;
}
2018-09-16 00:34:19 +00:00
Add_ColormapFader ( & sectors [ rover - > secnum ] , source_exc , d - > dest_exc , true ,
2018-09-12 20:57:35 +00:00
ticbased ? d - > timer :
FixedFloor ( FixedDiv ( abs ( d - > destvalue - d - > alpha ) , d - > speed ) ) / FRACUNIT ) ;
}
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & d - > thinker ) ;
2018-03-30 19:27:37 +00:00
}
/** Makes a FOF fade
*
* \ param d Fade thinker .
2018-08-17 19:13:05 +00:00
* \ sa P_AddFakeFloorFader
2018-03-30 19:27:37 +00:00
*/
void T_Fade ( fade_t * d )
{
2018-09-11 14:28:24 +00:00
if ( d - > rover & & ! P_FadeFakeFloor ( d - > rover , d - > sourcevalue , d - > destvalue , d - > speed , d - > ticbased , & d - > timer ,
2018-09-12 20:57:35 +00:00
d - > doexists , d - > dotranslucent , d - > dolighting , d - > docolormap , d - > docollision , d - > doghostfade , d - > exactalpha ) )
2018-09-09 21:19:49 +00:00
{
// Finalize lighting, copypasta from P_AddFakeFloorFader
2022-07-31 10:04:42 +00:00
if ( d - > dolighting & & ! ( d - > rover - > fofflags & FOF_NOSHADE ) & & d - > destlightlevel > - 1 )
2018-09-09 21:19:49 +00:00
sectors [ d - > rover - > secnum ] . lightlevel = d - > destlightlevel ;
2018-09-12 20:57:35 +00:00
// Finalize colormap
2022-07-31 10:04:42 +00:00
if ( d - > docolormap & & ! ( d - > rover - > fofflags & FOF_NOSHADE ) & & sectors [ d - > rover - > secnum ] . spawn_extra_colormap )
2018-09-12 20:57:35 +00:00
sectors [ d - > rover - > secnum ] . extra_colormap = d - > dest_exc ;
2018-08-17 19:13:05 +00:00
P_RemoveFakeFloorFader ( d - > rover ) ;
2018-09-09 21:19:49 +00:00
}
2018-03-30 19:27:37 +00:00
}
2018-09-12 13:06:38 +00:00
static void P_ResetColormapFader ( sector_t * sector )
{
if ( sector - > fadecolormapdata )
{
// The thinker is the first member in all the action structs,
// so just let the thinker get freed, and that will free the whole
// structure.
2021-09-20 06:18:59 +00:00
P_RemoveThinker ( & ( ( thinkerdata_t * ) sector - > fadecolormapdata ) - > thinker ) ;
2018-09-12 13:06:38 +00:00
sector - > fadecolormapdata = NULL ;
}
}
static void Add_ColormapFader ( sector_t * sector , extracolormap_t * source_exc , extracolormap_t * dest_exc ,
2018-09-13 14:38:15 +00:00
boolean ticbased , INT32 duration )
2018-09-12 13:06:38 +00:00
{
2018-09-15 04:51:55 +00:00
fadecolormap_t * d ;
2018-09-12 13:06:38 +00:00
P_ResetColormapFader ( sector ) ;
// nothing to do, set immediately
if ( ! duration | | R_CheckEqualColormaps ( source_exc , dest_exc , true , true , true ) )
{
sector - > extra_colormap = dest_exc ;
return ;
}
2018-09-15 04:51:55 +00:00
d = Z_Malloc ( sizeof * d , PU_LEVSPEC , NULL ) ;
2018-09-12 13:06:38 +00:00
d - > thinker . function . acp1 = ( actionf_p1 ) T_FadeColormap ;
d - > sector = sector ;
d - > source_exc = source_exc ;
d - > dest_exc = dest_exc ;
2018-09-13 14:38:15 +00:00
if ( ticbased )
{
d - > ticbased = true ;
d - > duration = d - > timer = duration ;
}
else
{
d - > ticbased = false ;
d - > timer = 256 ;
d - > duration = duration ; // use as speed
}
2018-09-12 13:06:38 +00:00
sector - > fadecolormapdata = d ;
2019-04-20 21:29:20 +00:00
P_AddThinker ( THINK_MAIN , & d - > thinker ) ;
2018-09-12 13:06:38 +00:00
}
void T_FadeColormap ( fadecolormap_t * d )
{
2018-09-13 14:38:15 +00:00
if ( ( d - > ticbased & & - - d - > timer < = 0 )
| | ( ! d - > ticbased & & ( d - > timer - = d - > duration ) < = 0 ) ) // d->duration used as speed decrement
2018-09-12 13:06:38 +00:00
{
d - > sector - > extra_colormap = d - > dest_exc ;
P_ResetColormapFader ( d - > sector ) ;
}
else
{
extracolormap_t * exc ;
2018-09-13 14:38:15 +00:00
INT32 duration = d - > ticbased ? d - > duration : 256 ;
fixed_t factor = min ( FixedDiv ( duration - d - > timer , duration ) , 1 * FRACUNIT ) ;
2020-02-16 19:19:24 +00:00
INT16 cr , cg , cb , ca , fadestart , fadeend , flags ;
2018-09-12 13:06:38 +00:00
INT32 rgba , fadergba ;
2018-09-12 15:11:22 +00:00
// NULL failsafes (or intentionally set to signify default)
if ( ! d - > sector - > extra_colormap )
d - > sector - > extra_colormap = R_GetDefaultColormap ( ) ;
if ( ! d - > source_exc )
d - > source_exc = R_GetDefaultColormap ( ) ;
if ( ! d - > dest_exc )
d - > dest_exc = R_GetDefaultColormap ( ) ;
2018-09-12 13:06:38 +00:00
// For each var (rgba + fadergba + params = 11 vars), we apply
// percentage fading: currentval = sourceval + (delta * percent of duration elapsed)
// delta is negative when fading out (destval is lower)
// max/min are used to ensure progressive calcs don't go backwards and to cap values to dest.
# define APPLYFADE(dest, src, cur) (\
( dest - src < 0 ) ? \
max ( \
min ( cur , \
2018-09-12 15:33:44 +00:00
src + ( INT16 ) FixedMul ( dest - src , factor ) ) , \
2018-09-12 13:06:38 +00:00
dest ) \
: ( dest - src > 0 ) ? \
min ( \
max ( cur , \
2018-09-12 15:33:44 +00:00
src + ( INT16 ) FixedMul ( dest - src , factor ) ) , \
2018-09-12 13:06:38 +00:00
dest ) \
: \
dest \
)
cr = APPLYFADE ( R_GetRgbaR ( d - > dest_exc - > rgba ) , R_GetRgbaR ( d - > source_exc - > rgba ) , R_GetRgbaR ( d - > sector - > extra_colormap - > rgba ) ) ;
cg = APPLYFADE ( R_GetRgbaG ( d - > dest_exc - > rgba ) , R_GetRgbaG ( d - > source_exc - > rgba ) , R_GetRgbaG ( d - > sector - > extra_colormap - > rgba ) ) ;
cb = APPLYFADE ( R_GetRgbaB ( d - > dest_exc - > rgba ) , R_GetRgbaB ( d - > source_exc - > rgba ) , R_GetRgbaB ( d - > sector - > extra_colormap - > rgba ) ) ;
ca = APPLYFADE ( R_GetRgbaA ( d - > dest_exc - > rgba ) , R_GetRgbaA ( d - > source_exc - > rgba ) , R_GetRgbaA ( d - > sector - > extra_colormap - > rgba ) ) ;
rgba = R_PutRgbaRGBA ( cr , cg , cb , ca ) ;
cr = APPLYFADE ( R_GetRgbaR ( d - > dest_exc - > fadergba ) , R_GetRgbaR ( d - > source_exc - > fadergba ) , R_GetRgbaR ( d - > sector - > extra_colormap - > fadergba ) ) ;
cg = APPLYFADE ( R_GetRgbaG ( d - > dest_exc - > fadergba ) , R_GetRgbaG ( d - > source_exc - > fadergba ) , R_GetRgbaG ( d - > sector - > extra_colormap - > fadergba ) ) ;
cb = APPLYFADE ( R_GetRgbaB ( d - > dest_exc - > fadergba ) , R_GetRgbaB ( d - > source_exc - > fadergba ) , R_GetRgbaB ( d - > sector - > extra_colormap - > fadergba ) ) ;
ca = APPLYFADE ( R_GetRgbaA ( d - > dest_exc - > fadergba ) , R_GetRgbaA ( d - > source_exc - > fadergba ) , R_GetRgbaA ( d - > sector - > extra_colormap - > fadergba ) ) ;
fadergba = R_PutRgbaRGBA ( cr , cg , cb , ca ) ;
fadestart = APPLYFADE ( d - > dest_exc - > fadestart , d - > source_exc - > fadestart , d - > sector - > extra_colormap - > fadestart ) ;
fadeend = APPLYFADE ( d - > dest_exc - > fadeend , d - > source_exc - > fadeend , d - > sector - > extra_colormap - > fadeend ) ;
2020-02-16 19:19:24 +00:00
flags = abs ( factor ) > FRACUNIT / 2 ? d - > dest_exc - > flags : d - > source_exc - > flags ; // set new flags halfway through fade
2018-09-12 13:06:38 +00:00
# undef APPLYFADE
//////////////////
// setup new colormap
//////////////////
2020-02-16 19:19:24 +00:00
if ( ! ( d - > sector - > extra_colormap = R_GetColormapFromListByValues ( rgba , fadergba , fadestart , fadeend , flags ) ) )
2018-09-12 13:06:38 +00:00
{
exc = R_CreateDefaultColormap ( false ) ;
exc - > fadestart = fadestart ;
exc - > fadeend = fadeend ;
2020-02-16 19:19:24 +00:00
exc - > flags = flags ;
2018-09-12 13:06:38 +00:00
exc - > rgba = rgba ;
exc - > fadergba = fadergba ;
exc - > colormap = R_CreateLightTable ( exc ) ;
R_AddColormapToList ( exc ) ;
d - > sector - > extra_colormap = exc ;
}
}
}
2014-03-15 16:59:03 +00:00
/*
SoM : 3 / 8 / 2000 : Friction functions start .
Add_Friction ,
T_Friction ,
P_SpawnFriction
*/
/** Adds friction thinker.
*
* \ param friction Friction value , 0xe800 is normal .
* \ param affectee Target sector .
* \ param roverfriction FOF or not
* \ sa T_Friction , P_SpawnFriction
*/
static void Add_Friction ( INT32 friction , INT32 movefactor , INT32 affectee , INT32 referrer )
{
friction_t * f = Z_Calloc ( sizeof * f , PU_LEVSPEC , NULL ) ;
f - > thinker . function . acp1 = ( actionf_p1 ) T_Friction ;
f - > friction = friction ;
f - > movefactor = movefactor ;
f - > affectee = affectee ;
if ( referrer ! = - 1 )
{
f - > roverfriction = true ;
f - > referrer = referrer ;
}
else
f - > roverfriction = false ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & f - > thinker ) ;
2014-03-15 16:59:03 +00:00
}
/** Applies friction to all things in a sector.
*
* \ param f Friction thinker .
* \ sa Add_Friction
*/
void T_Friction ( friction_t * f )
{
2015-10-10 16:57:35 +00:00
sector_t * sec , * referrer = NULL ;
2014-03-15 16:59:03 +00:00
mobj_t * thing ;
msecnode_t * node ;
sec = sectors + f - > affectee ;
2016-06-02 21:46:27 +00:00
// Get FOF control sector
2014-03-15 16:59:03 +00:00
if ( f - > roverfriction )
2015-05-24 17:53:30 +00:00
referrer = sectors + f - > referrer ;
2014-03-15 16:59:03 +00:00
// Assign the friction value to players on the floor, non-floating,
// and clipped. Normally the object's friction value is kept at
// ORIG_FRICTION and this thinker changes it for icy or muddy floors.
// When the object is straddling sectors with the same
// floorheight that have different frictions, use the lowest
// friction value (muddy has precedence over icy).
node = sec - > touching_thinglist ; // things touching this sector
while ( node )
{
thing = node - > m_thing ;
// apparently, all I had to do was comment out part of the next line and
// friction works for all mobj's
// (or at least MF_PUSHABLEs, which is all I care about anyway)
if ( ! ( thing - > flags & ( MF_NOGRAVITY | MF_NOCLIP ) ) & & thing - > z = = thing - > floorz )
{
if ( f - > roverfriction )
{
2015-05-24 17:53:30 +00:00
if ( thing - > floorz ! = P_GetSpecialTopZ ( thing , referrer , sec ) )
2014-03-15 16:59:03 +00:00
{
2016-06-09 13:16:02 +00:00
node = node - > m_thinglist_next ;
2014-03-15 16:59:03 +00:00
continue ;
}
if ( ( thing - > friction = = ORIG_FRICTION ) // normal friction?
| | ( f - > friction < thing - > friction ) )
{
thing - > friction = f - > friction ;
2016-06-03 16:26:50 +00:00
if ( thing - > player )
2016-08-09 23:05:46 +00:00
thing - > movefactor = f - > movefactor ;
2014-03-15 16:59:03 +00:00
}
}
2015-05-24 17:53:30 +00:00
else if ( P_GetSpecialBottomZ ( thing , sec , sec ) = = thing - > floorz & & ( thing - > friction = = ORIG_FRICTION // normal friction?
2014-03-15 16:59:03 +00:00
| | f - > friction < thing - > friction ) )
{
thing - > friction = f - > friction ;
2016-06-03 16:26:50 +00:00
if ( thing - > player )
2016-08-09 23:05:46 +00:00
thing - > movefactor = f - > movefactor ;
2014-03-15 16:59:03 +00:00
}
}
2016-06-09 13:16:02 +00:00
node = node - > m_thinglist_next ;
2014-03-15 16:59:03 +00:00
}
}
/** Spawns all friction effects.
*
* \ sa P_SpawnSpecials , Add_Friction
*/
static void P_SpawnFriction ( void )
{
size_t i ;
2021-12-30 14:32:28 +00:00
sector_t * s = sectors ;
2014-03-15 16:59:03 +00:00
fixed_t friction ; // friction value to be applied during movement
INT32 movefactor ; // applied to each player move to simulate inertia
2021-12-30 14:32:28 +00:00
for ( i = 0 ; i < numsectors ; i + + , s + + )
{
2022-01-04 20:25:34 +00:00
if ( s - > friction = = ORIG_FRICTION )
2021-12-30 14:32:28 +00:00
continue ;
2014-03-15 16:59:03 +00:00
2022-01-04 20:25:34 +00:00
friction = s - > friction ;
2021-12-30 14:32:28 +00:00
if ( friction > FRACUNIT )
friction = FRACUNIT ;
if ( friction < 0 )
friction = 0 ;
movefactor = FixedDiv ( ORIG_FRICTION , friction ) ;
if ( movefactor < FRACUNIT )
movefactor = 8 * movefactor - 7 * FRACUNIT ;
else
movefactor = FRACUNIT ;
Add_Friction ( friction , movefactor , ( INT32 ) ( s - sectors ) , - 1 ) ;
}
2014-03-15 16:59:03 +00:00
}
/*
SoM : 3 / 8 / 2000 : Push / Pull / Wind / Current functions .
Add_Pusher ,
PIT_PushThing ,
T_Pusher ,
P_GetPushThing ,
P_SpawnPushers
*/
# define PUSH_FACTOR 7
/** Adds a pusher.
*
* \ param type Type of push / pull effect .
* \ param x_mag X magnitude .
* \ param y_mag Y magnitude .
2021-07-01 18:05:23 +00:00
* \ param z_mag Z magnitude .
2014-03-15 16:59:03 +00:00
* \ param affectee Target sector .
* \ param referrer What sector set it
* \ sa T_Pusher , P_GetPushThing , P_SpawnPushers
*/
2021-07-01 18:05:23 +00:00
static void Add_Pusher ( pushertype_e type , fixed_t x_mag , fixed_t y_mag , fixed_t z_mag , INT32 affectee , INT32 referrer , INT32 exclusive , INT32 slider )
2014-03-15 16:59:03 +00:00
{
pusher_t * p = Z_Calloc ( sizeof * p , PU_LEVSPEC , NULL ) ;
p - > thinker . function . acp1 = ( actionf_p1 ) T_Pusher ;
p - > type = type ;
2021-07-01 18:05:23 +00:00
p - > x_mag = x_mag ;
p - > y_mag = y_mag ;
p - > z_mag = z_mag ;
2014-03-15 16:59:03 +00:00
p - > exclusive = exclusive ;
p - > slider = slider ;
if ( referrer ! = - 1 )
{
p - > roverpusher = true ;
p - > referrer = referrer ;
2021-12-31 12:58:41 +00:00
sectors [ referrer ] . specialflags | = SSF_WINDCURRENT ;
2014-03-15 16:59:03 +00:00
}
else
2021-12-31 12:58:41 +00:00
{
2014-03-15 16:59:03 +00:00
p - > roverpusher = false ;
2021-12-31 12:58:41 +00:00
sectors [ affectee ] . specialflags | = SSF_WINDCURRENT ;
}
2014-03-15 16:59:03 +00:00
p - > affectee = affectee ;
2019-04-20 20:37:19 +00:00
P_AddThinker ( THINK_MAIN , & p - > thinker ) ;
2014-03-15 16:59:03 +00:00
}
/** Applies a pusher to all affected objects.
*
* \ param p Thinker for the pusher effect .
* \ todo Split up into multiple functions .
* \ sa Add_Pusher , PIT_PushThing
*/
void T_Pusher ( pusher_t * p )
{
2015-10-10 16:57:35 +00:00
sector_t * sec , * referrer = NULL ;
2014-03-15 16:59:03 +00:00
mobj_t * thing ;
msecnode_t * node ;
2021-07-01 18:05:23 +00:00
fixed_t x_mag , y_mag , z_mag ;
fixed_t xspeed = 0 , yspeed = 0 , zspeed = 0 ;
2014-03-15 16:59:03 +00:00
boolean inFOF ;
boolean touching ;
boolean moved ;
2021-07-01 18:05:23 +00:00
x_mag = p - > x_mag > > PUSH_FACTOR ;
y_mag = p - > y_mag > > PUSH_FACTOR ;
z_mag = p - > z_mag > > PUSH_FACTOR ;
2014-03-15 16:59:03 +00:00
sec = sectors + p - > affectee ;
if ( p - > roverpusher )
2021-12-31 12:58:41 +00:00
referrer = sectors + p - > referrer ;
2014-03-15 16:59:03 +00:00
// 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.
//
// Apply the effect to clipped players only for now.
//
// In Phase II, you can apply these effects to Things other than players.
// constant pushers p_wind and p_current
node = sec - > touching_thinglist ; // things touching this sector
2016-06-09 13:16:02 +00:00
for ( ; node ; node = node - > m_thinglist_next )
2014-03-15 16:59:03 +00:00
{
thing = node - > m_thing ;
if ( thing - > flags & ( MF_NOGRAVITY | MF_NOCLIP )
& & ! ( thing - > type = = MT_SMALLBUBBLE
2014-11-12 00:55:07 +00:00
| | thing - > type = = MT_MEDIUMBUBBLE
| | thing - > type = = MT_EXTRALARGEBUBBLE ) )
2014-03-15 16:59:03 +00:00
continue ;
2020-01-08 22:36:29 +00:00
if ( ! ( ( thing - > flags & MF_PUSHABLE ) | | ( ( thing - > info - > flags & MF_PUSHABLE ) & & thing - > fuse ) )
& & ! ( thing - > type = = MT_PLAYER
2014-11-12 00:55:07 +00:00
| | thing - > type = = MT_SMALLBUBBLE
| | thing - > type = = MT_MEDIUMBUBBLE
| | thing - > type = = MT_EXTRALARGEBUBBLE
| | thing - > type = = MT_LITTLETUMBLEWEED
| | thing - > type = = MT_BIGTUMBLEWEED ) )
2014-03-15 16:59:03 +00:00
continue ;
2016-03-09 09:30:52 +00:00
if ( thing - > eflags & MFE_PUSHED )
2014-03-15 16:59:03 +00:00
continue ;
2016-09-23 22:48:48 +00:00
if ( thing - > player & & thing - > player - > powers [ pw_carry ] = = CR_ROPEHANG )
2014-03-15 16:59:03 +00:00
continue ;
if ( thing - > player & & ( thing - > state = = & states [ thing - > info - > painstate ] ) & & ( thing - > player - > powers [ pw_flashing ] > ( flashingtics / 4 ) * 3 & & thing - > player - > powers [ pw_flashing ] < = flashingtics ) )
continue ;
inFOF = touching = moved = false ;
// Find the area that the 'thing' is in
if ( p - > roverpusher )
{
2015-05-24 17:53:30 +00:00
fixed_t top , bottom ;
2014-03-15 16:59:03 +00:00
2015-05-24 17:53:30 +00:00
top = P_GetSpecialTopZ ( thing , referrer , sec ) ;
bottom = P_GetSpecialBottomZ ( thing , referrer , sec ) ;
2014-03-15 16:59:03 +00:00
if ( thing - > eflags & MFE_VERTICALFLIP )
{
2015-05-24 17:53:30 +00:00
if ( bottom > thing - > z + thing - > height
| | top < ( thing - > z + ( thing - > height > > 1 ) ) )
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
if ( thing - > z < bottom )
2014-03-15 16:59:03 +00:00
touching = true ;
2015-05-24 17:53:30 +00:00
if ( thing - > z + ( thing - > height > > 1 ) > bottom )
2014-03-15 16:59:03 +00:00
inFOF = true ;
}
else
{
2016-09-08 21:55:11 +00:00
if ( top < thing - > z | | bottom > ( thing - > z + ( thing - > height > > 1 ) ) )
2014-03-15 16:59:03 +00:00
continue ;
2015-05-24 17:53:30 +00:00
if ( thing - > z + thing - > height > top )
2014-03-15 16:59:03 +00:00
touching = true ;
2015-05-24 17:53:30 +00:00
if ( thing - > z + ( thing - > height > > 1 ) < top )
2014-03-15 16:59:03 +00:00
inFOF = true ;
}
}
else // Treat the entire sector as one big FOF
{
2015-05-24 17:53:30 +00:00
if ( thing - > z = = P_GetSpecialBottomZ ( thing , sec , sec ) )
2014-03-15 16:59:03 +00:00
touching = true ;
2022-10-08 08:20:20 +00:00
// Annoying backwards compatibility nonsense:
// In binary, horizontal currents require floor touch
else if ( udmf | | p - > type ! = p_current | | z_mag ! = 0 )
2014-03-15 16:59:03 +00:00
inFOF = true ;
}
if ( ! touching & & ! inFOF ) // Object is out of range of effect
continue ;
2021-07-01 18:05:23 +00:00
if ( inFOF | | ( p - > type = = p_current & & touching ) )
2014-03-15 16:59:03 +00:00
{
2021-07-01 18:05:23 +00:00
xspeed = x_mag ; // full force
yspeed = y_mag ;
zspeed = z_mag ;
moved = true ;
2014-03-15 16:59:03 +00:00
}
2021-07-01 18:05:23 +00:00
else if ( p - > type = = p_wind & & touching )
2014-03-15 16:59:03 +00:00
{
2021-07-01 18:05:23 +00:00
xspeed = x_mag > > 1 ; // half force
yspeed = y_mag > > 1 ;
zspeed = z_mag > > 1 ;
moved = true ;
2014-03-15 16:59:03 +00:00
}
2021-07-01 18:05:23 +00:00
thing - > momx + = xspeed ;
thing - > momy + = yspeed ;
2021-07-01 18:35:13 +00:00
thing - > momz + = zspeed ;
2021-07-01 18:05:23 +00:00
if ( thing - > player )
2014-03-15 16:59:03 +00:00
{
2021-07-01 18:05:23 +00:00
thing - > player - > cmomx + = xspeed ;
thing - > player - > cmomy + = yspeed ;
thing - > player - > cmomx = FixedMul ( thing - > player - > cmomx , ORIG_FRICTION ) ;
thing - > player - > cmomy = FixedMul ( thing - > player - > cmomy , ORIG_FRICTION ) ;
2014-03-15 16:59:03 +00:00
}
2021-07-01 18:05:23 +00:00
// Tumbleweeds bounce a bit...
if ( thing - > type = = MT_LITTLETUMBLEWEED | | thing - > type = = MT_BIGTUMBLEWEED )
thing - > momz + = P_AproxDistance ( xspeed , yspeed ) > > 2 ;
2014-03-15 16:59:03 +00:00
if ( moved )
{
if ( p - > slider & & thing - > player )
{
2017-03-18 21:06:06 +00:00
pflags_t jumped = ( thing - > player - > pflags & ( PF_JUMPED | PF_NOJUMPDAMAGE ) ) ;
2014-03-15 16:59:03 +00:00
P_ResetPlayer ( thing - > player ) ;
if ( jumped )
2017-03-18 21:06:06 +00:00
thing - > player - > pflags | = jumped ;
2014-03-15 16:59:03 +00:00
thing - > player - > pflags | = PF_SLIDING ;
2021-07-01 18:05:23 +00:00
thing - > angle = R_PointToAngle2 ( 0 , 0 , xspeed , yspeed ) ;
2014-03-15 16:59:03 +00:00
2019-12-30 21:05:24 +00:00
if ( ! demoplayback | | P_ControlStyle ( thing - > player ) = = CS_LMAOGALOG )
2014-03-15 16:59:03 +00:00
{
2020-05-28 09:03:35 +00:00
angle_t angle = thing - > player - > angleturn < < 16 ;
if ( thing - > angle - angle > ANGLE_180 )
P_SetPlayerAngle ( thing - > player , angle - ( angle - thing - > angle ) / 8 ) ;
else
P_SetPlayerAngle ( thing - > player , angle + ( thing - > angle - angle ) / 8 ) ;
//P_SetPlayerAngle(thing->player, thing->angle);
2014-03-15 16:59:03 +00:00
}
}
if ( p - > exclusive )
2016-03-09 09:30:52 +00:00
thing - > eflags | = MFE_PUSHED ;
2014-03-15 16:59:03 +00:00
}
}
}
/** Spawns pushers.
*
* \ sa P_SpawnSpecials , Add_Pusher
*/
static void P_SpawnPushers ( void )
{
size_t i ;
line_t * l = lines ;
register INT32 s ;
2021-12-16 06:00:30 +00:00
fixed_t length , hspeed , dx , dy ;
2014-03-15 16:59:03 +00:00
for ( i = 0 ; i < numlines ; i + + , l + + )
2020-04-17 16:15:25 +00:00
{
2021-12-16 06:00:30 +00:00
if ( l - > special ! = 541 )
continue ;
2021-07-01 18:35:13 +00:00
2021-12-16 06:00:30 +00:00
length = R_PointToDist2 ( l - > v2 - > x , l - > v2 - > y , l - > v1 - > x , l - > v1 - > y ) ;
hspeed = l - > args [ 1 ] < < FRACBITS ;
dx = FixedMul ( FixedDiv ( l - > dx , length ) , hspeed ) ;
dy = FixedMul ( FixedDiv ( l - > dy , length ) , hspeed ) ;
if ( l - > args [ 0 ] = = 0 )
Add_Pusher ( l - > args [ 3 ] , dx , dy , l - > args [ 2 ] < < FRACBITS , ( INT32 ) ( l - > frontsector - sectors ) , - 1 , ! ( l - > args [ 4 ] & TMPF_NONEXCLUSIVE ) , ! ! ( l - > args [ 4 ] & TMPF_SLIDE ) ) ;
else
{
TAG_ITER_SECTORS ( l - > args [ 0 ] , s )
Add_Pusher ( l - > args [ 3 ] , dx , dy , l - > args [ 2 ] < < FRACBITS , s , - 1 , ! ( l - > args [ 4 ] & TMPF_NONEXCLUSIVE ) , ! ! ( l - > args [ 4 ] & TMPF_SLIDE ) ) ;
2014-03-15 16:59:03 +00:00
}
2020-04-17 16:15:25 +00:00
}
2019-08-07 16:39:51 +00:00
}