2006-02-24 04:48:15 +00:00
// Emacs style mode select -*- C++ -*-
//-----------------------------------------------------------------------------
//
// $Id:$
//
// Copyright (C) 1993-1996 by id Software, Inc.
//
// This source is available for distribution and/or modification
// only under the terms of the DOOM Source Code License as
// published by id Software. All rights reserved.
//
// The source is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// FITNESS FOR A PARTICULAR PURPOSE. See the DOOM Source Code License
// for more details.
//
// $Log:$
//
// DESCRIPTION:
// Movement, collision handling.
// Shooting and aiming.
//
//-----------------------------------------------------------------------------
# include <stdlib.h>
# include <math.h>
# include "templates.h"
# include "m_bbox.h"
# include "m_random.h"
# include "i_system.h"
2008-03-18 18:18:18 +00:00
# include "c_dispatch.h"
2006-02-24 04:48:15 +00:00
# include "doomdef.h"
# include "p_local.h"
2016-02-15 01:14:34 +00:00
# include "p_spec.h"
# include "d_player.h"
# include "p_maputl.h"
2006-02-24 04:48:15 +00:00
# include "p_lnspec.h"
# include "p_effect.h"
# include "p_terrain.h"
# include "p_trace.h"
2016-02-15 12:40:31 +00:00
# include "p_checkposition.h"
2016-02-15 01:14:34 +00:00
# include "r_utility.h"
2006-02-24 04:48:15 +00:00
# include "s_sound.h"
# include "decallib.h"
// State.
# include "doomstat.h"
# include "r_state.h"
# include "gi.h"
# include "a_sharedglobal.h"
# include "p_conversation.h"
2011-07-06 08:50:15 +00:00
# include "r_data/r_translate.h"
2008-09-14 23:54:38 +00:00
# include "g_level.h"
2016-01-16 22:33:19 +00:00
# include "r_sky.h"
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
CVAR ( Bool , cl_bloodsplats , true , CVAR_ARCHIVE )
CVAR ( Int , sv_smartaim , 0 , CVAR_ARCHIVE | CVAR_SERVERINFO )
CVAR ( Bool , cl_doautoaim , false , CVAR_ARCHIVE )
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
static void CheckForPushSpecial ( line_t * line , int side , AActor * mobj , bool windowcheck ) ;
static void SpawnShootDecal ( AActor * t1 , const FTraceResults & trace ) ;
static void SpawnDeepSplash ( AActor * t1 , const FTraceResults & trace , AActor * puff ,
fixed_t vx , fixed_t vy , fixed_t vz , fixed_t shootz , bool ffloor = false ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
static FRandom pr_tracebleed ( " TraceBleed " ) ;
static FRandom pr_checkthing ( " CheckThing " ) ;
static FRandom pr_lineattack ( " LineAttack " ) ;
static FRandom pr_crunch ( " DoCrunch " ) ;
2006-02-24 04:48:15 +00:00
// keep track of special lines as they are hit,
// but don't process them until the move is proven valid
TArray < line_t * > spechit ;
// Temporary holder for thing_sectorlist threads
msecnode_t * sector_list = NULL ; // phares 3/16/98
2016-02-19 13:08:41 +00:00
2014-11-25 18:15:25 +00:00
//==========================================================================
//
// GetCoefficientClosestPointInLine24
//
// Formula: (dotProduct(ldv1 - tm, ld) << 24) / dotProduct(ld, ld)
// with: ldv1 = (ld->v1->x, ld->v1->y), tm = (tm.x, tm.y)
// and ld = (ld->dx, ld->dy)
// Returns truncated to range [0, 1 << 24].
//
//==========================================================================
2016-02-19 13:08:41 +00:00
static inline fixed_t GetCoefficientClosestPointInLine24 ( line_t * ld , fixedvec2 pos )
2014-11-25 18:15:25 +00:00
{
2016-02-19 13:08:41 +00:00
# ifndef USE_FLOAT
2014-11-25 18:15:25 +00:00
// [EP] Use 64 bit integers in order to keep the exact result of the
// multiplication, because in the case the vertexes have both the
// distance coordinates equal to the map limit (32767 units, which is
// 2147418112 in fixed_t notation), the product result would occupy
// 62 bits and the sum of two products would occupy 63 bits
// in the worst case. If instead the vertexes are very close (1 in
// fixed_t notation, which is 1.52587890625e-05 in float notation), the
// product and the sum can be 1 in the worst case, which is very tiny.
2016-02-19 13:08:41 +00:00
SQWORD r_num = ( ( SQWORD ( pos . x - ld - > v1 - > x ) * ld - > dx ) +
( SQWORD ( pos . y - ld - > v1 - > y ) * ld - > dy ) ) ;
2014-11-25 18:15:25 +00:00
// The denominator is always positive. Use this to avoid useless
// calculations.
SQWORD r_den = ( SQWORD ( ld - > dx ) * ld - > dx + SQWORD ( ld - > dy ) * ld - > dy ) ;
if ( r_num < = 0 ) {
// [EP] The numerator is less or equal to zero, hence the closest
// point on the line is the first vertex. Truncate the result to 0.
return 0 ;
}
if ( r_num > = r_den ) {
// [EP] The division is greater or equal to 1, hence the closest
// point on the line is the second vertex. Truncate the result to
// 1 << 24.
return ( 1 < < 24 ) ;
}
// [EP] Deal with the limited bits. The original formula is:
// r = (r_num << 24) / r_den,
// but r_num might be big enough to make the shift overflow.
// Since the numerator can't be saved in a 128bit integer,
// the denominator must be right shifted. If the denominator is
// less than (1 << 24), there would be a division by zero.
// Thanks to the fact that in this code path the denominator is greater
// than the numerator, it's possible to avoid this bad situation by
// just checking the last 24 bits of the numerator.
2016-02-19 13:08:41 +00:00
if ( ( r_num > > ( 63 - 24 ) ) ! = 0 ) {
2014-11-25 18:15:25 +00:00
// [EP] In fact, if the numerator is greater than
// (1 << (63-24)), the denominator must be greater than
// (1 << (63-24)), hence the denominator won't be zero after
// the right shift by 24 places.
2016-02-19 13:08:41 +00:00
return ( fixed_t ) ( r_num / ( r_den > > 24 ) ) ;
2014-11-25 18:15:25 +00:00
}
2014-11-25 18:22:56 +00:00
// [EP] Having the last 24 bits all zero allows left shifting
// the numerator by 24 bits without overflow.
2016-02-19 13:08:41 +00:00
return ( fixed_t ) ( ( r_num < < 24 ) / r_den ) ;
# else
double dx = ld - > dx ;
double dy = ld - > dy ;
return xs_CRoundToInt ( ( ( double ) ( pos . x - ld - > v1 - > x ) * dx + ( double ) ( pos . y - ld - > v1 - > y ) * dy ) / ( dx * dx + dy * dy ) * 16777216.f ) ;
# endif
}
//==========================================================================
//
// FindRefPoint
//
// Finds the point on the line closest to the given coordinate
//
//==========================================================================
static inline fixedvec2 FindRefPoint ( line_t * ld , fixedvec2 pos )
{
if ( ! ( ( ( ( ld - > frontsector - > floorplane . a | ld - > frontsector - > floorplane . b ) |
( ld - > backsector - > floorplane . a | ld - > backsector - > floorplane . b ) |
( ld - > frontsector - > ceilingplane . a | ld - > frontsector - > ceilingplane . b ) |
( ld - > backsector - > ceilingplane . a | ld - > backsector - > ceilingplane . b ) ) = = 0 )
& & ld - > backsector - > e - > XFloor . ffloors . Size ( ) = = 0 & & ld - > frontsector - > e - > XFloor . ffloors . Size ( ) = = 0 ) )
{
fixed_t r = GetCoefficientClosestPointInLine24 ( ld , pos ) ;
if ( r < = 0 )
{
pos . x = ld - > v1 - > x ;
pos . y = ld - > v1 - > y ;
}
else if ( r > = ( 1 < < 24 ) )
{
pos . x = ld - > v2 - > x ;
pos . y = ld - > v2 - > y ;
}
else
{
pos . x = ld - > v1 - > x + MulScale24 ( r , ld - > dx ) ;
pos . y = ld - > v1 - > y + MulScale24 ( r , ld - > dy ) ;
}
}
return pos ;
2014-11-25 18:15:25 +00:00
}
2006-02-24 04:48:15 +00:00
//==========================================================================
//
// PIT_FindFloorCeiling
//
2012-04-08 21:12:14 +00:00
// only3d set means to only check against 3D floors and midtexes.
//
2006-02-24 04:48:15 +00:00
//==========================================================================
2016-02-20 01:19:09 +00:00
bool ffcf_verbose ;
2006-02-24 04:48:15 +00:00
2016-02-20 01:19:09 +00:00
static bool PIT_FindFloorCeiling ( FMultiBlockLinesIterator & mit , FMultiBlockLinesIterator : : CheckResult & cres , const FBoundingBox & box , FCheckPosition & tmf , int flags )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
line_t * ld = cres . line ;
2008-04-06 17:33:43 +00:00
if ( box . Right ( ) < = ld - > bbox [ BOXLEFT ]
| | box . Left ( ) > = ld - > bbox [ BOXRIGHT ]
| | box . Top ( ) < = ld - > bbox [ BOXBOTTOM ]
2014-08-07 16:05:39 +00:00
| | box . Bottom ( ) > = ld - > bbox [ BOXTOP ] )
2006-02-24 04:48:15 +00:00
return true ;
2014-08-07 16:05:39 +00:00
if ( box . BoxOnLineSide ( ld ) ! = - 1 )
2006-02-24 04:48:15 +00:00
return true ;
// A line has been hit
2014-08-07 16:05:39 +00:00
2016-02-20 01:19:09 +00:00
if ( ffcf_verbose )
{
Printf ( " Hit line %d at position %f,%f, group %d \n " ,
int ( ld - lines ) , FIXED2FLOAT ( cres . position . x ) , FIXED2FLOAT ( cres . position . y ) , ld - > frontsector - > PortalGroup ) ;
}
2006-02-24 04:48:15 +00:00
if ( ! ld - > backsector )
{ // One sided line
return true ;
}
2016-02-19 13:08:41 +00:00
fixedvec2 refpoint = FindRefPoint ( ld , cres . position ) ;
2008-03-18 18:18:18 +00:00
FLineOpening open ;
2006-02-24 04:48:15 +00:00
2016-02-19 13:08:41 +00:00
P_LineOpening ( open , tmf . thing , ld , refpoint . x , refpoint . y , cres . position . x , cres . position . y , flags ) ;
// adjust floor / ceiling heights
if ( ! ( flags & FFCF_NOCEILING ) )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
if ( open . top < tmf . ceilingz )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
tmf . ceilingz = open . top ;
2016-02-19 22:54:55 +00:00
if ( open . topsec ! = NULL ) tmf . floorsector = open . topsec ;
2016-02-20 01:19:09 +00:00
if ( ffcf_verbose ) Printf ( " Adjust ceilingz to %f \n " , FIXED2FLOAT ( open . top ) ) ;
mit . StopUp ( ) ;
2006-02-24 04:48:15 +00:00
}
2016-02-19 13:08:41 +00:00
}
if ( ! ( flags & FFCF_NOFLOOR ) )
{
if ( open . bottom > tmf . floorz )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
tmf . floorz = open . bottom ;
if ( open . bottomsec ! = NULL ) tmf . floorsector = open . bottomsec ;
tmf . touchmidtex = open . touchmidtex ;
tmf . abovemidtex = open . abovemidtex ;
2016-02-20 01:19:09 +00:00
if ( ffcf_verbose ) Printf ( " Adjust floorz to %f \n " , FIXED2FLOAT ( open . bottom ) ) ;
if ( tmf . floorz > tmf . dropoffz + tmf . thing - > MaxDropOffHeight ) mit . StopDown ( ) ;
2006-02-24 04:48:15 +00:00
}
2016-02-19 13:08:41 +00:00
else if ( open . bottom = = tmf . floorz )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
tmf . touchmidtex | = open . touchmidtex ;
tmf . abovemidtex | = open . abovemidtex ;
2006-02-24 04:48:15 +00:00
}
2016-02-20 01:19:09 +00:00
if ( open . lowfloor < tmf . dropoffz & & open . lowfloor > FIXED_MIN )
2016-02-19 22:54:55 +00:00
{
2016-02-19 13:08:41 +00:00
tmf . dropoffz = open . lowfloor ;
2016-02-20 01:19:09 +00:00
if ( ffcf_verbose ) Printf ( " Adjust dropoffz to %f \n " , FIXED2FLOAT ( open . bottom ) ) ;
if ( tmf . floorz > tmf . dropoffz + tmf . thing - > MaxDropOffHeight ) mit . StopDown ( ) ;
2016-02-19 22:54:55 +00:00
}
2006-02-24 04:48:15 +00:00
}
return true ;
}
2009-01-25 21:59:38 +00:00
2009-10-11 17:44:50 +00:00
//==========================================================================
//
2016-02-19 13:08:41 +00:00
// calculates the actual floor and ceiling position at a given
// coordinate. Traverses through portals unless being told not to.
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2012-04-08 05:39:46 +00:00
void P_GetFloorCeilingZ ( FCheckPosition & tmf , int flags )
2009-01-25 21:59:38 +00:00
{
2016-02-19 13:08:41 +00:00
sector_t * sec = ( ! ( flags & FFCF_SAMESECTOR ) | | tmf . thing - > Sector = = NULL ) ? P_PointInSector ( tmf . x , tmf . y ) : tmf . thing - > Sector ;
2016-02-19 21:03:29 +00:00
F3DFloor * ffc , * fff ;
2009-03-22 11:37:56 +00:00
2016-02-20 01:19:09 +00:00
tmf . ceilingz = sec - > NextHighestCeilingAt ( tmf . thing , tmf . z + tmf . thing - > height , flags , & tmf . ceilingsector , & ffc ) ;
tmf . floorz = tmf . dropoffz = sec - > NextLowestFloorAt ( tmf . thing , tmf . z , flags , & tmf . floorsector , & fff ) ;
2016-02-19 21:03:29 +00:00
if ( fff )
2016-02-17 20:57:52 +00:00
{
2016-02-19 21:03:29 +00:00
tmf . floorpic = * fff - > top . texture ;
tmf . floorterrain = fff - > model - > GetTerrain ( fff - > top . isceiling ) ;
2009-03-22 11:37:56 +00:00
}
2012-04-08 05:39:46 +00:00
else
{
2016-02-19 21:03:29 +00:00
tmf . floorpic = tmf . floorsector - > GetTexture ( sector_t : : floor ) ;
tmf . floorterrain = tmf . floorsector - > GetTerrain ( sector_t : : floor ) ;
2012-04-08 05:39:46 +00:00
}
2016-02-19 21:03:29 +00:00
tmf . ceilingpic = ffc ? * ffc - > bottom . texture : tmf . ceilingsector - > GetTexture ( sector_t : : ceiling ) ;
2016-02-17 20:57:52 +00:00
tmf . sector = sec ;
2009-01-25 21:59:38 +00:00
}
2006-02-24 04:48:15 +00:00
//==========================================================================
//
// P_FindFloorCeiling
//
//==========================================================================
2014-08-07 16:05:39 +00:00
void P_FindFloorCeiling ( AActor * actor , int flags )
2006-02-24 04:48:15 +00:00
{
2008-04-08 10:47:28 +00:00
FCheckPosition tmf ;
2006-02-24 04:48:15 +00:00
2009-01-25 21:59:38 +00:00
tmf . thing = actor ;
2016-01-17 23:59:16 +00:00
tmf . x = actor - > X ( ) ;
tmf . y = actor - > Y ( ) ;
tmf . z = actor - > Z ( ) ;
2009-01-25 21:59:38 +00:00
2012-04-10 03:18:04 +00:00
if ( flags & FFCF_ONLYSPAWNPOS )
{
2012-04-14 03:55:46 +00:00
flags | = FFCF_3DRESTRICT ;
2012-04-10 03:18:04 +00:00
}
2016-02-17 20:57:52 +00:00
P_GetFloorCeilingZ ( tmf , flags ) ;
2016-02-19 13:08:41 +00:00
assert ( tmf . thing - > Sector ! = NULL ) ;
2016-02-17 20:57:52 +00:00
2009-03-22 11:37:56 +00:00
actor - > floorz = tmf . floorz ;
actor - > dropoffz = tmf . dropoffz ;
actor - > ceilingz = tmf . ceilingz ;
actor - > floorpic = tmf . floorpic ;
2016-01-09 11:10:36 +00:00
actor - > floorterrain = tmf . floorterrain ;
2009-03-22 11:37:56 +00:00
actor - > floorsector = tmf . floorsector ;
actor - > ceilingpic = tmf . ceilingpic ;
actor - > ceilingsector = tmf . ceilingsector ;
2016-02-20 01:19:09 +00:00
if ( ffcf_verbose ) Printf ( " Starting with ceilingz = %f, floorz = %f \n " , FIXED2FLOAT ( tmf . ceilingz ) , FIXED2FLOAT ( tmf . floorz ) ) ;
2006-02-24 04:48:15 +00:00
2008-04-08 20:52:49 +00:00
tmf . touchmidtex = false ;
2011-01-20 11:40:05 +00:00
tmf . abovemidtex = false ;
2006-02-24 04:48:15 +00:00
validcount + + ;
2016-02-19 13:08:41 +00:00
FPortalGroupArray grouplist ;
FMultiBlockLinesIterator mit ( grouplist , actor , tmf . x , tmf . y , actor - > radius ) ;
FMultiBlockLinesIterator : : CheckResult cres ;
2006-02-24 04:48:15 +00:00
2016-02-19 13:08:41 +00:00
// if we already have a valid floor/ceiling sector within the current sector,
// we do not need to iterate through plane portals to find a floor or ceiling.
if ( actor - > floorsector = = actor - > Sector ) mit . StopDown ( ) ;
if ( actor - > ceilingsector = = actor - > Sector ) mit . StopUp ( ) ;
while ( ( mit . Next ( & cres ) ) )
2008-04-06 17:33:43 +00:00
{
2016-02-20 01:19:09 +00:00
PIT_FindFloorCeiling ( mit , cres , mit . Box ( ) , tmf , flags | cres . portalflags ) ;
2008-04-06 17:33:43 +00:00
}
2008-03-18 18:18:18 +00:00
2008-04-08 10:47:28 +00:00
if ( tmf . touchmidtex ) tmf . dropoffz = tmf . floorz ;
2016-02-19 13:08:41 +00:00
bool usetmf = ! ( flags & FFCF_ONLYSPAWNPOS ) | | ( tmf . abovemidtex & & ( tmf . floorz < = actor - > Z ( ) ) ) ;
// when actual floor or ceiling are beyond a portal plane we also need to use the result of the blockmap iterator, regardless of the flags being specified.
if ( usetmf | | tmf . floorsector - > PortalGroup ! = actor - > Sector - > PortalGroup )
2008-05-14 07:45:40 +00:00
{
actor - > floorz = tmf . floorz ;
actor - > dropoffz = tmf . dropoffz ;
actor - > floorpic = tmf . floorpic ;
2016-01-09 11:10:36 +00:00
actor - > floorterrain = tmf . floorterrain ;
2008-05-14 07:45:40 +00:00
actor - > floorsector = tmf . floorsector ;
}
2016-02-19 13:08:41 +00:00
if ( usetmf | | tmf . ceilingsector - > PortalGroup ! = actor - > Sector - > PortalGroup )
2009-03-19 11:24:14 +00:00
{
2016-02-19 13:08:41 +00:00
actor - > ceilingz = tmf . ceilingz ;
actor - > ceilingpic = tmf . ceilingpic ;
actor - > ceilingsector = tmf . ceilingsector ;
2009-03-19 11:24:14 +00:00
}
2006-02-24 04:48:15 +00:00
}
2016-02-20 01:19:09 +00:00
// Debug CCMD for checking errors in the MultiBlockLinesIterator (needs to be removed when this code is complete)
CCMD ( ffcf )
{
ffcf_verbose = true ;
P_FindFloorCeiling ( players [ 0 ] . mo , 0 ) ;
ffcf_verbose = false ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// TELEPORT MOVE
//
//
// P_TeleportMove
//
// [RH] Added telefrag parameter: When true, anything in the spawn spot
// will always be telefragged, and the move will be successful.
// Added z parameter. Originally, the thing's z was set *after* the
// move was made, so the height checking I added for 1.13 could
// potentially erroneously indicate the move was okay if the thing
// was being teleported between two non-overlapping height ranges.
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2016-01-16 15:30:53 +00:00
bool P_TeleportMove ( AActor * thing , fixed_t x , fixed_t y , fixed_t z , bool telefrag , bool modifyactor )
2006-02-24 04:48:15 +00:00
{
2008-04-08 10:47:28 +00:00
FCheckPosition tmf ;
2012-03-18 03:52:18 +00:00
sector_t * oldsec = thing - > Sector ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
// kill anything occupying the position
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
// The base floor/ceiling is from the subsector that contains the point.
// Any contacted lines the step closer together will adjust them.
2008-04-08 10:47:28 +00:00
tmf . thing = thing ;
tmf . x = x ;
tmf . y = y ;
tmf . z = z ;
2011-01-20 11:40:05 +00:00
tmf . touchmidtex = false ;
tmf . abovemidtex = false ;
2012-04-08 20:01:43 +00:00
P_GetFloorCeilingZ ( tmf , 0 ) ;
2014-08-07 16:05:39 +00:00
spechit . Clear ( ) ;
2006-02-24 04:48:15 +00:00
2013-08-12 18:41:33 +00:00
bool StompAlwaysFrags = ( ( thing - > flags2 & MF2_TELESTOMP ) | | ( level . flags & LEVEL_MONSTERSTELEFRAG ) | | telefrag ) & & ! ( thing - > flags7 & MF7_NOTELESTOMP ) ;
2006-02-24 04:48:15 +00:00
2012-04-08 20:20:39 +00:00
// P_LineOpening requires the thing's z to be the destination z in order to work.
2016-01-17 23:59:16 +00:00
fixed_t savedz = thing - > Z ( ) ;
thing - > SetZ ( z ) ;
2016-02-19 13:08:41 +00:00
FPortalGroupArray grouplist ;
FMultiBlockLinesIterator mit ( grouplist , thing , tmf . x , tmf . y , thing - > radius ) ;
FMultiBlockLinesIterator : : CheckResult cres ;
while ( mit . Next ( & cres ) )
2006-02-24 04:48:15 +00:00
{
2016-02-20 01:19:09 +00:00
PIT_FindFloorCeiling ( mit , cres , mit . Box ( ) , tmf , 0 ) ;
2006-02-24 04:48:15 +00:00
}
2016-01-17 23:59:16 +00:00
thing - > SetZ ( savedz ) ;
2006-02-24 04:48:15 +00:00
2008-04-08 10:47:28 +00:00
if ( tmf . touchmidtex ) tmf . dropoffz = tmf . floorz ;
2006-02-24 04:48:15 +00:00
2008-04-15 10:04:41 +00:00
FBlockThingsIterator it2 ( FBoundingBox ( x , y , thing - > radius ) ) ;
2008-04-07 21:14:28 +00:00
AActor * th ;
2008-04-06 17:33:43 +00:00
2008-04-07 21:14:28 +00:00
while ( ( th = it2 . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
if ( ! ( th - > flags & MF_SHOOTABLE ) )
continue ;
// don't clip against self
if ( th = = thing )
continue ;
2008-04-15 10:04:41 +00:00
fixed_t blockdist = th - > radius + tmf . thing - > radius ;
2016-01-17 23:59:16 +00:00
if ( abs ( th - > X ( ) - tmf . x ) > = blockdist | | abs ( th - > Y ( ) - tmf . y ) > = blockdist )
2008-04-15 10:04:41 +00:00
continue ;
2016-01-29 10:29:47 +00:00
if ( ( th - > flags2 | tmf . thing - > flags2 ) & MF2_THRUACTORS )
continue ;
if ( tmf . thing - > flags6 & MF6_THRUSPECIES & & tmf . thing - > GetSpecies ( ) = = th - > GetSpecies ( ) )
continue ;
2008-04-07 21:14:28 +00:00
// [RH] Z-Check
// But not if not MF2_PASSMOBJ or MF3_DONTOVERLAP are set!
// Otherwise those things would get stuck inside each other.
if ( ( thing - > flags2 & MF2_PASSMOBJ | | th - > flags4 & MF4_ACTLIKEBRIDGE ) & & ! ( i_compatflags & COMPATF_NO_PASSMOBJ ) )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
if ( ! ( th - > flags3 & thing - > flags3 & MF3_DONTOVERLAP ) )
2006-02-24 04:48:15 +00:00
{
2016-01-17 23:59:16 +00:00
if ( z > th - > Top ( ) | | // overhead
z + thing - > height < th - > Z ( ) ) // underneath
2008-04-07 21:14:28 +00:00
continue ;
2006-02-24 04:48:15 +00:00
}
}
2008-04-07 21:14:28 +00:00
// monsters don't stomp things except on boss level
// [RH] Some Heretic/Hexen monsters can telestomp
2013-08-13 19:01:14 +00:00
// ... and some items can never be telefragged while others will be telefragged by everything that teleports upon them.
if ( ( StompAlwaysFrags & & ! ( th - > flags6 & MF6_NOTELEFRAG ) ) | | ( th - > flags7 & MF7_ALWAYSTELEFRAG ) )
2008-04-07 21:14:28 +00:00
{
2014-08-23 03:17:11 +00:00
// Don't actually damage if predicting a teleport
if ( thing - > player = = NULL | | ! ( thing - > player - > cheats & CF_PREDICTING ) )
P_DamageMobj ( th , thing , thing , TELEFRAG_DAMAGE , NAME_Telefrag , DMG_THRUSTLESS ) ;
2008-04-07 21:14:28 +00:00
continue ;
}
return false ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
2016-01-16 15:30:53 +00:00
if ( modifyactor )
2006-02-24 04:48:15 +00:00
{
2016-01-16 15:30:53 +00:00
// the move is ok, so link the thing into its new position
2016-01-17 23:59:16 +00:00
thing - > SetOrigin ( x , y , z , false ) ;
2016-01-16 15:30:53 +00:00
thing - > floorz = tmf . floorz ;
thing - > ceilingz = tmf . ceilingz ;
thing - > floorsector = tmf . floorsector ;
thing - > floorpic = tmf . floorpic ;
thing - > floorterrain = tmf . floorterrain ;
thing - > ceilingsector = tmf . ceilingsector ;
thing - > ceilingpic = tmf . ceilingpic ;
thing - > dropoffz = tmf . dropoffz ; // killough 11/98
thing - > BlockingLine = NULL ;
if ( thing - > flags2 & MF2_FLOORCLIP )
{
thing - > AdjustFloorClip ( ) ;
}
2006-02-24 04:48:15 +00:00
2016-01-16 15:30:53 +00:00
if ( thing = = players [ consoleplayer ] . camera )
{
R_ResetViewInterpolation ( ) ;
}
2006-02-24 04:48:15 +00:00
2016-01-16 15:30:53 +00:00
thing - > PrevX = x ;
thing - > PrevY = y ;
thing - > PrevZ = z ;
2006-02-24 04:48:15 +00:00
2016-01-16 15:30:53 +00:00
// If this teleport was caused by a move, P_TryMove() will handle the
// sector transition messages better than we can here.
if ( ! ( thing - > flags6 & MF6_INTRYMOVE ) )
{
thing - > CheckSectorTransition ( oldsec ) ;
}
2012-03-18 03:52:18 +00:00
}
2006-02-24 04:48:15 +00:00
return true ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// [RH] P_PlayerStartStomp
//
// Like P_TeleportMove, but it doesn't move anything, and only monsters and other
// players get telefragged.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2016-01-29 02:08:09 +00:00
void P_PlayerStartStomp ( AActor * actor , bool mononly )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
AActor * th ;
2016-01-17 23:59:16 +00:00
FBlockThingsIterator it ( FBoundingBox ( actor - > X ( ) , actor - > Y ( ) , actor - > radius ) ) ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
while ( ( th = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
if ( ! ( th - > flags & MF_SHOOTABLE ) )
continue ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
// don't clip against self, and don't kill your own voodoo dolls
if ( th = = actor | | ( th - > player = = actor - > player & & th - > player ! = NULL ) )
continue ;
2006-02-24 04:48:15 +00:00
2008-04-15 10:04:41 +00:00
if ( ! th - > intersects ( actor ) )
continue ;
2008-04-07 21:14:28 +00:00
// only kill monsters and other players
if ( th - > player = = NULL & & ! ( th - > flags3 & MF3_ISMONSTER ) )
continue ;
2006-02-24 04:48:15 +00:00
2016-01-29 02:08:09 +00:00
if ( th - > player ! = NULL & & mononly )
continue ;
2016-01-17 23:59:16 +00:00
if ( actor - > Z ( ) > th - > Top ( ) )
2008-04-07 21:14:28 +00:00
continue ; // overhead
2016-01-17 23:59:16 +00:00
if ( actor - > Top ( ) < th - > Z ( ) )
2008-04-07 21:14:28 +00:00
continue ; // underneath
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
P_DamageMobj ( th , actor , actor , TELEFRAG_DAMAGE , NAME_Telefrag ) ;
2006-02-24 04:48:15 +00:00
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2016-01-09 09:11:20 +00:00
inline fixed_t secfriction ( const sector_t * sec , int plane = sector_t : : floor )
2006-02-24 04:48:15 +00:00
{
2016-01-09 09:11:20 +00:00
if ( sec - > Flags & SECF_FRICTION ) return sec - > friction ;
2016-01-09 11:10:36 +00:00
fixed_t friction = Terrains [ sec - > GetTerrain ( plane ) ] . Friction ;
2016-01-09 09:11:20 +00:00
return friction ! = 0 ? friction : ORIG_FRICTION ;
2006-02-24 04:48:15 +00:00
}
2016-01-09 09:11:20 +00:00
inline fixed_t secmovefac ( const sector_t * sec , int plane = sector_t : : floor )
2006-02-24 04:48:15 +00:00
{
2016-01-16 08:05:46 +00:00
if ( sec - > Flags & SECF_FRICTION ) return sec - > movefactor ;
2016-01-09 11:10:36 +00:00
fixed_t movefactor = Terrains [ sec - > GetTerrain ( plane ) ] . MoveFactor ;
2016-01-09 09:11:20 +00:00
return movefactor ! = 0 ? movefactor : ORIG_FRICTION_FACTOR ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// killough 8/28/98:
//
// P_GetFriction()
//
// Returns the friction associated with a particular mobj.
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
int P_GetFriction ( const AActor * mo , int * frictionfactor )
2006-02-24 04:48:15 +00:00
{
int friction = ORIG_FRICTION ;
int movefactor = ORIG_FRICTION_FACTOR ;
2012-04-08 04:43:19 +00:00
fixed_t newfriction ;
2006-02-24 04:48:15 +00:00
const msecnode_t * m ;
2016-02-17 10:39:14 +00:00
sector_t * sec ;
2006-02-24 04:48:15 +00:00
2012-08-22 21:31:48 +00:00
if ( mo - > IsNoClip2 ( ) )
{
// The default values are fine for noclip2 mode
}
else if ( mo - > flags2 & MF2_FLY & & mo - > flags & MF_NOGRAVITY )
2006-02-24 04:48:15 +00:00
{
friction = FRICTION_FLY ;
}
2008-06-25 22:16:04 +00:00
else if ( ( ! ( mo - > flags & MF_NOGRAVITY ) & & mo - > waterlevel > 1 ) | |
2016-01-20 00:48:57 +00:00
( mo - > waterlevel = = 1 & & mo - > Z ( ) > mo - > floorz + 6 * FRACUNIT ) )
2006-02-24 04:48:15 +00:00
{
2012-04-08 04:43:19 +00:00
friction = secfriction ( mo - > Sector ) ;
movefactor = secmovefac ( mo - > Sector ) > > 1 ;
2015-06-06 22:27:03 +00:00
// Check 3D floors -- might be the source of the waterlevel
for ( unsigned i = 0 ; i < mo - > Sector - > e - > XFloor . ffloors . Size ( ) ; i + + )
{
F3DFloor * rover = mo - > Sector - > e - > XFloor . ffloors [ i ] ;
if ( ! ( rover - > flags & FF_EXISTS ) ) continue ;
if ( ! ( rover - > flags & FF_SWIMMABLE ) ) continue ;
2016-01-20 00:48:57 +00:00
if ( mo - > Z ( ) > rover - > top . plane - > ZatPoint ( mo ) | |
mo - > Z ( ) < rover - > bottom . plane - > ZatPoint ( mo ) )
2015-06-06 22:27:03 +00:00
continue ;
2016-01-09 09:11:20 +00:00
newfriction = secfriction ( rover - > model , rover - > top . isceiling ) ;
2015-06-06 22:27:03 +00:00
if ( newfriction < friction | | friction = = ORIG_FRICTION )
{
friction = newfriction ;
2016-01-09 09:11:20 +00:00
movefactor = secmovefac ( rover - > model , rover - > top . isceiling ) > > 1 ;
2015-06-06 22:27:03 +00:00
}
}
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
else if ( var_friction & & ! ( mo - > flags & ( MF_NOCLIP | MF_NOGRAVITY ) ) )
2006-02-24 04:48:15 +00:00
{ // When the object is straddling sectors with the same
// floor height that have different frictions, use the lowest
// friction value (muddy has precedence over icy).
for ( m = mo - > touching_sectorlist ; m ; m = m - > m_tnext )
{
sec = m - > m_sector ;
2016-02-17 10:39:14 +00:00
fixedvec3 pos = mo - > PosRelative ( sec ) ;
2009-01-04 15:00:29 +00:00
// 3D floors must be checked, too
2012-04-08 04:43:19 +00:00
for ( unsigned i = 0 ; i < sec - > e - > XFloor . ffloors . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
2012-04-08 04:43:19 +00:00
F3DFloor * rover = sec - > e - > XFloor . ffloors [ i ] ;
if ( ! ( rover - > flags & FF_EXISTS ) ) continue ;
2009-01-04 15:00:29 +00:00
2015-06-06 22:27:03 +00:00
if ( rover - > flags & FF_SOLID )
{
// Must be standing on a solid floor
2016-02-17 10:39:14 +00:00
if ( mo - > Z ( ) ! = rover - > top . plane - > ZatPoint ( pos ) ) continue ;
2015-06-06 22:27:03 +00:00
}
else if ( rover - > flags & FF_SWIMMABLE )
{
// Or on or inside a swimmable floor (e.g. in shallow water)
2016-02-17 10:39:14 +00:00
if ( mo - > Z ( ) > rover - > top . plane - > ZatPoint ( pos ) | |
( mo - > Top ( ) ) < rover - > bottom . plane - > ZatPoint ( pos ) )
2015-06-06 22:27:03 +00:00
continue ;
}
else
continue ;
2016-01-09 09:11:20 +00:00
newfriction = secfriction ( rover - > model , rover - > top . isceiling ) ;
2012-04-08 04:43:19 +00:00
if ( newfriction < friction | | friction = = ORIG_FRICTION )
2009-01-04 15:00:29 +00:00
{
friction = newfriction ;
2016-01-09 09:11:20 +00:00
movefactor = secmovefac ( rover - > model , rover - > top . isceiling ) ;
2009-01-04 15:00:29 +00:00
}
}
2016-01-06 01:01:59 +00:00
if ( ! ( sec - > Flags & SECF_FRICTION ) & &
2016-01-09 11:10:36 +00:00
Terrains [ sec - > GetTerrain ( sector_t : : floor ) ] . Friction = = 0 )
2006-02-24 04:48:15 +00:00
{
continue ;
}
2012-04-08 04:43:19 +00:00
newfriction = secfriction ( sec ) ;
if ( ( newfriction < friction | | friction = = ORIG_FRICTION ) & &
2016-02-17 10:39:14 +00:00
( mo - > Z ( ) < = sec - > floorplane . ZatPoint ( pos ) | |
2009-05-23 10:21:33 +00:00
( sec - > GetHeightSec ( ) ! = NULL & &
2016-02-17 10:39:14 +00:00
mo - > Z ( ) < = sec - > heightsec - > floorplane . ZatPoint ( pos ) ) ) )
2006-02-24 04:48:15 +00:00
{
2012-04-08 04:43:19 +00:00
friction = newfriction ;
movefactor = secmovefac ( sec ) ;
2006-02-24 04:48:15 +00:00
}
}
}
2014-08-07 16:05:39 +00:00
2014-07-29 04:15:50 +00:00
if ( mo - > Friction ! = FRACUNIT )
{
friction = clamp ( FixedMul ( friction , mo - > Friction ) , 0 , FRACUNIT ) ;
movefactor = FrictionToMoveFactor ( friction ) ;
}
2006-02-24 04:48:15 +00:00
if ( frictionfactor )
* frictionfactor = movefactor ;
return friction ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
2006-02-24 04:48:15 +00:00
// phares 3/19/98
// P_GetMoveFactor() returns the value by which the x,y
// movements are multiplied to add to player movement.
//
// killough 8/28/98: rewritten
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
int P_GetMoveFactor ( const AActor * mo , int * frictionp )
2006-02-24 04:48:15 +00:00
{
int movefactor , friction ;
// If the floor is icy or muddy, it's harder to get moving. This is where
// the different friction factors are applied to 'trying to move'. In
// p_mobj.c, the friction factors are applied as you coast and slow down.
if ( ( friction = P_GetFriction ( mo , & movefactor ) ) < ORIG_FRICTION )
{
// phares 3/11/98: you start off slowly, then increase as
// you get better footing
2009-06-30 20:57:51 +00:00
int velocity = P_AproxDistance ( mo - > velx , mo - > vely ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
if ( velocity > MORE_FRICTION_VELOCITY < < 2 )
2006-02-24 04:48:15 +00:00
movefactor < < = 3 ;
2014-08-07 16:05:39 +00:00
else if ( velocity > MORE_FRICTION_VELOCITY < < 1 )
2006-02-24 04:48:15 +00:00
movefactor < < = 2 ;
2009-06-30 20:57:51 +00:00
else if ( velocity > MORE_FRICTION_VELOCITY )
2006-02-24 04:48:15 +00:00
movefactor < < = 1 ;
}
if ( frictionp )
* frictionp = friction ;
return movefactor ;
}
2016-02-19 13:08:41 +00:00
2006-02-24 04:48:15 +00:00
//
// MOVEMENT ITERATOR FUNCTIONS
//
2009-10-11 17:44:50 +00:00
//==========================================================================
//
2006-02-24 04:48:15 +00:00
//
// PIT_CheckLine
// Adjusts tmfloorz and tmceilingz as lines are contacted
//
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2006-02-24 04:48:15 +00:00
static // killough 3/26/98: make static
2014-08-07 16:05:39 +00:00
bool PIT_CheckLine ( line_t * ld , const FBoundingBox & box , FCheckPosition & tm )
2006-02-24 04:48:15 +00:00
{
bool rail = false ;
2008-04-06 17:33:43 +00:00
if ( box . Right ( ) < = ld - > bbox [ BOXLEFT ]
| | box . Left ( ) > = ld - > bbox [ BOXRIGHT ]
| | box . Top ( ) < = ld - > bbox [ BOXBOTTOM ]
2014-08-07 16:05:39 +00:00
| | box . Bottom ( ) > = ld - > bbox [ BOXTOP ] )
2006-02-24 04:48:15 +00:00
return true ;
2014-08-07 16:05:39 +00:00
if ( box . BoxOnLineSide ( ld ) ! = - 1 )
2006-02-24 04:48:15 +00:00
return true ;
// A line has been hit
2014-08-07 16:05:39 +00:00
/*
=
= The moving thing ' s destination position will cross the given line .
= If this should not be allowed , return false .
= If the line is special , keep track of it to process later if the move
= is proven ok . NOTE : specials are NOT sorted by order , so two special lines
= that are only 8 pixels apart could be crossed in either order .
*/
2006-02-24 04:48:15 +00:00
if ( ! ld - > backsector )
{ // One sided line
2008-04-08 20:52:49 +00:00
if ( tm . thing - > flags2 & MF2_BLASTED )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_DamageMobj ( tm . thing , NULL , NULL , tm . thing - > Mass > > 5 , NAME_Melee ) ;
2006-02-24 04:48:15 +00:00
}
2008-04-08 20:52:49 +00:00
tm . thing - > BlockingLine = ld ;
2014-08-07 16:05:39 +00:00
CheckForPushSpecial ( ld , 0 , tm . thing , false ) ;
2006-02-24 04:48:15 +00:00
return false ;
}
2010-05-25 03:40:37 +00:00
// MBF bouncers are treated as missiles here.
bool Projectile = ( tm . thing - > flags & MF_MISSILE | | tm . thing - > BounceFlags & BOUNCE_MBF ) ;
// MBF considers that friendly monsters are not blocked by monster-blocking lines.
// This is added here as a compatibility option. Note that monsters that are dehacked
// into being friendly with the MBF flag automatically gain MF3_NOBLOCKMONST, so this
// just optionally generalizes the behavior to other friendly monsters.
bool NotBlocked = ( ( tm . thing - > flags3 & MF3_NOBLOCKMONST )
| | ( ( i_compatflags & COMPATF_NOBLOCKFRIENDS ) & & ( tm . thing - > flags & MF_FRIENDLY ) ) ) ;
2016-01-20 00:48:57 +00:00
fixedvec3 pos = tm . thing - > PosRelative ( ld ) ;
2014-08-07 16:05:39 +00:00
if ( ! ( Projectile ) | | ( ld - > flags & ( ML_BLOCKEVERYTHING | ML_BLOCKPROJECTILE ) ) )
2010-05-25 03:40:37 +00:00
{
2006-02-24 04:48:15 +00:00
if ( ld - > flags & ML_RAILING )
{
rail = true ;
}
2014-08-07 16:05:39 +00:00
else if ( ( ld - > flags & ( ML_BLOCKING | ML_BLOCKEVERYTHING ) ) | | // explicitly blocking everything
2009-09-14 20:47:53 +00:00
( ! ( NotBlocked ) & & ( ld - > flags & ML_BLOCKMONSTERS ) ) | | // block monsters only
( tm . thing - > player ! = NULL & & ( ld - > flags & ML_BLOCK_PLAYERS ) ) | | // block players
( ( Projectile ) & & ( ld - > flags & ML_BLOCKPROJECTILE ) ) | | // block projectiles
( ( tm . thing - > flags & MF_FLOAT ) & & ( ld - > flags & ML_BLOCK_FLOATERS ) ) ) // block floaters
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
if ( tm . thing - > flags2 & MF2_BLASTED )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_DamageMobj ( tm . thing , NULL , NULL , tm . thing - > Mass > > 5 , NAME_Melee ) ;
2006-02-24 04:48:15 +00:00
}
2008-04-08 20:52:49 +00:00
tm . thing - > BlockingLine = ld ;
2011-04-22 03:38:09 +00:00
// Calculate line side based on the actor's original position, not the new one.
2016-01-20 00:48:57 +00:00
CheckForPushSpecial ( ld , P_PointOnLineSide ( pos . x , pos . y , ld ) , tm . thing , false ) ;
2006-02-24 04:48:15 +00:00
return false ;
}
}
// [RH] Steep sectors count as dropoffs (unless already in one)
2008-04-08 20:52:49 +00:00
if ( ! ( tm . thing - > flags & MF_DROPOFF ) & &
2014-08-07 16:05:39 +00:00
! ( tm . thing - > flags & ( MF_NOGRAVITY | MF_NOCLIP ) ) )
2006-02-24 04:48:15 +00:00
{
2015-02-08 09:03:49 +00:00
secplane_t frontplane , backplane ;
2010-08-12 07:05:31 +00:00
// Check 3D floors as well
2016-01-20 00:48:57 +00:00
frontplane = P_FindFloorPlane ( ld - > frontsector , pos . x , pos . y , tm . thing - > floorz ) ;
backplane = P_FindFloorPlane ( ld - > backsector , pos . x , pos . y , tm . thing - > floorz ) ;
2010-08-12 07:05:31 +00:00
if ( frontplane . c < STEEPSLOPE | | backplane . c < STEEPSLOPE )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
const msecnode_t * node = tm . thing - > touching_sectorlist ;
2006-02-24 04:48:15 +00:00
bool allow = false ;
int count = 0 ;
while ( node ! = NULL )
{
count + + ;
if ( node - > m_sector - > floorplane . c < STEEPSLOPE )
{
allow = true ;
break ;
}
node = node - > m_tnext ;
}
if ( ! allow )
{
return false ;
}
}
}
2016-02-19 13:08:41 +00:00
fixedvec2 rpos = { tm . x , tm . y } ;
fixedvec2 ref = FindRefPoint ( ld , rpos ) ;
2008-03-18 18:18:18 +00:00
FLineOpening open ;
2006-02-24 04:48:15 +00:00
2016-02-19 18:14:49 +00:00
P_LineOpening ( open , tm . thing , ld , ref . x , ref . y , tm . x , tm . y , 0 ) ;
2016-02-19 13:08:41 +00:00
// the floorplane on both sides is identical with the current one
// so don't mess around with the z-position.
if ( ld - > frontsector - > floorplane = = ld - > backsector - > floorplane & &
ld - > frontsector - > floorplane = = tm . thing - > Sector - > floorplane & &
! ld - > frontsector - > e - > XFloor . ffloors . Size ( ) & & ! ld - > backsector - > e - > XFloor . ffloors . Size ( ) & &
! open . abovemidtex )
2006-02-24 04:48:15 +00:00
{
2016-02-19 13:08:41 +00:00
open . bottom = INT_MIN ;
2006-02-24 04:48:15 +00:00
}
if ( rail & &
// Eww! Gross! This check means the rail only exists if you stand on the
// high side of the rail. So if you're walking on the low side of the rail,
// it's possible to get stuck in the rail until you jump out. Unfortunately,
// there is an area on Strife MAP04 that requires this behavior. Still, it's
// better than Strife's handling of rails, which lets you jump into rails
// from either side. How long until somebody reports this as a bug and I'm
// forced to say, "It's not a bug. It's a feature?" Ugh.
2009-02-03 19:11:43 +00:00
( ! ( level . flags2 & LEVEL2_RAILINGHACK ) | |
2016-02-19 13:08:41 +00:00
open . bottom = = tm . thing - > Sector - > floorplane . ZatPoint ( ref . x , ref . y ) ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
open . bottom + = 32 * FRACUNIT ;
2006-02-24 04:48:15 +00:00
}
// adjust floor / ceiling heights
2008-04-08 20:52:49 +00:00
if ( open . top < tm . ceilingz )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
tm . ceilingz = open . top ;
tm . ceilingsector = open . topsec ;
tm . ceilingpic = open . ceilingpic ;
tm . ceilingline = ld ;
tm . thing - > BlockingLine = ld ;
2006-02-24 04:48:15 +00:00
}
2008-04-08 20:52:49 +00:00
if ( open . bottom > tm . floorz )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
tm . floorz = open . bottom ;
tm . floorsector = open . bottomsec ;
tm . floorpic = open . floorpic ;
2016-01-09 11:10:36 +00:00
tm . floorterrain = open . floorterrain ;
2008-04-08 20:52:49 +00:00
tm . touchmidtex = open . touchmidtex ;
2011-01-20 11:40:05 +00:00
tm . abovemidtex = open . abovemidtex ;
2008-04-08 20:52:49 +00:00
tm . thing - > BlockingLine = ld ;
2006-02-24 04:48:15 +00:00
}
2008-05-17 17:57:50 +00:00
else if ( open . bottom = = tm . floorz )
{
tm . touchmidtex | = open . touchmidtex ;
2011-01-20 11:40:05 +00:00
tm . abovemidtex | = open . abovemidtex ;
2008-05-17 17:57:50 +00:00
}
2006-02-24 04:48:15 +00:00
2008-04-08 20:52:49 +00:00
if ( open . lowfloor < tm . dropoffz )
tm . dropoffz = open . lowfloor ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
// if contacted a special line, add it to the list
if ( ld - > special )
{
2014-08-07 16:05:39 +00:00
spechit . Push ( ld ) ;
2006-02-24 04:48:15 +00:00
}
return true ;
}
2014-12-31 09:13:15 +00:00
//==========================================================================
//
// Isolated to keep the code readable and fix the logic
//
//==========================================================================
static bool CheckRipLevel ( AActor * victim , AActor * projectile )
{
if ( victim - > RipLevelMin > 0 & & projectile - > RipperLevel < victim - > RipLevelMin ) return false ;
if ( victim - > RipLevelMax > 0 & & projectile - > RipperLevel > victim - > RipLevelMax ) return false ;
return true ;
}
2015-03-02 09:02:49 +00:00
//==========================================================================
//
// Isolated to keep the code readable and allow reuse in other attacks
//
//==========================================================================
static bool CanAttackHurt ( AActor * victim , AActor * shooter )
{
// players are never subject to infighting settings and are always allowed
// to harm / be harmed by anything.
if ( ! victim - > player & & ! shooter - > player )
{
2015-11-29 11:10:12 +00:00
int infight = G_SkillProperty ( SKILLP_Infight ) ;
2015-03-02 09:02:49 +00:00
if ( infight < 0 )
{
// -1: Monsters cannot hurt each other, but make exceptions for
// friendliness and hate status.
if ( shooter - > flags & MF_SHOOTABLE )
{
// Question: Should monsters be allowed to shoot barrels in this mode?
// The old code does not.
if ( victim - > flags3 & MF3_ISMONSTER )
{
// Monsters that are clearly hostile can always hurt each other
if ( ! victim - > IsHostile ( shooter ) )
{
// The same if the shooter hates the target
if ( victim - > tid = = 0 | | shooter - > TIDtoHate ! = victim - > tid )
{
return false ;
}
}
}
}
}
else if ( infight = = 0 )
{
// 0: Monsters cannot hurt same species except
// cases where they are clearly supposed to do that
if ( victim - > IsFriend ( shooter ) )
{
// Friends never harm each other, unless the shooter has the HARMFRIENDS set.
if ( ! ( shooter - > flags7 & MF7_HARMFRIENDS ) ) return false ;
}
else
{
if ( victim - > TIDtoHate ! = 0 & & victim - > TIDtoHate = = shooter - > TIDtoHate )
{
// [RH] Don't hurt monsters that hate the same victim as you do
return false ;
}
if ( victim - > GetSpecies ( ) = = shooter - > GetSpecies ( ) & & ! ( victim - > flags6 & MF6_DOHARMSPECIES ) )
{
// Don't hurt same species or any relative -
// but only if the target isn't one's hostile.
if ( ! victim - > IsHostile ( shooter ) )
{
// Allow hurting monsters the shooter hates.
if ( victim - > tid = = 0 | | shooter - > TIDtoHate ! = victim - > tid )
{
return false ;
}
}
}
}
}
// else if (infight==1) every shot hurts anything - no further tests needed
}
return true ;
}
2006-02-24 04:48:15 +00:00
//==========================================================================
//
// PIT_CheckThing
//
//==========================================================================
2014-08-07 16:05:39 +00:00
bool PIT_CheckThing ( AActor * thing , FCheckPosition & tm )
2006-02-24 04:48:15 +00:00
{
fixed_t topz ;
2006-09-14 00:02:31 +00:00
bool solid ;
2006-02-24 04:48:15 +00:00
int damage ;
2008-04-15 10:04:41 +00:00
2015-08-30 10:02:36 +00:00
// don't clip against self
if ( thing = = tm . thing )
return true ;
2014-08-07 16:05:39 +00:00
if ( ! ( ( thing - > flags & ( MF_SOLID | MF_SPECIAL | MF_SHOOTABLE ) ) | | thing - > flags6 & MF6_TOUCHY ) )
2009-09-14 20:47:53 +00:00
return true ; // can't hit thing
2016-01-20 00:48:57 +00:00
fixedvec3 thingpos = thing - > PosRelative ( tm . thing ) ;
2009-09-14 20:47:53 +00:00
fixed_t blockdist = thing - > radius + tm . thing - > radius ;
2016-01-20 10:39:41 +00:00
if ( abs ( thingpos . x - tm . x ) > = blockdist | | abs ( thingpos . y - tm . y ) > = blockdist )
2009-09-14 20:47:53 +00:00
return true ;
2014-08-07 16:05:39 +00:00
if ( ( thing - > flags2 | tm . thing - > flags2 ) & MF2_THRUACTORS )
2009-05-30 08:56:40 +00:00
return true ;
2009-06-10 20:14:47 +00:00
if ( ( tm . thing - > flags6 & MF6_THRUSPECIES ) & & ( tm . thing - > GetSpecies ( ) = = thing - > GetSpecies ( ) ) )
2009-05-31 20:14:16 +00:00
return true ;
2008-04-08 20:52:49 +00:00
tm . thing - > BlockingMobj = thing ;
2016-01-20 00:48:57 +00:00
topz = thing - > Top ( ) ;
2014-08-07 16:05:39 +00:00
if ( ! ( i_compatflags & COMPATF_NO_PASSMOBJ ) & & ! ( tm . thing - > flags & ( MF_FLOAT | MF_MISSILE | MF_SKULLFLY | MF_NOGRAVITY ) ) & &
2006-02-24 04:48:15 +00:00
( thing - > flags & MF_SOLID ) & & ( thing - > flags4 & MF4_ACTLIKEBRIDGE ) )
{
// [RH] Let monsters walk on actors as well as floors
2008-04-08 20:52:49 +00:00
if ( ( tm . thing - > flags3 & MF3_ISMONSTER ) & &
2016-01-20 00:48:57 +00:00
topz > = tm . floorz & & topz < = tm . thing - > Z ( ) + tm . thing - > MaxStepHeight )
2006-02-24 04:48:15 +00:00
{
// The commented-out if is an attempt to prevent monsters from walking off a
// thing further than they would walk off a ledge. I can't think of an easy
// way to do this, so I restrict them to only walking on bridges instead.
// Uncommenting the if here makes it almost impossible for them to walk on
// anything, bridge or otherwise.
2014-08-07 16:05:39 +00:00
// if (abs(thing->x - tmx) <= thing->radius &&
// abs(thing->y - tmy) <= thing->radius)
2006-02-24 04:48:15 +00:00
{
2008-04-10 14:38:43 +00:00
tm . stepthing = thing ;
2008-04-08 20:52:49 +00:00
tm . floorz = topz ;
2006-02-24 04:48:15 +00:00
}
}
}
2009-12-06 22:10:25 +00:00
// Both things overlap in x or y direction
bool unblocking = false ;
2015-04-27 18:37:01 +00:00
if ( ( tm . FromPMove | | tm . thing - > player ! = NULL ) & & thing - > flags & MF_SOLID )
2009-12-06 22:10:25 +00:00
{
// Both actors already overlap. To prevent them from remaining stuck allow the move if it
2010-09-16 20:15:44 +00:00
// takes them further apart or the move does not change the position (when called from P_ChangeSector.)
2016-01-20 00:48:57 +00:00
if ( tm . x = = tm . thing - > X ( ) & & tm . y = = tm . thing - > Y ( ) )
2009-12-06 22:10:25 +00:00
{
2010-09-16 20:15:44 +00:00
unblocking = true ;
}
2016-01-20 00:48:57 +00:00
else if ( abs ( thingpos . x - tm . thing - > X ( ) ) < ( thing - > radius + tm . thing - > radius ) & &
abs ( thingpos . y - tm . thing - > Y ( ) ) < ( thing - > radius + tm . thing - > radius ) )
2015-04-27 18:37:01 +00:00
2010-09-16 20:15:44 +00:00
{
2016-01-10 16:52:41 +00:00
fixed_t newdist = thing - > AproxDistance ( tm . x , tm . y , tm . thing ) ;
fixed_t olddist = thing - > AproxDistance ( tm . thing ) ;
2010-09-16 20:15:44 +00:00
if ( newdist > olddist )
{
// ... but not if they did not overlap in z-direction before but would after the move.
2016-01-20 10:39:41 +00:00
unblocking = ! ( ( tm . thing - > Z ( ) > = topz & & tm . z < topz ) | |
2016-01-20 00:48:57 +00:00
( tm . thing - > Top ( ) < = thingpos . z & & tm . thing - > Top ( ) > thingpos . z ) ) ;
2010-09-16 20:15:44 +00:00
}
2009-12-06 22:10:25 +00:00
}
}
2006-02-24 04:48:15 +00:00
// [RH] If the other thing is a bridge, then treat the moving thing as if it had MF2_PASSMOBJ, so
// you can use a scrolling floor to move scenery items underneath a bridge.
2008-04-08 20:52:49 +00:00
if ( ( tm . thing - > flags2 & MF2_PASSMOBJ | | thing - > flags4 & MF4_ACTLIKEBRIDGE ) & & ! ( i_compatflags & COMPATF_NO_PASSMOBJ ) )
2006-02-24 04:48:15 +00:00
{ // check if a mobj passed over/under another object
2013-02-14 04:40:29 +00:00
if ( ! ( tm . thing - > flags & MF_MISSILE ) | |
! ( tm . thing - > flags2 & MF2_RIP ) | |
( thing - > flags5 & MF5_DONTRIP ) | |
2013-02-22 18:16:23 +00:00
( ( tm . thing - > flags6 & MF6_NOBOSSRIP ) & & ( thing - > flags2 & MF2_BOSS ) ) )
2013-02-14 04:40:29 +00:00
{
if ( tm . thing - > flags3 & thing - > flags3 & MF3_DONTOVERLAP )
{ // Some things prefer not to overlap each other, if possible
return unblocking ;
}
2016-01-20 00:48:57 +00:00
if ( ( tm . thing - > Z ( ) > = topz ) | | ( tm . thing - > Top ( ) < = thing - > Z ( ) ) )
2013-02-14 04:40:29 +00:00
return true ;
2006-02-24 04:48:15 +00:00
}
}
2009-09-14 20:47:53 +00:00
2014-03-14 15:15:30 +00:00
if ( tm . thing - > player = = NULL | | ! ( tm . thing - > player - > cheats & CF_PREDICTING ) )
2009-09-14 20:47:53 +00:00
{
2014-03-14 15:15:30 +00:00
// touchy object is alive, toucher is solid
if ( thing - > flags6 & MF6_TOUCHY & & tm . thing - > flags & MF_SOLID & & thing - > health > 0 & &
// Thing is an armed mine or a sentient thing
( thing - > flags6 & MF6_ARMED | | thing - > IsSentient ( ) ) & &
// either different classes or players
( thing - > player | | thing - > GetClass ( ) ! = tm . thing - > GetClass ( ) ) & &
// or different species if DONTHARMSPECIES
( ! ( thing - > flags6 & MF6_DONTHARMSPECIES ) | | thing - > GetSpecies ( ) ! = tm . thing - > GetSpecies ( ) ) & &
// touches vertically
2016-01-20 00:48:57 +00:00
topz > = tm . thing - > Z ( ) & & tm . thing - > Z ( ) + tm . thing - > height > = thingpos . z & &
2014-03-14 15:15:30 +00:00
// prevents lost souls from exploding when fired by pain elementals
( thing - > master ! = tm . thing & & tm . thing - > master ! = thing ) )
// Difference with MBF: MBF hardcodes the LS/PE check and lets actors of the same species
// but different classes trigger the touchiness, but that seems less straightforwards.
{
thing - > flags6 & = ~ MF6_ARMED ; // Disarm
P_DamageMobj ( thing , NULL , NULL , thing - > health , NAME_None , DMG_FORCED ) ; // kill object
return true ;
}
2009-09-15 14:16:55 +00:00
2014-03-14 15:15:30 +00:00
// Check for MF6_BUMPSPECIAL
// By default, only players can activate things by bumping into them
if ( ( thing - > flags6 & MF6_BUMPSPECIAL ) & & ( ( tm . thing - > player ! = NULL )
| | ( ( thing - > activationtype & THINGSPEC_MonsterTrigger ) & & ( tm . thing - > flags3 & MF3_ISMONSTER ) )
| | ( ( thing - > activationtype & THINGSPEC_MissileTrigger ) & & ( tm . thing - > flags & MF_MISSILE ) )
) & & ( level . maptime > thing - > lastbump ) ) // Leave the bumper enough time to go away
2012-08-28 02:52:53 +00:00
{
if ( P_ActivateThingSpecial ( thing , tm . thing ) )
thing - > lastbump = level . maptime + TICRATE ;
}
2009-09-15 14:16:55 +00:00
}
2006-02-24 04:48:15 +00:00
// Check for skulls slamming into things
2008-04-08 20:52:49 +00:00
if ( tm . thing - > flags & MF_SKULLFLY )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
bool res = tm . thing - > Slam ( tm . thing - > BlockingMobj ) ;
2008-04-08 20:52:49 +00:00
tm . thing - > BlockingMobj = NULL ;
2006-02-24 04:48:15 +00:00
return res ;
}
2014-03-14 15:15:30 +00:00
// [ED850] Player Prediction ends here. There is nothing else they could/should do.
if ( tm . thing - > player ! = NULL & & ( tm . thing - > player - > cheats & CF_PREDICTING ) )
{
solid = ( thing - > flags & MF_SOLID ) & &
! ( thing - > flags & MF_NOCLIP ) & &
( ( tm . thing - > flags & MF_SOLID ) | | ( tm . thing - > flags6 & MF6_BLOCKEDBYSOLIDACTORS ) ) ;
return ! solid | | unblocking ;
}
2006-02-24 04:48:15 +00:00
// Check for blasted thing running into another
2008-04-08 20:52:49 +00:00
if ( ( tm . thing - > flags2 & MF2_BLASTED ) & & ( thing - > flags & MF_SHOOTABLE ) )
2006-02-24 04:48:15 +00:00
{
2009-10-15 20:09:22 +00:00
if ( ! ( thing - > flags2 & MF2_BOSS ) & & ( thing - > flags3 & MF3_ISMONSTER ) & & ! ( thing - > flags3 & MF3_DONTBLAST ) )
2006-02-24 04:48:15 +00:00
{
2009-10-15 20:09:22 +00:00
// ideally this should take the mass factor into account
2009-06-30 20:57:51 +00:00
thing - > velx + = tm . thing - > velx ;
thing - > vely + = tm . thing - > vely ;
2014-08-07 16:05:39 +00:00
if ( ( thing - > velx + thing - > vely ) > 3 * FRACUNIT )
2006-02-24 04:48:15 +00:00
{
2013-01-02 04:39:59 +00:00
int newdam ;
2008-04-08 20:52:49 +00:00
damage = ( tm . thing - > Mass / 100 ) + 1 ;
2014-08-07 16:05:39 +00:00
newdam = P_DamageMobj ( thing , tm . thing , tm . thing , damage , tm . thing - > DamageType ) ;
P_TraceBleed ( newdam > 0 ? newdam : damage , thing , tm . thing ) ;
2006-02-24 04:48:15 +00:00
damage = ( thing - > Mass / 100 ) + 1 ;
2014-08-07 16:05:39 +00:00
newdam = P_DamageMobj ( tm . thing , thing , thing , damage > > 2 , tm . thing - > DamageType ) ;
P_TraceBleed ( newdam > 0 ? newdam : damage , tm . thing , thing ) ;
2006-02-24 04:48:15 +00:00
}
return false ;
}
}
2009-09-14 20:47:53 +00:00
// Check for missile or non-solid MBF bouncer
if ( tm . thing - > flags & MF_MISSILE | | ( ( tm . thing - > BounceFlags & BOUNCE_MBF ) & & ! ( tm . thing - > flags & MF_SOLID ) ) )
2006-02-24 04:48:15 +00:00
{
// Check for a non-shootable mobj
if ( thing - > flags2 & MF2_NONSHOOTABLE )
{
return true ;
}
// Check for passing through a ghost
2008-04-08 20:52:49 +00:00
if ( ( thing - > flags3 & MF3_GHOST ) & & ( tm . thing - > flags2 & MF2_THRUGHOST ) )
2006-02-24 04:48:15 +00:00
{
return true ;
2009-04-04 09:28:10 +00:00
}
2009-05-31 20:14:16 +00:00
2014-08-07 16:05:39 +00:00
if ( ( tm . thing - > flags6 & MF6_MTHRUSPECIES )
2009-05-31 20:14:16 +00:00
& & tm . thing - > target // NULL pointer check
& & ( tm . thing - > target - > GetSpecies ( ) = = thing - > GetSpecies ( ) ) )
return true ;
2009-04-04 09:28:10 +00:00
// Check for rippers passing through corpses
if ( ( thing - > flags & MF_CORPSE ) & & ( tm . thing - > flags2 & MF2_RIP ) & & ! ( thing - > flags & MF_SHOOTABLE ) )
{
return true ;
2006-02-24 04:48:15 +00:00
}
2008-10-05 11:23:41 +00:00
int clipheight ;
2014-08-07 16:05:39 +00:00
if ( thing - > projectilepassheight > 0 )
2008-10-05 11:23:41 +00:00
{
clipheight = thing - > projectilepassheight ;
}
else if ( thing - > projectilepassheight < 0 & & ( i_compatflags & COMPATF_MISSILECLIP ) )
{
clipheight = - thing - > projectilepassheight ;
}
else
{
clipheight = thing - > height ;
}
2006-02-24 04:48:15 +00:00
// Check if it went over / under
2016-01-20 00:48:57 +00:00
if ( tm . thing - > Z ( ) > thingpos . z + clipheight )
2006-02-24 04:48:15 +00:00
{ // Over thing
return true ;
}
2016-01-20 00:48:57 +00:00
if ( tm . thing - > Top ( ) < thingpos . z )
2006-02-24 04:48:15 +00:00
{ // Under thing
return true ;
}
2009-09-06 01:49:17 +00:00
// [RH] What is the point of this check, again? In Hexen, it is unconditional,
// but here we only do it if the missile's damage is 0.
2009-09-14 20:47:53 +00:00
// MBF bouncer might have a non-0 damage value, but they must not deal damage on impact either.
if ( ( tm . thing - > BounceFlags & BOUNCE_Actors ) & & ( tm . thing - > Damage = = 0 | | ! ( tm . thing - > flags & MF_MISSILE ) ) )
2006-02-24 04:48:15 +00:00
{
2009-09-06 01:49:17 +00:00
return ( tm . thing - > target = = thing | | ! ( thing - > flags & MF_SOLID ) ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
switch ( tm . thing - > SpecialMissileHit ( thing ) )
2006-02-24 04:48:15 +00:00
{
case 0 : return false ;
case 1 : return true ;
default : break ;
}
// [RH] Extend DeHacked infighting to allow for monsters
// to never fight each other
2008-04-08 20:52:49 +00:00
if ( tm . thing - > target ! = NULL )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
if ( thing = = tm . thing - > target )
2006-02-24 04:48:15 +00:00
{ // Don't missile self
return true ;
}
2015-03-02 09:02:49 +00:00
if ( ! CanAttackHurt ( thing , tm . thing - > target ) )
2006-02-24 04:48:15 +00:00
{
2015-03-02 09:02:49 +00:00
return false ;
2006-02-24 04:48:15 +00:00
}
}
if ( ! ( thing - > flags & MF_SHOOTABLE ) )
{ // Didn't do any damage
return ! ( thing - > flags & MF_SOLID ) ;
}
2008-04-08 20:52:49 +00:00
if ( ( thing - > flags4 & MF4_SPECTRAL ) & & ! ( tm . thing - > flags4 & MF4_SPECTRAL ) )
2006-02-24 04:48:15 +00:00
{
return true ;
}
2014-12-31 09:13:15 +00:00
if ( ( tm . DoRipping & & ! ( thing - > flags5 & MF5_DONTRIP ) ) & & CheckRipLevel ( thing , tm . thing ) )
2006-02-24 04:48:15 +00:00
{
2009-05-30 08:56:40 +00:00
if ( ! ( tm . thing - > flags6 & MF6_NOBOSSRIP ) | | ! ( thing - > flags2 & MF2_BOSS ) )
2006-02-24 04:48:15 +00:00
{
2016-01-30 21:06:04 +00:00
bool * check = tm . LastRipped . CheckKey ( thing ) ;
if ( check = = NULL | | ! * check )
2006-02-24 04:48:15 +00:00
{
2016-01-30 21:06:04 +00:00
tm . LastRipped [ thing ] = true ;
2009-05-30 08:56:40 +00:00
if ( ! ( thing - > flags & MF_NOBLOOD ) & &
! ( thing - > flags2 & MF2_REFLECTIVE ) & &
! ( tm . thing - > flags3 & MF3_BLOODLESSIMPACT ) & &
2014-08-07 16:05:39 +00:00
! ( thing - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) )
2009-05-30 08:56:40 +00:00
{ // Ok to spawn blood
2014-08-07 16:05:39 +00:00
P_RipperBlood ( tm . thing , thing ) ;
2009-05-30 08:56:40 +00:00
}
2014-08-07 16:05:39 +00:00
S_Sound ( tm . thing , CHAN_BODY , " misc/ripslop " , 1 , ATTN_IDLE ) ;
2010-07-23 21:36:17 +00:00
// Do poisoning (if using new style poison)
2010-07-29 06:53:20 +00:00
if ( tm . thing - > PoisonDamage > 0 & & tm . thing - > PoisonDuration ! = INT_MIN )
2010-07-23 21:36:17 +00:00
{
2011-06-13 10:39:14 +00:00
P_PoisonMobj ( thing , tm . thing , tm . thing - > target , tm . thing - > PoisonDamage , tm . thing - > PoisonDuration , tm . thing - > PoisonPeriod , tm . thing - > PoisonDamageType ) ;
2010-07-23 21:36:17 +00:00
}
2014-08-07 16:05:39 +00:00
damage = tm . thing - > GetMissileDamage ( 3 , 2 ) ;
int newdam = P_DamageMobj ( thing , tm . thing , tm . thing - > target , damage , tm . thing - > DamageType ) ;
2009-05-30 08:56:40 +00:00
if ( ! ( tm . thing - > flags3 & MF3_BLOODLESSIMPACT ) )
{
2014-08-07 16:05:39 +00:00
P_TraceBleed ( newdam > 0 ? newdam : damage , thing , tm . thing ) ;
2009-05-30 08:56:40 +00:00
}
if ( thing - > flags2 & MF2_PUSHABLE
& & ! ( tm . thing - > flags2 & MF2_CANNOTPUSH ) )
{ // Push thing
2009-06-14 13:47:38 +00:00
if ( thing - > lastpush ! = tm . PushTime )
{
2009-06-30 20:57:51 +00:00
thing - > velx + = FixedMul ( tm . thing - > velx , thing - > pushfactor ) ;
thing - > vely + = FixedMul ( tm . thing - > vely , thing - > pushfactor ) ;
2009-06-14 13:47:38 +00:00
thing - > lastpush = tm . PushTime ;
}
2009-05-30 08:56:40 +00:00
}
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
spechit . Clear ( ) ;
2009-05-30 08:56:40 +00:00
return true ;
2006-02-24 04:48:15 +00:00
}
}
2010-07-23 21:36:17 +00:00
// Do poisoning (if using new style poison)
2010-07-29 06:53:20 +00:00
if ( tm . thing - > PoisonDamage > 0 & & tm . thing - > PoisonDuration ! = INT_MIN )
2010-07-23 21:36:17 +00:00
{
2011-06-13 10:39:14 +00:00
P_PoisonMobj ( thing , tm . thing , tm . thing - > target , tm . thing - > PoisonDamage , tm . thing - > PoisonDuration , tm . thing - > PoisonPeriod , tm . thing - > PoisonDamageType ) ;
2010-07-23 21:36:17 +00:00
}
2006-02-24 04:48:15 +00:00
// Do damage
2014-08-07 16:05:39 +00:00
damage = tm . thing - > GetMissileDamage ( ( tm . thing - > flags4 & MF4_STRIFEDAMAGE ) ? 3 : 7 , 1 ) ;
2015-03-25 19:19:50 +00:00
if ( ( damage > 0 ) | | ( tm . thing - > flags6 & MF6_FORCEPAIN ) | | ( tm . thing - > flags7 & MF7_CAUSEPAIN ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
int newdam = P_DamageMobj ( thing , tm . thing , tm . thing - > target , damage , tm . thing - > DamageType ) ;
2009-06-07 17:04:40 +00:00
if ( damage > 0 )
2006-02-24 04:48:15 +00:00
{
2009-06-07 17:04:40 +00:00
if ( ( tm . thing - > flags5 & MF5_BLOODSPLATTER ) & &
! ( thing - > flags & MF_NOBLOOD ) & &
! ( thing - > flags2 & MF2_REFLECTIVE ) & &
2014-08-07 16:05:39 +00:00
! ( thing - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) & &
2009-06-07 17:04:40 +00:00
! ( tm . thing - > flags3 & MF3_BLOODLESSIMPACT ) & &
( pr_checkthing ( ) < 192 ) )
{
2016-01-20 00:48:57 +00:00
P_BloodSplatter ( tm . thing - > X ( ) , tm . thing - > Y ( ) , tm . thing - > Z ( ) , thing ) ;
2009-06-07 17:04:40 +00:00
}
if ( ! ( tm . thing - > flags3 & MF3_BLOODLESSIMPACT ) )
{
2014-08-07 16:05:39 +00:00
P_TraceBleed ( newdam > 0 ? newdam : damage , thing , tm . thing ) ;
2009-06-07 17:04:40 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
2009-10-28 23:28:09 +00:00
else
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_GiveBody ( thing , - damage ) ;
2006-02-24 04:48:15 +00:00
}
2014-12-09 18:09:36 +00:00
if ( ( thing - > flags7 & MF7_THRUREFLECT ) & & ( thing - > flags2 & MF2_REFLECTIVE ) & & ( tm . thing - > flags & MF_MISSILE ) )
{
if ( tm . thing - > flags2 & MF2_SEEKERMISSILE )
{
tm . thing - > tracer = tm . thing - > target ;
}
tm . thing - > target = thing ;
return true ;
}
2006-02-24 04:48:15 +00:00
return false ; // don't traverse any more
}
2014-03-14 15:15:30 +00:00
if ( thing - > flags2 & MF2_PUSHABLE & & ! ( tm . thing - > flags2 & MF2_CANNOTPUSH ) )
2006-02-24 04:48:15 +00:00
{ // Push thing
2009-06-14 13:47:38 +00:00
if ( thing - > lastpush ! = tm . PushTime )
{
2009-06-30 20:57:51 +00:00
thing - > velx + = FixedMul ( tm . thing - > velx , thing - > pushfactor ) ;
thing - > vely + = FixedMul ( tm . thing - > vely , thing - > pushfactor ) ;
2009-06-14 13:47:38 +00:00
thing - > lastpush = tm . PushTime ;
}
2006-02-24 04:48:15 +00:00
}
solid = ( thing - > flags & MF_SOLID ) & &
2014-08-07 16:05:39 +00:00
! ( thing - > flags & MF_NOCLIP ) & &
( ( tm . thing - > flags & MF_SOLID ) | | ( tm . thing - > flags6 & MF6_BLOCKEDBYSOLIDACTORS ) ) ;
2010-05-30 20:12:32 +00:00
2006-02-24 04:48:15 +00:00
// Check for special pickup
2008-04-08 20:52:49 +00:00
if ( ( thing - > flags & MF_SPECIAL ) & & ( tm . thing - > flags & MF_PICKUP )
2006-02-24 04:48:15 +00:00
// [RH] The next condition is to compensate for the extra height
// that gets added by P_CheckPosition() so that you cannot pick
// up things that are above your true height.
2016-01-20 00:48:57 +00:00
& & thingpos . z < tm . thing - > Top ( ) - tm . thing - > MaxStepHeight )
2006-02-24 04:48:15 +00:00
{ // Can be picked up by tmthing
2014-08-07 16:05:39 +00:00
P_TouchSpecialThing ( thing , tm . thing ) ; // can remove thing
2006-02-24 04:48:15 +00:00
}
// killough 3/16/98: Allow non-solid moving objects to move through solid
// ones, by allowing the moving thing (tmthing) to move if it's non-solid,
// despite another solid thing being in the way.
// killough 4/11/98: Treat no-clipping things as not blocking
2009-12-06 22:10:25 +00:00
return ! solid | | unblocking ;
2006-02-24 04:48:15 +00:00
// return !(thing->flags & MF_SOLID); // old code -- killough
}
/*
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
2014-08-07 16:05:39 +00:00
MOVEMENT CLIPPING
2006-02-24 04:48:15 +00:00
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
*/
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// P_CheckPosition
// This is purely informative, nothing is modified
// (except things picked up and missile damage applied).
//
// in:
// a AActor (can be valid or invalid)
// a position to be checked
// (doesn't need to be related to the AActor->x,y)
//
// during:
// special things are touched if MF_PICKUP
// early out on solid lines?
//
// out:
// newsubsec
// floorz
// ceilingz
// tmdropoffz = the lowest point contacted (monsters won't move to a dropoff)
// speciallines[]
// numspeciallines
// AActor *BlockingMobj = pointer to thing that blocked position (NULL if not
// blocked, or blocked by a line).
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2014-08-07 16:05:39 +00:00
bool P_CheckPosition ( AActor * thing , fixed_t x , fixed_t y , FCheckPosition & tm , bool actorsonly )
2006-02-24 04:48:15 +00:00
{
2007-12-25 10:07:58 +00:00
sector_t * newsec ;
2006-02-24 04:48:15 +00:00
AActor * thingblocker ;
fixed_t realheight = thing - > height ;
2008-04-08 20:52:49 +00:00
tm . thing = thing ;
2006-02-24 04:48:15 +00:00
2008-04-08 20:52:49 +00:00
tm . x = x ;
tm . y = y ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
newsec = P_PointInSector ( x , y ) ;
2008-04-08 20:52:49 +00:00
tm . ceilingline = thing - > BlockingLine = NULL ;
2014-08-07 16:05:39 +00:00
// The base floor / ceiling is from the subsector that contains the point.
// Any contacted lines the step closer together will adjust them.
2016-02-17 10:39:14 +00:00
tm . floorz = tm . dropoffz = newsec - > LowestFloorAt ( x , y , & tm . floorsector ) ;
tm . ceilingz = newsec - > HighestCeilingAt ( x , y , & tm . ceilingsector ) ;
tm . floorpic = tm . floorsector - > GetTexture ( sector_t : : floor ) ;
tm . floorterrain = tm . floorsector - > GetTerrain ( sector_t : : floor ) ;
tm . ceilingpic = tm . ceilingsector - > GetTexture ( sector_t : : ceiling ) ;
2008-04-08 20:52:49 +00:00
tm . touchmidtex = false ;
2011-01-20 11:40:05 +00:00
tm . abovemidtex = false ;
2006-02-24 04:48:15 +00:00
//Added by MC: Fill the tmsector.
2008-04-08 20:52:49 +00:00
tm . sector = newsec ;
2014-08-07 16:05:39 +00:00
2009-01-04 15:00:29 +00:00
//Check 3D floors
2012-08-22 21:31:48 +00:00
if ( ! thing - > IsNoClip2 ( ) & & newsec - > e - > XFloor . ffloors . Size ( ) )
2009-01-04 15:00:29 +00:00
{
F3DFloor * rover ;
fixed_t delta1 ;
fixed_t delta2 ;
2016-01-20 00:48:57 +00:00
int thingtop = thing - > Z ( ) + ( thing - > height = = 0 ? 1 : thing - > height ) ;
2014-08-07 16:05:39 +00:00
for ( unsigned i = 0 ; i < newsec - > e - > XFloor . ffloors . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
rover = newsec - > e - > XFloor . ffloors [ i ] ;
2014-08-07 16:05:39 +00:00
if ( ! ( rover - > flags & FF_SOLID ) | | ! ( rover - > flags & FF_EXISTS ) ) continue ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
fixed_t ff_bottom = rover - > bottom . plane - > ZatPoint ( x , y ) ;
fixed_t ff_top = rover - > top . plane - > ZatPoint ( x , y ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
delta1 = thing - > Z ( ) - ( ff_bottom + ( ( ff_top - ff_bottom ) / 2 ) ) ;
2014-08-07 16:05:39 +00:00
delta2 = thingtop - ( ff_bottom + ( ( ff_top - ff_bottom ) / 2 ) ) ;
if ( ff_top > tm . floorz & & abs ( delta1 ) < abs ( delta2 ) )
2009-01-04 15:00:29 +00:00
{
tm . floorz = tm . dropoffz = ff_top ;
tm . floorpic = * rover - > top . texture ;
2016-01-09 11:10:36 +00:00
tm . floorterrain = rover - > model - > GetTerrain ( rover - > top . isceiling ) ;
2009-01-04 15:00:29 +00:00
}
2014-08-07 16:05:39 +00:00
if ( ff_bottom < tm . ceilingz & & abs ( delta1 ) > = abs ( delta2 ) )
2009-01-04 15:00:29 +00:00
{
tm . ceilingz = ff_bottom ;
tm . ceilingpic = * rover - > bottom . texture ;
}
}
}
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
validcount + + ;
2014-08-07 16:05:39 +00:00
spechit . Clear ( ) ;
2006-02-24 04:48:15 +00:00
2008-04-08 20:52:49 +00:00
if ( ( thing - > flags & MF_NOCLIP ) & & ! ( thing - > flags & MF_SKULLFLY ) )
2006-02-24 04:48:15 +00:00
return true ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
// Check things first, possibly picking things up.
2008-04-08 20:52:49 +00:00
thing - > BlockingMobj = NULL ;
2006-02-24 04:48:15 +00:00
thingblocker = NULL ;
if ( thing - > player )
{ // [RH] Fake taller height to catch stepping up into things.
thing - > height = realheight + thing - > MaxStepHeight ;
}
2006-04-11 08:36:23 +00:00
2008-04-10 14:38:43 +00:00
tm . stepthing = NULL ;
2008-04-19 00:55:55 +00:00
FBoundingBox box ( x , y , thing - > radius ) ;
2008-04-07 21:14:28 +00:00
{
2008-04-19 00:55:55 +00:00
FBlockThingsIterator it2 ( box ) ;
AActor * th ;
while ( ( th = it2 . Next ( ) ) )
{
if ( ! PIT_CheckThing ( th , tm ) )
{ // [RH] If a thing can be stepped up on, we need to continue checking
2014-08-07 16:05:39 +00:00
// other things in the blocks and see if we hit something that is
// definitely blocking. Otherwise, we need to check the lines, or we
// could end up stuck inside a wall.
2008-04-19 00:55:55 +00:00
AActor * BlockingMobj = thing - > BlockingMobj ;
if ( BlockingMobj = = NULL | | ( i_compatflags & COMPATF_NO_PASSMOBJ ) )
{ // Thing slammed into something; don't let it move now.
thing - > height = realheight ;
return false ;
}
2014-08-07 16:05:39 +00:00
else if ( ! BlockingMobj - > player & & ! ( thing - > flags & ( MF_FLOAT | MF_MISSILE | MF_SKULLFLY ) ) & &
2016-01-20 00:48:57 +00:00
BlockingMobj - > Top ( ) - thing - > Z ( ) < = thing - > MaxStepHeight )
2006-02-24 04:48:15 +00:00
{
2008-04-19 00:55:55 +00:00
if ( thingblocker = = NULL | |
2016-01-20 00:48:57 +00:00
BlockingMobj - > Z ( ) > thingblocker - > Z ( ) )
2008-04-19 00:55:55 +00:00
{
thingblocker = BlockingMobj ;
}
thing - > BlockingMobj = NULL ;
2006-02-24 04:48:15 +00:00
}
2008-04-19 00:55:55 +00:00
else if ( thing - > player & &
2016-01-20 00:48:57 +00:00
thing - > Top ( ) - BlockingMobj - > Z ( ) < = thing - > MaxStepHeight )
2008-04-19 00:55:55 +00:00
{
if ( thingblocker )
{ // There is something to step up on. Return this thing as
2014-08-07 16:05:39 +00:00
// the blocker so that we don't step up.
2008-04-19 00:55:55 +00:00
thing - > height = realheight ;
return false ;
}
// Nothing is blocking us, but this actor potentially could
// if there is something else to step on.
thing - > BlockingMobj = NULL ;
}
else
{ // Definitely blocking
2008-04-07 21:14:28 +00:00
thing - > height = realheight ;
return false ;
}
}
2006-02-24 04:48:15 +00:00
}
}
// check lines
// [RH] We need to increment validcount again, because a function above may
// have already set some lines to equal the current validcount.
//
// Specifically, when DehackedPickup spawns a new item in its TryPickup()
// function, that new actor will set the lines around it to match validcount
// when it links itself into the world. If we just leave validcount alone,
// that will give the player the freedom to walk through walls at will near
// a pickup they cannot get, because their validcount will prevent them from
// being considered for collision with the player.
validcount + + ;
2008-04-08 20:52:49 +00:00
thing - > BlockingMobj = NULL ;
2006-02-24 04:48:15 +00:00
thing - > height = realheight ;
2012-04-06 04:46:45 +00:00
if ( actorsonly | | ( thing - > flags & MF_NOCLIP ) )
2008-04-08 20:52:49 +00:00
return ( thing - > BlockingMobj = thingblocker ) = = NULL ;
2008-04-06 17:33:43 +00:00
FBlockLinesIterator it ( box ) ;
line_t * ld ;
2006-02-24 04:48:15 +00:00
2008-04-08 20:52:49 +00:00
fixed_t thingdropoffz = tm . floorz ;
2006-02-24 04:48:15 +00:00
//bool onthing = (thingdropoffz != tmdropoffz);
2008-04-08 20:52:49 +00:00
tm . floorz = tm . dropoffz ;
2006-02-24 04:48:15 +00:00
2011-05-08 16:58:53 +00:00
bool good = true ;
2008-04-06 17:33:43 +00:00
while ( ( ld = it . Next ( ) ) )
{
2011-05-08 16:58:53 +00:00
good & = PIT_CheckLine ( ld , box , tm ) ;
}
if ( ! good )
{
return false ;
2008-04-06 17:33:43 +00:00
}
2008-04-08 20:52:49 +00:00
if ( tm . ceilingz - tm . floorz < thing - > height )
2011-05-08 16:58:53 +00:00
{
2006-02-24 04:48:15 +00:00
return false ;
2011-05-08 16:58:53 +00:00
}
2008-05-17 17:57:50 +00:00
if ( tm . touchmidtex )
{
tm . dropoffz = tm . floorz ;
}
else if ( tm . stepthing ! = NULL )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
tm . dropoffz = thingdropoffz ;
2006-02-24 04:48:15 +00:00
}
2008-04-08 20:52:49 +00:00
return ( thing - > BlockingMobj = thingblocker ) = = NULL ;
}
2014-08-07 16:05:39 +00:00
bool P_CheckPosition ( AActor * thing , fixed_t x , fixed_t y , bool actorsonly )
2008-04-08 20:52:49 +00:00
{
FCheckPosition tm ;
2012-04-06 04:46:45 +00:00
return P_CheckPosition ( thing , x , y , tm , actorsonly ) ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//----------------------------------------------------------------------------
//
// FUNC P_TestMobjLocation
//
// Returns true if the mobj is not blocked by anything at its current
// location, otherwise returns false.
//
//----------------------------------------------------------------------------
2014-08-07 16:05:39 +00:00
bool P_TestMobjLocation ( AActor * mobj )
2009-10-11 17:44:50 +00:00
{
2015-04-04 16:40:43 +00:00
ActorFlags flags ;
2009-10-11 17:44:50 +00:00
flags = mobj - > flags ;
mobj - > flags & = ~ MF_PICKUP ;
2016-01-20 00:48:57 +00:00
if ( P_CheckPosition ( mobj , mobj - > X ( ) , mobj - > Y ( ) ) )
2009-10-11 17:44:50 +00:00
{ // XY is ok, now check Z
mobj - > flags = flags ;
2016-01-20 00:48:57 +00:00
if ( ( mobj - > Z ( ) < mobj - > floorz ) | | ( mobj - > Top ( ) > mobj - > ceilingz ) )
2009-10-11 17:44:50 +00:00
{ // Bad Z
return false ;
}
return true ;
}
mobj - > flags = flags ;
return false ;
}
2006-02-24 04:48:15 +00:00
//=============================================================================
//
// P_CheckOnmobj(AActor *thing)
//
// Checks if the new Z position is legal
//=============================================================================
2014-08-07 16:05:39 +00:00
AActor * P_CheckOnmobj ( AActor * thing )
2006-02-24 04:48:15 +00:00
{
fixed_t oldz ;
bool good ;
2008-04-10 14:38:43 +00:00
AActor * onmobj ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
oldz = thing - > Z ( ) ;
2014-08-07 16:05:39 +00:00
P_FakeZMovement ( thing ) ;
good = P_TestMobjZ ( thing , false , & onmobj ) ;
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2006-02-24 04:48:15 +00:00
return good ? NULL : onmobj ;
}
//=============================================================================
//
// P_TestMobjZ
//
//=============================================================================
2014-08-07 16:05:39 +00:00
bool P_TestMobjZ ( AActor * actor , bool quick , AActor * * pOnmobj )
2006-02-24 04:48:15 +00:00
{
2008-04-10 14:38:43 +00:00
AActor * onmobj = NULL ;
2006-02-24 04:48:15 +00:00
if ( actor - > flags & MF_NOCLIP )
2008-04-10 14:38:43 +00:00
{
if ( pOnmobj ) * pOnmobj = NULL ;
2006-02-24 04:48:15 +00:00
return true ;
2008-04-10 14:38:43 +00:00
}
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
FBlockThingsIterator it ( FBoundingBox ( actor - > X ( ) , actor - > Y ( ) , actor - > radius ) ) ;
2008-04-07 21:14:28 +00:00
AActor * thing ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
while ( ( thing = it . Next ( ) ) )
{
2008-04-15 10:04:41 +00:00
if ( ! thing - > intersects ( actor ) )
{
continue ;
}
2009-08-11 01:03:58 +00:00
if ( ( actor - > flags2 | thing - > flags2 ) & MF2_THRUACTORS )
{
continue ;
}
if ( ( actor - > flags6 & MF6_THRUSPECIES ) & & ( thing - > GetSpecies ( ) = = actor - > GetSpecies ( ) ) )
{
continue ;
}
2008-04-07 21:14:28 +00:00
if ( ! ( thing - > flags & MF_SOLID ) )
{ // Can't hit thing
continue ;
}
2014-08-07 16:05:39 +00:00
if ( thing - > flags & ( MF_SPECIAL | MF_NOCLIP ) )
2009-10-09 20:18:31 +00:00
{ // [RH] Specials and noclippers don't block moves
2008-04-07 21:14:28 +00:00
continue ;
}
2009-10-09 20:18:31 +00:00
if ( thing - > flags & ( MF_CORPSE ) )
{ // Corpses need a few more checks
if ( ! ( actor - > flags & MF_ICECORPSE ) )
continue ;
}
2008-04-07 21:14:28 +00:00
if ( ! ( thing - > flags4 & MF4_ACTLIKEBRIDGE ) & & ( actor - > flags & MF_SPECIAL ) )
{ // [RH] Only bridges block pickup items
continue ;
}
if ( thing = = actor )
{ // Don't clip against self
continue ;
}
2014-12-09 18:09:36 +00:00
if ( ( actor - > flags & MF_MISSILE ) & & ( thing = = actor - > target ) )
2012-04-19 02:50:43 +00:00
{ // Don't clip against whoever shot the missile.
continue ;
}
2016-01-20 00:48:57 +00:00
if ( actor - > Z ( ) > thing - > Top ( ) )
2008-04-07 21:14:28 +00:00
{ // over thing
continue ;
}
2016-01-20 00:48:57 +00:00
else if ( actor - > Top ( ) < = thing - > Z ( ) )
2008-04-07 21:14:28 +00:00
{ // under thing
continue ;
}
2016-01-20 00:48:57 +00:00
else if ( ! quick & & onmobj ! = NULL & & thing - > Top ( ) < onmobj - > Top ( ) )
2008-04-07 21:14:28 +00:00
{ // something higher is in the way
continue ;
}
onmobj = thing ;
if ( quick ) break ;
}
2006-02-24 04:48:15 +00:00
2008-04-10 14:38:43 +00:00
if ( pOnmobj ) * pOnmobj = onmobj ;
2006-08-12 21:02:46 +00:00
return onmobj = = NULL ;
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// P_FakeZMovement
//
// Fake the zmovement so that we can check if a move is legal
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_FakeZMovement ( AActor * mo )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
//
// adjust height
//
2016-01-20 00:48:57 +00:00
mo - > AddZ ( mo - > velz ) ;
2006-02-24 04:48:15 +00:00
if ( ( mo - > flags & MF_FLOAT ) & & mo - > target )
{ // float down towards target if too close
if ( ! ( mo - > flags & MF_SKULLFLY ) & & ! ( mo - > flags & MF_INFLOAT ) )
{
2016-01-10 16:52:41 +00:00
fixed_t dist = mo - > AproxDistance ( mo - > target ) ;
2016-01-20 00:48:57 +00:00
fixed_t delta = ( mo - > target - > Z ( ) + ( mo - > height > > 1 ) ) - mo - > Z ( ) ;
2014-08-07 16:05:39 +00:00
if ( delta < 0 & & dist < - ( delta * 3 ) )
2016-01-20 00:48:57 +00:00
mo - > AddZ ( - mo - > FloatSpeed ) ;
2014-08-07 16:05:39 +00:00
else if ( delta > 0 & & dist < ( delta * 3 ) )
2016-01-20 00:48:57 +00:00
mo - > AddZ ( mo - > FloatSpeed ) ;
2006-02-24 04:48:15 +00:00
}
}
2016-01-20 00:48:57 +00:00
if ( mo - > player & & mo - > flags & MF_NOGRAVITY & & ( mo - > Z ( ) > mo - > floorz ) & & ! mo - > IsNoClip2 ( ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
mo - > AddZ ( finesine [ ( FINEANGLES / 80 * level . maptime ) & FINEMASK ] / 8 ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
//
// clip movement
//
2016-01-20 00:48:57 +00:00
if ( mo - > Z ( ) < = mo - > floorz )
2006-02-24 04:48:15 +00:00
{ // hit the floor
2016-01-20 00:48:57 +00:00
mo - > SetZ ( mo - > floorz ) ;
2006-02-24 04:48:15 +00:00
}
2016-01-20 00:48:57 +00:00
if ( mo - > Top ( ) > mo - > ceilingz )
2006-02-24 04:48:15 +00:00
{ // hit the ceiling
2016-01-20 00:48:57 +00:00
mo - > SetZ ( mo - > ceilingz - mo - > height ) ;
2006-02-24 04:48:15 +00:00
}
}
//===========================================================================
//
// CheckForPushSpecial
//
//===========================================================================
2014-08-07 16:05:39 +00:00
static void CheckForPushSpecial ( line_t * line , int side , AActor * mobj , bool windowcheck )
2006-02-24 04:48:15 +00:00
{
2009-10-10 12:42:57 +00:00
if ( line - > special & & ! ( mobj - > flags6 & MF6_NOTRIGGER ) )
2006-02-24 04:48:15 +00:00
{
2014-02-24 23:43:28 +00:00
if ( windowcheck & & ! ( ib_compatflags & BCOMPATF_NOWINDOWCHECK ) & & line - > backsector ! = NULL )
2012-03-09 01:49:26 +00:00
{ // Make sure this line actually blocks us and is not a window
2014-08-07 16:05:39 +00:00
// or similar construct we are standing inside of.
2016-02-17 10:39:14 +00:00
fixedvec3 pos = mobj - > PosRelative ( line ) ;
fixed_t fzt = line - > frontsector - > ceilingplane . ZatPoint ( pos ) ;
fixed_t fzb = line - > frontsector - > floorplane . ZatPoint ( pos ) ;
fixed_t bzt = line - > backsector - > ceilingplane . ZatPoint ( pos ) ;
fixed_t bzb = line - > backsector - > floorplane . ZatPoint ( pos ) ;
2016-01-20 00:48:57 +00:00
if ( fzt > = mobj - > Top ( ) & & bzt > = mobj - > Top ( ) & &
fzb < = mobj - > Z ( ) & & bzb < = mobj - > Z ( ) )
2012-03-09 01:49:26 +00:00
{
2012-05-31 10:07:30 +00:00
// we must also check if some 3D floor in the backsector may be blocking
2014-08-07 16:05:39 +00:00
for ( unsigned int i = 0 ; i < line - > backsector - > e - > XFloor . ffloors . Size ( ) ; i + + )
{
F3DFloor * rover = line - > backsector - > e - > XFloor . ffloors [ i ] ;
2012-05-31 10:07:30 +00:00
2014-08-07 16:05:39 +00:00
if ( ! ( rover - > flags & FF_SOLID ) | | ! ( rover - > flags & FF_EXISTS ) ) continue ;
2012-05-31 10:07:30 +00:00
2016-02-17 10:39:14 +00:00
fixed_t ff_bottom = rover - > bottom . plane - > ZatPoint ( pos ) ;
fixed_t ff_top = rover - > top . plane - > ZatPoint ( pos ) ;
2012-05-31 10:07:30 +00:00
2016-01-20 00:48:57 +00:00
if ( ff_bottom < mobj - > Top ( ) & & ff_top > mobj - > Z ( ) )
2014-08-07 16:05:39 +00:00
{
goto isblocking ;
2012-05-31 10:07:30 +00:00
}
2014-08-07 16:05:39 +00:00
}
2012-03-09 01:49:26 +00:00
return ;
}
}
2014-08-07 16:05:39 +00:00
isblocking :
2006-02-24 04:48:15 +00:00
if ( mobj - > flags2 & MF2_PUSHWALL )
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( line , mobj , side , SPAC_Push ) ;
2006-02-24 04:48:15 +00:00
}
else if ( mobj - > flags2 & MF2_IMPACT )
{
2009-02-03 19:11:43 +00:00
if ( ( level . flags2 & LEVEL2_MISSILESACTIVATEIMPACT ) | |
2007-02-04 02:12:54 +00:00
! ( mobj - > flags & MF_MISSILE ) | |
( mobj - > target = = NULL ) )
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( line , mobj , side , SPAC_Impact ) ;
2007-02-04 02:12:54 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( line , mobj - > target , side , SPAC_Impact ) ;
2007-02-04 02:12:54 +00:00
}
2014-08-07 16:05:39 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// P_TryMove
// Attempt to move to a new position,
// crossing special lines unless MF_TELEPORT is set.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
bool P_TryMove ( AActor * thing , fixed_t x , fixed_t y ,
int dropoff , // killough 3/15/98: allow dropoff as option
const secplane_t * onfloor , // [RH] Let P_TryMove keep the thing on the floor
FCheckPosition & tm ,
bool missileCheck ) // [GZ] Fired missiles ignore the drop-off test
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
fixedvec3 oldpos ;
sector_t * oldsector ;
2006-02-24 04:48:15 +00:00
fixed_t oldz ;
int side ;
int oldside ;
line_t * ld ;
sector_t * oldsec = thing - > Sector ; // [RH] for sector actions
sector_t * newsec ;
2008-04-08 20:52:49 +00:00
tm . floatok = false ;
2016-01-20 00:48:57 +00:00
oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
if ( onfloor )
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( onfloor - > ZatPoint ( x , y ) ) ;
2006-02-24 04:48:15 +00:00
}
2012-03-18 03:52:18 +00:00
thing - > flags6 | = MF6_INTRYMOVE ;
2014-08-07 16:05:39 +00:00
if ( ! P_CheckPosition ( thing , x , y , tm ) )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
AActor * BlockingMobj = thing - > BlockingMobj ;
2006-02-24 04:48:15 +00:00
// Solid wall or thing
if ( ! BlockingMobj | | BlockingMobj - > player | | ! thing - > player )
{
goto pushline ;
}
else
{
if ( BlockingMobj - > player | | ! thing - > player )
{
goto pushline ;
}
2016-01-20 00:48:57 +00:00
else if ( BlockingMobj - > Top ( ) - thing - > Z ( ) > thing - > MaxStepHeight
| | ( BlockingMobj - > Sector - > ceilingplane . ZatPoint ( x , y ) - ( BlockingMobj - > Top ( ) ) < thing - > height )
| | ( tm . ceilingz - ( BlockingMobj - > Top ( ) ) < thing - > height ) )
2006-02-24 04:48:15 +00:00
{
goto pushline ;
}
}
2008-04-08 20:52:49 +00:00
if ( ! ( tm . thing - > flags2 & MF2_PASSMOBJ ) | | ( i_compatflags & COMPATF_NO_PASSMOBJ ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2012-03-18 03:52:18 +00:00
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-02-24 04:48:15 +00:00
return false ;
}
}
2007-11-08 09:22:06 +00:00
if ( thing - > flags3 & MF3_FLOORHUGGER )
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( tm . floorz ) ;
2007-11-08 09:22:06 +00:00
}
else if ( thing - > flags3 & MF3_CEILINGHUGGER )
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( tm . ceilingz - thing - > height ) ;
2007-11-08 09:22:06 +00:00
}
2008-04-08 20:52:49 +00:00
if ( onfloor & & tm . floorsector = = thing - > floorsector )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( tm . floorz ) ;
2006-02-24 04:48:15 +00:00
}
if ( ! ( thing - > flags & MF_NOCLIP ) )
{
2008-04-08 20:52:49 +00:00
if ( tm . ceilingz - tm . floorz < thing - > height )
2006-02-24 04:48:15 +00:00
{
goto pushline ; // doesn't fit
}
2008-04-08 20:52:49 +00:00
tm . floatok = true ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
if ( ! ( thing - > flags & MF_TELEPORT )
2016-01-20 00:48:57 +00:00
& & tm . ceilingz - thing - > Z ( ) < thing - > height
2006-02-24 04:48:15 +00:00
& & ! ( thing - > flags3 & MF3_CEILINGHUGGER )
& & ( ! ( thing - > flags2 & MF2_FLY ) | | ! ( thing - > flags & MF_NOGRAVITY ) ) )
{
goto pushline ; // mobj must lower itself to fit
}
if ( thing - > flags2 & MF2_FLY & & thing - > flags & MF_NOGRAVITY )
{
# if 1
2016-01-20 00:48:57 +00:00
if ( thing - > Top ( ) > tm . ceilingz )
2006-02-24 04:48:15 +00:00
goto pushline ;
# else
// When flying, slide up or down blocking lines until the actor
// is not blocked.
2016-01-20 00:48:57 +00:00
if ( thing - > Top ( ) > tm . ceilingz )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
thing - > velz = - 8 * FRACUNIT ;
2006-02-24 04:48:15 +00:00
goto pushline ;
}
2016-01-20 00:48:57 +00:00
else if ( thing - > Z ( ) < tm . floorz & & tm . floorz - tm . dropoffz > thing - > MaxDropOffHeight )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
thing - > velz = 8 * FRACUNIT ;
2006-02-24 04:48:15 +00:00
goto pushline ;
}
# endif
}
if ( ! ( thing - > flags & MF_TELEPORT ) & & ! ( thing - > flags3 & MF3_FLOORHUGGER ) )
{
2016-01-20 00:48:57 +00:00
if ( ( thing - > flags & MF_MISSILE ) & & ! ( thing - > flags6 & MF6_STEPMISSILE ) & & tm . floorz > thing - > Z ( ) )
2012-04-22 02:30:26 +00:00
{ // [RH] Don't let normal missiles climb steps
2006-02-24 04:48:15 +00:00
goto pushline ;
}
2016-01-20 00:48:57 +00:00
if ( tm . floorz - thing - > Z ( ) > thing - > MaxStepHeight )
2012-04-22 02:30:26 +00:00
{ // too big a step up
2006-02-24 04:48:15 +00:00
goto pushline ;
}
2016-01-20 00:48:57 +00:00
else if ( thing - > Z ( ) < tm . floorz )
2006-02-24 04:48:15 +00:00
{ // [RH] Check to make sure there's nothing in the way for the step up
2016-01-20 00:48:57 +00:00
fixed_t savedz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
bool good ;
2016-01-20 00:48:57 +00:00
thing - > SetZ ( tm . floorz ) ;
2014-08-07 16:05:39 +00:00
good = P_TestMobjZ ( thing ) ;
2016-01-20 00:48:57 +00:00
thing - > SetZ ( savedz ) ;
2006-02-24 04:48:15 +00:00
if ( ! good )
{
goto pushline ;
}
2012-06-10 10:17:49 +00:00
if ( thing - > flags6 & MF6_STEPMISSILE )
2012-03-29 05:23:04 +00:00
{
2016-01-20 00:48:57 +00:00
thing - > SetZ ( tm . floorz ) ;
2012-03-29 05:23:04 +00:00
// If moving down, cancel vertical component of the velocity
if ( thing - > velz < 0 )
{
2013-05-12 19:02:15 +00:00
// If it's a bouncer, let it bounce off its new floor, too.
if ( thing - > BounceFlags & BOUNCE_Floors )
{
2014-08-07 16:05:39 +00:00
thing - > FloorBounceMissile ( tm . floorsector - > floorplane ) ;
2013-05-12 19:02:15 +00:00
}
2013-05-12 19:05:00 +00:00
else
{
thing - > velz = 0 ;
}
2012-03-29 05:23:04 +00:00
}
}
2006-02-24 04:48:15 +00:00
}
}
2009-01-25 21:59:38 +00:00
// compatibility check: Doom originally did not allow monsters to cross dropoffs at all.
2009-06-30 20:57:51 +00:00
// If the compatibility flag is on, only allow this when the velocity comes from a scroller
2009-01-25 21:59:38 +00:00
if ( ( i_compatflags & COMPATF_CROSSDROPOFF ) & & ! ( thing - > flags4 & MF4_SCROLLMOVE ) )
{
dropoff = false ;
}
2014-08-07 16:05:39 +00:00
if ( dropoff = = 2 & & // large jump down (e.g. dogs)
2016-01-20 00:48:57 +00:00
( tm . floorz - tm . dropoffz > 128 * FRACUNIT | | thing - > target = = NULL | | thing - > target - > Z ( ) > tm . dropoffz ) )
2009-09-15 14:16:55 +00:00
{
2011-12-06 08:36:28 +00:00
dropoff = false ;
2009-09-15 14:16:55 +00:00
}
2006-02-24 04:48:15 +00:00
// killough 3/15/98: Allow certain objects to drop off
2014-08-07 16:05:39 +00:00
if ( ( ! dropoff & & ! ( thing - > flags & ( MF_DROPOFF | MF_FLOAT | MF_MISSILE ) ) ) | | ( thing - > flags5 & MF5_NODROPOFF ) )
2006-02-24 04:48:15 +00:00
{
2006-04-24 14:26:06 +00:00
if ( ! ( thing - > flags5 & MF5_AVOIDINGDROPOFF ) )
{
2008-04-08 20:52:49 +00:00
fixed_t floorz = tm . floorz ;
2006-12-23 12:12:06 +00:00
// [RH] If the thing is standing on something, use its current z as the floorz.
// This is so that it does not walk off of things onto a drop off.
if ( thing - > flags2 & MF2_ONMOBJ )
{
2016-01-20 00:48:57 +00:00
floorz = MAX ( thing - > Z ( ) , tm . floorz ) ;
2006-12-23 12:12:06 +00:00
}
2014-08-07 16:05:39 +00:00
2008-04-08 20:52:49 +00:00
if ( floorz - tm . dropoffz > thing - > MaxDropOffHeight & &
2014-08-07 16:05:39 +00:00
! ( thing - > flags2 & MF2_BLASTED ) & & ! missileCheck )
2006-04-24 14:26:06 +00:00
{ // Can't move over a dropoff unless it's been blasted
2014-08-07 16:05:39 +00:00
// [GZ] Or missile-spawned
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2012-03-18 03:52:18 +00:00
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-04-24 14:26:06 +00:00
return false ;
}
}
else
{
// special logic to move a monster off a dropoff
2006-12-23 12:12:06 +00:00
// this intentionally does not check for standing on things.
2008-04-08 20:52:49 +00:00
if ( thing - > floorz - tm . floorz > thing - > MaxDropOffHeight | |
2012-03-18 03:52:18 +00:00
thing - > dropoffz - tm . dropoffz > thing - > MaxDropOffHeight )
{
thing - > flags6 & = ~ MF6_INTRYMOVE ;
return false ;
}
2006-02-24 04:48:15 +00:00
}
}
if ( thing - > flags2 & MF2_CANTLEAVEFLOORPIC
2008-04-08 20:52:49 +00:00
& & ( tm . floorpic ! = thing - > floorpic
2016-01-20 00:48:57 +00:00
| | tm . floorz - thing - > Z ( ) ! = 0 ) )
2006-02-24 04:48:15 +00:00
{ // must stay within a sector of a certain floor type
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2012-03-18 03:52:18 +00:00
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-02-24 04:48:15 +00:00
return false ;
}
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
//Added by MC: To prevent bot from getting into dangerous sectors.
2014-10-14 18:57:11 +00:00
if ( thing - > player & & thing - > player - > Bot ! = NULL & & thing - > flags & MF_SHOOTABLE )
2006-02-24 04:48:15 +00:00
{
2008-04-08 20:52:49 +00:00
if ( tm . sector ! = thing - > Sector
2014-08-07 16:05:39 +00:00
& & bglobal . IsDangerous ( tm . sector ) )
2006-02-24 04:48:15 +00:00
{
2014-10-14 18:57:11 +00:00
thing - > player - > Bot - > prev = thing - > player - > Bot - > dest ;
thing - > player - > Bot - > dest = NULL ;
2009-06-30 20:57:51 +00:00
thing - > velx = 0 ;
thing - > vely = 0 ;
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2012-03-18 03:52:18 +00:00
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-02-24 04:48:15 +00:00
return false ;
}
2014-08-07 16:05:39 +00:00
}
2006-02-24 04:48:15 +00:00
}
// [RH] Check status of eyes against fake floor/ceiling in case
// it slopes or the player's eyes are bobbing in and out.
bool oldAboveFakeFloor , oldAboveFakeCeiling ;
fixed_t viewheight ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
viewheight = thing - > player ? thing - > player - > viewheight : thing - > height / 2 ;
oldAboveFakeFloor = oldAboveFakeCeiling = false ; // pacify GCC
if ( oldsec - > heightsec )
{
fixed_t eyez = oldz + viewheight ;
2016-01-17 13:08:20 +00:00
oldAboveFakeFloor = eyez > oldsec - > heightsec - > floorplane . ZatPoint ( thing ) ;
oldAboveFakeCeiling = eyez > oldsec - > heightsec - > ceilingplane . ZatPoint ( thing ) ;
2006-02-24 04:48:15 +00:00
}
2009-09-14 20:47:53 +00:00
// Borrowed from MBF:
if ( thing - > BounceFlags & BOUNCE_MBF & & // killough 8/13/98
2014-08-07 16:05:39 +00:00
! ( thing - > flags & ( MF_MISSILE | MF_NOGRAVITY ) ) & &
2016-01-20 00:48:57 +00:00
! thing - > IsSentient ( ) & & tm . floorz - thing - > Z ( ) > 16 * FRACUNIT )
2012-03-18 03:52:18 +00:00
{ // too big a step up for MBF bouncers under gravity
thing - > flags6 & = ~ MF6_INTRYMOVE ;
return false ;
}
2009-09-14 20:47:53 +00:00
2006-02-24 04:48:15 +00:00
// the move is ok, so link the thing into its new position
2014-08-07 16:05:39 +00:00
thing - > UnlinkFromWorld ( ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
oldpos = thing - > Pos ( ) ;
oldsector = thing - > Sector ;
2008-04-08 20:52:49 +00:00
thing - > floorz = tm . floorz ;
thing - > ceilingz = tm . ceilingz ;
thing - > dropoffz = tm . dropoffz ; // killough 11/98: keep track of dropoffs
thing - > floorpic = tm . floorpic ;
2016-01-09 11:10:36 +00:00
thing - > floorterrain = tm . floorterrain ;
2008-04-08 20:52:49 +00:00
thing - > floorsector = tm . floorsector ;
thing - > ceilingpic = tm . ceilingpic ;
thing - > ceilingsector = tm . ceilingsector ;
2016-01-20 00:48:57 +00:00
thing - > SetXY ( x , y ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
thing - > LinkToWorld ( ) ;
2006-02-24 04:48:15 +00:00
if ( thing - > flags2 & MF2_FLOORCLIP )
{
2014-08-07 16:05:39 +00:00
thing - > AdjustFloorClip ( ) ;
2006-02-24 04:48:15 +00:00
}
// if any special lines were hit, do the effect
2014-08-07 16:05:39 +00:00
if ( ! ( thing - > flags & ( MF_TELEPORT | MF_NOCLIP ) ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
while ( spechit . Pop ( ld ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
fixedvec3 thingpos = thing - > PosRelative ( ld ) ;
fixedvec3 oldrelpos = PosRelative ( oldpos , ld , oldsector ) ;
2006-02-24 04:48:15 +00:00
// see if the line was crossed
2016-01-20 00:48:57 +00:00
side = P_PointOnLineSide ( thingpos . x , thingpos . y , ld ) ;
oldside = P_PointOnLineSide ( oldrelpos . x , oldrelpos . y , ld ) ;
2009-10-10 12:42:57 +00:00
if ( side ! = oldside & & ld - > special & & ! ( thing - > flags6 & MF6_NOTRIGGER ) )
2006-02-24 04:48:15 +00:00
{
2014-08-23 03:17:11 +00:00
if ( thing - > player & & ( thing - > player - > cheats & CF_PREDICTING ) )
{
P_PredictLine ( ld , thing , oldside , SPAC_Cross ) ;
}
else if ( thing - > player )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( ld , thing , oldside , SPAC_Cross ) ;
2006-02-24 04:48:15 +00:00
}
else if ( thing - > flags2 & MF2_MCROSS )
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( ld , thing , oldside , SPAC_MCross ) ;
2006-02-24 04:48:15 +00:00
}
else if ( thing - > flags2 & MF2_PCROSS )
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( ld , thing , oldside , SPAC_PCross ) ;
2006-02-24 04:48:15 +00:00
}
else if ( ( ld - > special = = Teleport | |
2014-08-07 16:05:39 +00:00
ld - > special = = Teleport_NoFog | |
ld - > special = = Teleport_Line ) )
2006-02-24 04:48:15 +00:00
{ // [RH] Just a little hack for BOOM compatibility
2014-08-07 16:05:39 +00:00
P_ActivateLine ( ld , thing , oldside , SPAC_MCross ) ;
2006-02-24 04:48:15 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( ld , thing , oldside , SPAC_AnyCross ) ;
2006-02-24 04:48:15 +00:00
}
}
}
}
2014-08-23 03:17:11 +00:00
// [RH] Don't activate anything if just predicting
if ( thing - > player & & ( thing - > player - > cheats & CF_PREDICTING ) )
{
thing - > flags6 & = ~ MF6_INTRYMOVE ;
return true ;
}
2006-02-24 04:48:15 +00:00
// [RH] Check for crossing fake floor/ceiling
newsec = thing - > Sector ;
if ( newsec - > heightsec & & oldsec - > heightsec & & newsec - > SecActTarget )
{
const sector_t * hs = newsec - > heightsec ;
2016-01-20 00:48:57 +00:00
fixed_t eyez = thing - > Z ( ) + viewheight ;
2014-08-07 16:05:39 +00:00
fixed_t fakez = hs - > floorplane . ZatPoint ( x , y ) ;
2006-02-24 04:48:15 +00:00
if ( ! oldAboveFakeFloor & & eyez > fakez )
{ // View went above fake floor
2014-08-07 16:05:39 +00:00
newsec - > SecActTarget - > TriggerAction ( thing , SECSPAC_EyesSurface ) ;
2006-02-24 04:48:15 +00:00
}
else if ( oldAboveFakeFloor & & eyez < = fakez )
{ // View went below fake floor
2014-08-07 16:05:39 +00:00
newsec - > SecActTarget - > TriggerAction ( thing , SECSPAC_EyesDive ) ;
2006-02-24 04:48:15 +00:00
}
if ( ! ( hs - > MoreFlags & SECF_FAKEFLOORONLY ) )
{
2014-08-07 16:05:39 +00:00
fakez = hs - > ceilingplane . ZatPoint ( x , y ) ;
2006-02-24 04:48:15 +00:00
if ( ! oldAboveFakeCeiling & & eyez > fakez )
{ // View went above fake ceiling
2014-08-07 16:05:39 +00:00
newsec - > SecActTarget - > TriggerAction ( thing , SECSPAC_EyesAboveC ) ;
2006-02-24 04:48:15 +00:00
}
else if ( oldAboveFakeCeiling & & eyez < = fakez )
{ // View went below fake ceiling
2014-08-07 16:05:39 +00:00
newsec - > SecActTarget - > TriggerAction ( thing , SECSPAC_EyesBelowC ) ;
2006-02-24 04:48:15 +00:00
}
}
}
// [RH] If changing sectors, trigger transitions
2012-03-18 03:52:18 +00:00
thing - > CheckSectorTransition ( oldsec ) ;
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-02-24 04:48:15 +00:00
return true ;
pushline :
2012-03-18 03:52:18 +00:00
thing - > flags6 & = ~ MF6_INTRYMOVE ;
2006-02-24 04:48:15 +00:00
// [RH] Don't activate anything if just predicting
if ( thing - > player & & ( thing - > player - > cheats & CF_PREDICTING ) )
{
return false ;
}
2016-01-20 00:48:57 +00:00
thing - > SetZ ( oldz ) ;
2014-08-07 16:05:39 +00:00
if ( ! ( thing - > flags & ( MF_TELEPORT | MF_NOCLIP ) ) )
2006-02-24 04:48:15 +00:00
{
int numSpecHitTemp ;
2008-04-08 20:52:49 +00:00
if ( tm . thing - > flags2 & MF2_BLASTED )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_DamageMobj ( tm . thing , NULL , NULL , tm . thing - > Mass > > 5 , NAME_Melee ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
numSpecHitTemp = ( int ) spechit . Size ( ) ;
2006-02-24 04:48:15 +00:00
while ( numSpecHitTemp > 0 )
{
// see which lines were pushed
ld = spechit [ - - numSpecHitTemp ] ;
2016-01-20 00:48:57 +00:00
fixedvec3 pos = thing - > PosRelative ( ld ) ;
side = P_PointOnLineSide ( pos . x , pos . y , ld ) ;
2014-08-07 16:05:39 +00:00
CheckForPushSpecial ( ld , side , thing , true ) ;
2006-02-24 04:48:15 +00:00
}
}
return false ;
}
2014-08-07 16:05:39 +00:00
bool P_TryMove ( AActor * thing , fixed_t x , fixed_t y ,
int dropoff , // killough 3/15/98: allow dropoff as option
const secplane_t * onfloor ) // [RH] Let P_TryMove keep the thing on the floor
2008-04-08 20:52:49 +00:00
{
FCheckPosition tm ;
return P_TryMove ( thing , x , y , dropoff , onfloor , tm ) ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2009-01-07 18:45:39 +00:00
//
// P_CheckMove
// Similar to P_TryMove but doesn't actually move the actor. Used for polyobject crushing
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2009-01-07 18:45:39 +00:00
bool P_CheckMove ( AActor * thing , fixed_t x , fixed_t y )
{
FCheckPosition tm ;
2016-01-20 00:48:57 +00:00
fixed_t newz = thing - > Z ( ) ;
2009-01-07 18:45:39 +00:00
2014-08-07 16:05:39 +00:00
if ( ! P_CheckPosition ( thing , x , y , tm ) )
2009-01-07 18:45:39 +00:00
{
return false ;
}
if ( thing - > flags3 & MF3_FLOORHUGGER )
{
newz = tm . floorz ;
}
else if ( thing - > flags3 & MF3_CEILINGHUGGER )
{
newz = tm . ceilingz - thing - > height ;
}
if ( ! ( thing - > flags & MF_NOCLIP ) )
{
if ( tm . ceilingz - tm . floorz < thing - > height )
{
return false ;
}
if ( ! ( thing - > flags & MF_TELEPORT )
& & tm . ceilingz - newz < thing - > height
& & ! ( thing - > flags3 & MF3_CEILINGHUGGER )
& & ( ! ( thing - > flags2 & MF2_FLY ) | | ! ( thing - > flags & MF_NOGRAVITY ) ) )
{
return false ;
}
if ( thing - > flags2 & MF2_FLY & & thing - > flags & MF_NOGRAVITY )
{
2016-01-20 00:48:57 +00:00
if ( thing - > Top ( ) > tm . ceilingz )
2009-01-07 18:45:39 +00:00
return false ;
}
if ( ! ( thing - > flags & MF_TELEPORT ) & & ! ( thing - > flags3 & MF3_FLOORHUGGER ) )
{
2014-08-07 16:05:39 +00:00
if ( tm . floorz - newz > thing - > MaxStepHeight )
2009-01-07 18:45:39 +00:00
{ // too big a step up
return false ;
}
2014-10-24 22:01:04 +00:00
else if ( ( thing - > flags & MF_MISSILE ) & & ! ( thing - > flags6 & MF6_STEPMISSILE ) & & tm . floorz > newz )
2009-01-07 18:45:39 +00:00
{ // [RH] Don't let normal missiles climb steps
return false ;
}
else if ( newz < tm . floorz )
{ // [RH] Check to make sure there's nothing in the way for the step up
2016-01-20 00:48:57 +00:00
fixed_t savedz = thing - > Z ( ) ;
thing - > SetZ ( newz = tm . floorz ) ;
2014-08-07 16:05:39 +00:00
bool good = P_TestMobjZ ( thing ) ;
2016-01-20 00:48:57 +00:00
thing - > SetZ ( savedz ) ;
2009-01-07 18:45:39 +00:00
if ( ! good )
{
return false ;
}
}
}
if ( thing - > flags2 & MF2_CANTLEAVEFLOORPIC
& & ( tm . floorpic ! = thing - > floorpic
2014-08-07 16:05:39 +00:00
| | tm . floorz - newz ! = 0 ) )
2009-01-07 18:45:39 +00:00
{ // must stay within a sector of a certain floor type
return false ;
}
}
return true ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// SLIDE MOVE
// Allows the player to slide along any angled walls.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2008-04-09 18:35:21 +00:00
struct FSlide
{
fixed_t bestslidefrac ;
fixed_t secondslidefrac ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
line_t * bestslideline ;
line_t * secondslideline ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
AActor * slidemo ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
fixed_t tmxmove ;
fixed_t tmymove ;
void HitSlideLine ( line_t * ld ) ;
2014-08-07 16:05:39 +00:00
void SlideTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy ) ;
void SlideMove ( AActor * mo , fixed_t tryx , fixed_t tryy , int numsteps ) ;
2008-04-09 18:35:21 +00:00
// The bouncing code uses the same data structure
2014-08-07 16:05:39 +00:00
bool BounceTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy ) ;
bool BounceWall ( AActor * mo ) ;
2008-04-09 18:35:21 +00:00
} ;
2006-02-24 04:48:15 +00:00
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// P_HitSlideLine
// Adjusts the xmove / ymove
// so that the next move will slide along the wall.
// If the floor is icy, then you can bounce off a wall. // phares
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
void FSlide : : HitSlideLine ( line_t * ld )
2006-02-24 04:48:15 +00:00
{
int side ;
angle_t lineangle ;
angle_t moveangle ;
angle_t deltaangle ;
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
fixed_t movelen ;
2006-09-14 00:02:31 +00:00
bool icyfloor ; // is floor icy? // phares
2014-08-07 16:05:39 +00:00
// |
2006-02-24 04:48:15 +00:00
// Under icy conditions, if the angle of approach to the wall // V
// is more than 45 degrees, then you'll bounce and lose half
2009-06-30 20:57:51 +00:00
// your velocity. If less than 45 degrees, you'll slide along
2006-02-24 04:48:15 +00:00
// the wall. 45 is arbitrary and is believable.
// Check for the special cases of horz or vert walls.
// killough 10/98: only bounce if hit hard (prevents wobbling)
2014-08-07 16:05:39 +00:00
icyfloor =
( P_AproxDistance ( tmxmove , tmymove ) > 4 * FRACUNIT ) & &
2006-02-24 04:48:15 +00:00
var_friction & & // killough 8/28/98: calc friction on demand
2016-01-20 00:48:57 +00:00
slidemo - > Z ( ) < = slidemo - > floorz & &
2014-08-07 16:05:39 +00:00
P_GetFriction ( slidemo , NULL ) > ORIG_FRICTION ;
2006-02-24 04:48:15 +00:00
2014-08-01 00:15:08 +00:00
if ( ld - > dx = = 0 )
{ // ST_VERTICAL
2006-02-24 04:48:15 +00:00
if ( icyfloor & & ( abs ( tmxmove ) > abs ( tmymove ) ) )
{
2014-08-07 16:05:39 +00:00
tmxmove = - tmxmove / 2 ; // absorb half the velocity
2006-02-24 04:48:15 +00:00
tmymove / = 2 ;
if ( slidemo - > player & & slidemo - > health > 0 & & ! ( slidemo - > player - > cheats & CF_PREDICTING ) )
{
2014-08-07 16:05:39 +00:00
S_Sound ( slidemo , CHAN_VOICE , " *grunt " , 1 , ATTN_IDLE ) ; // oooff!// ^
2006-02-24 04:48:15 +00:00
}
} // |
else // phares
tmxmove = 0 ; // no more movement in the X direction
return ;
}
2014-08-01 00:15:08 +00:00
if ( ld - > dy = = 0 )
{ // ST_HORIZONTAL
if ( icyfloor & & ( abs ( tmymove ) > abs ( tmxmove ) ) )
{
tmxmove / = 2 ; // absorb half the velocity
2014-08-07 16:05:39 +00:00
tmymove = - tmymove / 2 ;
2014-08-01 00:15:08 +00:00
if ( slidemo - > player & & slidemo - > health > 0 & & ! ( slidemo - > player - > cheats & CF_PREDICTING ) )
{
2014-08-07 16:05:39 +00:00
S_Sound ( slidemo , CHAN_VOICE , " *grunt " , 1 , ATTN_IDLE ) ; // oooff!
2014-08-01 00:15:08 +00:00
}
}
else
tmymove = 0 ; // no more movement in the Y direction
return ;
}
2006-02-24 04:48:15 +00:00
// The wall is angled. Bounce if the angle of approach is // phares
// less than 45 degrees. // phares
2016-01-20 00:48:57 +00:00
fixedvec3 pos = slidemo - > PosRelative ( ld ) ;
side = P_PointOnLineSide ( pos . x , pos . y , ld ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
lineangle = R_PointToAngle2 ( 0 , 0 , ld - > dx , ld - > dy ) ;
2006-02-24 04:48:15 +00:00
if ( side = = 1 )
lineangle + = ANG180 ;
2014-08-07 16:05:39 +00:00
moveangle = R_PointToAngle2 ( 0 , 0 , tmxmove , tmymove ) ;
2006-02-24 04:48:15 +00:00
moveangle + = 10 ; // prevents sudden path reversal due to // phares
2014-08-07 16:05:39 +00:00
// rounding error // |
deltaangle = moveangle - lineangle ; // V
movelen = P_AproxDistance ( tmxmove , tmymove ) ;
if ( icyfloor & & ( deltaangle > ANG45 ) & & ( deltaangle < ANG90 + ANG45 ) )
2006-02-24 04:48:15 +00:00
{
moveangle = lineangle - deltaangle ;
movelen / = 2 ; // absorb
if ( slidemo - > player & & slidemo - > health > 0 & & ! ( slidemo - > player - > cheats & CF_PREDICTING ) )
{
2014-08-07 16:05:39 +00:00
S_Sound ( slidemo , CHAN_VOICE , " *grunt " , 1 , ATTN_IDLE ) ; // oooff!
2006-02-24 04:48:15 +00:00
}
moveangle > > = ANGLETOFINESHIFT ;
2014-08-07 16:05:39 +00:00
tmxmove = FixedMul ( movelen , finecosine [ moveangle ] ) ;
tmymove = FixedMul ( movelen , finesine [ moveangle ] ) ;
2006-02-24 04:48:15 +00:00
} // ^
else // |
{ // phares
2010-05-28 11:15:05 +00:00
// Doom's original algorithm here does not work well due to imprecisions of the sine table.
// However, keep it active if the wallrunning compatibility flag is on
if ( i_compatflags & COMPATF_WALLRUN )
{
fixed_t newlen ;
2014-08-07 16:05:39 +00:00
2010-05-28 11:15:05 +00:00
if ( deltaangle > ANG180 )
deltaangle + = ANG180 ;
// I_Error ("SlideLine: ang>ANG180");
2006-02-24 04:48:15 +00:00
2010-05-28 11:15:05 +00:00
lineangle > > = ANGLETOFINESHIFT ;
deltaangle > > = ANGLETOFINESHIFT ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
newlen = FixedMul ( movelen , finecosine [ deltaangle ] ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
tmxmove = FixedMul ( newlen , finecosine [ lineangle ] ) ;
tmymove = FixedMul ( newlen , finesine [ lineangle ] ) ;
2010-05-28 11:15:05 +00:00
}
else
{
divline_t dll , dlv ;
fixed_t inter1 , inter2 , inter3 ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
P_MakeDivline ( ld , & dll ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
dlv . x = pos . x ;
dlv . y = pos . y ;
2010-05-28 11:15:05 +00:00
dlv . dx = dll . dy ;
dlv . dy = - dll . dx ;
2006-02-24 04:48:15 +00:00
2010-05-28 11:15:05 +00:00
inter1 = P_InterceptVector ( & dll , & dlv ) ;
2006-02-24 04:48:15 +00:00
2010-05-28 11:15:05 +00:00
dlv . dx = tmxmove ;
dlv . dy = tmymove ;
2014-08-07 16:05:39 +00:00
inter2 = P_InterceptVector ( & dll , & dlv ) ;
inter3 = P_InterceptVector ( & dlv , & dll ) ;
2006-02-24 04:48:15 +00:00
2010-05-28 11:15:05 +00:00
if ( inter3 ! = 0 )
{
2014-08-07 16:05:39 +00:00
tmxmove = Scale ( inter2 - inter1 , dll . dx , inter3 ) ;
tmymove = Scale ( inter2 - inter1 , dll . dy , inter3 ) ;
2010-05-28 11:15:05 +00:00
}
else
{
tmxmove = tmymove = 0 ;
}
2006-02-24 04:48:15 +00:00
}
} // phares
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// PTR_SlideTraverse
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
void FSlide : : SlideTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy )
2006-02-24 04:48:15 +00:00
{
2008-06-15 18:36:26 +00:00
FLineOpening open ;
2008-04-09 18:35:21 +00:00
FPathTraverse it ( startx , starty , endx , endy , PT_ADDLINES ) ;
intercept_t * in ;
while ( ( in = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
line_t * li ;
2014-08-07 16:05:39 +00:00
2008-04-09 18:35:21 +00:00
if ( ! in - > isaline )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
// should never happen
2014-08-07 16:05:39 +00:00
Printf ( " PTR_SlideTraverse: not a line? " ) ;
2008-04-09 18:35:21 +00:00
continue ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
2008-04-09 18:35:21 +00:00
li = in - > d . line ;
2014-08-07 16:05:39 +00:00
if ( ! ( li - > flags & ML_TWOSIDED ) | | ! li - > backsector )
2008-04-09 18:35:21 +00:00
{
2016-01-20 00:48:57 +00:00
fixedvec3 pos = slidemo - > PosRelative ( li ) ;
if ( P_PointOnLineSide ( pos . x , pos . y , li ) )
2008-04-09 18:35:21 +00:00
{
// don't hit the back side
2014-08-07 16:05:39 +00:00
continue ;
2008-04-09 18:35:21 +00:00
}
goto isblocking ;
}
2014-08-07 16:05:39 +00:00
if ( li - > flags & ( ML_BLOCKING | ML_BLOCKEVERYTHING ) )
2008-04-09 18:35:21 +00:00
{
goto isblocking ;
}
if ( li - > flags & ML_BLOCK_PLAYERS & & slidemo - > player ! = NULL )
{
goto isblocking ;
}
2009-09-14 20:47:53 +00:00
if ( li - > flags & ML_BLOCKMONSTERS & & ! ( ( slidemo - > flags3 & MF3_NOBLOCKMONST )
| | ( ( i_compatflags & COMPATF_NOBLOCKFRIENDS ) & & ( slidemo - > flags & MF_FRIENDLY ) ) ) )
2006-02-24 04:48:15 +00:00
{
goto isblocking ;
}
2008-04-09 18:35:21 +00:00
// set openrange, opentop, openbottom
2014-08-07 16:05:39 +00:00
P_LineOpening ( open , slidemo , li , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ;
2008-04-09 18:35:21 +00:00
if ( open . range < slidemo - > height )
goto isblocking ; // doesn't fit
2014-08-07 16:05:39 +00:00
2016-01-20 00:48:57 +00:00
if ( open . top - slidemo - > Z ( ) < slidemo - > height )
2008-04-09 18:35:21 +00:00
goto isblocking ; // mobj is too high
2016-01-20 00:48:57 +00:00
if ( open . bottom - slidemo - > Z ( ) > slidemo - > MaxStepHeight )
2008-04-09 18:35:21 +00:00
{
goto isblocking ; // too big a step up
}
2016-01-20 00:48:57 +00:00
else if ( slidemo - > Z ( ) < open . bottom )
2008-04-09 18:35:21 +00:00
{ // [RH] Check to make sure there's nothing in the way for the step up
2016-01-20 00:48:57 +00:00
fixed_t savedz = slidemo - > Z ( ) ;
slidemo - > SetZ ( open . bottom ) ;
2014-08-07 16:05:39 +00:00
bool good = P_TestMobjZ ( slidemo ) ;
2016-01-20 00:48:57 +00:00
slidemo - > SetZ ( savedz ) ;
2008-04-09 18:35:21 +00:00
if ( ! good )
{
goto isblocking ;
}
}
// this line doesn't block movement
2014-08-07 16:05:39 +00:00
continue ;
2008-04-09 18:35:21 +00:00
// the line does block movement,
// see if it is closer than best so far
2014-08-07 16:05:39 +00:00
isblocking :
2008-04-09 18:35:21 +00:00
if ( in - > frac < bestslidefrac )
{
secondslidefrac = bestslidefrac ;
secondslideline = bestslideline ;
bestslidefrac = in - > frac ;
bestslideline = li ;
}
2014-08-07 16:05:39 +00:00
2008-04-09 18:35:21 +00:00
return ; // stop
2006-02-24 04:48:15 +00:00
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// P_SlideMove
//
2009-06-30 20:57:51 +00:00
// The velx / vely move is bad, so try to slide along a wall.
2006-02-24 04:48:15 +00:00
//
// Find the first line hit, move flush to it, and slide along it
//
// This is a kludgy mess.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
void FSlide : : SlideMove ( AActor * mo , fixed_t tryx , fixed_t tryy , int numsteps )
2006-02-24 04:48:15 +00:00
{
fixed_t leadx , leady ;
fixed_t trailx , traily ;
fixed_t newx , newy ;
fixed_t xmove , ymove ;
2009-01-04 15:00:29 +00:00
const secplane_t * walkplane ;
2006-02-24 04:48:15 +00:00
int hitcount ;
hitcount = 3 ;
2008-04-09 18:35:21 +00:00
slidemo = mo ;
2007-10-29 20:27:40 +00:00
2010-02-11 17:10:01 +00:00
if ( mo - > player & & mo - > player - > mo = = mo & & mo - > reactiontime > 0 )
2007-10-29 20:27:40 +00:00
return ; // player coming right out of a teleporter.
2014-08-07 16:05:39 +00:00
retry :
2006-02-24 04:48:15 +00:00
if ( ! - - hitcount )
goto stairstep ; // don't loop forever
2014-08-07 16:05:39 +00:00
2006-02-24 04:48:15 +00:00
// trace along the three leading corners
if ( tryx > 0 )
{
2016-01-20 00:48:57 +00:00
leadx = mo - > X ( ) + mo - > radius ;
trailx = mo - > X ( ) - mo - > radius ;
2006-02-24 04:48:15 +00:00
}
else
{
2016-01-20 00:48:57 +00:00
leadx = mo - > X ( ) - mo - > radius ;
trailx = mo - > X ( ) + mo - > radius ;
2006-02-24 04:48:15 +00:00
}
if ( tryy > 0 )
{
2016-01-20 00:48:57 +00:00
leady = mo - > Y ( ) + mo - > radius ;
traily = mo - > Y ( ) - mo - > radius ;
2006-02-24 04:48:15 +00:00
}
else
{
2016-01-20 00:48:57 +00:00
leady = mo - > Y ( ) - mo - > radius ;
traily = mo - > Y ( ) + mo - > radius ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
bestslidefrac = FRACUNIT + 1 ;
SlideTraverse ( leadx , leady , leadx + tryx , leady + tryy ) ;
SlideTraverse ( trailx , leady , trailx + tryx , leady + tryy ) ;
SlideTraverse ( leadx , traily , leadx + tryx , traily + tryy ) ;
2008-04-09 18:35:21 +00:00
2006-02-24 04:48:15 +00:00
// move up to the wall
2014-08-07 16:05:39 +00:00
if ( bestslidefrac = = FRACUNIT + 1 )
2006-02-24 04:48:15 +00:00
{
// the move must have hit the middle, so stairstep
2014-08-07 16:05:39 +00:00
stairstep :
2006-02-24 04:48:15 +00:00
// killough 3/15/98: Allow objects to drop off ledges
xmove = 0 , ymove = tryy ;
2014-08-07 16:05:39 +00:00
walkplane = P_CheckSlopeWalk ( mo , xmove , ymove ) ;
2016-01-20 00:48:57 +00:00
if ( ! P_TryMove ( mo , mo - > X ( ) + xmove , mo - > Y ( ) + ymove , true , walkplane ) )
2006-02-24 04:48:15 +00:00
{
xmove = tryx , ymove = 0 ;
2014-08-07 16:05:39 +00:00
walkplane = P_CheckSlopeWalk ( mo , xmove , ymove ) ;
2016-01-20 00:48:57 +00:00
P_TryMove ( mo , mo - > X ( ) + xmove , mo - > Y ( ) + ymove , true , walkplane ) ;
2006-02-24 04:48:15 +00:00
}
return ;
}
// fudge a bit to make sure it doesn't hit
2014-08-07 16:05:39 +00:00
bestslidefrac - = FRACUNIT / 32 ;
2006-02-24 04:48:15 +00:00
if ( bestslidefrac > 0 )
{
2014-08-07 16:05:39 +00:00
newx = FixedMul ( tryx , bestslidefrac ) ;
newy = FixedMul ( tryy , bestslidefrac ) ;
2013-02-14 23:27:09 +00:00
// [BL] We need to abandon this function if we end up going through a teleporter
const fixed_t startvelx = mo - > velx ;
const fixed_t startvely = mo - > vely ;
2006-02-24 04:48:15 +00:00
// killough 3/15/98: Allow objects to drop off ledges
2016-01-20 00:48:57 +00:00
if ( ! P_TryMove ( mo , mo - > X ( ) + newx , mo - > Y ( ) + newy , true ) )
2006-02-24 04:48:15 +00:00
goto stairstep ;
2013-02-14 23:27:09 +00:00
if ( mo - > velx ! = startvelx | | mo - > vely ! = startvely )
return ;
2006-02-24 04:48:15 +00:00
}
// Now continue along the wall.
2014-08-07 16:05:39 +00:00
bestslidefrac = FRACUNIT - ( bestslidefrac + FRACUNIT / 32 ) ; // remainder
2006-02-24 04:48:15 +00:00
if ( bestslidefrac > FRACUNIT )
bestslidefrac = FRACUNIT ;
else if ( bestslidefrac < = 0 )
return ;
2014-08-07 16:05:39 +00:00
tryx = tmxmove = FixedMul ( tryx , bestslidefrac ) ;
tryy = tmymove = FixedMul ( tryy , bestslidefrac ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
HitSlideLine ( bestslideline ) ; // clip the moves
2006-02-24 04:48:15 +00:00
2009-06-30 20:57:51 +00:00
mo - > velx = tmxmove * numsteps ;
mo - > vely = tmymove * numsteps ;
2006-02-24 04:48:15 +00:00
// killough 10/98: affect the bobbing the same way (but not voodoo dolls)
if ( mo - > player & & mo - > player - > mo = = mo )
{
2009-06-30 20:57:51 +00:00
if ( abs ( mo - > player - > velx ) > abs ( mo - > velx ) )
mo - > player - > velx = mo - > velx ;
if ( abs ( mo - > player - > vely ) > abs ( mo - > vely ) )
mo - > player - > vely = mo - > vely ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
walkplane = P_CheckSlopeWalk ( mo , tmxmove , tmymove ) ;
2006-02-24 04:48:15 +00:00
// killough 3/15/98: Allow objects to drop off ledges
2016-01-20 00:48:57 +00:00
if ( ! P_TryMove ( mo , mo - > X ( ) + tmxmove , mo - > Y ( ) + tmymove , true , walkplane ) )
2006-02-24 04:48:15 +00:00
{
goto retry ;
}
}
2014-08-07 16:05:39 +00:00
void P_SlideMove ( AActor * mo , fixed_t tryx , fixed_t tryy , int numsteps )
2008-04-09 18:35:21 +00:00
{
FSlide slide ;
slide . SlideMove ( mo , tryx , tryy , numsteps ) ;
}
2008-03-18 18:18:18 +00:00
2006-02-24 04:48:15 +00:00
//============================================================================
//
// P_CheckSlopeWalk
//
//============================================================================
2014-08-07 16:05:39 +00:00
const secplane_t * P_CheckSlopeWalk ( AActor * actor , fixed_t & xmove , fixed_t & ymove )
2006-02-24 04:48:15 +00:00
{
2009-01-04 15:00:29 +00:00
static secplane_t copyplane ;
2006-02-24 04:48:15 +00:00
if ( actor - > flags & MF_NOGRAVITY )
{
2009-01-04 15:00:29 +00:00
return NULL ;
2006-02-24 04:48:15 +00:00
}
2016-02-17 10:39:14 +00:00
fixedvec3 pos = actor - > PosRelative ( actor - > floorsector ) ;
2006-02-24 04:48:15 +00:00
const secplane_t * plane = & actor - > floorsector - > floorplane ;
2016-02-17 10:39:14 +00:00
fixed_t planezhere = plane - > ZatPoint ( pos ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
for ( unsigned int i = 0 ; i < actor - > floorsector - > e - > XFloor . ffloors . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
F3DFloor * rover = actor - > floorsector - > e - > XFloor . ffloors [ i ] ;
if ( ! ( rover - > flags & FF_SOLID ) | | ! ( rover - > flags & FF_EXISTS ) ) continue ;
2009-01-04 15:00:29 +00:00
2016-02-17 10:39:14 +00:00
fixed_t thisplanez = rover - > top . plane - > ZatPoint ( pos ) ;
2009-01-04 15:00:29 +00:00
2016-01-20 00:48:57 +00:00
if ( thisplanez > planezhere & & thisplanez < = actor - > Z ( ) + actor - > MaxStepHeight )
2009-01-04 15:00:29 +00:00
{
copyplane = * rover - > top . plane ;
if ( copyplane . c < 0 ) copyplane . FlipVert ( ) ;
plane = & copyplane ;
2014-08-07 16:05:39 +00:00
planezhere = thisplanez ;
2009-01-04 15:00:29 +00:00
}
}
if ( actor - > floorsector ! = actor - > Sector )
{
2014-08-07 16:05:39 +00:00
for ( unsigned int i = 0 ; i < actor - > Sector - > e - > XFloor . ffloors . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
F3DFloor * rover = actor - > Sector - > e - > XFloor . ffloors [ i ] ;
if ( ! ( rover - > flags & FF_SOLID ) | | ! ( rover - > flags & FF_EXISTS ) ) continue ;
2009-01-04 15:00:29 +00:00
2016-01-17 13:08:20 +00:00
fixed_t thisplanez = rover - > top . plane - > ZatPoint ( actor ) ;
2009-01-04 15:00:29 +00:00
2016-01-20 00:48:57 +00:00
if ( thisplanez > planezhere & & thisplanez < = actor - > Z ( ) + actor - > MaxStepHeight )
2009-01-04 15:00:29 +00:00
{
copyplane = * rover - > top . plane ;
if ( copyplane . c < 0 ) copyplane . FlipVert ( ) ;
plane = & copyplane ;
2014-08-07 16:05:39 +00:00
planezhere = thisplanez ;
2009-01-04 15:00:29 +00:00
}
}
}
2006-02-24 04:48:15 +00:00
if ( actor - > floorsector ! = actor - > Sector )
{
2006-04-11 08:36:23 +00:00
// this additional check prevents sliding on sloped dropoffs
2014-08-07 16:05:39 +00:00
if ( planezhere > actor - > floorz + 4 * FRACUNIT )
2009-01-04 15:00:29 +00:00
return NULL ;
2006-02-24 04:48:15 +00:00
}
2016-01-20 00:48:57 +00:00
if ( actor - > Z ( ) - planezhere > FRACUNIT )
2006-02-24 04:48:15 +00:00
{ // not on floor
2009-01-04 15:00:29 +00:00
return NULL ;
2006-02-24 04:48:15 +00:00
}
if ( ( plane - > a | plane - > b ) ! = 0 )
{
fixed_t destx , desty ;
fixed_t t ;
2016-01-20 00:48:57 +00:00
destx = actor - > X ( ) + xmove ;
desty = actor - > Y ( ) + ymove ;
t = TMulScale16 ( plane - > a , destx , plane - > b , desty , plane - > c , actor - > Z ( ) ) + plane - > d ;
2006-02-24 04:48:15 +00:00
if ( t < 0 )
{ // Desired location is behind (below) the plane
2014-08-07 16:05:39 +00:00
// (i.e. Walking up the plane)
2006-02-24 04:48:15 +00:00
if ( plane - > c < STEEPSLOPE )
{ // Can't climb up slopes of ~45 degrees or more
if ( actor - > flags & MF_NOCLIP )
{
2009-01-04 15:00:29 +00:00
return ( actor - > floorsector = = actor - > Sector ) ? plane : NULL ;
2006-02-24 04:48:15 +00:00
}
else
{
const msecnode_t * node ;
bool dopush = true ;
2014-08-07 16:05:39 +00:00
if ( plane - > c > STEEPSLOPE * 2 / 3 )
2006-02-24 04:48:15 +00:00
{
for ( node = actor - > touching_sectorlist ; node ; node = node - > m_tnext )
{
2016-02-17 10:39:14 +00:00
sector_t * sec = node - > m_sector ;
2006-02-24 04:48:15 +00:00
if ( sec - > floorplane . c > = STEEPSLOPE )
{
2016-02-17 10:39:14 +00:00
fixedvec3 pos = actor - > PosRelative ( sec ) ;
pos . x + = xmove ;
pos . y + = ymove ;
if ( sec - > floorplane . ZatPoint ( pos ) > = actor - > Z ( ) - actor - > MaxStepHeight )
2006-02-24 04:48:15 +00:00
{
dopush = false ;
break ;
}
}
}
}
if ( dopush )
{
2009-06-30 20:57:51 +00:00
xmove = actor - > velx = plane - > a * 2 ;
ymove = actor - > vely = plane - > b * 2 ;
2006-02-24 04:48:15 +00:00
}
2009-01-04 15:00:29 +00:00
return ( actor - > floorsector = = actor - > Sector ) ? plane : NULL ;
2006-02-24 04:48:15 +00:00
}
}
// Slide the desired location along the plane's normal
// so that it lies on the plane's surface
2014-08-07 16:05:39 +00:00
destx - = FixedMul ( plane - > a , t ) ;
desty - = FixedMul ( plane - > b , t ) ;
2016-01-20 00:48:57 +00:00
xmove = destx - actor - > X ( ) ;
ymove = desty - actor - > Y ( ) ;
2009-01-04 15:00:29 +00:00
return ( actor - > floorsector = = actor - > Sector ) ? plane : NULL ;
2006-02-24 04:48:15 +00:00
}
else if ( t > 0 )
{ // Desired location is in front of (above) the plane
2016-01-20 00:48:57 +00:00
if ( planezhere = = actor - > Z ( ) )
2006-02-24 04:48:15 +00:00
{ // Actor's current spot is on/in the plane, so walk down it
2014-08-07 16:05:39 +00:00
// Same principle as walking up, except reversed
destx + = FixedMul ( plane - > a , t ) ;
desty + = FixedMul ( plane - > b , t ) ;
2016-01-20 00:48:57 +00:00
xmove = destx - actor - > X ( ) ;
ymove = desty - actor - > Y ( ) ;
2009-01-04 15:00:29 +00:00
return ( actor - > floorsector = = actor - > Sector ) ? plane : NULL ;
2006-02-24 04:48:15 +00:00
}
}
}
2009-01-04 15:00:29 +00:00
return NULL ;
2006-02-24 04:48:15 +00:00
}
//============================================================================
//
// PTR_BounceTraverse
//
//============================================================================
2014-08-07 16:05:39 +00:00
bool FSlide : : BounceTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy )
2006-02-24 04:48:15 +00:00
{
2008-06-15 18:36:26 +00:00
FLineOpening open ;
2008-04-09 18:35:21 +00:00
FPathTraverse it ( startx , starty , endx , endy , PT_ADDLINES ) ;
intercept_t * in ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
while ( ( in = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
line_t * li ;
2008-03-18 18:18:18 +00:00
2008-04-09 18:35:21 +00:00
if ( ! in - > isaline )
{
2014-08-07 16:05:39 +00:00
Printf ( " PTR_BounceTraverse: not a line? " ) ;
2008-04-09 18:35:21 +00:00
continue ;
}
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
li = in - > d . line ;
assert ( ( ( size_t ) li - ( size_t ) lines ) % sizeof ( line_t ) = = 0 ) ;
if ( li - > flags & ML_BLOCKEVERYTHING )
{
goto bounceblocking ;
}
if ( ! ( li - > flags & ML_TWOSIDED ) | | ! li - > backsector )
{
2016-01-20 00:48:57 +00:00
if ( P_PointOnLineSide ( slidemo - > X ( ) , slidemo - > Y ( ) , li ) )
2008-04-09 18:35:21 +00:00
continue ; // don't hit the back side
goto bounceblocking ;
}
2008-03-04 00:56:22 +00:00
2014-08-07 16:05:39 +00:00
P_LineOpening ( open , slidemo , li , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ; // set openrange, opentop, openbottom
2008-04-09 18:35:21 +00:00
if ( open . range < slidemo - > height )
goto bounceblocking ; // doesn't fit
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
if ( open . top - slidemo - > Z ( ) < slidemo - > height )
2008-04-09 18:35:21 +00:00
goto bounceblocking ; // mobj is too high
2016-01-20 00:48:57 +00:00
if ( open . bottom > slidemo - > Z ( ) )
2008-04-09 18:35:21 +00:00
goto bounceblocking ; // mobj is too low
continue ; // this line doesn't block movement
2014-08-07 16:05:39 +00:00
// the line does block movement, see if it is closer than best so far
2008-04-09 18:35:21 +00:00
bounceblocking :
if ( in - > frac < bestslidefrac )
{
secondslidefrac = bestslidefrac ;
secondslideline = bestslideline ;
bestslidefrac = in - > frac ;
bestslideline = li ;
}
return false ; // stop
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
return true ;
2006-02-24 04:48:15 +00:00
}
//============================================================================
//
// P_BounceWall
//
//============================================================================
2014-08-07 16:05:39 +00:00
bool FSlide : : BounceWall ( AActor * mo )
2006-02-24 04:48:15 +00:00
{
fixed_t leadx , leady ;
int side ;
angle_t lineangle , moveangle , deltaangle ;
fixed_t movelen ;
line_t * line ;
2009-09-06 01:49:17 +00:00
if ( ! ( mo - > BounceFlags & BOUNCE_Walls ) )
2006-02-24 04:48:15 +00:00
{
return false ;
}
2008-01-08 03:21:01 +00:00
slidemo = mo ;
2014-08-07 16:05:39 +00:00
//
// trace along the three leading corners
//
2009-06-30 20:57:51 +00:00
if ( mo - > velx > 0 )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
leadx = mo - > X ( ) + mo - > radius ;
2006-02-24 04:48:15 +00:00
}
else
{
2016-01-20 00:48:57 +00:00
leadx = mo - > X ( ) - mo - > radius ;
2008-01-08 03:21:01 +00:00
}
2009-06-30 20:57:51 +00:00
if ( mo - > vely > 0 )
2008-01-08 03:21:01 +00:00
{
2016-01-20 00:48:57 +00:00
leady = mo - > Y ( ) + mo - > radius ;
2008-01-08 03:21:01 +00:00
}
else
{
2016-01-20 00:48:57 +00:00
leady = mo - > Y ( ) - mo - > radius ;
2008-01-08 03:21:01 +00:00
}
2014-08-07 16:05:39 +00:00
bestslidefrac = FRACUNIT + 1 ;
2008-04-08 20:52:49 +00:00
bestslideline = mo - > BlockingLine ;
2014-08-07 16:05:39 +00:00
if ( BounceTraverse ( leadx , leady , leadx + mo - > velx , leady + mo - > vely ) & & mo - > BlockingLine = = NULL )
2008-01-08 03:21:01 +00:00
{ // Could not find a wall, so bounce off the floor/ceiling instead.
2016-01-20 00:48:57 +00:00
fixed_t floordist = mo - > Z ( ) - mo - > floorz ;
fixed_t ceildist = mo - > ceilingz - mo - > Z ( ) ;
2008-01-08 03:21:01 +00:00
if ( floordist < = ceildist )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
mo - > FloorBounceMissile ( mo - > Sector - > floorplane ) ;
2008-01-08 03:21:01 +00:00
return true ;
2006-02-24 04:48:15 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
mo - > FloorBounceMissile ( mo - > Sector - > ceilingplane ) ;
2008-01-08 03:21:01 +00:00
return true ;
2006-02-24 04:48:15 +00:00
}
}
2008-03-16 01:32:54 +00:00
line = bestslideline ;
2006-02-24 04:48:15 +00:00
2006-06-28 11:14:20 +00:00
if ( line - > special = = Line_Horizon )
{
mo - > SeeSound = 0 ; // it might make a sound otherwise
mo - > Destroy ( ) ;
return true ;
}
2008-03-01 16:59:17 +00:00
// The amount of bounces is limited
2014-08-07 16:05:39 +00:00
if ( mo - > bouncecount > 0 & & - - mo - > bouncecount = = 0 )
2008-03-01 16:59:17 +00:00
{
2009-09-14 20:47:53 +00:00
if ( mo - > flags & MF_MISSILE )
2012-03-16 01:17:12 +00:00
P_ExplodeMissile ( mo , line , NULL ) ;
2009-09-14 20:47:53 +00:00
else
mo - > Die ( NULL , NULL ) ;
2008-03-01 16:59:17 +00:00
return true ;
}
2016-01-20 00:48:57 +00:00
side = P_PointOnLineSide ( mo - > X ( ) , mo - > Y ( ) , line ) ;
2014-08-07 16:05:39 +00:00
lineangle = R_PointToAngle2 ( 0 , 0 , line - > dx , line - > dy ) ;
2006-02-24 04:48:15 +00:00
if ( side = = 1 )
{
lineangle + = ANG180 ;
}
2014-08-07 16:05:39 +00:00
moveangle = R_PointToAngle2 ( 0 , 0 , mo - > velx , mo - > vely ) ;
deltaangle = ( 2 * lineangle ) - moveangle ;
2006-02-24 04:48:15 +00:00
mo - > angle = deltaangle ;
deltaangle > > = ANGLETOFINESHIFT ;
2009-09-06 01:49:17 +00:00
movelen = fixed_t ( sqrt ( double ( mo - > velx ) * mo - > velx + double ( mo - > vely ) * mo - > vely ) ) ;
2008-06-10 09:16:01 +00:00
movelen = FixedMul ( movelen , mo - > wallbouncefactor ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
FBoundingBox box ( mo - > X ( ) , mo - > Y ( ) , mo - > radius ) ;
2014-08-07 16:05:39 +00:00
if ( box . BoxOnLineSide ( line ) = = - 1 )
2006-02-24 04:48:15 +00:00
{
2016-01-20 00:48:57 +00:00
fixedvec3 pos = mo - > Vec3Offset (
FixedMul ( mo - > radius , finecosine [ deltaangle ] ) ,
FixedMul ( mo - > radius , finesine [ deltaangle ] ) , 0 ) ;
mo - > SetOrigin ( pos , true ) ;
2006-02-24 04:48:15 +00:00
}
if ( movelen < FRACUNIT )
{
2014-08-07 16:05:39 +00:00
movelen = 2 * FRACUNIT ;
2006-02-24 04:48:15 +00:00
}
2009-06-30 20:57:51 +00:00
mo - > velx = FixedMul ( movelen , finecosine [ deltaangle ] ) ;
mo - > vely = FixedMul ( movelen , finesine [ deltaangle ] ) ;
2013-05-04 22:52:37 +00:00
if ( mo - > BounceFlags & BOUNCE_UseBounceState )
{
FState * bouncestate = mo - > FindState ( NAME_Bounce , NAME_Wall ) ;
if ( bouncestate ! = NULL )
{
mo - > SetState ( bouncestate ) ;
}
}
2006-02-24 04:48:15 +00:00
return true ;
}
2014-08-07 16:05:39 +00:00
bool P_BounceWall ( AActor * mo )
2008-04-09 18:35:21 +00:00
{
FSlide slide ;
return slide . BounceWall ( mo ) ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2009-09-15 06:35:46 +00:00
extern FRandom pr_bounce ;
2014-08-07 16:05:39 +00:00
bool P_BounceActor ( AActor * mo , AActor * BlockingMobj , bool ontop )
2009-09-14 20:47:53 +00:00
{
2014-12-09 18:09:36 +00:00
//Don't go through all of this if the actor is reflective and wants things to pass through them.
if ( BlockingMobj & & ( ( BlockingMobj - > flags2 & MF2_REFLECTIVE ) & & ( BlockingMobj - > flags7 & MF7_THRUREFLECT ) ) ) return true ;
2009-09-14 20:47:53 +00:00
if ( mo & & BlockingMobj & & ( ( mo - > BounceFlags & BOUNCE_AllActors )
2014-12-09 18:09:36 +00:00
| | ( ( mo - > flags & MF_MISSILE ) & & ( ! ( mo - > flags2 & MF2_RIP )
| | ( BlockingMobj - > flags5 & MF5_DONTRIP )
| | ( ( mo - > flags6 & MF6_NOBOSSRIP ) & & ( BlockingMobj - > flags2 & MF2_BOSS ) ) ) & & ( BlockingMobj - > flags2 & MF2_REFLECTIVE ) )
| | ( ( BlockingMobj - > player = = NULL ) & & ( ! ( BlockingMobj - > flags3 & MF3_ISMONSTER ) ) ) ) )
2009-09-14 20:47:53 +00:00
{
2012-04-07 08:21:44 +00:00
if ( mo - > bouncecount > 0 & & - - mo - > bouncecount = = 0 ) return false ;
2012-04-26 03:50:11 +00:00
2014-12-19 19:43:24 +00:00
if ( mo - > flags7 & MF7_HITTARGET ) mo - > target = BlockingMobj ;
if ( mo - > flags7 & MF7_HITMASTER ) mo - > master = BlockingMobj ;
if ( mo - > flags7 & MF7_HITTRACER ) mo - > tracer = BlockingMobj ;
2012-04-26 03:50:11 +00:00
if ( ! ontop )
{
fixed_t speed ;
2016-01-10 19:46:26 +00:00
angle_t angle = BlockingMobj - > AngleTo ( mo ) + ANGLE_1 * ( ( pr_bounce ( ) % 16 ) - 8 ) ;
2014-08-07 16:05:39 +00:00
speed = P_AproxDistance ( mo - > velx , mo - > vely ) ;
speed = FixedMul ( speed , mo - > wallbouncefactor ) ; // [GZ] was 0.75, using wallbouncefactor seems more consistent
2012-04-26 03:50:11 +00:00
mo - > angle = angle ;
angle > > = ANGLETOFINESHIFT ;
2014-08-07 16:05:39 +00:00
mo - > velx = FixedMul ( speed , finecosine [ angle ] ) ;
mo - > vely = FixedMul ( speed , finesine [ angle ] ) ;
2012-04-26 03:50:11 +00:00
mo - > PlayBounceSound ( true ) ;
2013-05-04 22:52:37 +00:00
if ( mo - > BounceFlags & BOUNCE_UseBounceState )
{
FName names [ ] = { NAME_Bounce , NAME_Actor , NAME_Creature } ;
FState * bouncestate ;
int count = 2 ;
2014-08-07 16:05:39 +00:00
2013-05-04 22:52:37 +00:00
if ( ( BlockingMobj - > flags & MF_SHOOTABLE ) & & ! ( BlockingMobj - > flags & MF_NOBLOOD ) )
{
count = 3 ;
}
bouncestate = mo - > FindState ( count , names ) ;
if ( bouncestate ! = NULL )
{
mo - > SetState ( bouncestate ) ;
}
}
2012-04-26 03:50:11 +00:00
}
else
{
fixed_t dot = mo - > velz ;
if ( mo - > BounceFlags & ( BOUNCE_HereticType | BOUNCE_MBF ) )
{
2014-08-07 16:05:39 +00:00
mo - > velz - = MulScale15 ( FRACUNIT , dot ) ;
2012-04-26 03:50:11 +00:00
if ( ! ( mo - > BounceFlags & BOUNCE_MBF ) ) // Heretic projectiles die, MBF projectiles don't.
{
mo - > flags | = MF_INBOUNCE ;
mo - > SetState ( mo - > FindState ( NAME_Death ) ) ;
mo - > flags & = ~ MF_INBOUNCE ;
return false ;
}
else
{
mo - > velz = FixedMul ( mo - > velz , mo - > bouncefactor ) ;
}
}
else // Don't run through this for MBF-style bounces
{
// The reflected velocity keeps only about 70% of its original speed
mo - > velz = FixedMul ( mo - > velz - MulScale15 ( FRACUNIT , dot ) , mo - > bouncefactor ) ;
}
mo - > PlayBounceSound ( true ) ;
if ( mo - > BounceFlags & BOUNCE_MBF ) // Bring it to rest below a certain speed
{
if ( abs ( mo - > velz ) < ( fixed_t ) ( mo - > Mass * mo - > GetGravity ( ) / 64 ) )
mo - > velz = 0 ;
}
2014-08-07 16:05:39 +00:00
else if ( mo - > BounceFlags & ( BOUNCE_AutoOff | BOUNCE_AutoOffFloorOnly ) )
2012-04-26 03:50:11 +00:00
{
2014-08-07 16:05:39 +00:00
if ( ! ( mo - > flags & MF_NOGRAVITY ) & & ( mo - > velz < 3 * FRACUNIT ) )
2012-04-26 03:50:11 +00:00
mo - > BounceFlags & = ~ BOUNCE_TypeMask ;
}
}
2012-04-07 08:21:44 +00:00
return true ;
2009-09-14 20:47:53 +00:00
}
2012-04-01 11:02:05 +00:00
return false ;
2009-09-14 20:47:53 +00:00
}
2006-02-24 04:48:15 +00:00
2006-12-16 11:49:48 +00:00
//============================================================================
2006-02-24 04:48:15 +00:00
//
2006-12-16 11:49:48 +00:00
// Aiming
2006-02-24 04:48:15 +00:00
//
2006-12-16 11:49:48 +00:00
//============================================================================
struct aim_t
{
2008-04-09 18:35:21 +00:00
fixed_t aimpitch ;
fixed_t attackrange ;
fixed_t shootz ; // Height if not aiming up or down
AActor * shootthing ;
2011-11-07 01:23:23 +00:00
AActor * friender ; // actor to check friendliness again
2006-02-24 04:48:15 +00:00
2006-12-16 11:49:48 +00:00
fixed_t toppitch , bottompitch ;
2008-04-10 14:38:43 +00:00
AActor * linetarget ;
2014-08-07 16:05:39 +00:00
AActor * thing_friend , * thing_other ;
2006-12-16 11:49:48 +00:00
angle_t pitch_friend , pitch_other ;
2010-03-28 10:51:35 +00:00
int flags ;
2009-01-04 15:00:29 +00:00
sector_t * lastsector ;
secplane_t * lastfloorplane ;
secplane_t * lastceilingplane ;
bool crossedffloors ;
bool AimTraverse3DFloors ( const divline_t & trace , intercept_t * in ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
void AimTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy , AActor * target = NULL ) ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
} ;
2006-02-24 04:48:15 +00:00
2009-01-04 15:00:29 +00:00
//============================================================================
//
// AimTraverse3DFloors
//
//============================================================================
bool aim_t : : AimTraverse3DFloors ( const divline_t & trace , intercept_t * in )
{
sector_t * nextsector ;
2014-08-07 16:05:39 +00:00
secplane_t * nexttopplane , * nextbottomplane ;
line_t * li = in - > d . line ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
nextsector = NULL ;
nexttopplane = nextbottomplane = NULL ;
2009-01-04 15:00:29 +00:00
2011-01-20 11:40:05 +00:00
if ( li - > backsector = = NULL ) return true ; // shouldn't really happen but crashed once for me...
2014-08-07 16:05:39 +00:00
if ( li - > frontsector - > e - > XFloor . ffloors . Size ( ) | | li - > backsector - > e - > XFloor . ffloors . Size ( ) )
{
2009-01-04 15:00:29 +00:00
int frontflag ;
F3DFloor * rover ;
int highpitch , lowpitch ;
2014-08-07 16:05:39 +00:00
fixed_t trX = trace . x + FixedMul ( trace . dx , in - > frac ) ;
fixed_t trY = trace . y + FixedMul ( trace . dy , in - > frac ) ;
fixed_t dist = FixedMul ( attackrange , in - > frac ) ;
2016-01-20 00:48:57 +00:00
frontflag = P_PointOnLineSide ( shootthing - > X ( ) , shootthing - > Y ( ) , li ) ;
2014-08-07 16:05:39 +00:00
2009-01-04 15:00:29 +00:00
// 3D floor check. This is not 100% accurate but normally sufficient when
// combined with a final sight check
2014-08-07 16:05:39 +00:00
for ( int i = 1 ; i < = 2 ; i + + )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
sector_t * s = i = = 1 ? li - > frontsector : li - > backsector ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
for ( unsigned k = 0 ; k < s - > e - > XFloor . ffloors . Size ( ) ; k + + )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
crossedffloors = true ;
rover = s - > e - > XFloor . ffloors [ k ] ;
if ( ( rover - > flags & FF_SHOOTTHROUGH ) | | ! ( rover - > flags & FF_EXISTS ) ) continue ;
fixed_t ff_bottom = rover - > bottom . plane - > ZatPoint ( trX , trY ) ;
fixed_t ff_top = rover - > top . plane - > ZatPoint ( trX , trY ) ;
highpitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , ff_top ) ;
lowpitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , ff_bottom ) ;
if ( highpitch < = toppitch )
2009-01-04 15:00:29 +00:00
{
// blocks completely
2014-08-07 16:05:39 +00:00
if ( lowpitch > = bottompitch ) return false ;
2009-01-04 15:00:29 +00:00
// blocks upper edge of view
2014-08-07 16:05:39 +00:00
if ( lowpitch > toppitch )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
toppitch = lowpitch ;
if ( frontflag ! = i - 1 )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
nexttopplane = rover - > bottom . plane ;
2009-01-04 15:00:29 +00:00
}
}
}
2014-08-07 16:05:39 +00:00
else if ( lowpitch > = bottompitch )
2009-01-04 15:00:29 +00:00
{
// blocks lower edge of view
2014-08-07 16:05:39 +00:00
if ( highpitch < bottompitch )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
bottompitch = highpitch ;
if ( frontflag ! = i - 1 )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
nextbottomplane = rover - > top . plane ;
2009-01-04 15:00:29 +00:00
}
}
}
// trace is leaving a sector with a 3d-floor
2014-08-07 16:05:39 +00:00
if ( frontflag = = i - 1 )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
if ( s = = lastsector )
2009-01-04 15:00:29 +00:00
{
// upper slope intersects with this 3d-floor
2014-08-07 16:05:39 +00:00
if ( rover - > bottom . plane = = lastceilingplane & & lowpitch > toppitch )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
toppitch = lowpitch ;
2009-01-04 15:00:29 +00:00
}
// lower slope intersects with this 3d-floor
2014-08-07 16:05:39 +00:00
if ( rover - > top . plane = = lastfloorplane & & highpitch < bottompitch )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
bottompitch = highpitch ;
2009-01-04 15:00:29 +00:00
}
}
}
if ( toppitch > = bottompitch ) return false ; // stop
}
}
2014-08-07 16:05:39 +00:00
}
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
lastsector = nextsector ;
lastceilingplane = nexttopplane ;
lastfloorplane = nextbottomplane ;
2009-01-04 15:00:29 +00:00
return true ;
}
2006-12-16 11:49:48 +00:00
//============================================================================
2006-02-24 04:48:15 +00:00
//
// PTR_AimTraverse
// Sets linetaget and aimpitch when a target is aimed at.
//
2006-12-16 11:49:48 +00:00
//============================================================================
2014-08-07 16:05:39 +00:00
void aim_t : : AimTraverse ( fixed_t startx , fixed_t starty , fixed_t endx , fixed_t endy , AActor * target )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
FPathTraverse it ( startx , starty , endx , endy , PT_ADDLINES | PT_ADDTHINGS | PT_COMPATIBLE ) ;
2008-04-09 18:35:21 +00:00
intercept_t * in ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
while ( ( in = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
line_t * li ;
AActor * th ;
fixed_t pitch ;
fixed_t thingtoppitch ;
fixed_t thingbottompitch ;
fixed_t dist ;
int thingpitch ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
if ( in - > isaline )
2008-04-09 18:35:21 +00:00
{
li = in - > d . line ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
if ( ! ( li - > flags & ML_TWOSIDED ) | | ( li - > flags & ML_BLOCKEVERYTHING ) )
2008-04-09 18:35:21 +00:00
return ; // stop
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
// Crosses a two sided line.
// A two sided line will restrict the possible target ranges.
FLineOpening open ;
2014-08-07 16:05:39 +00:00
P_LineOpening ( open , NULL , li , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
if ( open . bottom > = open . top )
return ; // stop
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
dist = FixedMul ( attackrange , in - > frac ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
pitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , open . bottom ) ;
2008-04-09 18:35:21 +00:00
if ( pitch < bottompitch )
bottompitch = pitch ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
pitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , open . top ) ;
2008-04-09 18:35:21 +00:00
if ( pitch > toppitch )
toppitch = pitch ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
if ( toppitch > = bottompitch )
return ; // stop
2014-08-07 16:05:39 +00:00
2009-01-04 15:00:29 +00:00
if ( ! AimTraverse3DFloors ( it . Trace ( ) , in ) ) return ;
2008-04-09 18:35:21 +00:00
continue ; // shot continues
}
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
// shoot a thing
th = in - > d . thing ;
if ( th = = shootthing )
continue ; // can't shoot self
2006-02-24 04:48:15 +00:00
2009-10-27 03:59:29 +00:00
if ( target ! = NULL & & th ! = target )
continue ; // only care about target, and you're not it
2010-03-28 10:51:35 +00:00
// If we want to start a conversation anything that has one should be
// found, regardless of other settings.
if ( ! ( flags & ALF_CHECKCONVERSATION ) | | th - > Conversation = = NULL )
2008-04-09 18:35:21 +00:00
{
2010-03-28 10:51:35 +00:00
if ( ! ( flags & ALF_CHECKNONSHOOTABLE ) ) // For info CCMD, ignore stuff about GHOST and SHOOTABLE flags
2009-09-14 19:44:14 +00:00
{
2014-08-07 16:05:39 +00:00
if ( ! ( th - > flags & MF_SHOOTABLE ) )
2010-03-28 10:51:35 +00:00
continue ; // corpse or something
// check for physical attacks on a ghost
2014-08-07 16:05:39 +00:00
if ( ( th - > flags3 & MF3_GHOST ) & &
2010-03-28 10:51:35 +00:00
shootthing - > player & & // [RH] Be sure shootthing is a player
shootthing - > player - > ReadyWeapon & &
( shootthing - > player - > ReadyWeapon - > flags2 & MF2_THRUGHOST ) )
{
continue ;
}
2009-09-14 19:44:14 +00:00
}
2008-04-09 18:35:21 +00:00
}
2014-08-07 16:05:39 +00:00
dist = FixedMul ( attackrange , in - > frac ) ;
2009-01-04 15:00:29 +00:00
2012-04-07 12:40:50 +00:00
// Don't autoaim certain special actors
if ( ! cl_doautoaim & & th - > flags6 & MF6_NOTAUTOAIMED )
{
continue ;
}
2009-01-04 15:00:29 +00:00
// we must do one last check whether the trace has crossed a 3D floor
2014-08-07 16:05:39 +00:00
if ( lastsector = = th - > Sector & & th - > Sector - > e - > XFloor . ffloors . Size ( ) )
2009-01-04 15:00:29 +00:00
{
if ( lastceilingplane )
{
2016-01-20 00:48:57 +00:00
fixed_t ff_top = lastceilingplane - > ZatPoint ( th ) ;
2014-08-07 16:05:39 +00:00
fixed_t pitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , ff_top ) ;
2009-01-04 15:00:29 +00:00
// upper slope intersects with this 3d-floor
if ( pitch > toppitch )
{
2014-08-07 16:05:39 +00:00
toppitch = pitch ;
2009-01-04 15:00:29 +00:00
}
}
if ( lastfloorplane )
{
2016-01-20 00:48:57 +00:00
fixed_t ff_bottom = lastfloorplane - > ZatPoint ( th ) ;
2014-08-07 16:05:39 +00:00
fixed_t pitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , ff_bottom ) ;
2009-01-04 15:00:29 +00:00
// lower slope intersects with this 3d-floor
if ( pitch < bottompitch )
{
2014-08-07 16:05:39 +00:00
bottompitch = pitch ;
2009-01-04 15:00:29 +00:00
}
}
}
2008-04-09 18:35:21 +00:00
// check angles to see if the thing can be aimed at
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
thingtoppitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , th - > Z ( ) + th - > height ) ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
if ( thingtoppitch > bottompitch )
continue ; // shot over the thing
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
thingbottompitch = - ( int ) R_PointToAngle2 ( 0 , shootz , dist , th - > Z ( ) ) ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
if ( thingbottompitch < toppitch )
continue ; // shot under the thing
2014-08-07 16:05:39 +00:00
2009-01-04 15:00:29 +00:00
if ( crossedffloors )
{
// if 3D floors were in the way do an extra visibility check for safety
2014-08-07 16:05:39 +00:00
if ( ! P_CheckSight ( shootthing , th , SF_IGNOREVISIBILITY | SF_IGNOREWATERBOUNDARY ) )
2009-01-04 15:00:29 +00:00
{
// the thing can't be seen so we can safely exclude its range from our aiming field
2014-08-07 16:05:39 +00:00
if ( thingtoppitch < toppitch )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
if ( thingbottompitch > toppitch ) toppitch = thingbottompitch ;
2009-01-04 15:00:29 +00:00
}
else if ( thingbottompitch > bottompitch )
{
2014-08-07 16:05:39 +00:00
if ( thingtoppitch < bottompitch ) bottompitch = thingtoppitch ;
2009-01-04 15:00:29 +00:00
}
if ( toppitch < bottompitch ) continue ;
else return ;
}
}
2008-04-09 18:35:21 +00:00
// this thing can be hit!
if ( thingtoppitch < toppitch )
thingtoppitch = toppitch ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
if ( thingbottompitch > bottompitch )
thingbottompitch = bottompitch ;
2014-08-07 16:05:39 +00:00
thingpitch = thingtoppitch / 2 + thingbottompitch / 2 ;
2010-03-28 10:51:35 +00:00
if ( flags & ALF_CHECK3D )
2008-12-30 23:27:27 +00:00
{
// We need to do a 3D distance check here because this is nearly always used in
// combination with P_LineAttack. P_LineAttack uses 3D distance but FPathTraverse
// only 2D. This causes some problems with Hexen's weapons that use different
// attack modes based on distance to target
fixed_t cosine = finecosine [ thingpitch > > ANGLETOFINESHIFT ] ;
if ( cosine ! = 0 )
{
2014-08-07 16:05:39 +00:00
fixed_t d3 = FixedDiv ( FixedMul ( P_AproxDistance ( it . Trace ( ) . dx , it . Trace ( ) . dy ) , in - > frac ) , cosine ) ;
2008-12-30 23:27:27 +00:00
if ( d3 > attackrange )
{
return ;
}
}
}
2006-02-24 04:48:15 +00:00
2011-11-07 01:23:23 +00:00
if ( ( flags & ALF_NOFRIENDS ) & & th - > IsFriend ( friender ) )
{
continue ;
}
else if ( sv_smartaim ! = 0 & & ! ( flags & ALF_FORCENOSMART ) )
2006-12-16 11:49:48 +00:00
{
2008-04-09 18:35:21 +00:00
// try to be a little smarter about what to aim at!
2011-11-07 01:23:23 +00:00
// In particular avoid autoaiming at friends and barrels.
if ( th - > IsFriend ( friender ) )
2008-04-09 18:35:21 +00:00
{
if ( sv_smartaim < 2 )
{
// friends don't aim at friends (except players), at least not first
2009-10-27 03:59:29 +00:00
thing_friend = th ;
pitch_friend = thingpitch ;
2008-04-09 18:35:21 +00:00
}
}
2011-11-07 01:23:23 +00:00
else if ( ! ( th - > flags3 & MF3_ISMONSTER ) & & th - > player = = NULL )
2006-12-16 16:34:39 +00:00
{
2008-04-09 18:35:21 +00:00
if ( sv_smartaim < 3 )
{
// don't autoaim at barrels and other shootable stuff unless no monsters have been found
2009-10-27 03:59:29 +00:00
thing_other = th ;
pitch_other = thingpitch ;
2008-04-09 18:35:21 +00:00
}
2006-12-16 16:34:39 +00:00
}
2008-04-09 18:35:21 +00:00
else
2006-12-16 16:34:39 +00:00
{
2009-10-27 03:59:29 +00:00
linetarget = th ;
aimpitch = thingpitch ;
2008-04-09 18:35:21 +00:00
return ;
2006-12-16 16:34:39 +00:00
}
2006-12-16 11:49:48 +00:00
}
else
{
2009-10-27 03:59:29 +00:00
linetarget = th ;
aimpitch = thingpitch ;
2008-04-09 18:35:21 +00:00
return ;
2006-12-16 11:49:48 +00:00
}
}
2006-02-24 04:48:15 +00:00
}
2006-12-16 11:49:48 +00:00
//============================================================================
2006-02-24 04:48:15 +00:00
//
// P_AimLineAttack
//
2006-12-16 11:49:48 +00:00
//============================================================================
2009-10-11 17:44:50 +00:00
2014-08-07 16:05:39 +00:00
fixed_t P_AimLineAttack ( AActor * t1 , angle_t angle , fixed_t distance , AActor * * pLineTarget , fixed_t vrange ,
int flags , AActor * target , AActor * friender )
2006-02-24 04:48:15 +00:00
{
fixed_t x2 ;
fixed_t y2 ;
2008-04-09 18:35:21 +00:00
aim_t aim ;
2006-02-24 04:48:15 +00:00
angle > > = ANGLETOFINESHIFT ;
2010-03-28 10:51:35 +00:00
aim . flags = flags ;
2008-04-09 18:35:21 +00:00
aim . shootthing = t1 ;
2011-11-07 01:23:23 +00:00
aim . friender = ( friender = = NULL ) ? t1 : friender ;
2006-02-24 04:48:15 +00:00
2016-01-20 00:48:57 +00:00
x2 = t1 - > X ( ) + ( distance > > FRACBITS ) * finecosine [ angle ] ;
y2 = t1 - > Y ( ) + ( distance > > FRACBITS ) * finesine [ angle ] ;
aim . shootz = t1 - > Z ( ) + ( t1 - > height > > 1 ) - t1 - > floorclip ;
2007-03-07 17:31:40 +00:00
if ( t1 - > player ! = NULL )
{
2014-08-07 16:05:39 +00:00
aim . shootz + = FixedMul ( t1 - > player - > mo - > AttackZOffset , t1 - > player - > crouchfactor ) ;
2007-03-07 17:31:40 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
aim . shootz + = 8 * FRACUNIT ;
2007-03-07 17:31:40 +00:00
}
2006-02-24 04:48:15 +00:00
// can't shoot outside view angles
if ( vrange = = 0 )
{
2007-12-11 04:03:40 +00:00
if ( t1 - > player = = NULL | | ! level . IsFreelookAllowed ( ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
vrange = ANGLE_1 * 35 ;
2006-02-24 04:48:15 +00:00
}
else
{
2009-11-24 06:55:38 +00:00
// [BB] Disable autoaim on weapons with WIF_NOAUTOAIM.
AWeapon * weapon = t1 - > player - > ReadyWeapon ;
2014-08-07 16:05:39 +00:00
if ( weapon & & ( weapon - > WeaponFlags & WIF_NOAUTOAIM ) )
2009-11-24 06:55:38 +00:00
{
2014-08-07 16:05:39 +00:00
vrange = ANGLE_1 / 2 ;
2009-11-24 06:55:38 +00:00
}
else
{
// 35 degrees is approximately what Doom used. You cannot have a
// vrange of 0 degrees, because then toppitch and bottompitch will
// be equal, and PTR_AimTraverse will never find anything to shoot at
// if it crosses a line.
2014-08-07 16:05:39 +00:00
vrange = clamp ( t1 - > player - > userinfo . GetAimDist ( ) , ANGLE_1 / 2 , ANGLE_1 * 35 ) ;
2009-11-24 06:55:38 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
2006-12-16 11:49:48 +00:00
aim . toppitch = t1 - > pitch - vrange ;
aim . bottompitch = t1 - > pitch + vrange ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
aim . attackrange = distance ;
2008-04-10 14:38:43 +00:00
aim . linetarget = NULL ;
2006-02-24 04:48:15 +00:00
2006-12-16 11:49:48 +00:00
// for smart aiming
2014-08-07 16:05:39 +00:00
aim . thing_friend = aim . thing_other = NULL ;
2006-02-24 04:48:15 +00:00
2006-12-16 11:49:48 +00:00
// Information for tracking crossed 3D floors
2014-08-07 16:05:39 +00:00
aim . aimpitch = t1 - > pitch ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
aim . crossedffloors = t1 - > Sector - > e - > XFloor . ffloors . Size ( ) ! = 0 ;
aim . lastsector = t1 - > Sector ;
aim . lastfloorplane = aim . lastceilingplane = NULL ;
2009-01-04 15:00:29 +00:00
// set initial 3d-floor info
2014-08-07 16:05:39 +00:00
for ( unsigned i = 0 ; i < t1 - > Sector - > e - > XFloor . ffloors . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
F3DFloor * rover = t1 - > Sector - > e - > XFloor . ffloors [ i ] ;
2016-01-20 00:48:57 +00:00
fixed_t bottomz = rover - > bottom . plane - > ZatPoint ( t1 ) ;
2009-01-04 15:00:29 +00:00
2016-01-20 00:48:57 +00:00
if ( bottomz > = t1 - > Top ( ) ) aim . lastceilingplane = rover - > bottom . plane ;
2009-01-04 15:00:29 +00:00
2016-01-20 00:48:57 +00:00
bottomz = rover - > top . plane - > ZatPoint ( t1 ) ;
if ( bottomz < = t1 - > Z ( ) ) aim . lastfloorplane = rover - > top . plane ;
2009-01-04 15:00:29 +00:00
}
2016-01-20 00:48:57 +00:00
aim . AimTraverse ( t1 - > X ( ) , t1 - > Y ( ) , x2 , y2 , target ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
if ( ! aim . linetarget )
2006-12-16 11:49:48 +00:00
{
if ( aim . thing_other )
{
2009-10-27 03:59:29 +00:00
aim . linetarget = aim . thing_other ;
aim . aimpitch = aim . pitch_other ;
2006-12-16 11:49:48 +00:00
}
else if ( aim . thing_friend )
{
2009-10-27 03:59:29 +00:00
aim . linetarget = aim . thing_friend ;
aim . aimpitch = aim . pitch_friend ;
2006-12-16 11:49:48 +00:00
}
}
2009-10-27 03:59:29 +00:00
if ( pLineTarget )
2010-03-28 10:51:35 +00:00
{
2009-10-27 03:59:29 +00:00
* pLineTarget = aim . linetarget ;
2010-03-28 10:51:35 +00:00
}
2008-04-10 14:38:43 +00:00
return aim . linetarget ? aim . aimpitch : t1 - > pitch ;
2006-02-24 04:48:15 +00:00
}
2006-12-16 11:49:48 +00:00
2006-02-24 04:48:15 +00:00
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2006-02-24 04:48:15 +00:00
2014-08-07 13:50:21 +00:00
struct Origin
{
AActor * Caller ;
2014-08-07 16:05:39 +00:00
bool hitGhosts ;
2014-08-08 03:40:12 +00:00
bool hitSameSpecies ;
2014-08-07 13:50:21 +00:00
} ;
2014-08-07 16:05:39 +00:00
static ETraceStatus CheckForActor ( FTraceResults & res , void * userdata )
2014-08-07 13:50:21 +00:00
{
if ( res . HitType ! = TRACE_HitActor )
{
return TRACE_Stop ;
}
Origin * data = ( Origin * ) userdata ;
2014-08-07 16:05:39 +00:00
// check for physical attacks on spectrals
if ( res . Actor - > flags4 & MF4_SPECTRAL )
2014-08-07 13:50:21 +00:00
{
return TRACE_Skip ;
}
2014-08-08 03:40:12 +00:00
if ( data - > hitSameSpecies & & res . Actor - > GetSpecies ( ) = = data - > Caller - > GetSpecies ( ) )
2006-02-24 04:48:15 +00:00
{
2013-03-23 02:54:13 +00:00
return TRACE_Skip ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
if ( data - > hitGhosts & & res . Actor - > flags3 & MF3_GHOST )
2006-02-24 04:48:15 +00:00
{
2013-03-23 02:54:13 +00:00
return TRACE_Skip ;
2006-02-24 04:48:15 +00:00
}
2013-03-23 02:54:13 +00:00
return TRACE_Stop ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
// P_LineAttack
//
// if damage == 0, it is just a test trace that will leave linetarget set
//
//==========================================================================
2014-08-07 16:05:39 +00:00
AActor * P_LineAttack ( AActor * t1 , angle_t angle , fixed_t distance ,
2014-12-21 01:11:39 +00:00
int pitch , int damage , FName damageType , PClassActor * pufftype , int flags , AActor * * victim , int * actualdamage )
2006-02-24 04:48:15 +00:00
{
fixed_t vx , vy , vz , shootz ;
FTraceResults trace ;
2014-08-07 13:50:21 +00:00
Origin TData ;
TData . Caller = t1 ;
2006-02-24 04:48:15 +00:00
angle_t srcangle = angle ;
int srcpitch = pitch ;
2007-05-26 10:50:32 +00:00
bool killPuff = false ;
AActor * puff = NULL ;
2012-05-13 07:54:44 +00:00
int pflag = 0 ;
2014-08-07 16:05:39 +00:00
int puffFlags = ( flags & LAF_ISMELEEATTACK ) ? PF_MELEERANGE : 0 ;
2013-02-27 10:35:44 +00:00
if ( flags & LAF_NORANDOMPUFFZ )
puffFlags | = PF_NORANDOMZ ;
2006-02-24 04:48:15 +00:00
2010-06-13 11:14:01 +00:00
if ( victim ! = NULL )
{
* victim = NULL ;
}
2013-06-24 14:42:43 +00:00
if ( actualdamage ! = NULL )
{
* actualdamage = 0 ;
}
2010-06-13 11:14:01 +00:00
2006-02-24 04:48:15 +00:00
angle > > = ANGLETOFINESHIFT ;
pitch = ( angle_t ) ( pitch ) > > ANGLETOFINESHIFT ;
2014-08-07 16:05:39 +00:00
vx = FixedMul ( finecosine [ pitch ] , finecosine [ angle ] ) ;
vy = FixedMul ( finecosine [ pitch ] , finesine [ angle ] ) ;
2006-02-24 04:48:15 +00:00
vz = - finesine [ pitch ] ;
2016-01-20 00:48:57 +00:00
shootz = t1 - > Z ( ) - t1 - > floorclip + ( t1 - > height > > 1 ) ;
2007-03-07 17:31:40 +00:00
if ( t1 - > player ! = NULL )
{
2014-08-07 16:05:39 +00:00
shootz + = FixedMul ( t1 - > player - > mo - > AttackZOffset , t1 - > player - > crouchfactor ) ;
2012-05-13 07:54:44 +00:00
if ( damageType = = NAME_Melee | | damageType = = NAME_Hitscan )
{
// this is coming from a weapon attack function which needs to transfer information to the obituary code,
// We need to preserve this info from the damage type because the actual damage type can get overridden by the puff
pflag = DMG_PLAYERATTACK ;
}
2007-03-07 17:31:40 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
shootz + = 8 * FRACUNIT ;
2007-03-07 17:31:40 +00:00
}
2006-02-24 04:48:15 +00:00
2009-12-16 16:09:48 +00:00
// We need to check the defaults of the replacement here
2010-08-26 20:59:15 +00:00
AActor * puffDefaults = GetDefaultByType ( pufftype - > GetReplacement ( ) ) ;
2009-10-10 12:42:57 +00:00
2014-08-07 16:05:39 +00:00
TData . hitGhosts = ( t1 - > player ! = NULL & &
2011-06-13 10:27:24 +00:00
t1 - > player - > ReadyWeapon ! = NULL & &
( t1 - > player - > ReadyWeapon - > flags2 & MF2_THRUGHOST ) ) | |
( puffDefaults & & ( puffDefaults - > flags2 & MF2_THRUGHOST ) ) ;
2014-08-08 03:40:12 +00:00
TData . hitSameSpecies = ( puffDefaults & & ( puffDefaults - > flags6 & MF6_MTHRUSPECIES ) ) ;
2014-08-07 13:50:21 +00:00
2013-05-13 02:30:55 +00:00
// if the puff uses a non-standard damage type, this will override default, hitscan and melee damage type.
2009-12-16 16:09:48 +00:00
// All other explicitly passed damage types (currenty only MDK) will be preserved.
2013-05-13 02:30:55 +00:00
if ( ( damageType = = NAME_None | | damageType = = NAME_Melee | | damageType = = NAME_Hitscan ) & &
puffDefaults ! = NULL & & puffDefaults - > DamageType ! = NAME_None )
2009-12-16 16:09:48 +00:00
{
damageType = puffDefaults - > DamageType ;
}
2009-10-10 12:42:57 +00:00
int tflags ;
if ( puffDefaults ! = NULL & & puffDefaults - > flags6 & MF6_NOTRIGGER ) tflags = TRACE_NoSky ;
2014-08-07 16:05:39 +00:00
else tflags = TRACE_NoSky | TRACE_Impact ;
2009-10-10 12:42:57 +00:00
2016-01-20 00:48:57 +00:00
if ( ! Trace ( t1 - > X ( ) , t1 - > Y ( ) , shootz , t1 - > Sector , vx , vy , vz , distance ,
2014-08-07 16:05:39 +00:00
MF_SHOOTABLE , ML_BLOCKEVERYTHING | ML_BLOCKHITSCAN , t1 , trace ,
tflags , CheckForActor , & TData ) )
2006-02-24 04:48:15 +00:00
{ // hit nothing
2009-10-10 12:42:57 +00:00
if ( puffDefaults = = NULL )
{
}
else if ( puffDefaults - > ActiveSound )
2006-02-24 04:48:15 +00:00
{ // Play miss sound
2014-08-07 16:05:39 +00:00
S_Sound ( t1 , CHAN_WEAPON , puffDefaults - > ActiveSound , 1 , ATTN_NORM ) ;
2006-02-24 04:48:15 +00:00
}
2013-05-13 02:30:55 +00:00
if ( puffDefaults ! = NULL & & puffDefaults - > flags3 & MF3_ALWAYSPUFF )
2006-02-24 04:48:15 +00:00
{ // Spawn the puff anyway
2014-08-07 16:05:39 +00:00
puff = P_SpawnPuff ( t1 , pufftype , trace . X , trace . Y , trace . Z , angle - ANG180 , 2 , puffFlags ) ;
2006-02-24 04:48:15 +00:00
}
else
{
2008-02-12 05:54:03 +00:00
return NULL ;
2006-02-24 04:48:15 +00:00
}
}
else
{
fixed_t hitx = 0 , hity = 0 , hitz = 0 ;
if ( trace . HitType ! = TRACE_HitActor )
{
// position a bit closer for puffs
if ( trace . HitType ! = TRACE_HitWall | | trace . Line - > special ! = Line_Horizon )
{
2014-08-07 16:05:39 +00:00
fixed_t closer = trace . Distance - 4 * FRACUNIT ;
2016-01-20 00:48:57 +00:00
fixedvec2 pos = t1 - > Vec2Offset ( FixedMul ( vx , closer ) , FixedMul ( vy , closer ) ) ;
puff = P_SpawnPuff ( t1 , pufftype , pos . x , pos . y ,
2014-08-07 16:05:39 +00:00
shootz + FixedMul ( vz , closer ) , angle - ANG90 , 0 , puffFlags ) ;
2006-02-24 04:48:15 +00:00
}
// [RH] Spawn a decal
2015-01-20 01:35:00 +00:00
if ( trace . HitType = = TRACE_HitWall & & trace . Line - > special ! = Line_Horizon & & ! ( flags & LAF_NOIMPACTDECAL ) & & ! ( puffDefaults - > flags7 & MF7_NODECAL ) )
2014-08-07 16:05:39 +00:00
{
2008-03-24 22:06:26 +00:00
// [TN] If the actor or weapon has a decal defined, use that one.
2014-08-07 16:05:39 +00:00
if ( t1 - > DecalGenerator ! = NULL | |
2008-03-24 22:06:26 +00:00
( t1 - > player ! = NULL & & t1 - > player - > ReadyWeapon ! = NULL & & t1 - > player - > ReadyWeapon - > DecalGenerator ! = NULL ) )
{
2015-01-20 01:35:00 +00:00
// [ZK] If puff has FORCEDECAL set, do not use the weapon's decal
if ( puffDefaults - > flags7 & MF7_FORCEDECAL & & puff ! = NULL & & puff - > DecalGenerator )
SpawnShootDecal ( puff , trace ) ;
else
SpawnShootDecal ( t1 , trace ) ;
2008-03-24 22:06:26 +00:00
}
// Else, look if the bulletpuff has a decal defined.
2014-08-07 16:05:39 +00:00
else if ( puff ! = NULL & & puff - > DecalGenerator )
2008-03-24 22:06:26 +00:00
{
2014-08-07 16:05:39 +00:00
SpawnShootDecal ( puff , trace ) ;
}
2008-03-24 22:06:26 +00:00
else
{
2014-08-07 16:05:39 +00:00
SpawnShootDecal ( t1 , trace ) ;
}
2006-02-24 04:48:15 +00:00
}
else if ( puff ! = NULL & &
trace . CrossedWater = = NULL & &
trace . Sector - > heightsec = = NULL & &
trace . HitType = = TRACE_HitFloor )
{
2009-01-04 12:25:22 +00:00
// Using the puff's position is not accurate enough.
// Instead make it splash at the actual hit position
2016-01-20 00:48:57 +00:00
hitx = t1 - > X ( ) + FixedMul ( vx , trace . Distance ) ;
hity = t1 - > Y ( ) + FixedMul ( vy , trace . Distance ) ;
2014-08-07 16:05:39 +00:00
hitz = shootz + FixedMul ( vz , trace . Distance ) ;
P_HitWater ( puff , P_PointInSector ( hitx , hity ) , hitx , hity , hitz ) ;
2006-02-24 04:48:15 +00:00
}
}
else
{
2006-05-13 12:41:15 +00:00
bool bloodsplatter = ( t1 - > flags5 & MF5_BLOODSPLATTER ) | |
2014-08-07 16:05:39 +00:00
( t1 - > player ! = NULL & & t1 - > player - > ReadyWeapon ! = NULL & &
( t1 - > player - > ReadyWeapon - > WeaponFlags & WIF_AXEBLOOD ) ) ;
2006-02-24 04:48:15 +00:00
2006-05-13 12:41:15 +00:00
bool axeBlood = ( t1 - > player ! = NULL & &
2006-02-24 04:48:15 +00:00
t1 - > player - > ReadyWeapon ! = NULL & &
( t1 - > player - > ReadyWeapon - > WeaponFlags & WIF_AXEBLOOD ) ) ;
// Hit a thing, so it could be either a puff or blood
2010-05-28 21:07:45 +00:00
fixed_t dist = trace . Distance ;
// position a bit closer for puffs/blood if using compatibility mode.
2014-08-07 16:05:39 +00:00
if ( i_compatflags & COMPATF_HITSCAN ) dist - = 10 * FRACUNIT ;
2016-01-20 12:48:05 +00:00
hitx = t1 - > X ( ) + FixedMul ( vx , dist ) ;
hity = t1 - > Y ( ) + FixedMul ( vy , dist ) ;
2014-08-07 16:05:39 +00:00
hitz = shootz + FixedMul ( vz , dist ) ;
2006-02-24 04:48:15 +00:00
2014-12-16 18:37:31 +00:00
2006-02-24 04:48:15 +00:00
// Spawn bullet puffs or blood spots, depending on target type.
2013-05-26 20:56:20 +00:00
if ( ( puffDefaults ! = NULL & & puffDefaults - > flags3 & MF3_PUFFONACTORS ) | |
2006-02-24 04:48:15 +00:00
( trace . Actor - > flags & MF_NOBLOOD ) | |
2014-08-07 16:05:39 +00:00
( trace . Actor - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) )
2006-02-24 04:48:15 +00:00
{
2011-06-13 10:34:46 +00:00
if ( ! ( trace . Actor - > flags & MF_NOBLOOD ) )
2013-02-27 10:35:44 +00:00
puffFlags | = PF_HITTHINGBLEED ;
2011-06-13 10:34:46 +00:00
2009-12-16 16:09:48 +00:00
// We must pass the unreplaced puff type here
2014-12-27 20:15:14 +00:00
puff = P_SpawnPuff ( t1 , pufftype , hitx , hity , hitz , angle - ANG180 , 2 , puffFlags | PF_HITTHING , trace . Actor ) ;
2014-12-16 18:37:31 +00:00
}
2013-01-02 04:39:59 +00:00
// Allow puffs to inflict poison damage, so that hitscans can poison, too.
2013-05-26 20:56:20 +00:00
if ( puffDefaults ! = NULL & & puffDefaults - > PoisonDamage > 0 & & puffDefaults - > PoisonDuration ! = INT_MIN )
2013-01-02 04:39:59 +00:00
{
P_PoisonMobj ( trace . Actor , puff ? puff : t1 , t1 , puffDefaults - > PoisonDamage , puffDefaults - > PoisonDuration , puffDefaults - > PoisonPeriod , puffDefaults - > PoisonDamageType ) ;
}
// [GZ] If MF6_FORCEPAIN is set, we need to call P_DamageMobj even if damage is 0!
// Note: The puff may not yet be spawned here so we must check the class defaults, not the actor.
int newdam = damage ;
2014-11-12 20:08:26 +00:00
if ( damage | | ( puffDefaults ! = NULL & & ( ( puffDefaults - > flags6 & MF6_FORCEPAIN ) | | ( puffDefaults - > flags7 & MF7_CAUSEPAIN ) ) ) )
2013-01-02 04:39:59 +00:00
{
int dmgflags = DMG_INFLICTOR_IS_PUFF | pflag ;
// Allow MF5_PIERCEARMOR on a weapon as well.
if ( t1 - > player ! = NULL & & ( dmgflags & DMG_PLAYERATTACK ) & & t1 - > player - > ReadyWeapon ! = NULL & &
t1 - > player - > ReadyWeapon - > flags5 & MF5_PIERCEARMOR )
{
dmgflags | = DMG_NO_ARMOR ;
}
2014-12-16 18:37:31 +00:00
2013-01-02 04:39:59 +00:00
if ( puff = = NULL )
2014-08-07 16:05:39 +00:00
{
2013-01-02 04:39:59 +00:00
// Since the puff is the damage inflictor we need it here
// regardless of whether it is displayed or not.
2014-08-07 16:05:39 +00:00
puff = P_SpawnPuff ( t1 , pufftype , hitx , hity , hitz , angle - ANG180 , 2 , puffFlags | PF_HITTHING | PF_TEMPORARY ) ;
2013-01-02 04:39:59 +00:00
killPuff = true ;
}
2014-08-07 16:05:39 +00:00
newdam = P_DamageMobj ( trace . Actor , puff ? puff : t1 , t1 , damage , damageType , dmgflags ) ;
2013-06-24 14:42:43 +00:00
if ( actualdamage ! = NULL )
{
* actualdamage = newdam ;
}
2013-01-02 04:39:59 +00:00
}
2013-05-26 20:56:20 +00:00
if ( ! ( puffDefaults ! = NULL & & puffDefaults - > flags3 & MF3_BLOODLESSIMPACT ) )
2006-02-24 04:48:15 +00:00
{
2006-05-13 12:41:15 +00:00
if ( ! bloodsplatter & & ! axeBlood & &
2006-04-11 08:36:23 +00:00
! ( trace . Actor - > flags & MF_NOBLOOD ) & &
2014-08-07 16:05:39 +00:00
! ( trace . Actor - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_SpawnBlood ( hitx , hity , hitz , angle - ANG180 , newdam > 0 ? newdam : damage , trace . Actor ) ;
2006-04-11 08:36:23 +00:00
}
2014-08-07 16:05:39 +00:00
2006-04-11 08:36:23 +00:00
if ( damage )
{
2006-05-13 12:41:15 +00:00
if ( bloodsplatter | | axeBlood )
2006-02-24 04:48:15 +00:00
{
2006-04-11 08:36:23 +00:00
if ( ! ( trace . Actor - > flags & MF_NOBLOOD ) & &
2014-08-07 16:05:39 +00:00
! ( trace . Actor - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) )
2006-02-24 04:48:15 +00:00
{
2006-04-11 08:36:23 +00:00
if ( axeBlood )
{
2014-08-07 16:05:39 +00:00
P_BloodSplatter2 ( hitx , hity , hitz , trace . Actor ) ;
2006-04-11 08:36:23 +00:00
}
if ( pr_lineattack ( ) < 192 )
{
2014-08-07 16:05:39 +00:00
P_BloodSplatter ( hitx , hity , hitz , trace . Actor ) ;
2006-04-11 08:36:23 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
2006-04-11 08:36:23 +00:00
// [RH] Stick blood to walls
2014-08-07 16:05:39 +00:00
P_TraceBleed ( newdam > 0 ? newdam : damage , trace . X , trace . Y , trace . Z ,
2006-04-11 08:36:23 +00:00
trace . Actor , srcangle , srcpitch ) ;
2006-02-24 04:48:15 +00:00
}
}
2010-06-13 11:14:01 +00:00
if ( victim ! = NULL )
{
* victim = trace . Actor ;
}
2006-02-24 04:48:15 +00:00
}
2011-09-21 19:39:12 +00:00
if ( trace . Crossed3DWater | | trace . CrossedWater )
2006-02-24 04:48:15 +00:00
{
if ( puff = = NULL )
{ // Spawn puff just to get a mass for the splash
2014-08-07 16:05:39 +00:00
puff = P_SpawnPuff ( t1 , pufftype , hitx , hity , hitz , angle - ANG180 , 2 , puffFlags | PF_HITTHING | PF_TEMPORARY ) ;
2006-02-24 04:48:15 +00:00
killPuff = true ;
}
2014-08-07 16:05:39 +00:00
SpawnDeepSplash ( t1 , trace , puff , vx , vy , vz , shootz , trace . Crossed3DWater ! = NULL ) ;
2006-02-24 04:48:15 +00:00
}
}
2007-05-26 10:50:32 +00:00
if ( killPuff & & puff ! = NULL )
{
puff - > Destroy ( ) ;
2008-02-12 05:54:03 +00:00
puff = NULL ;
2007-05-26 10:50:32 +00:00
}
2008-02-12 05:54:03 +00:00
return puff ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
AActor * P_LineAttack ( AActor * t1 , angle_t angle , fixed_t distance ,
int pitch , int damage , FName damageType , FName pufftype , int flags , AActor * * victim , int * actualdamage )
2006-11-04 13:06:42 +00:00
{
2010-03-24 02:49:37 +00:00
PClassActor * type = PClass : : FindActor ( pufftype ) ;
2010-06-13 11:14:01 +00:00
if ( victim ! = NULL )
{
* victim = NULL ;
}
2006-11-04 13:06:42 +00:00
if ( type = = NULL )
{
Printf ( " Attempt to spawn unknown actor type '%s' \n " , pufftype . GetChars ( ) ) ;
}
else
{
2013-06-24 14:42:43 +00:00
return P_LineAttack ( t1 , angle , distance , pitch , damage , damageType , type , flags , victim , actualdamage ) ;
2006-11-04 13:06:42 +00:00
}
2008-02-12 05:54:03 +00:00
return NULL ;
2006-11-04 13:06:42 +00:00
}
2014-09-21 14:43:17 +00:00
//==========================================================================
//
// P_LinePickActor
//
//==========================================================================
AActor * P_LinePickActor ( AActor * t1 , angle_t angle , fixed_t distance , int pitch ,
2015-10-12 22:30:06 +00:00
ActorFlags actorMask , DWORD wallMask )
2014-09-21 14:43:17 +00:00
{
fixed_t vx , vy , vz , shootz ;
angle > > = ANGLETOFINESHIFT ;
pitch = ( angle_t ) ( pitch ) > > ANGLETOFINESHIFT ;
vx = FixedMul ( finecosine [ pitch ] , finecosine [ angle ] ) ;
vy = FixedMul ( finecosine [ pitch ] , finesine [ angle ] ) ;
vz = - finesine [ pitch ] ;
2016-01-20 12:48:05 +00:00
shootz = t1 - > Z ( ) - t1 - > floorclip + ( t1 - > height > > 1 ) ;
2014-09-21 14:43:17 +00:00
if ( t1 - > player ! = NULL )
{
shootz + = FixedMul ( t1 - > player - > mo - > AttackZOffset , t1 - > player - > crouchfactor ) ;
}
else
{
shootz + = 8 * FRACUNIT ;
}
FTraceResults trace ;
Origin TData ;
TData . Caller = t1 ;
TData . hitGhosts = true ;
2016-01-20 12:48:05 +00:00
if ( Trace ( t1 - > X ( ) , t1 - > Y ( ) , shootz , t1 - > Sector , vx , vy , vz , distance ,
2014-09-21 14:43:17 +00:00
actorMask , wallMask , t1 , trace , TRACE_NoSky , CheckForActor , & TData ) )
{
if ( trace . HitType = = TRACE_HitActor )
{
return trace . Actor ;
}
}
return NULL ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2014-08-07 16:05:39 +00:00
void P_TraceBleed ( int damage , fixed_t x , fixed_t y , fixed_t z , AActor * actor , angle_t angle , int pitch )
2006-02-24 04:48:15 +00:00
{
if ( ! cl_bloodsplats )
return ;
const char * bloodType = " BloodSplat " ;
int count ;
int noise ;
2012-04-11 04:44:12 +00:00
2006-02-24 04:48:15 +00:00
if ( ( actor - > flags & MF_NOBLOOD ) | |
2006-12-24 12:51:24 +00:00
( actor - > flags5 & MF5_NOBLOODDECALS ) | |
2014-08-07 16:05:39 +00:00
( actor - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) | |
2006-02-24 04:48:15 +00:00
( actor - > player & & actor - > player - > cheats & CF_GODMODE ) )
{
return ;
}
if ( damage < 15 )
{ // For low damages, there is a chance to not spray blood at all
if ( damage < = 10 )
{
if ( pr_tracebleed ( ) < 160 )
{
return ;
}
}
count = 1 ;
noise = 18 ;
}
else if ( damage < 25 )
{
count = 2 ;
noise = 19 ;
}
else
{ // For high damages, there is a chance to spray just one big glob of blood
if ( pr_tracebleed ( ) < 24 )
{
bloodType = " BloodSmear " ;
count = 1 ;
noise = 20 ;
}
else
{
count = 3 ;
noise = 20 ;
}
}
for ( ; count ; - - count )
{
FTraceResults bleedtrace ;
2014-08-07 16:05:39 +00:00
angle_t bleedang = ( angle + ( ( pr_tracebleed ( ) - 128 ) < < noise ) ) > > ANGLETOFINESHIFT ;
angle_t bleedpitch = ( angle_t ) ( pitch + ( ( pr_tracebleed ( ) - 128 ) < < noise ) ) > > ANGLETOFINESHIFT ;
fixed_t vx = FixedMul ( finecosine [ bleedpitch ] , finecosine [ bleedang ] ) ;
fixed_t vy = FixedMul ( finecosine [ bleedpitch ] , finesine [ bleedang ] ) ;
2006-02-24 04:48:15 +00:00
fixed_t vz = - finesine [ bleedpitch ] ;
2014-08-07 16:05:39 +00:00
if ( Trace ( x , y , z , actor - > Sector ,
vx , vy , vz , 172 * FRACUNIT , 0 , ML_BLOCKEVERYTHING , actor ,
bleedtrace , TRACE_NoSky ) )
2006-02-24 04:48:15 +00:00
{
if ( bleedtrace . HitType = = TRACE_HitWall )
{
2010-03-21 08:09:45 +00:00
PalEntry bloodcolor = actor - > GetBloodColor ( ) ;
2006-02-24 04:48:15 +00:00
if ( bloodcolor ! = 0 )
{
2014-08-07 16:05:39 +00:00
bloodcolor . r > > = 1 ; // the full color is too bright for blood decals
bloodcolor . g > > = 1 ;
bloodcolor . b > > = 1 ;
bloodcolor . a = 1 ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
DImpactDecal : : StaticCreate ( bloodType ,
2006-02-24 04:48:15 +00:00
bleedtrace . X , bleedtrace . Y , bleedtrace . Z ,
2009-09-06 20:45:56 +00:00
bleedtrace . Line - > sidedef [ bleedtrace . Side ] ,
2009-01-04 15:00:29 +00:00
bleedtrace . ffloor ,
2006-02-24 04:48:15 +00:00
bloodcolor ) ;
}
}
}
}
2014-08-07 16:05:39 +00:00
void P_TraceBleed ( int damage , AActor * target , angle_t angle , int pitch )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
P_TraceBleed ( damage , target - > X ( ) , target - > Y ( ) , target - > Z ( ) + target - > height / 2 ,
2006-02-24 04:48:15 +00:00
target , angle , pitch ) ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2014-08-07 16:05:39 +00:00
void P_TraceBleed ( int damage , AActor * target , AActor * missile )
2006-02-24 04:48:15 +00:00
{
int pitch ;
if ( target = = NULL | | missile - > flags3 & MF3_BLOODLESSIMPACT )
{
return ;
}
2009-06-30 20:57:51 +00:00
if ( missile - > velz ! = 0 )
2006-02-24 04:48:15 +00:00
{
double aim ;
2016-01-20 12:48:05 +00:00
aim = atan ( ( double ) missile - > velz / ( double ) target - > AproxDistance ( missile ) ) ;
2014-08-07 16:05:39 +00:00
pitch = - ( int ) ( aim * ANGLE_180 / PI ) ;
2006-02-24 04:48:15 +00:00
}
else
{
pitch = 0 ;
}
2016-01-20 12:48:05 +00:00
P_TraceBleed ( damage , target - > X ( ) , target - > Y ( ) , target - > Z ( ) + target - > height / 2 ,
2016-01-10 19:46:26 +00:00
target , missile - > AngleTo ( target ) ,
2006-02-24 04:48:15 +00:00
pitch ) ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2014-08-07 16:05:39 +00:00
void P_TraceBleed ( int damage , AActor * target )
2006-02-24 04:48:15 +00:00
{
2009-05-11 21:05:40 +00:00
if ( target ! = NULL )
{
fixed_t one = pr_tracebleed ( ) < < 24 ;
2014-08-07 16:05:39 +00:00
fixed_t two = ( pr_tracebleed ( ) - 128 ) < < 16 ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
P_TraceBleed ( damage , target - > X ( ) , target - > Y ( ) , target - > Z ( ) + target - > height / 2 ,
2009-05-11 21:05:40 +00:00
target , one , two ) ;
}
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// [RH] Rail gun stuffage
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
struct SRailHit
{
AActor * HitActor ;
fixed_t Distance ;
} ;
2013-03-23 03:05:01 +00:00
struct RailData
{
2015-03-02 18:01:40 +00:00
AActor * Caller ;
2013-03-23 03:05:01 +00:00
TArray < SRailHit > RailHits ;
bool StopAtOne ;
2013-04-28 03:45:19 +00:00
bool StopAtInvul ;
2015-03-02 18:01:40 +00:00
bool ThruSpecies ;
2013-03-23 03:05:01 +00:00
} ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
static ETraceStatus ProcessRailHit ( FTraceResults & res , void * userdata )
2006-02-24 04:48:15 +00:00
{
2013-03-23 03:05:01 +00:00
RailData * data = ( RailData * ) userdata ;
2006-02-24 04:48:15 +00:00
if ( res . HitType ! = TRACE_HitActor )
{
2013-03-23 02:54:13 +00:00
return TRACE_Stop ;
2006-02-24 04:48:15 +00:00
}
// Invulnerable things completely block the shot
2013-04-28 03:45:19 +00:00
if ( data - > StopAtInvul & & res . Actor - > flags2 & MF2_INVULNERABLE )
2006-02-24 04:48:15 +00:00
{
2013-03-23 02:54:13 +00:00
return TRACE_Stop ;
2006-02-24 04:48:15 +00:00
}
2015-03-02 18:01:40 +00:00
// Skip actors with the same species if the puff has MTHRUSPECIES.
if ( data - > ThruSpecies & & res . Actor - > GetSpecies ( ) = = data - > Caller - > GetSpecies ( ) )
{
return TRACE_Skip ;
}
2006-02-24 04:48:15 +00:00
// Save this thing for damaging later, and continue the trace
SRailHit newhit ;
newhit . HitActor = res . Actor ;
2014-08-07 16:05:39 +00:00
newhit . Distance = res . Distance - 10 * FRACUNIT ; // put blood in front
data - > RailHits . Push ( newhit ) ;
2006-02-24 04:48:15 +00:00
2013-03-23 03:05:01 +00:00
return data - > StopAtOne ? TRACE_Stop : TRACE_Continue ;
2009-07-04 18:17:44 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2015-08-01 07:42:31 +00:00
void P_RailAttack ( AActor * source , int damage , int offset_xy , fixed_t offset_z , int color1 , int color2 , double maxdiff , int railflags , PClassActor * puffclass , angle_t angleoffset , angle_t pitchoffset , fixed_t distance , int duration , double sparsity , double drift , PClassActor * spawnclass , int SpiralOffset )
2006-02-24 04:48:15 +00:00
{
fixed_t vx , vy , vz ;
angle_t angle , pitch ;
2015-06-25 01:57:36 +00:00
TVector3 < double > start , end ;
2006-02-24 04:48:15 +00:00
FTraceResults trace ;
2008-04-09 18:35:21 +00:00
fixed_t shootz ;
2006-02-24 04:48:15 +00:00
2010-03-24 02:49:37 +00:00
if ( puffclass = = NULL )
{
puffclass = PClass : : FindActor ( NAME_BulletPuff ) ;
}
2009-01-04 12:25:22 +00:00
2010-07-24 06:27:13 +00:00
pitch = ( ( angle_t ) ( - source - > pitch ) + pitchoffset ) > > ANGLETOFINESHIFT ;
angle = ( source - > angle + angleoffset ) > > ANGLETOFINESHIFT ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
vx = FixedMul ( finecosine [ pitch ] , finecosine [ angle ] ) ;
vy = FixedMul ( finecosine [ pitch ] , finesine [ angle ] ) ;
2006-02-24 04:48:15 +00:00
vz = finesine [ pitch ] ;
2016-01-20 12:48:05 +00:00
shootz = source - > Z ( ) - source - > floorclip + ( source - > height > > 1 ) + offset_z ;
2008-01-30 18:45:15 +00:00
2012-06-22 03:56:08 +00:00
if ( ! ( railflags & RAF_CENTERZ ) )
2008-01-30 18:45:15 +00:00
{
2012-06-22 03:56:08 +00:00
if ( source - > player ! = NULL )
{
2014-08-07 16:05:39 +00:00
shootz + = FixedMul ( source - > player - > mo - > AttackZOffset , source - > player - > crouchfactor ) ;
2012-06-22 03:56:08 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
shootz + = 8 * FRACUNIT ;
2012-06-22 03:56:08 +00:00
}
2008-01-30 18:45:15 +00:00
}
2006-02-24 04:48:15 +00:00
2010-07-24 06:27:13 +00:00
angle = ( ( source - > angle + angleoffset ) - ANG90 ) > > ANGLETOFINESHIFT ;
2016-01-20 12:48:05 +00:00
fixedvec2 xy = source - > Vec2Offset ( offset_xy * finecosine [ angle ] , offset_xy * finesine [ angle ] ) ;
2006-02-24 04:48:15 +00:00
2013-03-23 03:05:01 +00:00
RailData rail_data ;
2015-03-02 18:01:40 +00:00
rail_data . Caller = source ;
2013-03-23 03:05:01 +00:00
rail_data . StopAtOne = ! ! ( railflags & RAF_NOPIERCE ) ;
2016-02-11 20:33:30 +00:00
start . X = FIXED2DBL ( xy . x ) ;
start . Y = FIXED2DBL ( xy . y ) ;
start . Z = FIXED2DBL ( shootz ) ;
2006-02-24 04:48:15 +00:00
2009-10-10 12:42:57 +00:00
int flags ;
2013-04-28 03:45:19 +00:00
assert ( puffclass ! = NULL ) ; // Because we set it to a default above
2014-12-16 18:37:31 +00:00
AActor * puffDefaults = GetDefaultByType ( puffclass - > GetReplacement ( ) ) ; //Contains all the flags such as FOILINVUL, etc.
2009-10-10 12:42:57 +00:00
2014-08-07 16:05:39 +00:00
flags = ( puffDefaults - > flags6 & MF6_NOTRIGGER ) ? 0 : TRACE_PCross | TRACE_Impact ;
2013-04-28 03:45:19 +00:00
rail_data . StopAtInvul = ( puffDefaults - > flags3 & MF3_FOILINVUL ) ? false : true ;
2015-03-02 18:01:40 +00:00
rail_data . ThruSpecies = ( puffDefaults - > flags6 & MF6_MTHRUSPECIES ) ? true : false ;
2016-01-20 12:48:05 +00:00
Trace ( xy . x , xy . y , shootz , source - > Sector , vx , vy , vz ,
2012-06-22 03:56:08 +00:00
distance , MF_SHOOTABLE , ML_BLOCKEVERYTHING , source , trace ,
2013-03-23 03:05:01 +00:00
flags , ProcessRailHit , & rail_data ) ;
2006-02-24 04:48:15 +00:00
2009-03-20 01:40:28 +00:00
// Hurt anything the trace hit
unsigned int i ;
FName damagetype = ( puffDefaults = = NULL | | puffDefaults - > DamageType = = NAME_None ) ? FName ( NAME_Railgun ) : puffDefaults - > DamageType ;
2009-08-30 10:43:51 +00:00
// used as damage inflictor
AActor * thepuff = NULL ;
2016-01-20 12:48:05 +00:00
if ( puffclass ! = NULL ) thepuff = Spawn ( puffclass , source - > Pos ( ) , ALLOW_REPLACE ) ;
2014-08-07 16:05:39 +00:00
for ( i = 0 ; i < rail_data . RailHits . Size ( ) ; i + + )
2009-03-20 01:40:28 +00:00
{
2015-03-02 18:01:40 +00:00
2009-03-20 01:40:28 +00:00
fixed_t x , y , z ;
2009-10-16 16:04:19 +00:00
bool spawnpuff ;
2013-01-02 04:39:59 +00:00
bool bleed = false ;
2013-02-27 10:35:44 +00:00
2012-04-11 04:50:23 +00:00
int puffflags = PF_HITTHING ;
2013-03-23 03:05:01 +00:00
AActor * hitactor = rail_data . RailHits [ i ] . HitActor ;
2015-03-02 18:01:40 +00:00
fixed_t hitdist = rail_data . RailHits [ i ] . Distance ;
2009-03-20 01:40:28 +00:00
2016-01-20 12:48:05 +00:00
x = xy . x + FixedMul ( hitdist , vx ) ;
y = xy . y + FixedMul ( hitdist , vy ) ;
2013-03-23 03:05:01 +00:00
z = shootz + FixedMul ( hitdist , vz ) ;
2009-03-20 01:40:28 +00:00
2013-03-23 03:05:01 +00:00
if ( ( hitactor - > flags & MF_NOBLOOD ) | |
2014-11-06 05:16:01 +00:00
( hitactor - > flags2 & MF2_DORMANT | | ( ( hitactor - > flags2 & MF2_INVULNERABLE ) & & ! ( puffDefaults - > flags3 & MF3_FOILINVUL ) ) ) )
2009-03-20 01:40:28 +00:00
{
2012-04-07 12:57:44 +00:00
spawnpuff = ( puffclass ! = NULL ) ;
2009-03-20 01:40:28 +00:00
}
else
{
2009-10-16 16:04:19 +00:00
spawnpuff = ( puffclass ! = NULL & & puffDefaults - > flags3 & MF3_ALWAYSPUFF ) ;
2012-04-11 04:50:23 +00:00
puffflags | = PF_HITTHINGBLEED ; // [XA] Allow for puffs to jump to XDeath state.
2014-08-07 16:05:39 +00:00
if ( ! ( puffDefaults - > flags3 & MF3_BLOODLESSIMPACT ) )
2012-04-15 07:59:50 +00:00
{
2013-01-02 04:39:59 +00:00
bleed = true ;
2012-04-15 07:59:50 +00:00
}
2009-03-20 01:40:28 +00:00
}
2012-04-11 04:50:23 +00:00
if ( spawnpuff )
2013-03-23 03:05:01 +00:00
{
2014-12-27 20:15:14 +00:00
P_SpawnPuff ( source , puffclass , x , y , z , ( source - > angle + angleoffset ) - ANG90 , 1 , puffflags , hitactor ) ;
2014-12-16 18:37:31 +00:00
}
2014-12-27 20:15:14 +00:00
2015-03-27 09:10:22 +00:00
int dmgFlagPass = DMG_INFLICTOR_IS_PUFF ;
if ( puffDefaults ! = NULL ) // is this even possible?
2013-03-23 03:05:01 +00:00
{
2015-03-27 09:10:22 +00:00
if ( puffDefaults - > PoisonDamage > 0 & & puffDefaults - > PoisonDuration ! = INT_MIN )
{
P_PoisonMobj ( hitactor , thepuff ? thepuff : source , source , puffDefaults - > PoisonDamage , puffDefaults - > PoisonDuration , puffDefaults - > PoisonPeriod , puffDefaults - > PoisonDamageType ) ;
}
if ( puffDefaults - > flags3 & MF3_FOILINVUL ) dmgFlagPass | = DMG_FOILINVUL ;
if ( puffDefaults - > flags7 & MF7_FOILBUDDHA ) dmgFlagPass | = DMG_FOILBUDDHA ;
2013-03-23 03:05:01 +00:00
}
2014-11-06 05:16:01 +00:00
int newdam = P_DamageMobj ( hitactor , thepuff ? thepuff : source , source , damage , damagetype , dmgFlagPass ) ;
2014-12-16 18:37:31 +00:00
2013-01-02 04:39:59 +00:00
if ( bleed )
{
2013-03-23 03:05:01 +00:00
P_SpawnBlood ( x , y , z , ( source - > angle + angleoffset ) - ANG180 , newdam > 0 ? newdam : damage , hitactor ) ;
P_TraceBleed ( newdam > 0 ? newdam : damage , x , y , z , hitactor , source - > angle , pitch ) ;
2013-01-02 04:39:59 +00:00
}
2009-03-20 01:40:28 +00:00
}
// Spawn a decal or puff at the point where the trace ended.
2006-02-24 04:48:15 +00:00
if ( trace . HitType = = TRACE_HitWall )
{
2015-06-25 00:53:46 +00:00
AActor * puff = NULL ;
2014-08-07 16:05:39 +00:00
if ( puffclass ! = NULL & & puffDefaults - > flags3 & MF3_ALWAYSPUFF )
2009-10-16 16:04:19 +00:00
{
2015-06-25 00:53:46 +00:00
puff = P_SpawnPuff ( source , puffclass , trace . X , trace . Y , trace . Z , ( source - > angle + angleoffset ) - ANG90 , 1 , 0 ) ;
2016-01-16 22:36:42 +00:00
if ( puff & & ( trace . Line ! = NULL ) & & ( trace . Line - > special = = Line_Horizon ) & & ! ( puff - > flags3 & MF3_SKYEXPLODE ) )
2016-01-16 22:33:19 +00:00
puff - > Destroy ( ) ;
2009-10-16 16:04:19 +00:00
}
2015-06-25 00:53:46 +00:00
if ( puff ! = NULL & & puffDefaults - > flags7 & MF7_FORCEDECAL & & puff - > DecalGenerator )
SpawnShootDecal ( puff , trace ) ;
else
SpawnShootDecal ( source , trace ) ;
2009-10-16 16:04:19 +00:00
2006-02-24 04:48:15 +00:00
}
2016-01-14 17:21:08 +00:00
if ( trace . HitType = = TRACE_HitFloor | | trace . HitType = = TRACE_HitCeiling )
{
AActor * puff = NULL ;
if ( puffclass ! = NULL & & puffDefaults - > flags3 & MF3_ALWAYSPUFF )
{
puff = P_SpawnPuff ( source , puffclass , trace . X , trace . Y , trace . Z , ( source - > angle + angleoffset ) - ANG90 , 1 , 0 ) ;
2016-01-16 22:33:19 +00:00
if ( puff & & ! ( puff - > flags3 & MF3_SKYEXPLODE ) & &
( ( ( trace . HitType = = TRACE_HitFloor ) & & ( puff - > floorpic = = skyflatnum ) ) | |
( ( trace . HitType = = TRACE_HitCeiling ) & & ( puff - > ceilingpic = = skyflatnum ) ) ) )
{
puff - > Destroy ( ) ;
}
2016-01-14 17:21:08 +00:00
}
}
2014-08-07 16:05:39 +00:00
if ( thepuff ! = NULL )
2006-02-24 04:48:15 +00:00
{
2013-05-20 20:47:55 +00:00
if ( trace . HitType = = TRACE_HitFloor & &
trace . CrossedWater = = NULL & &
trace . Sector - > heightsec = = NULL )
2009-01-04 12:25:22 +00:00
{
2016-01-20 12:48:05 +00:00
thepuff - > SetOrigin ( trace . X , trace . Y , trace . Z , false ) ;
2014-08-07 16:05:39 +00:00
P_HitWater ( thepuff , trace . Sector ) ;
2009-01-04 12:25:22 +00:00
}
2013-05-20 20:47:55 +00:00
if ( trace . Crossed3DWater | | trace . CrossedWater )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
SpawnDeepSplash ( source , trace , thepuff , vx , vy , vz , shootz , trace . Crossed3DWater ! = NULL ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
thepuff - > Destroy ( ) ;
2006-02-24 04:48:15 +00:00
}
2009-03-20 01:40:28 +00:00
// Draw the slug's trail.
2015-06-25 01:57:36 +00:00
end . X = FIXED2DBL ( trace . X ) ;
end . Y = FIXED2DBL ( trace . Y ) ;
end . Z = FIXED2DBL ( trace . Z ) ;
2015-04-23 18:11:54 +00:00
P_DrawRailTrail ( source , start , end , color1 , color2 , maxdiff , railflags , spawnclass , source - > angle + angleoffset , duration , sparsity , drift , SpiralOffset ) ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// [RH] P_AimCamera
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
CVAR ( Float , chase_height , - 8.f , CVAR_ARCHIVE | CVAR_GLOBALCONFIG )
CVAR ( Float , chase_dist , 90.f , CVAR_ARCHIVE | CVAR_GLOBALCONFIG )
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
void P_AimCamera ( AActor * t1 , fixed_t & CameraX , fixed_t & CameraY , fixed_t & CameraZ , sector_t * & CameraSector )
2006-02-24 04:48:15 +00:00
{
2014-12-26 21:15:42 +00:00
fixed_t distance = ( fixed_t ) ( clamp < double > ( chase_dist , 0 , 30000 ) * FRACUNIT ) ;
2006-02-24 04:48:15 +00:00
angle_t angle = ( t1 - > angle - ANG180 ) > > ANGLETOFINESHIFT ;
angle_t pitch = ( angle_t ) ( t1 - > pitch ) > > ANGLETOFINESHIFT ;
FTraceResults trace ;
fixed_t vx , vy , vz , sz ;
2014-08-07 16:05:39 +00:00
vx = FixedMul ( finecosine [ pitch ] , finecosine [ angle ] ) ;
vy = FixedMul ( finecosine [ pitch ] , finesine [ angle ] ) ;
2006-02-24 04:48:15 +00:00
vz = finesine [ pitch ] ;
2016-01-20 12:48:05 +00:00
sz = t1 - > Z ( ) - t1 - > floorclip + t1 - > height + ( fixed_t ) ( clamp < double > ( chase_height , - 1000 , 1000 ) * FRACUNIT ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
if ( Trace ( t1 - > X ( ) , t1 - > Y ( ) , sz , t1 - > Sector ,
2006-02-24 04:48:15 +00:00
vx , vy , vz , distance , 0 , 0 , NULL , trace ) & &
2014-08-07 16:05:39 +00:00
trace . Distance > 10 * FRACUNIT )
2006-02-24 04:48:15 +00:00
{
// Position camera slightly in front of hit thing
2014-08-07 16:05:39 +00:00
fixed_t dist = trace . Distance - 5 * FRACUNIT ;
2016-01-20 12:48:05 +00:00
CameraX = t1 - > X ( ) + FixedMul ( vx , dist ) ;
CameraY = t1 - > Y ( ) + FixedMul ( vy , dist ) ;
2014-08-07 16:05:39 +00:00
CameraZ = sz + FixedMul ( vz , dist ) ;
2006-02-24 04:48:15 +00:00
}
else
{
CameraX = trace . X ;
CameraY = trace . Y ;
CameraZ = trace . Z ;
}
CameraSector = trace . Sector ;
}
2010-01-06 04:36:38 +00:00
//==========================================================================
//
// P_TalkFacing
//
// Looks for something within 5.625 degrees left or right of the player
// to talk to.
//
//==========================================================================
bool P_TalkFacing ( AActor * player )
{
AActor * linetarget ;
2014-08-07 16:05:39 +00:00
P_AimLineAttack ( player , player - > angle , TALKRANGE , & linetarget , ANGLE_1 * 35 , ALF_FORCENOSMART | ALF_CHECKCONVERSATION ) ;
2010-01-06 04:36:38 +00:00
if ( linetarget = = NULL )
{
2014-08-07 16:05:39 +00:00
P_AimLineAttack ( player , player - > angle + ( ANGLE_90 > > 4 ) , TALKRANGE , & linetarget , ANGLE_1 * 35 , ALF_FORCENOSMART | ALF_CHECKCONVERSATION ) ;
2010-01-06 04:36:38 +00:00
if ( linetarget = = NULL )
{
2014-08-07 16:05:39 +00:00
P_AimLineAttack ( player , player - > angle - ( ANGLE_90 > > 4 ) , TALKRANGE , & linetarget , ANGLE_1 * 35 , ALF_FORCENOSMART | ALF_CHECKCONVERSATION ) ;
2010-01-06 04:36:38 +00:00
if ( linetarget = = NULL )
{
return false ;
}
}
}
// Dead things can't talk.
if ( linetarget - > health < = 0 )
{
return false ;
}
// Fighting things don't talk either.
if ( linetarget - > flags4 & MF4_INCOMBAT )
{
return false ;
}
if ( linetarget - > Conversation ! = NULL )
{
// Give the NPC a chance to play a brief animation
2014-08-07 16:05:39 +00:00
linetarget - > ConversationAnimation ( 0 ) ;
P_StartConversation ( linetarget , player , true , true ) ;
2010-01-06 04:36:38 +00:00
return true ;
}
return false ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// USE LINES
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
bool P_UseTraverse ( AActor * usething , fixed_t endx , fixed_t endy , bool & foundline )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
FPathTraverse it ( usething - > X ( ) , usething - > Y ( ) , endx , endy , PT_ADDLINES | PT_ADDTHINGS ) ;
2008-04-09 18:35:21 +00:00
intercept_t * in ;
2006-02-24 04:48:15 +00:00
2008-04-09 18:35:21 +00:00
while ( ( in = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
// [RH] Check for things to talk with or use a puzzle item on
if ( ! in - > isaline )
2006-02-24 04:48:15 +00:00
{
2010-01-06 04:36:38 +00:00
if ( usething = = in - > d . thing )
continue ;
2008-04-09 18:35:21 +00:00
// Check thing
// Check for puzzle item use or USESPECIAL flag
2009-09-15 14:16:55 +00:00
// Extended to use the same activationtype mechanism as BUMPSPECIAL does
2008-04-09 18:35:21 +00:00
if ( in - > d . thing - > flags5 & MF5_USESPECIAL | | in - > d . thing - > special = = UsePuzzleItem )
2009-10-09 20:35:07 +00:00
{
if ( P_ActivateThingSpecial ( in - > d . thing , usething ) )
2008-04-09 18:35:21 +00:00
return true ;
}
continue ;
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
FLineOpening open ;
2014-08-07 16:05:39 +00:00
if ( in - > d . line - > special = = 0 | | ! ( in - > d . line - > activation & ( SPAC_Use | SPAC_UseThrough | SPAC_UseBack ) ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
blocked :
if ( in - > d . line - > flags & ( ML_BLOCKEVERYTHING | ML_BLOCKUSE ) )
2008-04-09 18:35:21 +00:00
{
open . range = 0 ;
}
else
{
2014-08-07 16:05:39 +00:00
P_LineOpening ( open , NULL , in - > d . line , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ;
2008-04-09 18:35:21 +00:00
}
if ( open . range < = 0 | |
( in - > d . line - > special ! = 0 & & ( i_compatflags & COMPATF_USEBLOCKING ) ) )
{
// [RH] Give sector a chance to intercept the use
2006-04-29 12:40:09 +00:00
2008-04-09 18:35:21 +00:00
sector_t * sec ;
2006-04-29 12:40:09 +00:00
2008-04-09 18:35:21 +00:00
sec = usething - > Sector ;
2006-04-29 12:40:09 +00:00
2014-08-07 16:05:39 +00:00
if ( sec - > SecActTarget & & sec - > SecActTarget - > TriggerAction ( usething , SECSPAC_Use ) )
2008-04-09 18:35:21 +00:00
{
return true ;
}
2006-04-29 12:40:09 +00:00
2016-01-20 12:48:05 +00:00
sec = P_PointOnLineSide ( usething - > X ( ) , usething - > Y ( ) , in - > d . line ) = = 0 ?
2008-04-09 18:35:21 +00:00
in - > d . line - > frontsector : in - > d . line - > backsector ;
2006-04-29 12:40:09 +00:00
2008-04-09 18:35:21 +00:00
if ( sec ! = NULL & & sec - > SecActTarget & &
2014-08-07 16:05:39 +00:00
sec - > SecActTarget - > TriggerAction ( usething , SECSPAC_UseWall ) )
2008-04-09 18:35:21 +00:00
{
return true ;
}
2006-04-29 12:40:09 +00:00
2008-04-09 18:35:21 +00:00
if ( usething - > player )
{
2014-08-07 16:05:39 +00:00
S_Sound ( usething , CHAN_VOICE , " *usefail " , 1 , ATTN_IDLE ) ;
2008-04-09 18:35:21 +00:00
}
return true ; // can't use through a wall
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
foundline = true ;
continue ; // not a special line, but keep checking
}
2014-08-07 16:05:39 +00:00
2016-01-20 12:48:05 +00:00
if ( P_PointOnLineSide ( usething - > X ( ) , usething - > Y ( ) , in - > d . line ) = = 1 )
2008-04-09 18:35:21 +00:00
{
2010-04-25 07:21:35 +00:00
if ( ! ( in - > d . line - > activation & SPAC_UseBack ) )
{
// [RH] continue traversal for two-sided lines
//return in->d.line->backsector != NULL; // don't use back side
goto blocked ; // do a proper check for back sides of triggers
}
else
{
2014-08-07 16:05:39 +00:00
P_ActivateLine ( in - > d . line , usething , 1 , SPAC_UseBack ) ;
2010-04-25 07:21:35 +00:00
return true ;
}
2008-04-09 18:35:21 +00:00
}
2014-08-07 16:05:39 +00:00
else
2008-04-09 18:35:21 +00:00
{
2014-08-07 16:05:39 +00:00
if ( ( in - > d . line - > activation & ( SPAC_Use | SPAC_UseThrough | SPAC_UseBack ) ) = = SPAC_UseBack )
2010-04-25 07:21:35 +00:00
{
goto blocked ; // Line cannot be used from front side so treat it as a non-trigger line
}
2014-08-07 16:05:39 +00:00
P_ActivateLine ( in - > d . line , usething , 0 , SPAC_Use ) ;
2010-04-25 07:21:35 +00:00
//WAS can't use more than one special line in a row
//jff 3/21/98 NOW multiple use allowed with enabling line flag
//[RH] And now I've changed it again. If the line is of type
// SPAC_USE, then it eats the use. Everything else passes
// it through, including SPAC_USETHROUGH.
if ( i_compatflags & COMPATF_USEBLOCKING )
{
if ( in - > d . line - > activation & SPAC_UseThrough ) continue ;
else return true ;
}
else
{
if ( ! ( in - > d . line - > activation & SPAC_Use ) ) continue ;
else return true ;
}
2006-02-24 04:48:15 +00:00
}
2010-04-25 07:21:35 +00:00
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
return false ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
2006-02-24 04:48:15 +00:00
// Returns false if a "oof" sound should be made because of a blocking
// linedef. Makes 2s middles which are impassable, as well as 2s uppers
// and lowers which block the player, cause the sound effect when the
// player tries to activate them. Specials are excluded, although it is
// assumed that all special linedefs within reach have been considered
// and rejected already (see P_UseLines).
//
// by Lee Killough
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
bool P_NoWayTraverse ( AActor * usething , fixed_t endx , fixed_t endy )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
FPathTraverse it ( usething - > X ( ) , usething - > Y ( ) , endx , endy , PT_ADDLINES ) ;
2008-04-09 18:35:21 +00:00
intercept_t * in ;
2006-04-11 08:36:23 +00:00
2008-04-09 18:35:21 +00:00
while ( ( in = it . Next ( ) ) )
{
line_t * ld = in - > d . line ;
FLineOpening open ;
2009-10-11 17:44:50 +00:00
// [GrafZahl] de-obfuscated. Was I the only one who was unable to make sense out of
2008-04-09 18:35:21 +00:00
// this convoluted mess?
if ( ld - > special ) continue ;
2014-08-07 16:05:39 +00:00
if ( ld - > flags & ( ML_BLOCKING | ML_BLOCKEVERYTHING | ML_BLOCK_PLAYERS ) ) return true ;
P_LineOpening ( open , NULL , ld , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ;
2008-04-09 18:35:21 +00:00
if ( open . range < = 0 | |
2016-01-20 12:48:05 +00:00
open . bottom > usething - > Z ( ) + usething - > MaxStepHeight | |
open . top < usething - > Top ( ) ) return true ;
2008-04-09 18:35:21 +00:00
}
return false ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
// P_UseLines
//
// Looks for special lines in front of the player to activate
//
//==========================================================================
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
void P_UseLines ( player_t * player )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
bool foundline = false ;
2013-07-22 20:37:50 +00:00
// [NS] Now queries the Player's UseRange.
2016-01-20 12:48:05 +00:00
fixedvec2 end = player - > mo - > Vec2Angle ( player - > mo - > UseRange , player - > mo - > angle , true ) ;
2009-11-17 23:09:21 +00:00
2006-02-24 04:48:15 +00:00
// old code:
//
// P_PathTraverse ( x1, y1, x2, y2, PT_ADDLINES, PTR_UseTraverse );
//
// This added test makes the "oof" sound work on 2s lines -- killough:
2016-01-20 12:48:05 +00:00
if ( ! P_UseTraverse ( player - > mo , end . x , end . y , foundline ) )
2006-02-24 04:48:15 +00:00
{ // [RH] Give sector a chance to eat the use
2008-04-09 18:35:21 +00:00
sector_t * sec = player - > mo - > Sector ;
2006-02-24 04:48:15 +00:00
int spac = SECSPAC_Use ;
2008-04-09 18:35:21 +00:00
if ( foundline ) spac | = SECSPAC_UseWall ;
2014-08-07 16:05:39 +00:00
if ( ( ! sec - > SecActTarget | | ! sec - > SecActTarget - > TriggerAction ( player - > mo , spac ) ) & &
2016-01-20 12:48:05 +00:00
P_NoWayTraverse ( player - > mo , end . x , end . y ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
S_Sound ( player - > mo , CHAN_VOICE , " *usefail " , 1 , ATTN_IDLE ) ;
2006-02-24 04:48:15 +00:00
}
}
}
//==========================================================================
//
2008-04-09 18:35:21 +00:00
// P_UsePuzzleItem
//
// Returns true if the puzzle item was used on a line or a thing.
2006-02-24 04:48:15 +00:00
//
//==========================================================================
2014-08-07 16:05:39 +00:00
bool P_UsePuzzleItem ( AActor * PuzzleItemUser , int PuzzleItemType )
2006-02-24 04:48:15 +00:00
{
2008-04-09 18:35:21 +00:00
int angle ;
2013-07-22 20:37:50 +00:00
fixed_t x1 , y1 , x2 , y2 , usedist ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
angle = PuzzleItemUser - > angle > > ANGLETOFINESHIFT ;
2016-01-20 12:48:05 +00:00
x1 = PuzzleItemUser - > X ( ) ;
y1 = PuzzleItemUser - > Y ( ) ;
2013-07-22 20:37:50 +00:00
// [NS] If it's a Player, get their UseRange.
if ( PuzzleItemUser - > player )
usedist = PuzzleItemUser - > player - > mo - > UseRange ;
else
usedist = USERANGE ;
x2 = x1 + FixedMul ( usedist , finecosine [ angle ] ) ;
y2 = y1 + FixedMul ( usedist , finesine [ angle ] ) ;
2008-04-09 18:35:21 +00:00
2014-08-07 16:05:39 +00:00
FPathTraverse it ( x1 , y1 , x2 , y2 , PT_ADDLINES | PT_ADDTHINGS ) ;
2008-04-09 18:35:21 +00:00
intercept_t * in ;
while ( ( in = it . Next ( ) ) )
{
AActor * mobj ;
FLineOpening open ;
if ( in - > isaline )
{ // Check line
if ( in - > d . line - > special ! = UsePuzzleItem )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
P_LineOpening ( open , NULL , in - > d . line , it . Trace ( ) . x + FixedMul ( it . Trace ( ) . dx , in - > frac ) ,
it . Trace ( ) . y + FixedMul ( it . Trace ( ) . dy , in - > frac ) ) ;
2008-04-09 18:35:21 +00:00
if ( open . range < = 0 )
{
return false ; // can't use through a wall
}
continue ;
}
2016-01-20 12:48:05 +00:00
if ( P_PointOnLineSide ( PuzzleItemUser - > X ( ) , PuzzleItemUser - > Y ( ) , in - > d . line ) = = 1 )
2008-04-09 18:35:21 +00:00
{ // Don't use back sides
return false ;
}
if ( PuzzleItemType ! = in - > d . line - > args [ 0 ] )
{ // Item type doesn't match
return false ;
2006-02-24 04:48:15 +00:00
}
2012-02-26 03:36:05 +00:00
int args [ 3 ] = { in - > d . line - > args [ 2 ] , in - > d . line - > args [ 3 ] , in - > d . line - > args [ 4 ] } ;
2014-08-07 16:05:39 +00:00
P_StartScript ( PuzzleItemUser , in - > d . line , in - > d . line - > args [ 1 ] , NULL , args , 3 , ACS_ALWAYS ) ;
2008-04-09 18:35:21 +00:00
in - > d . line - > special = 0 ;
return true ;
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
// Check thing
mobj = in - > d . thing ;
if ( mobj - > special ! = UsePuzzleItem )
{ // Wrong special
continue ;
2006-02-24 04:48:15 +00:00
}
2008-04-09 18:35:21 +00:00
if ( PuzzleItemType ! = mobj - > args [ 0 ] )
2006-02-24 04:48:15 +00:00
{ // Item type doesn't match
2008-04-09 18:35:21 +00:00
continue ;
2006-02-24 04:48:15 +00:00
}
2012-02-26 03:36:05 +00:00
int args [ 3 ] = { mobj - > args [ 2 ] , mobj - > args [ 3 ] , mobj - > args [ 4 ] } ;
2014-08-07 16:05:39 +00:00
P_StartScript ( PuzzleItemUser , NULL , mobj - > args [ 1 ] , NULL , args , 3 , ACS_ALWAYS ) ;
2008-04-09 18:35:21 +00:00
mobj - > special = 0 ;
2006-02-24 04:48:15 +00:00
return true ;
}
2008-04-09 18:35:21 +00:00
return false ;
2006-02-24 04:48:15 +00:00
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// RADIUS ATTACK
//
2009-10-11 17:44:50 +00:00
//
//==========================================================================
2006-02-24 04:48:15 +00:00
// [RH] Damage scale to apply to thing that shot the missile.
static float selfthrustscale ;
2014-08-07 16:05:39 +00:00
CUSTOM_CVAR ( Float , splashfactor , 1.f , CVAR_SERVERINFO )
2006-02-24 04:48:15 +00:00
{
if ( self < = 0.f )
self = 1.f ;
else
selfthrustscale = 1.f / self ;
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2008-04-07 21:14:28 +00:00
//
// P_RadiusAttack
// Source is the creature that caused the explosion at spot.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2014-08-07 16:05:39 +00:00
void P_RadiusAttack ( AActor * bombspot , AActor * bombsource , int bombdamage , int bombdistance , FName bombmod ,
2012-08-30 04:01:50 +00:00
int flags , int fulldamagedistance )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
if ( bombdistance < = 0 )
return ;
2014-08-07 16:05:39 +00:00
fulldamagedistance = clamp < int > ( fulldamagedistance , 0 , bombdistance - 1 ) ;
2006-02-24 04:48:15 +00:00
2010-11-07 14:39:09 +00:00
double bombdistancefloat = 1.f / ( double ) ( bombdistance - fulldamagedistance ) ;
double bombdamagefloat = ( double ) bombdamage ;
2009-06-27 19:37:53 +00:00
2016-01-20 12:48:05 +00:00
FBlockThingsIterator it ( FBoundingBox ( bombspot - > X ( ) , bombspot - > Y ( ) , bombdistance < < FRACBITS ) ) ;
2008-04-07 21:14:28 +00:00
AActor * thing ;
2006-02-24 04:48:15 +00:00
2012-08-30 04:01:50 +00:00
if ( flags & RADF_SOURCEISSPOT )
2013-03-20 02:41:59 +00:00
{ // The source is actually the same as the spot, even if that wasn't what we received.
2012-08-30 04:01:50 +00:00
bombsource = bombspot ;
}
2008-04-07 21:14:28 +00:00
while ( ( thing = it . Next ( ) ) )
2006-02-24 04:48:15 +00:00
{
2009-09-14 20:47:53 +00:00
// Vulnerable actors can be damaged by radius attacks even if not shootable
// Used to emulate MBF's vulnerability of non-missile bouncers to explosions.
if ( ! ( ( thing - > flags & MF_SHOOTABLE ) | | ( thing - > flags6 & MF6_VULNERABLE ) ) )
2008-04-07 21:14:28 +00:00
continue ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
// Boss spider and cyborg and Heretic's ep >= 2 bosses
// take no damage from concussion.
if ( thing - > flags3 & MF3_NORADIUSDMG & & ! ( bombspot - > flags4 & MF4_FORCERADIUSDMG ) )
continue ;
2006-02-24 04:48:15 +00:00
2012-08-30 04:01:50 +00:00
if ( ! ( flags & RADF_HURTSOURCE ) & & ( thing = = bombsource | | thing = = bombspot ) )
2008-04-07 21:14:28 +00:00
{ // don't damage the source of the explosion
continue ;
}
// a much needed option: monsters that fire explosive projectiles cannot
// be hurt by projectiles fired by a monster of the same type.
2009-06-27 19:37:53 +00:00
// Controlled by the DONTHARMCLASS and DONTHARMSPECIES flags.
if ( ( bombsource & & ! thing - > player ) // code common to both checks
2014-08-07 16:05:39 +00:00
& & ( // Class check first
2009-06-27 19:37:53 +00:00
( ( bombsource - > flags4 & MF4_DONTHARMCLASS ) & & ( thing - > GetClass ( ) = = bombsource - > GetClass ( ) ) )
| | // Nigh-identical species check second
( ( bombsource - > flags6 & MF6_DONTHARMSPECIES ) & & ( thing - > GetSpecies ( ) = = bombsource - > GetSpecies ( ) ) )
)
2014-08-07 16:05:39 +00:00
) continue ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
// Barrels always use the original code, since this makes
// them far too "active." BossBrains also use the old code
// because some user levels require they have a height of 16,
// which can make them near impossible to hit with the new code.
2012-08-30 04:01:50 +00:00
if ( ( flags & RADF_NODAMAGE ) | | ! ( ( bombspot - > flags5 | thing - > flags5 ) & MF5_OLDRADIUSDMG ) )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
// [RH] New code. The bounding box only covers the
// height of the thing and not the height of the map.
2010-11-07 14:39:09 +00:00
double points ;
double len ;
2008-04-07 21:14:28 +00:00
fixed_t dx , dy ;
2010-11-07 14:39:09 +00:00
double boxradius ;
2008-04-07 21:14:28 +00:00
2016-01-20 12:48:05 +00:00
fixedvec2 vec = bombspot - > Vec2To ( thing ) ;
dx = abs ( vec . x ) ;
dy = abs ( vec . y ) ;
2014-08-07 16:05:39 +00:00
boxradius = double ( thing - > radius ) ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
// The damage pattern is square, not circular.
2014-08-07 16:05:39 +00:00
len = double ( dx > dy ? dx : dy ) ;
2008-04-07 21:14:28 +00:00
2016-01-20 12:48:05 +00:00
if ( bombspot - > Z ( ) < thing - > Z ( ) | | bombspot - > Z ( ) > = thing - > Top ( ) )
2006-02-24 04:48:15 +00:00
{
2010-11-07 14:39:09 +00:00
double dz ;
2008-04-07 21:14:28 +00:00
2016-01-20 12:48:05 +00:00
if ( bombspot - > Z ( ) > thing - > Z ( ) )
2008-04-07 21:14:28 +00:00
{
2016-01-20 12:48:05 +00:00
dz = double ( bombspot - > Z ( ) - thing - > Top ( ) ) ;
2008-04-07 21:14:28 +00:00
}
else
{
2016-01-20 12:48:05 +00:00
dz = double ( thing - > Z ( ) - bombspot - > Z ( ) ) ;
2008-04-07 21:14:28 +00:00
}
if ( len < = boxradius )
{
len = dz ;
}
else
{
len - = boxradius ;
2014-08-07 16:05:39 +00:00
len = sqrt ( len * len + dz * dz ) ;
2008-04-07 21:14:28 +00:00
}
2006-02-24 04:48:15 +00:00
}
else
{
2008-04-07 21:14:28 +00:00
len - = boxradius ;
if ( len < 0.f )
len = 0.f ;
2006-02-24 04:48:15 +00:00
}
2008-04-07 21:14:28 +00:00
len / = FRACUNIT ;
2010-11-07 14:39:09 +00:00
len = clamp < double > ( len - ( double ) fulldamagedistance , 0 , len ) ;
2008-04-07 21:14:28 +00:00
points = bombdamagefloat * ( 1.f - len * bombdistancefloat ) ;
if ( thing = = bombsource )
2006-02-24 04:48:15 +00:00
{
2008-04-07 21:14:28 +00:00
points = points * splashfactor ;
2006-02-24 04:48:15 +00:00
}
2012-07-14 03:04:41 +00:00
points * = thing - > GetClass ( ) - > RDFactor / ( float ) FRACUNIT ;
2006-06-03 12:30:11 +00:00
2013-03-20 02:41:59 +00:00
// points and bombdamage should be the same sign
2015-04-01 08:01:48 +00:00
if ( ( ( points * bombdamage ) > 0 ) & & P_CheckSight ( thing , bombspot , SF_IGNOREVISIBILITY | SF_IGNOREWATERBOUNDARY ) )
2008-04-07 21:14:28 +00:00
{ // OK to damage; target is in direct path
2010-11-07 14:39:09 +00:00
double velz ;
double thrust ;
2013-03-20 02:41:59 +00:00
int damage = abs ( ( int ) points ) ;
2013-01-02 04:39:59 +00:00
int newdam = damage ;
2006-02-24 04:48:15 +00:00
2012-08-30 04:01:50 +00:00
if ( ! ( flags & RADF_NODAMAGE ) )
2014-08-07 16:05:39 +00:00
newdam = P_DamageMobj ( thing , bombspot , bombsource , damage , bombmod ) ;
2014-10-28 07:40:34 +00:00
else if ( thing - > player = = NULL & & ( ! ( flags & RADF_NOIMPACTDAMAGE ) & & ! ( thing - > flags7 & MF7_DONTTHRUST ) ) )
2012-08-30 04:01:50 +00:00
thing - > flags2 | = MF2_BLASTED ;
2006-08-10 21:03:17 +00:00
2008-04-07 21:14:28 +00:00
if ( ! ( thing - > flags & MF_ICECORPSE ) )
2006-02-24 04:48:15 +00:00
{
2012-08-30 04:01:50 +00:00
if ( ! ( flags & RADF_NODAMAGE ) & & ! ( bombspot - > flags3 & MF3_BLOODLESSIMPACT ) )
2014-08-07 16:05:39 +00:00
P_TraceBleed ( newdam > 0 ? newdam : damage , thing , bombspot ) ;
2008-04-07 21:14:28 +00:00
2013-02-22 03:06:00 +00:00
if ( ( flags & RADF_NODAMAGE ) | | ! ( bombspot - > flags2 & MF2_NODMGTHRUST ) )
2006-02-24 04:48:15 +00:00
{
2011-05-11 04:29:19 +00:00
if ( bombsource = = NULL | | ! ( bombsource - > flags2 & MF2_NODMGTHRUST ) )
2008-04-07 21:14:28 +00:00
{
2014-10-28 07:40:34 +00:00
if ( ! ( thing - > flags7 & MF7_DONTTHRUST ) )
2009-08-12 18:57:31 +00:00
{
2014-10-28 02:35:55 +00:00
thrust = points * 0.5f / ( double ) thing - > Mass ;
if ( bombsource = = thing )
{
thrust * = selfthrustscale ;
}
2016-01-20 12:48:05 +00:00
velz = ( double ) ( thing - > Z ( ) + ( thing - > height > > 1 ) - bombspot - > Z ( ) ) * thrust ;
2014-10-28 02:35:55 +00:00
if ( bombsource ! = thing )
{
velz * = 0.5f ;
}
else
{
velz * = 0.8f ;
}
2016-01-10 19:46:26 +00:00
angle_t ang = bombspot - > AngleTo ( thing ) > > ANGLETOFINESHIFT ;
2014-10-28 02:35:55 +00:00
thing - > velx + = fixed_t ( finecosine [ ang ] * thrust ) ;
thing - > vely + = fixed_t ( finesine [ ang ] * thrust ) ;
if ( ! ( flags & RADF_NODAMAGE ) )
thing - > velz + = ( fixed_t ) velz ; // this really doesn't work well
2009-08-12 18:57:31 +00:00
}
2008-04-07 21:14:28 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
}
}
2008-04-07 21:14:28 +00:00
else
{
// [RH] Old code just for barrels
fixed_t dx , dy , dist ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
fixedvec2 vec = bombspot - > Vec2To ( thing ) ;
dx = abs ( vec . x ) ;
dy = abs ( vec . y ) ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
dist = dx > dy ? dx : dy ;
dist = ( dist - thing - > radius ) > > FRACBITS ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
if ( dist < 0 )
dist = 0 ;
2006-02-24 04:48:15 +00:00
2008-04-07 21:14:28 +00:00
if ( dist > = bombdistance )
continue ; // out of range
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
if ( P_CheckSight ( thing , bombspot , SF_IGNOREVISIBILITY | SF_IGNOREWATERBOUNDARY ) )
2008-04-07 21:14:28 +00:00
{ // OK to damage; target is in direct path
2009-06-27 19:37:53 +00:00
dist = clamp < int > ( dist - fulldamagedistance , 0 , dist ) ;
2014-08-07 16:05:39 +00:00
int damage = Scale ( bombdamage , bombdistance - dist , bombdistance ) ;
2010-11-07 14:39:09 +00:00
damage = ( int ) ( ( double ) damage * splashfactor ) ;
2006-06-03 12:30:11 +00:00
2010-03-25 20:38:00 +00:00
damage = Scale ( damage , thing - > GetClass ( ) - > RDFactor , FRACUNIT ) ;
2008-04-07 21:14:28 +00:00
if ( damage > 0 )
{
2014-08-07 16:05:39 +00:00
int newdam = P_DamageMobj ( thing , bombspot , bombsource , damage , bombmod ) ;
P_TraceBleed ( newdam > 0 ? newdam : damage , thing , bombspot ) ;
2008-04-07 21:14:28 +00:00
}
2006-02-24 04:48:15 +00:00
}
}
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
2006-02-24 04:48:15 +00:00
//
// SECTOR HEIGHT CHANGING
// After modifying a sector's floor or ceiling height,
// call this routine to adjust the positions
// of all things that touch the sector.
//
// If anything doesn't fit anymore, true will be returned.
//
// [RH] If crushchange is non-negative, they will take the
// specified amount of damage as they are being crushed.
// If crushchange is negative, you should set the sector
// height back the way it was and call P_ChangeSector()
// again to undo the changes.
// Note that this is very different from the original
// true/false usage of crushchange! If you want regular
// DOOM crushing behavior set crushchange to 10 or -1
// if no crushing is desired.
//
2009-10-11 17:44:50 +00:00
//==========================================================================
2008-03-18 18:18:18 +00:00
struct FChangePosition
{
2008-05-22 19:35:38 +00:00
sector_t * sector ;
2008-03-18 18:18:18 +00:00
int moveamt ;
int crushchange ;
bool nofit ;
bool movemidtex ;
} ;
2006-02-24 04:48:15 +00:00
TArray < AActor * > intersectors ;
2014-08-07 16:05:39 +00:00
EXTERN_CVAR ( Int , cl_bloodtype )
2006-02-24 04:48:15 +00:00
//=============================================================================
//
// P_AdjustFloorCeil
//
//=============================================================================
2014-08-07 16:05:39 +00:00
bool P_AdjustFloorCeil ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
2015-04-04 16:40:43 +00:00
ActorFlags2 flags2 = thing - > flags2 & MF2_PASSMOBJ ;
2008-04-08 20:52:49 +00:00
FCheckPosition tm ;
2008-03-18 18:18:18 +00:00
2010-09-16 20:15:44 +00:00
if ( ( thing - > flags2 & MF2_PASSMOBJ ) & & ( thing - > flags3 & MF3_ISMONSTER ) )
{
tm . FromPMove = true ;
}
2008-03-18 18:18:18 +00:00
if ( cpos - > movemidtex )
{
// From Eternity:
// ALL things must be treated as PASSMOBJ when moving
// 3DMidTex lines, otherwise you get stuck in them.
thing - > flags2 | = MF2_PASSMOBJ ;
}
2016-01-20 12:48:05 +00:00
bool isgood = P_CheckPosition ( thing , thing - > X ( ) , thing - > Y ( ) , tm ) ;
2008-04-08 20:52:49 +00:00
thing - > floorz = tm . floorz ;
thing - > ceilingz = tm . ceilingz ;
thing - > dropoffz = tm . dropoffz ; // killough 11/98: remember dropoffs
thing - > floorpic = tm . floorpic ;
2016-01-09 11:10:36 +00:00
thing - > floorterrain = tm . floorterrain ;
2008-04-08 20:52:49 +00:00
thing - > floorsector = tm . floorsector ;
thing - > ceilingpic = tm . ceilingpic ;
thing - > ceilingsector = tm . ceilingsector ;
2008-03-18 18:18:18 +00:00
// restore the PASSMOBJ flag but leave the other flags alone.
thing - > flags2 = ( thing - > flags2 & ~ MF2_PASSMOBJ ) | flags2 ;
2006-02-24 04:48:15 +00:00
return isgood ;
}
//=============================================================================
//
// P_FindAboveIntersectors
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_FindAboveIntersectors ( AActor * actor )
2006-02-24 04:48:15 +00:00
{
if ( actor - > flags & MF_NOCLIP )
return ;
if ( ! ( actor - > flags & MF_SOLID ) )
return ;
2008-04-07 21:14:28 +00:00
AActor * thing ;
2016-01-20 12:48:05 +00:00
FBlockThingsIterator it ( FBoundingBox ( actor - > X ( ) , actor - > Y ( ) , actor - > radius ) ) ;
2008-04-07 21:14:28 +00:00
while ( ( thing = it . Next ( ) ) )
{
2008-04-15 10:04:41 +00:00
if ( ! thing - > intersects ( actor ) )
{
continue ;
}
2008-04-07 21:14:28 +00:00
if ( ! ( thing - > flags & MF_SOLID ) )
{ // Can't hit thing
continue ;
}
2009-10-09 20:18:31 +00:00
if ( thing - > flags & ( MF_SPECIAL ) )
2008-04-07 21:14:28 +00:00
{ // [RH] Corpses and specials don't block moves
continue ;
}
2009-10-09 20:18:31 +00:00
if ( thing - > flags & ( MF_CORPSE ) )
{ // Corpses need a few more checks
if ( ! ( actor - > flags & MF_ICECORPSE ) )
continue ;
}
2008-04-07 21:14:28 +00:00
if ( thing = = actor )
{ // Don't clip against self
continue ;
}
2009-10-01 14:54:29 +00:00
if ( ! ( ( thing - > flags2 | actor - > flags2 ) & MF2_PASSMOBJ ) & & ! ( ( thing - > flags3 | actor - > flags3 ) & MF3_ISMONSTER ) )
{
// Don't bother if both things don't have MF2_PASSMOBJ set and aren't monsters.
// These things would always block each other which in nearly every situation is
// not what is wanted here.
continue ;
}
2016-01-20 12:48:05 +00:00
if ( thing - > Z ( ) > = actor - > Z ( ) & &
thing - > Z ( ) < = actor - > Top ( ) )
2008-04-07 21:14:28 +00:00
{ // Thing intersects above the base
2014-08-07 16:05:39 +00:00
intersectors . Push ( thing ) ;
2008-04-07 21:14:28 +00:00
}
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// P_FindBelowIntersectors
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_FindBelowIntersectors ( AActor * actor )
2006-02-24 04:48:15 +00:00
{
if ( actor - > flags & MF_NOCLIP )
return ;
if ( ! ( actor - > flags & MF_SOLID ) )
return ;
2008-04-07 21:14:28 +00:00
AActor * thing ;
2016-01-20 12:48:05 +00:00
FBlockThingsIterator it ( FBoundingBox ( actor - > X ( ) , actor - > Y ( ) , actor - > radius ) ) ;
2008-04-07 21:14:28 +00:00
while ( ( thing = it . Next ( ) ) )
{
2008-04-15 10:04:41 +00:00
if ( ! thing - > intersects ( actor ) )
{
continue ;
}
2008-04-07 21:14:28 +00:00
if ( ! ( thing - > flags & MF_SOLID ) )
{ // Can't hit thing
continue ;
}
2009-10-09 20:18:31 +00:00
if ( thing - > flags & ( MF_SPECIAL ) )
2008-04-07 21:14:28 +00:00
{ // [RH] Corpses and specials don't block moves
continue ;
}
2009-10-09 20:18:31 +00:00
if ( thing - > flags & ( MF_CORPSE ) )
{ // Corpses need a few more checks
if ( ! ( actor - > flags & MF_ICECORPSE ) )
continue ;
}
2008-04-08 20:52:49 +00:00
if ( thing = = actor )
2008-04-07 21:14:28 +00:00
{ // Don't clip against self
continue ;
}
2009-10-01 14:54:29 +00:00
if ( ! ( ( thing - > flags2 | actor - > flags2 ) & MF2_PASSMOBJ ) & & ! ( ( thing - > flags3 | actor - > flags3 ) & MF3_ISMONSTER ) )
{
// Don't bother if both things don't have MF2_PASSMOBJ set and aren't monsters.
// These things would always block each other which in nearly every situation is
// not what is wanted here.
continue ;
}
2016-01-20 12:48:05 +00:00
if ( thing - > Top ( ) < = actor - > Top ( ) & &
thing - > Top ( ) > actor - > Z ( ) )
2008-04-07 21:14:28 +00:00
{ // Thing intersects below the base
2014-08-07 16:05:39 +00:00
intersectors . Push ( thing ) ;
2008-04-07 21:14:28 +00:00
}
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// P_DoCrunch
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_DoCrunch ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
2009-05-15 17:21:45 +00:00
if ( ! ( thing & & thing - > Grind ( true ) & & cpos ) ) return ;
2008-03-18 18:18:18 +00:00
cpos - > nofit = true ;
2006-02-24 04:48:15 +00:00
2008-03-18 18:18:18 +00:00
if ( ( cpos - > crushchange > 0 ) & & ! ( level . maptime & 3 ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
int newdam = P_DamageMobj ( thing , NULL , NULL , cpos - > crushchange , NAME_Crush ) ;
2006-02-24 04:48:15 +00:00
// spray blood in a random direction
2014-08-07 16:05:39 +00:00
if ( ! ( thing - > flags2 & ( MF2_INVULNERABLE | MF2_DORMANT ) ) )
2006-02-24 04:48:15 +00:00
{
2010-03-21 08:09:45 +00:00
if ( ! ( thing - > flags & MF_NOBLOOD ) )
2006-02-24 04:48:15 +00:00
{
2010-03-21 08:09:45 +00:00
PalEntry bloodcolor = thing - > GetBloodColor ( ) ;
2010-04-04 04:09:24 +00:00
PClassActor * bloodcls = thing - > GetBloodType ( ) ;
2010-03-21 08:09:45 +00:00
2013-01-02 04:39:59 +00:00
P_TraceBleed ( newdam > 0 ? newdam : cpos - > crushchange , thing ) ;
2013-12-01 09:06:48 +00:00
if ( bloodcls ! = NULL )
2010-03-21 08:09:45 +00:00
{
AActor * mo ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
mo = Spawn ( bloodcls , thing - > PosPlusZ ( thing - > height / 2 ) , ALLOW_REPLACE ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
mo - > velx = pr_crunch . Random2 ( ) < < 12 ;
mo - > vely = pr_crunch . Random2 ( ) < < 12 ;
2010-03-21 08:09:45 +00:00
if ( bloodcolor ! = 0 & & ! ( mo - > flags2 & MF2_DONTTRANSLATE ) )
{
mo - > Translation = TRANSLATION ( TRANSLATION_Blood , bloodcolor . a ) ;
}
2013-12-01 09:06:48 +00:00
if ( ! ( cl_bloodtype < = 1 ) ) mo - > renderflags | = RF_INVISIBLE ;
2010-03-21 08:09:45 +00:00
}
2013-12-01 09:06:48 +00:00
angle_t an ;
2014-08-07 16:05:39 +00:00
an = ( M_Random ( ) - 128 ) < < 24 ;
2010-03-21 08:09:45 +00:00
if ( cl_bloodtype > = 1 )
2006-11-29 10:03:35 +00:00
{
2016-01-20 12:48:05 +00:00
P_DrawSplash2 ( 32 , thing - > X ( ) , thing - > Y ( ) , thing - > Z ( ) + thing - > height / 2 , an , 2 , bloodcolor ) ;
2006-11-29 10:03:35 +00:00
}
2006-02-24 04:48:15 +00:00
}
2010-03-21 08:09:45 +00:00
if ( thing - > CrushPainSound ! = 0 & & ! S_GetSoundPlayingInfo ( thing , thing - > CrushPainSound ) )
2006-02-24 04:48:15 +00:00
{
2010-03-21 08:09:45 +00:00
S_Sound ( thing , CHAN_VOICE , thing - > CrushPainSound , 1.f , ATTN_NORM ) ;
2006-02-24 04:48:15 +00:00
}
}
}
// keep checking (crush other things)
return ;
}
//=============================================================================
//
// P_PushUp
//
// Returns 0 if thing fits, 1 if ceiling got in the way, or 2 if something
// above it didn't fit.
//=============================================================================
2014-08-07 16:05:39 +00:00
int P_PushUp ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
unsigned int firstintersect = intersectors . Size ( ) ;
2006-02-24 04:48:15 +00:00
unsigned int lastintersect ;
int mymass = thing - > Mass ;
2016-01-20 12:48:05 +00:00
if ( thing - > Top ( ) > thing - > ceilingz )
2006-02-24 04:48:15 +00:00
{
return 1 ;
}
2012-02-21 20:50:38 +00:00
// [GZ] Skip thing intersect test for THRUACTORS things.
if ( thing - > flags2 & MF2_THRUACTORS )
return 0 ;
2014-08-07 16:05:39 +00:00
P_FindAboveIntersectors ( thing ) ;
lastintersect = intersectors . Size ( ) ;
2006-02-24 04:48:15 +00:00
for ( ; firstintersect < lastintersect ; firstintersect + + )
{
AActor * intersect = intersectors [ firstintersect ] ;
2012-02-21 20:50:38 +00:00
// [GZ] Skip this iteration for THRUSPECIES things
// Should there be MF2_THRUGHOST / MF3_GHOST checks there too for consistency?
// Or would that risk breaking established behavior? THRUGHOST, like MTHRUSPECIES,
// is normally for projectiles which would have exploded by now anyway...
if ( thing - > flags6 & MF6_THRUSPECIES & & thing - > GetSpecies ( ) = = intersect - > GetSpecies ( ) )
continue ;
2014-12-09 18:09:36 +00:00
if ( ( thing - > flags & MF_MISSILE ) & & ( intersect - > flags2 & MF2_REFLECTIVE ) & & ( intersect - > flags7 & MF7_THRUREFLECT ) )
continue ;
2006-02-24 04:48:15 +00:00
if ( ! ( intersect - > flags2 & MF2_PASSMOBJ ) | |
2010-11-30 08:18:11 +00:00
( ! ( intersect - > flags3 & MF3_ISMONSTER ) & & intersect - > Mass > mymass ) | |
( intersect - > flags4 & MF4_ACTLIKEBRIDGE )
2014-08-07 16:05:39 +00:00
)
{
2010-11-30 08:18:11 +00:00
// Can't push bridges or things more massive than ourself
2006-02-24 04:48:15 +00:00
return 2 ;
}
2014-12-09 18:09:36 +00:00
fixed_t oldz ;
2016-01-20 12:48:05 +00:00
oldz = intersect - > Z ( ) ;
2014-08-07 16:05:39 +00:00
P_AdjustFloorCeil ( intersect , cpos ) ;
2016-01-20 12:48:05 +00:00
intersect - > SetZ ( thing - > Top ( ) + 1 ) ;
2014-08-07 16:05:39 +00:00
if ( P_PushUp ( intersect , cpos ) )
2006-02-24 04:48:15 +00:00
{ // Move blocked
2014-08-07 16:05:39 +00:00
P_DoCrunch ( intersect , cpos ) ;
2016-01-20 12:48:05 +00:00
intersect - > SetZ ( oldz ) ;
2006-02-24 04:48:15 +00:00
return 2 ;
}
}
return 0 ;
}
//=============================================================================
//
// P_PushDown
//
// Returns 0 if thing fits, 1 if floor got in the way, or 2 if something
// below it didn't fit.
//=============================================================================
2014-08-07 16:05:39 +00:00
int P_PushDown ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
unsigned int firstintersect = intersectors . Size ( ) ;
2006-02-24 04:48:15 +00:00
unsigned int lastintersect ;
int mymass = thing - > Mass ;
2016-01-20 12:48:05 +00:00
if ( thing - > Z ( ) < = thing - > floorz )
2006-02-24 04:48:15 +00:00
{
return 1 ;
}
2014-08-07 16:05:39 +00:00
P_FindBelowIntersectors ( thing ) ;
lastintersect = intersectors . Size ( ) ;
2006-02-24 04:48:15 +00:00
for ( ; firstintersect < lastintersect ; firstintersect + + )
{
AActor * intersect = intersectors [ firstintersect ] ;
if ( ! ( intersect - > flags2 & MF2_PASSMOBJ ) | |
2010-11-30 08:18:11 +00:00
( ! ( intersect - > flags3 & MF3_ISMONSTER ) & & intersect - > Mass > mymass ) | |
( intersect - > flags4 & MF4_ACTLIKEBRIDGE )
2014-08-07 16:05:39 +00:00
)
{
2010-11-30 08:18:11 +00:00
// Can't push bridges or things more massive than ourself
2006-02-24 04:48:15 +00:00
return 2 ;
}
2016-01-20 12:48:05 +00:00
fixed_t oldz = intersect - > Z ( ) ;
2014-08-07 16:05:39 +00:00
P_AdjustFloorCeil ( intersect , cpos ) ;
2016-01-20 12:48:05 +00:00
if ( oldz > thing - > Z ( ) - intersect - > height )
2006-02-24 04:48:15 +00:00
{ // Only push things down, not up.
2016-01-20 12:48:05 +00:00
intersect - > SetZ ( thing - > Z ( ) - intersect - > height ) ;
2014-08-07 16:05:39 +00:00
if ( P_PushDown ( intersect , cpos ) )
2006-02-24 04:48:15 +00:00
{ // Move blocked
2014-08-07 16:05:39 +00:00
P_DoCrunch ( intersect , cpos ) ;
2016-01-20 12:48:05 +00:00
intersect - > SetZ ( oldz ) ;
2006-02-24 04:48:15 +00:00
return 2 ;
}
}
}
return 0 ;
}
//=============================================================================
//
// PIT_FloorDrop
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void PIT_FloorDrop ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
fixed_t oldfloorz = thing - > floorz ;
2016-01-21 09:23:20 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
P_AdjustFloorCeil ( thing , cpos ) ;
2006-02-24 04:48:15 +00:00
2008-06-12 10:08:50 +00:00
if ( oldfloorz = = thing - > floorz ) return ;
2010-11-30 08:18:11 +00:00
if ( thing - > flags4 & MF4_ACTLIKEBRIDGE ) return ; // do not move bridge things
2008-06-12 10:08:50 +00:00
2009-06-30 20:57:51 +00:00
if ( thing - > velz = = 0 & &
2006-02-24 04:48:15 +00:00
( ! ( thing - > flags & MF_NOGRAVITY ) | |
2016-01-20 12:48:05 +00:00
( thing - > Z ( ) = = oldfloorz & & ! ( thing - > flags & MF_NOLIFTDROP ) ) ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
2012-07-06 03:42:03 +00:00
if ( ( thing - > flags & MF_NOGRAVITY ) | | ( thing - > flags5 & MF5_MOVEWITHSECTOR ) | |
2014-08-07 16:05:39 +00:00
( ( ( cpos - > sector - > Flags & SECF_FLOORDROP ) | | cpos - > moveamt < 9 * FRACUNIT )
2016-01-20 12:48:05 +00:00
& & thing - > Z ( ) - thing - > floorz < = cpos - > moveamt ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > floorz ) ;
2014-08-07 16:05:39 +00:00
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2006-02-24 04:48:15 +00:00
}
}
2016-01-20 12:48:05 +00:00
else if ( ( thing - > Z ( ) ! = oldfloorz & & ! ( thing - > flags & MF_NOLIFTDROP ) ) )
2013-08-10 07:32:55 +00:00
{
2016-01-20 12:48:05 +00:00
fixed_t oldz = thing - > Z ( ) ;
2013-08-10 07:32:55 +00:00
if ( ( thing - > flags & MF_NOGRAVITY ) & & ( thing - > flags6 & MF6_RELATIVETOFLOOR ) )
{
2016-01-20 12:48:05 +00:00
thing - > AddZ ( - oldfloorz + thing - > floorz ) ;
2014-08-07 16:05:39 +00:00
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2013-08-10 07:32:55 +00:00
}
}
2016-01-21 09:23:20 +00:00
if ( thing - > player & & thing - > player - > mo = = thing )
{
thing - > player - > viewz + = thing - > Z ( ) - oldz ;
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// PIT_FloorRaise
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void PIT_FloorRaise ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
fixed_t oldfloorz = thing - > floorz ;
2016-01-20 12:48:05 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
P_AdjustFloorCeil ( thing , cpos ) ;
2006-02-24 04:48:15 +00:00
2008-06-12 10:08:50 +00:00
if ( oldfloorz = = thing - > floorz ) return ;
2006-02-24 04:48:15 +00:00
// Move things intersecting the floor up
2016-01-20 12:48:05 +00:00
if ( thing - > Z ( ) < = thing - > floorz )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
if ( thing - > flags4 & MF4_ACTLIKEBRIDGE )
2010-11-30 08:18:11 +00:00
{
cpos - > nofit = true ;
return ; // do not move bridge things
}
2014-08-07 16:05:39 +00:00
intersectors . Clear ( ) ;
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > floorz ) ;
2013-08-10 07:32:55 +00:00
}
else
{
2014-08-07 16:05:39 +00:00
if ( ( thing - > flags & MF_NOGRAVITY ) & & ( thing - > flags6 & MF6_RELATIVETOFLOOR ) )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
intersectors . Clear ( ) ;
2016-01-20 12:48:05 +00:00
thing - > AddZ ( - oldfloorz + thing - > floorz ) ;
2006-02-24 04:48:15 +00:00
}
2013-08-10 07:32:55 +00:00
else return ;
}
2014-08-07 16:05:39 +00:00
switch ( P_PushUp ( thing , cpos ) )
2013-08-10 07:32:55 +00:00
{
default :
2014-08-07 16:05:39 +00:00
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2013-08-10 07:32:55 +00:00
break ;
case 1 :
2014-08-07 16:05:39 +00:00
P_DoCrunch ( thing , cpos ) ;
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2013-08-10 07:32:55 +00:00
break ;
case 2 :
2014-08-07 16:05:39 +00:00
P_DoCrunch ( thing , cpos ) ;
2016-01-20 12:48:05 +00:00
thing - > SetZ ( oldz ) ;
2013-08-10 07:32:55 +00:00
break ;
2006-02-24 04:48:15 +00:00
}
2016-01-21 09:23:20 +00:00
if ( thing - > player & & thing - > player - > mo = = thing )
{
thing - > player - > viewz + = thing - > Z ( ) - oldz ;
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// PIT_CeilingLower
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void PIT_CeilingLower ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
bool onfloor ;
2016-01-21 09:23:20 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
onfloor = thing - > Z ( ) < = thing - > floorz ;
2014-08-07 16:05:39 +00:00
P_AdjustFloorCeil ( thing , cpos ) ;
2006-02-24 04:48:15 +00:00
2016-01-20 12:48:05 +00:00
if ( thing - > Top ( ) > thing - > ceilingz )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
if ( thing - > flags4 & MF4_ACTLIKEBRIDGE )
2010-11-30 08:18:11 +00:00
{
cpos - > nofit = true ;
return ; // do not move bridge things
}
2014-08-07 16:05:39 +00:00
intersectors . Clear ( ) ;
2016-01-20 12:48:05 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
if ( thing - > ceilingz - thing - > height > = thing - > floorz )
{
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > ceilingz - thing - > height ) ;
2006-02-24 04:48:15 +00:00
}
else
{
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > floorz ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
switch ( P_PushDown ( thing , cpos ) )
2006-02-24 04:48:15 +00:00
{
case 2 :
// intentional fall-through
case 1 :
if ( onfloor )
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > floorz ) ;
2014-08-07 16:05:39 +00:00
P_DoCrunch ( thing , cpos ) ;
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2006-02-24 04:48:15 +00:00
break ;
default :
2014-08-07 16:05:39 +00:00
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2006-02-24 04:48:15 +00:00
break ;
}
}
2016-01-21 09:23:20 +00:00
if ( thing - > player & & thing - > player - > mo = = thing )
{
thing - > player - > viewz + = thing - > Z ( ) - oldz ;
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// PIT_CeilingRaise
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void PIT_CeilingRaise ( AActor * thing , FChangePosition * cpos )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
bool isgood = P_AdjustFloorCeil ( thing , cpos ) ;
2016-01-21 09:23:20 +00:00
fixed_t oldz = thing - > Z ( ) ;
2006-02-24 04:48:15 +00:00
2010-11-30 08:18:11 +00:00
if ( thing - > flags4 & MF4_ACTLIKEBRIDGE ) return ; // do not move bridge things
2006-02-24 04:48:15 +00:00
// For DOOM compatibility, only move things that are inside the floor.
// (or something else?) Things marked as hanging from the ceiling will
// stay where they are.
2016-01-20 12:48:05 +00:00
if ( thing - > Z ( ) < thing - > floorz & &
thing - > Top ( ) > = thing - > ceilingz - cpos - > moveamt & &
2006-05-22 09:53:09 +00:00
! ( thing - > flags & MF_NOLIFTDROP ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
fixed_t oldz = thing - > Z ( ) ;
thing - > SetZ ( thing - > floorz ) ;
if ( thing - > Top ( ) > thing - > ceilingz )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
thing - > SetZ ( thing - > ceilingz - thing - > height ) ;
2006-02-24 04:48:15 +00:00
}
2014-08-07 16:05:39 +00:00
P_CheckFakeFloorTriggers ( thing , oldz ) ;
2006-02-24 04:48:15 +00:00
}
2016-01-20 12:48:05 +00:00
else if ( ( thing - > flags2 & MF2_PASSMOBJ ) & & ! isgood & & thing - > Top ( ) < thing - > ceilingz )
2006-02-24 04:48:15 +00:00
{
2008-04-10 14:38:43 +00:00
AActor * onmobj ;
2016-01-20 12:48:05 +00:00
if ( ! P_TestMobjZ ( thing , true , & onmobj ) & & onmobj - > Z ( ) < = thing - > Z ( ) )
2006-02-24 04:48:15 +00:00
{
2016-01-20 12:48:05 +00:00
thing - > SetZ ( MIN ( thing - > ceilingz - thing - > height , onmobj - > Top ( ) ) ) ;
2006-02-24 04:48:15 +00:00
}
}
2016-01-21 09:23:20 +00:00
if ( thing - > player & & thing - > player - > mo = = thing )
{
thing - > player - > viewz + = thing - > Z ( ) - oldz ;
}
2006-02-24 04:48:15 +00:00
}
//=============================================================================
//
// P_ChangeSector [RH] Was P_CheckSector in BOOM
//
// jff 3/19/98 added to just check monsters on the periphery
// of a moving sector instead of all in bounding box of the
// sector. Both more accurate and faster.
//
//=============================================================================
2014-08-07 16:05:39 +00:00
bool P_ChangeSector ( sector_t * sector , int crunch , int amt , int floorOrCeil , bool isreset )
2006-02-24 04:48:15 +00:00
{
2008-03-18 18:18:18 +00:00
FChangePosition cpos ;
2014-08-07 16:05:39 +00:00
void ( * iterator ) ( AActor * , FChangePosition * ) ;
void ( * iterator2 ) ( AActor * , FChangePosition * ) = NULL ;
2006-02-24 04:48:15 +00:00
msecnode_t * n ;
2008-03-18 18:18:18 +00:00
cpos . nofit = false ;
cpos . crushchange = crunch ;
2014-08-07 16:05:39 +00:00
cpos . moveamt = abs ( amt ) ;
2008-03-18 18:18:18 +00:00
cpos . movemidtex = false ;
2008-05-22 19:35:38 +00:00
cpos . sector = sector ;
2006-02-24 04:48:15 +00:00
2009-01-04 15:00:29 +00:00
// Also process all sectors that have 3D floors transferred from the
// changed sector.
2014-08-07 16:05:39 +00:00
if ( sector - > e - > XFloor . attached . Size ( ) )
2009-01-04 15:00:29 +00:00
{
unsigned i ;
sector_t * sec ;
// Use different functions for the four different types of sector movement.
// for 3D-floors the meaning of floor and ceiling is inverted!!!
if ( floorOrCeil = = 1 )
2014-08-07 16:05:39 +00:00
{
2009-01-04 15:00:29 +00:00
iterator = ( amt > = 0 ) ? PIT_FloorRaise : PIT_FloorDrop ;
}
else
2014-08-07 16:05:39 +00:00
{
iterator = ( amt > = 0 ) ? PIT_CeilingRaise : PIT_CeilingLower ;
2009-01-04 15:00:29 +00:00
}
2014-08-07 16:05:39 +00:00
for ( i = 0 ; i < sector - > e - > XFloor . attached . Size ( ) ; i + + )
2009-01-04 15:00:29 +00:00
{
sec = sector - > e - > XFloor . attached [ i ] ;
P_Recalculate3DFloors ( sec ) ; // Must recalculate the 3d floor and light lists
// no thing checks for attached sectors because of heightsec
2014-08-07 16:05:39 +00:00
if ( sec - > heightsec = = sector ) continue ;
2009-01-04 15:00:29 +00:00
2014-08-07 16:05:39 +00:00
for ( n = sec - > touching_thinglist ; n ; n = n - > m_snext ) n - > visited = false ;
do
2009-01-04 15:00:29 +00:00
{
2014-08-07 16:05:39 +00:00
for ( n = sec - > touching_thinglist ; n ; n = n - > m_snext )
2009-01-04 15:00:29 +00:00
{
if ( ! n - > visited )
{
2014-08-07 16:05:39 +00:00
n - > visited = true ;
2009-04-10 06:50:39 +00:00
if ( ! ( n - > m_thing - > flags & MF_NOBLOCKMAP ) | | //jff 4/7/98 don't do these
( n - > m_thing - > flags5 & MF5_MOVEWITHSECTOR ) )
{
iterator ( n - > m_thing , & cpos ) ;
}
2009-01-04 15:00:29 +00:00
break ;
}
}
2014-08-07 16:05:39 +00:00
} while ( n ) ;
2016-02-14 15:26:27 +00:00
sec - > CheckPortalPlane ( ! floorOrCeil ) ;
2009-01-04 15:00:29 +00:00
}
}
P_Recalculate3DFloors ( sector ) ; // Must recalculate the 3d floor and light lists
// [RH] Use different functions for the four different types of sector
2009-04-09 02:25:37 +00:00
// movement.
2008-03-18 18:18:18 +00:00
switch ( floorOrCeil )
{
case 0 :
// floor
2006-02-24 04:48:15 +00:00
iterator = ( amt < 0 ) ? PIT_FloorDrop : PIT_FloorRaise ;
2008-03-18 18:18:18 +00:00
break ;
case 1 :
// ceiling
2006-02-24 04:48:15 +00:00
iterator = ( amt < 0 ) ? PIT_CeilingLower : PIT_CeilingRaise ;
2008-03-18 18:18:18 +00:00
break ;
case 2 :
// 3dmidtex
// This must check both floor and ceiling
iterator = ( amt < 0 ) ? PIT_FloorDrop : PIT_FloorRaise ;
iterator2 = ( amt < 0 ) ? PIT_CeilingLower : PIT_CeilingRaise ;
cpos . movemidtex = true ;
break ;
default :
// invalid
assert ( floorOrCeil > 0 & & floorOrCeil < 2 ) ;
return false ;
2006-02-24 04:48:15 +00:00
}
// killough 4/4/98: scan list front-to-back until empty or exhausted,
// restarting from beginning after each thing is processed. Avoids
// crashes, and is sure to examine all things in the sector, and only
// the things which are in the sector, until a steady-state is reached.
// Things can arbitrarily be inserted and removed and it won't mess up.
//
// killough 4/7/98: simplified to avoid using complicated counter
// Mark all things invalid
for ( n = sector - > touching_thinglist ; n ; n = n - > m_snext )
n - > visited = false ;
do
{
for ( n = sector - > touching_thinglist ; n ; n = n - > m_snext ) // go through list
{
if ( ! n - > visited ) // unprocessed thing found
{
n - > visited = true ; // mark thing as processed
2009-04-09 02:25:37 +00:00
if ( ! ( n - > m_thing - > flags & MF_NOBLOCKMAP ) | | //jff 4/7/98 don't do these
( n - > m_thing - > flags5 & MF5_MOVEWITHSECTOR ) )
2008-03-18 18:18:18 +00:00
{
2014-08-07 16:05:39 +00:00
iterator ( n - > m_thing , & cpos ) ; // process it
if ( iterator2 ! = NULL ) iterator2 ( n - > m_thing , & cpos ) ;
2008-03-18 18:18:18 +00:00
}
2006-02-24 04:48:15 +00:00
break ; // exit and start over
}
}
} while ( n ) ; // repeat from scratch until all things left are marked valid
2016-02-14 15:26:27 +00:00
sector - > CheckPortalPlane ( floorOrCeil ) ; // check for portal obstructions after everything is done.
2008-03-22 12:17:52 +00:00
if ( ! cpos . nofit & & ! isreset /* && sector->MoreFlags & (SECF_UNDERWATERMASK)*/ )
{
// If this is a control sector for a deep water transfer, all actors in affected
// sectors need to have their waterlevel information updated and if applicable,
// execute appropriate sector actions.
// Only check if the sector move was successful.
TArray < sector_t * > & secs = sector - > e - > FakeFloor . Sectors ;
2014-08-07 16:05:39 +00:00
for ( unsigned i = 0 ; i < secs . Size ( ) ; i + + )
2008-03-22 12:17:52 +00:00
{
sector_t * s = secs [ i ] ;
for ( n = s - > touching_thinglist ; n ; n = n - > m_snext )
n - > visited = false ;
do
{
for ( n = s - > touching_thinglist ; n ; n = n - > m_snext ) // go through list
{
if ( ! n - > visited & & n - > m_thing - > Sector = = s ) // unprocessed thing found
{
n - > visited = true ; // mark thing as processed
2016-01-20 12:48:05 +00:00
n - > m_thing - > UpdateWaterLevel ( n - > m_thing - > Z ( ) , false ) ;
P_CheckFakeFloorTriggers ( n - > m_thing , n - > m_thing - > Z ( ) - amt ) ;
2008-03-22 12:17:52 +00:00
}
}
} while ( n ) ; // repeat from scratch until all things left are marked valid
}
}
2008-03-18 18:18:18 +00:00
return cpos . nofit ;
2006-02-24 04:48:15 +00:00
}
//=============================================================================
// phares 3/21/98
//
// Maintain a freelist of msecnode_t's to reduce memory allocs and frees.
//=============================================================================
msecnode_t * headsecnode = NULL ;
//=============================================================================
//
// P_GetSecnode
//
// Retrieve a node from the freelist. The calling routine
// should make sure it sets all fields properly.
//
//=============================================================================
msecnode_t * P_GetSecnode ( )
{
msecnode_t * node ;
if ( headsecnode )
{
node = headsecnode ;
headsecnode = headsecnode - > m_snext ;
}
else
{
2014-08-07 16:05:39 +00:00
node = ( msecnode_t * ) M_Malloc ( sizeof ( * node ) ) ;
2006-02-24 04:48:15 +00:00
}
return node ;
}
//=============================================================================
//
// P_PutSecnode
//
// Returns a node to the freelist.
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_PutSecnode ( msecnode_t * node )
2006-02-24 04:48:15 +00:00
{
node - > m_snext = headsecnode ;
headsecnode = node ;
}
//=============================================================================
// phares 3/16/98
//
// P_AddSecnode
//
// Searches the current list to see if this sector is
// already there. If not, it adds a sector node at the head of the list of
// sectors this object appears in. This is called when creating a list of
// nodes that will get linked in later. Returns a pointer to the new node.
//
//=============================================================================
2014-08-07 16:05:39 +00:00
msecnode_t * P_AddSecnode ( sector_t * s , AActor * thing , msecnode_t * nextnode )
2006-02-24 04:48:15 +00:00
{
msecnode_t * node ;
if ( s = = 0 )
{
2014-12-21 01:11:39 +00:00
I_FatalError ( " AddSecnode of 0 for %s \n " , thing - > GetClass ( ) - > TypeName . GetChars ( ) ) ;
2006-02-24 04:48:15 +00:00
}
node = nextnode ;
while ( node )
{
if ( node - > m_sector = = s ) // Already have a node for this sector?
{
node - > m_thing = thing ; // Yes. Setting m_thing says 'keep it'.
return nextnode ;
}
node = node - > m_tnext ;
}
// Couldn't find an existing node for this sector. Add one at the head
// of the list.
node = P_GetSecnode ( ) ;
// killough 4/4/98, 4/7/98: mark new nodes unvisited.
node - > visited = 0 ;
node - > m_sector = s ; // sector
2014-08-07 16:05:39 +00:00
node - > m_thing = thing ; // mobj
node - > m_tprev = NULL ; // prev node on Thing thread
node - > m_tnext = nextnode ; // next node on Thing thread
2006-02-24 04:48:15 +00:00
if ( nextnode )
nextnode - > m_tprev = node ; // set back link on Thing
// Add new node at head of sector thread starting at s->touching_thinglist
2014-08-07 16:05:39 +00:00
node - > m_sprev = NULL ; // prev node on sector thread
node - > m_snext = s - > touching_thinglist ; // next node on sector thread
2006-02-24 04:48:15 +00:00
if ( s - > touching_thinglist )
node - > m_snext - > m_sprev = node ;
s - > touching_thinglist = node ;
return node ;
}
//=============================================================================
//
// P_DelSecnode
//
// Deletes a sector node from the list of
// sectors this object appears in. Returns a pointer to the next node
// on the linked list, or NULL.
//
//=============================================================================
2014-08-07 16:05:39 +00:00
msecnode_t * P_DelSecnode ( msecnode_t * node )
2006-02-24 04:48:15 +00:00
{
msecnode_t * tp ; // prev node on thing thread
msecnode_t * tn ; // next node on thing thread
msecnode_t * sp ; // prev node on sector thread
msecnode_t * sn ; // next node on sector thread
if ( node )
{
// Unlink from the Thing thread. The Thing thread begins at
// sector_list and not from AActor->touching_sectorlist.
tp = node - > m_tprev ;
tn = node - > m_tnext ;
if ( tp )
tp - > m_tnext = tn ;
if ( tn )
tn - > m_tprev = tp ;
// Unlink from the sector thread. This thread begins at
// sector_t->touching_thinglist.
sp = node - > m_sprev ;
sn = node - > m_snext ;
if ( sp )
sp - > m_snext = sn ;
else
node - > m_sector - > touching_thinglist = sn ;
if ( sn )
sn - > m_sprev = sp ;
// Return this node to the freelist
P_PutSecnode ( node ) ;
return tn ;
}
return NULL ;
} // phares 3/13/98
2006-05-18 04:25:26 +00:00
//=============================================================================
//
// P_DelSector_List
//
// Deletes the sector_list and NULLs it.
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_DelSector_List ( )
2006-05-18 04:25:26 +00:00
{
if ( sector_list ! = NULL )
{
2014-08-07 16:05:39 +00:00
P_DelSeclist ( sector_list ) ;
2006-05-18 04:25:26 +00:00
sector_list = NULL ;
}
}
2006-02-24 04:48:15 +00:00
//=============================================================================
//
// P_DelSeclist
//
// Delete an entire sector list
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_DelSeclist ( msecnode_t * node )
2006-02-24 04:48:15 +00:00
{
while ( node )
2014-08-07 16:05:39 +00:00
node = P_DelSecnode ( node ) ;
2006-02-24 04:48:15 +00:00
}
//=============================================================================
// phares 3/14/98
//
// P_CreateSecNodeList
//
// Alters/creates the sector_list that shows what sectors the object resides in
//
//=============================================================================
2014-08-07 16:05:39 +00:00
void P_CreateSecNodeList ( AActor * thing , fixed_t x , fixed_t y )
2006-02-24 04:48:15 +00:00
{
msecnode_t * node ;
// First, clear out the existing m_thing fields. As each node is
// added or verified as needed, m_thing will be set properly. When
// finished, delete all nodes where m_thing is still NULL. These
// represent the sectors the Thing has vacated.
node = sector_list ;
while ( node )
{
node - > m_thing = NULL ;
node = node - > m_tnext ;
}
2016-01-20 12:48:05 +00:00
FBoundingBox box ( thing - > X ( ) , thing - > Y ( ) , thing - > radius ) ;
2008-04-06 17:33:43 +00:00
FBlockLinesIterator it ( box ) ;
line_t * ld ;
2006-02-24 04:48:15 +00:00
2008-04-06 17:33:43 +00:00
while ( ( ld = it . Next ( ) ) )
{
2014-08-07 16:05:39 +00:00
if ( box . Right ( ) < = ld - > bbox [ BOXLEFT ] | |
box . Left ( ) > = ld - > bbox [ BOXRIGHT ] | |
box . Top ( ) < = ld - > bbox [ BOXBOTTOM ] | |
2008-04-06 17:33:43 +00:00
box . Bottom ( ) > = ld - > bbox [ BOXTOP ] )
continue ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
if ( box . BoxOnLineSide ( ld ) ! = - 1 )
2008-04-06 17:33:43 +00:00
continue ;
2006-02-24 04:48:15 +00:00
2008-04-06 17:33:43 +00:00
// This line crosses through the object.
2006-02-24 04:48:15 +00:00
2008-04-06 17:33:43 +00:00
// Collect the sector(s) from the line and add to the
// sector_list you're examining. If the Thing ends up being
// allowed to move to this position, then the sector_list
// will be attached to the Thing's AActor at touching_sectorlist.
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
sector_list = P_AddSecnode ( ld - > frontsector , thing , sector_list ) ;
2008-04-06 17:33:43 +00:00
// Don't assume all lines are 2-sided, since some Things
// like MT_TFOG are allowed regardless of whether their radius takes
// them beyond an impassable linedef.
// killough 3/27/98, 4/4/98:
// Use sidedefs instead of 2s flag to determine two-sidedness.
if ( ld - > backsector )
sector_list = P_AddSecnode ( ld - > backsector , thing , sector_list ) ;
}
2006-02-24 04:48:15 +00:00
// Add the sector of the (x,y) point to sector_list.
2014-08-07 16:05:39 +00:00
sector_list = P_AddSecnode ( thing - > Sector , thing , sector_list ) ;
2006-02-24 04:48:15 +00:00
// Now delete any nodes that won't be used. These are the ones where
// m_thing is still NULL.
node = sector_list ;
while ( node )
{
if ( node - > m_thing = = NULL )
{
if ( node = = sector_list )
sector_list = node - > m_tnext ;
2014-08-07 16:05:39 +00:00
node = P_DelSecnode ( node ) ;
2006-02-24 04:48:15 +00:00
}
else
{
node = node - > m_tnext ;
}
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2014-08-07 16:05:39 +00:00
void SpawnShootDecal ( AActor * t1 , const FTraceResults & trace )
2006-02-24 04:48:15 +00:00
{
FDecalBase * decalbase = NULL ;
if ( t1 - > player ! = NULL & & t1 - > player - > ReadyWeapon ! = NULL )
{
decalbase = t1 - > player - > ReadyWeapon - > GetDefault ( ) - > DecalGenerator ;
}
else
{
decalbase = t1 - > DecalGenerator ;
}
if ( decalbase ! = NULL )
{
2014-08-07 16:05:39 +00:00
DImpactDecal : : StaticCreate ( decalbase - > GetDecal ( ) ,
2009-09-06 20:45:56 +00:00
trace . X , trace . Y , trace . Z , trace . Line - > sidedef [ trace . Side ] , trace . ffloor ) ;
2006-02-24 04:48:15 +00:00
}
}
2009-10-11 17:44:50 +00:00
//==========================================================================
//
//
//
//==========================================================================
2014-08-07 16:05:39 +00:00
static void SpawnDeepSplash ( AActor * t1 , const FTraceResults & trace , AActor * puff ,
2011-09-21 19:39:12 +00:00
fixed_t vx , fixed_t vy , fixed_t vz , fixed_t shootz , bool ffloor )
2006-02-24 04:48:15 +00:00
{
2014-08-07 16:05:39 +00:00
const secplane_t * plane ;
2011-09-21 19:39:12 +00:00
if ( ffloor & & trace . Crossed3DWater )
plane = trace . Crossed3DWater - > top . plane ;
else if ( trace . CrossedWater & & trace . CrossedWater - > heightsec )
plane = & trace . CrossedWater - > heightsec - > floorplane ;
else return ;
2006-02-24 04:48:15 +00:00
2011-09-21 19:39:12 +00:00
fixed_t num , den , hitdist ;
2014-08-07 16:05:39 +00:00
den = TMulScale16 ( plane - > a , vx , plane - > b , vy , plane - > c , vz ) ;
2006-02-24 04:48:15 +00:00
if ( den ! = 0 )
{
2016-01-20 12:48:05 +00:00
num = TMulScale16 ( plane - > a , t1 - > X ( ) , plane - > b , t1 - > Y ( ) , plane - > c , shootz ) + plane - > d ;
2014-08-07 16:05:39 +00:00
hitdist = FixedDiv ( - num , den ) ;
2006-02-24 04:48:15 +00:00
if ( hitdist > = 0 & & hitdist < = trace . Distance )
{
2016-01-20 12:48:05 +00:00
fixed_t hitx = t1 - > X ( ) + FixedMul ( vx , hitdist ) ;
fixed_t hity = t1 - > Y ( ) + FixedMul ( vy , hitdist ) ;
2014-08-07 16:05:39 +00:00
fixed_t hitz = shootz + FixedMul ( vz , hitdist ) ;
2006-02-24 04:48:15 +00:00
2014-08-07 16:05:39 +00:00
P_HitWater ( puff ! = NULL ? puff : t1 , P_PointInSector ( hitx , hity ) , hitx , hity , hitz ) ;
2006-02-24 04:48:15 +00:00
}
}
}
2009-10-09 20:35:07 +00:00
//=============================================================================
//
// P_ActivateThingSpecial
//
// Handles the code for things activated by death, USESPECIAL or BUMPSPECIAL
//
//=============================================================================
bool P_ActivateThingSpecial ( AActor * thing , AActor * trigger , bool death )
{
bool res = false ;
// Target switching mechanism
if ( thing - > activationtype & THINGSPEC_ThingTargets ) thing - > target = trigger ;
if ( thing - > activationtype & THINGSPEC_TriggerTargets ) trigger - > target = thing ;
// State change mechanism. The thing needs to be not dead and to have at least one of the relevant flags
2014-08-07 16:05:39 +00:00
if ( ! death & & ( thing - > activationtype & ( THINGSPEC_Activate | THINGSPEC_Deactivate | THINGSPEC_Switch ) ) )
2009-10-09 20:35:07 +00:00
{
// If a switchable thing does not know whether it should be activated
// or deactivated, the default is to activate it.
2014-08-07 16:05:39 +00:00
if ( ( thing - > activationtype & THINGSPEC_Switch )
& & ! ( thing - > activationtype & ( THINGSPEC_Activate | THINGSPEC_Deactivate ) ) )
2009-10-09 20:35:07 +00:00
{
thing - > activationtype | = THINGSPEC_Activate ;
}
// Can it be activated?
if ( thing - > activationtype & THINGSPEC_Activate )
{
thing - > activationtype & = ~ THINGSPEC_Activate ; // Clear flag
if ( thing - > activationtype & THINGSPEC_Switch ) // Set other flag if switching
thing - > activationtype | = THINGSPEC_Deactivate ;
thing - > Activate ( trigger ) ;
res = true ;
}
// If not, can it be deactivated?
else if ( thing - > activationtype & THINGSPEC_Deactivate )
{
thing - > activationtype & = ~ THINGSPEC_Deactivate ; // Clear flag
if ( thing - > activationtype & THINGSPEC_Switch ) // Set other flag if switching
thing - > activationtype | = THINGSPEC_Activate ;
thing - > Deactivate ( trigger ) ;
res = true ;
}
}
// Run the special, if any
if ( thing - > special )
{
2014-08-07 16:05:39 +00:00
res = ! ! P_ExecuteSpecial ( thing - > special , NULL ,
2009-10-09 20:35:07 +00:00
// TriggerActs overrides the level flag, which only concerns thing activated by death
( ( ( death & & level . flags & LEVEL_ACTOWNSPECIAL & & ! ( thing - > activationtype & THINGSPEC_TriggerActs ) )
| | ( thing - > activationtype & THINGSPEC_ThingActs ) ) // Who triggers?
2014-08-07 16:05:39 +00:00
? thing : trigger ) ,
2009-10-09 20:35:07 +00:00
false , thing - > args [ 0 ] , thing - > args [ 1 ] , thing - > args [ 2 ] , thing - > args [ 3 ] , thing - > args [ 4 ] ) ;
// Clears the special if it was run on thing's death or if flag is set.
if ( death | | ( thing - > activationtype & THINGSPEC_ClearSpecial & & res ) ) thing - > special = 0 ;
}
// Returns the result
return res ;
}