2019-08-21 19:31:12 +00:00
//
//-----------------------------------------------------------------------------
//
// Copyright 2016 ZZYZX, Christoph Oelckers, et. al.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see http://www.gnu.org/licenses/
//
//--------------------------------------------------------------------------
//
// DESCRIPTION:
// Everything that has to do with portals
// (both of the line and sector variety)
//
//-----------------------------------------------------------------------------
2016-03-01 15:47:10 +00:00
# include "p_local.h"
# include "p_blockmap.h"
# include "p_lnspec.h"
# include "c_cvars.h"
# include "m_bbox.h"
# include "p_tags.h"
# include "v_text.h"
# include "a_sharedglobal.h"
# include "i_system.h"
# include "c_dispatch.h"
# include "p_maputl.h"
# include "p_spec.h"
# include "p_checkposition.h"
2016-03-11 14:45:47 +00:00
# include "math/cmath.h"
2017-01-08 17:45:30 +00:00
# include "g_levellocals.h"
2017-04-12 23:12:04 +00:00
# include "vm.h"
2016-03-01 15:47:10 +00:00
// simulation recurions maximum
CVAR ( Int , sv_portal_recursions , 4 , CVAR_ARCHIVE | CVAR_SERVERINFO )
FDisplacementTable Displacements ;
FPortalBlockmap PortalBlockmap ;
TArray < FLinePortal > linePortals ;
TArray < FLinePortal * > linkedPortals ; // only the linked portals, this is used to speed up looking for them in P_CollectConnectedGroups.
2017-01-14 15:05:40 +00:00
DEFINE_FIELD ( FSectorPortal , mType ) ;
DEFINE_FIELD ( FSectorPortal , mFlags ) ;
DEFINE_FIELD ( FSectorPortal , mPartner ) ;
DEFINE_FIELD ( FSectorPortal , mPlane ) ;
DEFINE_FIELD ( FSectorPortal , mOrigin ) ;
DEFINE_FIELD ( FSectorPortal , mDestination ) ;
DEFINE_FIELD ( FSectorPortal , mDisplacement ) ;
DEFINE_FIELD ( FSectorPortal , mPlaneZ ) ;
DEFINE_FIELD ( FSectorPortal , mSkybox ) ;
2016-04-20 17:20:11 +00:00
2016-03-01 15:47:10 +00:00
//============================================================================
//
// This is used to mark processed portals for some collection functions.
//
//============================================================================
struct FPortalBits
{
2017-03-08 14:20:00 +00:00
TArray < uint32_t > data ;
2016-03-01 15:47:10 +00:00
void setSize ( int num )
{
data . Resize ( ( num + 31 ) / 32 ) ;
clear ( ) ;
}
void clear ( )
{
2017-03-08 14:20:00 +00:00
memset ( & data [ 0 ] , 0 , data . Size ( ) * sizeof ( uint32_t ) ) ;
2016-03-01 15:47:10 +00:00
}
void setBit ( int group )
{
data [ group > > 5 ] | = ( 1 < < ( group & 31 ) ) ;
}
int getBit ( int group )
{
return data [ group > > 5 ] & ( 1 < < ( group & 31 ) ) ;
}
} ;
//============================================================================
//
// BuildBlockmap
//
//============================================================================
static void BuildBlockmap ( )
{
2017-03-17 13:24:21 +00:00
auto bmapwidth = level . blockmap . bmapwidth ;
auto bmapheight = level . blockmap . bmapheight ;
2016-03-01 15:47:10 +00:00
PortalBlockmap . Clear ( ) ;
PortalBlockmap . Create ( bmapwidth , bmapheight ) ;
for ( int y = 0 ; y < bmapheight ; y + + )
{
for ( int x = 0 ; x < bmapwidth ; x + + )
{
2017-03-17 13:24:21 +00:00
int * list = level . blockmap . GetLines ( x , y ) ;
2016-03-01 15:47:10 +00:00
FPortalBlock & block = PortalBlockmap ( x , y ) ;
while ( * list ! = - 1 )
{
2017-01-08 13:39:16 +00:00
line_t * ld = & level . lines [ * list + + ] ;
2016-04-21 20:59:07 +00:00
FLinePortal * port = ld - > getPortal ( ) ;
if ( port & & port - > mType ! = PORTT_VISUAL )
2016-03-01 15:47:10 +00:00
{
PortalBlockmap . containsLines = true ;
block . portallines . Push ( ld ) ;
block . neighborContainsLines = true ;
2016-03-07 20:58:34 +00:00
if ( ld - > getPortal ( ) - > mType = = PORTT_LINKED ) block . containsLinkedPortals = true ;
2016-03-01 15:47:10 +00:00
if ( x > 0 ) PortalBlockmap ( x - 1 , y ) . neighborContainsLines = true ;
if ( y > 0 ) PortalBlockmap ( x , y - 1 ) . neighborContainsLines = true ;
if ( x < PortalBlockmap . dx - 1 ) PortalBlockmap ( x + 1 , y ) . neighborContainsLines = true ;
if ( y < PortalBlockmap . dy - 1 ) PortalBlockmap ( x , y + 1 ) . neighborContainsLines = true ;
}
2016-03-07 20:58:34 +00:00
else
{
bool yes = ld - > frontsector - > PortalIsLinked ( sector_t : : ceiling ) | | ld - > frontsector - > PortalIsLinked ( sector_t : : floor ) ;
if ( ld - > backsector )
{
yes | = ld - > backsector - > PortalIsLinked ( sector_t : : ceiling ) | | ld - > backsector - > PortalIsLinked ( sector_t : : floor ) ;
}
block . containsLinkedPortals | = yes ;
PortalBlockmap . hasLinkedSectorPortals | = yes ;
}
2016-03-01 15:47:10 +00:00
}
}
}
}
//===========================================================================
//
// FLinePortalTraverse :: AddLineIntercepts.
//
// Similar to AddLineIntercepts but checks the portal blockmap for line-to-line portals
//
//===========================================================================
void FLinePortalTraverse : : AddLineIntercepts ( int bx , int by )
{
2017-03-27 07:55:51 +00:00
if ( by < 0 | | by > = PortalBlockmap . dy | | bx < 0 | | bx > = PortalBlockmap . dx ) return ;
2016-03-24 17:26:27 +00:00
2016-03-01 15:47:10 +00:00
FPortalBlock & block = PortalBlockmap ( bx , by ) ;
for ( unsigned i = 0 ; i < block . portallines . Size ( ) ; i + + )
{
line_t * ld = block . portallines [ i ] ;
2016-03-31 14:52:25 +00:00
double frac ;
divline_t dl ;
2016-03-01 15:47:10 +00:00
if ( ld - > validcount = = validcount ) continue ; // already processed
2016-03-31 14:52:25 +00:00
if ( P_PointOnDivlineSide ( ld - > v1 - > fPos ( ) , & trace ) = =
P_PointOnDivlineSide ( ld - > v2 - > fPos ( ) , & trace ) )
2016-03-01 15:47:10 +00:00
{
continue ; // line isn't crossed
}
P_MakeDivline ( ld , & dl ) ;
2016-03-31 14:52:25 +00:00
if ( P_PointOnDivlineSide ( trace . x , trace . y , & dl ) ! = 0 | |
P_PointOnDivlineSide ( trace . x + trace . dx , trace . y + trace . dy , & dl ) ! = 1 )
2016-03-01 15:47:10 +00:00
{
continue ; // line isn't crossed from the front side
}
// hit the line
P_MakeDivline ( ld , & dl ) ;
frac = P_InterceptVector ( & trace , & dl ) ;
2016-03-31 14:52:25 +00:00
if ( frac < 0 | | frac > 1. ) continue ; // behind source
2016-03-01 15:47:10 +00:00
intercept_t newintercept ;
newintercept . frac = frac ;
newintercept . isaline = true ;
newintercept . done = false ;
newintercept . d . line = ld ;
intercepts . Push ( newintercept ) ;
}
}
//============================================================================
//
// finds the destination for a line portal for spawning
//
//============================================================================
static line_t * FindDestination ( line_t * src , int tag )
{
if ( tag )
{
int lineno = - 1 ;
FLineIdIterator it ( tag ) ;
while ( ( lineno = it . Next ( ) ) > = 0 )
{
2017-01-08 13:39:16 +00:00
if ( & level . lines [ lineno ] ! = src )
2016-03-01 15:47:10 +00:00
{
2017-01-08 13:39:16 +00:00
return & level . lines [ lineno ] ;
2016-03-01 15:47:10 +00:00
}
}
}
2017-02-14 22:13:29 +00:00
return nullptr ;
2016-03-01 15:47:10 +00:00
}
2016-03-04 13:09:26 +00:00
//============================================================================
//
//
//
//============================================================================
static void SetRotation ( FLinePortal * port )
{
2016-06-19 10:32:45 +00:00
if ( port ! = nullptr & & port - > mDestination ! = nullptr )
2016-03-12 21:37:43 +00:00
{
2016-06-19 10:32:45 +00:00
if ( port - > mType ! = PORTT_LINKED )
{
line_t * dst = port - > mDestination ;
line_t * line = port - > mOrigin ;
DAngle angle = dst - > Delta ( ) . Angle ( ) - line - > Delta ( ) . Angle ( ) + 180. ;
port - > mSinRot = sindeg ( angle . Degrees ) ; // Here precision matters so use the slower but more precise versions.
port - > mCosRot = cosdeg ( angle . Degrees ) ;
port - > mAngleDiff = angle ;
if ( ( line - > sidedef [ 0 ] - > Flags & WALLF_POLYOBJ ) | | ( dst - > sidedef [ 0 ] - > Flags & WALLF_POLYOBJ ) )
{
port - > mFlags | = PORTF_POLYOBJ ;
}
else
{
2016-06-29 10:13:24 +00:00
port - > mFlags & = ~ PORTF_POLYOBJ ;
2016-06-19 10:32:45 +00:00
}
}
else
{
// Linked portals have no angular difference.
2016-08-08 11:06:29 +00:00
port - > mSinRot = 0. ;
port - > mCosRot = 1. ;
2016-06-19 10:32:45 +00:00
port - > mAngleDiff = 0. ;
}
2016-03-12 21:37:43 +00:00
}
2016-03-04 13:09:26 +00:00
}
2016-03-01 15:47:10 +00:00
//============================================================================
//
// Spawns a single line portal
//
//============================================================================
void P_SpawnLinePortal ( line_t * line )
{
// portal destination is special argument #0
2017-02-14 22:13:29 +00:00
line_t * dst = nullptr ;
2016-03-01 15:47:10 +00:00
if ( line - > args [ 2 ] > = PORTT_VISUAL & & line - > args [ 2 ] < = PORTT_LINKED )
{
dst = FindDestination ( line , line - > args [ 0 ] ) ;
line - > portalindex = linePortals . Reserve ( 1 ) ;
FLinePortal * port = & linePortals . Last ( ) ;
port - > mOrigin = line ;
port - > mDestination = dst ;
2017-03-08 14:20:00 +00:00
port - > mType = uint8_t ( line - > args [ 2 ] ) ; // range check is done above.
2016-03-01 15:47:10 +00:00
if ( port - > mType = = PORTT_LINKED )
{
// Linked portals have no z-offset ever.
port - > mAlign = PORG_ABSOLUTE ;
}
else
{
2017-03-08 14:20:00 +00:00
port - > mAlign = uint8_t ( line - > args [ 3 ] > = PORG_ABSOLUTE & & line - > args [ 3 ] < = PORG_CEILING ? line - > args [ 3 ] : PORG_ABSOLUTE ) ;
2016-05-25 09:36:23 +00:00
if ( port - > mType = = PORTT_INTERACTIVE & & port - > mAlign ! = PORG_ABSOLUTE )
2016-03-01 15:47:10 +00:00
{
// Due to the way z is often handled, these pose a major issue for parts of the code that needs to transparently handle interactive portals.
2017-01-08 13:39:16 +00:00
Printf ( TEXTCOLOR_RED " Warning: z-offsetting not allowed for interactive portals. Changing line %d to teleport-portal! \n " , line - > Index ( ) ) ;
2016-03-01 15:47:10 +00:00
port - > mType = PORTT_TELEPORT ;
}
}
2017-09-30 05:59:30 +00:00
port - > mDefFlags = port - > mType = = PORTT_VISUAL ? PORTF_VISIBLE :
port - > mType = = PORTT_TELEPORT ? PORTF_TYPETELEPORT :
PORTF_TYPEINTERACTIVE ;
2016-03-01 15:47:10 +00:00
}
else if ( line - > args [ 2 ] = = PORTT_LINKEDEE & & line - > args [ 0 ] = = 0 )
{
// EE-style portals require that the first line ID is identical and the first arg of the two linked linedefs are 0 and 1 respectively.
int mytag = tagManager . GetFirstLineID ( line ) ;
2017-01-08 13:39:16 +00:00
for ( auto & ln : level . lines )
2016-03-01 15:47:10 +00:00
{
2017-01-08 13:39:16 +00:00
if ( tagManager . GetFirstLineID ( & ln ) = = mytag & & ln . args [ 0 ] = = 1 & & ln . special = = Line_SetPortal )
2016-03-01 15:47:10 +00:00
{
line - > portalindex = linePortals . Reserve ( 1 ) ;
FLinePortal * port = & linePortals . Last ( ) ;
memset ( port , 0 , sizeof ( FLinePortal ) ) ;
port - > mOrigin = line ;
2017-01-08 13:39:16 +00:00
port - > mDestination = & ln ;
2016-03-01 15:47:10 +00:00
port - > mType = PORTT_LINKED ;
port - > mAlign = PORG_ABSOLUTE ;
port - > mDefFlags = PORTF_TYPEINTERACTIVE ;
// we need to create the backlink here, too.
2017-01-08 13:39:16 +00:00
ln . portalindex = linePortals . Reserve ( 1 ) ;
2016-03-01 15:47:10 +00:00
port = & linePortals . Last ( ) ;
memset ( port , 0 , sizeof ( FLinePortal ) ) ;
2017-01-08 13:39:16 +00:00
port - > mOrigin = & ln ;
2016-03-01 15:47:10 +00:00
port - > mDestination = line ;
port - > mType = PORTT_LINKED ;
port - > mAlign = PORG_ABSOLUTE ;
port - > mDefFlags = PORTF_TYPEINTERACTIVE ;
}
}
}
else
{
// undefined type
return ;
}
}
//============================================================================
//
// Update a line portal's state after all have been spawned
//
//============================================================================
void P_UpdatePortal ( FLinePortal * port )
{
2017-09-30 05:59:30 +00:00
if ( port - > mType ! = PORTT_VISUAL & & port - > mOrigin - > backsector = = nullptr & & ! ( port - > mOrigin - > sidedef [ 0 ] - > Flags & WALLF_POLYOBJ ) )
{
Printf ( TEXTCOLOR_RED " Warning: Traversable portals must have a back-sector and empty space behind them (or be on a polyobject)! Changing line %d to visual-portal! \n " , port - > mOrigin - > Index ( ) ) ;
port - > mType = PORTT_VISUAL ;
2017-10-02 14:07:47 +00:00
port - > mDefFlags & = ~ ( PORTF_PASSABLE | PORTF_SOUNDTRAVERSE | PORTF_INTERACTIVE ) ;
2017-09-30 05:59:30 +00:00
}
2017-02-14 22:13:29 +00:00
if ( port - > mDestination = = nullptr )
2016-03-01 15:47:10 +00:00
{
// Portal has no destination: switch it off
port - > mFlags = 0 ;
}
else if ( port - > mDestination - > getPortalDestination ( ) ! = port - > mOrigin )
{
//portal doesn't link back. This will be a simple teleporter portal.
port - > mFlags = port - > mDefFlags & ~ PORTF_INTERACTIVE ;
if ( port - > mType = = PORTT_LINKED )
{
// this is illegal. Demote the type to TELEPORT
port - > mType = PORTT_TELEPORT ;
port - > mDefFlags & = ~ PORTF_INTERACTIVE ;
2017-09-30 05:59:30 +00:00
Printf ( TEXTCOLOR_RED " Warning: linked portal did not have matching reverse portal. Changing line %d to teleport-portal! \n " , port - > mOrigin - > Index ( ) ) ;
2016-03-01 15:47:10 +00:00
}
}
else
{
port - > mFlags = port - > mDefFlags ;
if ( port - > mType = = PORTT_LINKED )
{
if ( linePortals [ port - > mDestination - > portalindex ] . mType ! = PORTT_LINKED )
{
port - > mType = PORTT_INTERACTIVE ; // linked portals must be two-way.
}
else
{
2016-04-17 15:10:11 +00:00
port - > mDisplacement = port - > mDestination - > v2 - > fPos ( ) - port - > mOrigin - > v1 - > fPos ( ) ;
2016-03-01 15:47:10 +00:00
}
}
}
2016-06-19 10:40:38 +00:00
// Cache the angle between the two linedefs, for rotating.
SetRotation ( port ) ;
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// Collect a separate list of linked portals so that these can be
// processed faster without the simpler types interfering.
//
//============================================================================
void P_CollectLinkedPortals ( )
{
linkedPortals . Clear ( ) ;
for ( unsigned i = 0 ; i < linePortals . Size ( ) ; i + + )
{
FLinePortal * port = & linePortals [ i ] ;
if ( port - > mType = = PORTT_LINKED )
{
linkedPortals . Push ( port ) ;
}
}
}
//============================================================================
//
// Post-process all line portals
//
//============================================================================
void P_FinalizePortals ( )
{
for ( unsigned i = 0 ; i < linePortals . Size ( ) ; i + + )
{
FLinePortal * port = & linePortals [ i ] ;
P_UpdatePortal ( port ) ;
}
P_CollectLinkedPortals ( ) ;
BuildBlockmap ( ) ;
P_CreateLinkedPortals ( ) ;
}
//============================================================================
//
// Change the destination of a portal
//
//============================================================================
static bool ChangePortalLine ( line_t * line , int destid )
{
if ( line - > portalindex > = linePortals . Size ( ) ) return false ;
FLinePortal * port = & linePortals [ line - > portalindex ] ;
if ( port - > mType = = PORTT_LINKED ) return false ; // linked portals cannot be changed.
2017-02-14 22:13:29 +00:00
if ( destid = = 0 ) port - > mDestination = nullptr ;
2016-03-01 15:47:10 +00:00
port - > mDestination = FindDestination ( line , destid ) ;
2017-02-14 22:13:29 +00:00
if ( port - > mDestination = = nullptr )
2016-03-01 15:47:10 +00:00
{
port - > mFlags = 0 ;
}
else if ( port - > mType = = PORTT_INTERACTIVE )
{
2017-02-14 22:13:29 +00:00
FLinePortal * portd = port - > mDestination - > portalindex < linePortals . Size ( ) ? & linePortals [ port - > mDestination - > portalindex ] : nullptr ;
if ( portd ! = nullptr & & portd - > mType = = PORTT_INTERACTIVE & & portd - > mDestination = = line )
2016-03-01 15:47:10 +00:00
{
// this is a 2-way interactive portal
port - > mFlags = port - > mDefFlags | PORTF_INTERACTIVE ;
portd - > mFlags = portd - > mDefFlags | PORTF_INTERACTIVE ;
}
else
{
port - > mFlags = port - > mDefFlags ;
}
2016-03-04 13:09:26 +00:00
SetRotation ( portd ) ;
2016-03-01 15:47:10 +00:00
}
2017-09-30 05:59:30 +00:00
else
{
port - > mFlags = port - > mDefFlags ;
}
2016-03-04 13:09:26 +00:00
SetRotation ( port ) ;
2016-03-01 15:47:10 +00:00
return true ;
}
//============================================================================
//
// Change the destination of a group of portals
//
//============================================================================
bool P_ChangePortal ( line_t * ln , int thisid , int destid )
{
int lineno ;
if ( thisid = = 0 ) return ChangePortalLine ( ln , destid ) ;
FLineIdIterator it ( thisid ) ;
bool res = false ;
while ( ( lineno = it . Next ( ) ) > = 0 )
{
2017-01-08 13:39:16 +00:00
res | = ChangePortalLine ( & level . lines [ lineno ] , destid ) ;
2016-03-01 15:47:10 +00:00
}
return res ;
}
//============================================================================
//
2016-04-20 17:20:11 +00:00
// clears all portal data for a new level start
2016-03-01 15:47:10 +00:00
//
//============================================================================
void P_ClearPortals ( )
{
Displacements . Create ( 1 ) ;
linePortals . Clear ( ) ;
linkedPortals . Clear ( ) ;
2017-01-14 15:05:40 +00:00
level . sectorPortals . Resize ( 2 ) ;
2016-04-20 17:20:11 +00:00
// The first entry must always be the default skybox. This is what every sector gets by default.
2017-01-14 15:05:40 +00:00
memset ( & level . sectorPortals [ 0 ] , 0 , sizeof ( level . sectorPortals [ 0 ] ) ) ;
level . sectorPortals [ 0 ] . mType = PORTS_SKYVIEWPOINT ;
level . sectorPortals [ 0 ] . mFlags = PORTSF_SKYFLATONLY ;
2016-04-20 17:20:11 +00:00
// The second entry will be the default sky. This is for forcing a regular sky through the skybox picker
2017-01-14 15:05:40 +00:00
memset ( & level . sectorPortals [ 1 ] , 0 , sizeof ( level . sectorPortals [ 0 ] ) ) ;
level . sectorPortals [ 1 ] . mType = PORTS_SKYVIEWPOINT ;
level . sectorPortals [ 1 ] . mFlags = PORTSF_SKYFLATONLY ;
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// check if this line is between portal and the viewer. clip away if it is.
//
//============================================================================
2016-03-31 14:52:25 +00:00
inline int P_GetLineSide ( const DVector2 & pos , const line_t * line )
2016-03-03 12:07:11 +00:00
{
2016-03-31 14:52:25 +00:00
double v = ( pos . Y - line - > v1 - > fY ( ) ) * line - > Delta ( ) . X + ( line - > v1 - > fX ( ) - pos . X ) * line - > Delta ( ) . Y ;
return v < - 1. / 65536. ? - 1 : v > 1. / 65536 ? 1 : 0 ;
2016-03-03 12:07:11 +00:00
}
2016-03-31 14:52:25 +00:00
bool P_ClipLineToPortal ( line_t * line , line_t * portal , DVector2 view , bool partial , bool samebehind )
2016-03-01 15:47:10 +00:00
{
2016-03-31 14:52:25 +00:00
int behind1 = P_GetLineSide ( line - > v1 - > fPos ( ) , portal ) ;
int behind2 = P_GetLineSide ( line - > v2 - > fPos ( ) , portal ) ;
2016-03-03 12:07:11 +00:00
if ( behind1 = = 0 & & behind2 = = 0 )
2016-03-01 15:47:10 +00:00
{
2016-03-03 12:07:11 +00:00
// The line is parallel to the portal and cannot possibly be visible.
return true ;
2016-03-01 15:47:10 +00:00
}
2016-03-03 12:07:11 +00:00
// If one point lies on the same straight line than the portal, the other vertex will determine sidedness alone.
else if ( behind2 = = 0 ) behind2 = behind1 ;
else if ( behind1 = = 0 ) behind1 = behind2 ;
2016-03-01 15:47:10 +00:00
2016-03-03 12:07:11 +00:00
if ( behind1 > 0 & & behind2 > 0 )
{
// The line is behind the portal, i.e. between viewer and portal line, and must be rejected
return true ;
}
else if ( behind1 < 0 & & behind2 < 0 )
{
// The line is in front of the portal, i.e. the portal is between viewer and line. This line must not be rejected
return false ;
}
else
{
// The line intersects with the portal straight, so we need to do another check to see how both ends of the portal lie in relation to the viewer.
2016-03-31 14:52:25 +00:00
int viewside = P_GetLineSide ( view , line ) ;
int p1side = P_GetLineSide ( portal - > v1 - > fPos ( ) , line ) ;
int p2side = P_GetLineSide ( portal - > v2 - > fPos ( ) , line ) ;
// Do the same handling of points on the portal straight as above.
2016-03-03 12:07:11 +00:00
if ( p1side = = 0 ) p1side = p2side ;
else if ( p2side = = 0 ) p2side = p1side ;
// If the portal is on the other side of the line than the viewpoint, there is no possibility to see this line inside the portal.
return ( p1side = = p2side & & viewside ! = p1side ) ;
}
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// Translates a coordinate by a portal's displacement
//
//============================================================================
2016-03-31 14:52:25 +00:00
void P_TranslatePortalXY ( line_t * src , double & x , double & y )
2016-03-01 15:47:10 +00:00
{
2016-03-03 10:58:04 +00:00
if ( ! src ) return ;
FLinePortal * port = src - > getPortal ( ) ;
if ( ! port ) return ;
2016-06-19 10:32:45 +00:00
if ( port - > mFlags & PORTF_POLYOBJ ) SetRotation ( port ) ; // update the angle for polyportals.
2016-03-01 15:47:10 +00:00
2016-03-03 10:58:04 +00:00
// offsets from line
2016-03-31 14:52:25 +00:00
double nposx = x - src - > v1 - > fX ( ) ;
double nposy = y - src - > v1 - > fY ( ) ;
2016-03-01 15:47:10 +00:00
// Rotate position along normal to match exit linedef
2016-03-31 14:52:25 +00:00
double tx = nposx * port - > mCosRot - nposy * port - > mSinRot ;
double ty = nposy * port - > mCosRot + nposx * port - > mSinRot ;
2016-03-01 15:47:10 +00:00
2016-03-31 14:52:25 +00:00
tx + = port - > mDestination - > v2 - > fX ( ) ;
ty + = port - > mDestination - > v2 - > fY ( ) ;
2016-03-01 15:47:10 +00:00
x = tx ;
y = ty ;
}
//============================================================================
//
// Translates a velocity vector by a portal's displacement
//
//============================================================================
2016-03-31 14:52:25 +00:00
void P_TranslatePortalVXVY ( line_t * src , double & velx , double & vely )
2016-03-01 15:47:10 +00:00
{
2016-03-03 10:58:04 +00:00
if ( ! src ) return ;
FLinePortal * port = src - > getPortal ( ) ;
if ( ! port ) return ;
2016-06-19 10:32:45 +00:00
if ( port - > mFlags & PORTF_POLYOBJ ) SetRotation ( port ) ; // update the angle for polyportals.
2016-03-01 15:47:10 +00:00
2016-03-31 14:52:25 +00:00
double orig_velx = velx ;
double orig_vely = vely ;
velx = orig_velx * port - > mCosRot - orig_vely * port - > mSinRot ;
2016-05-09 19:59:13 +00:00
vely = orig_vely * port - > mCosRot + orig_velx * port - > mSinRot ;
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// Translates an angle by a portal's displacement
//
//============================================================================
2016-03-16 11:41:26 +00:00
void P_TranslatePortalAngle ( line_t * src , DAngle & angle )
2016-03-01 15:47:10 +00:00
{
2016-03-03 10:58:04 +00:00
if ( ! src ) return ;
FLinePortal * port = src - > getPortal ( ) ;
if ( ! port ) return ;
2016-06-19 10:32:45 +00:00
if ( port - > mFlags & PORTF_POLYOBJ ) SetRotation ( port ) ; // update the angle for polyportals.
2016-03-16 11:41:26 +00:00
angle = ( angle + port - > mAngleDiff ) . Normalized360 ( ) ;
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// Translates a z-coordinate by a portal's displacement
//
//============================================================================
2016-03-31 14:52:25 +00:00
void P_TranslatePortalZ ( line_t * src , double & z )
2016-03-01 15:47:10 +00:00
{
// args[2] = 0 - no adjustment
// args[2] = 1 - adjust by floor difference
// args[2] = 2 - adjust by ceiling difference
2016-03-03 10:58:04 +00:00
// This cannot be precalculated because heights may change.
line_t * dst = src - > getPortalDestination ( ) ;
2016-03-01 15:47:10 +00:00
switch ( src - > getPortalAlignment ( ) )
{
case PORG_FLOOR :
2016-03-31 14:52:25 +00:00
z = z - src - > frontsector - > floorplane . ZatPoint ( src - > v1 ) + dst - > frontsector - > floorplane . ZatPoint ( dst - > v2 ) ;
2016-03-01 15:47:10 +00:00
return ;
case PORG_CEILING :
2016-03-31 14:52:25 +00:00
z = z - src - > frontsector - > ceilingplane . ZatPoint ( src - > v1 ) + dst - > frontsector - > ceilingplane . ZatPoint ( dst - > v2 ) ;
2016-03-01 15:47:10 +00:00
return ;
default :
return ;
}
}
2016-04-20 17:20:11 +00:00
//============================================================================
//
// P_GetSkyboxPortal
//
// Gets a portal for a SkyViewpoint
// If none exists yet, it will create a new one.
//
//============================================================================
2017-01-14 15:05:40 +00:00
unsigned P_GetSkyboxPortal ( AActor * actor )
2016-04-20 17:20:11 +00:00
{
2017-02-14 22:13:29 +00:00
if ( actor = = nullptr ) return 1 ; // this means a regular sky.
2017-01-14 15:05:40 +00:00
for ( unsigned i = 0 ; i < level . sectorPortals . Size ( ) ; i + + )
2016-04-20 17:20:11 +00:00
{
2017-01-14 15:05:40 +00:00
if ( level . sectorPortals [ i ] . mSkybox = = actor ) return i ;
2016-04-20 17:20:11 +00:00
}
2017-01-14 15:05:40 +00:00
unsigned i = level . sectorPortals . Reserve ( 1 ) ;
level . sectorPortals [ i ] . mType = PORTS_SKYVIEWPOINT ;
2017-02-08 14:47:22 +00:00
level . sectorPortals [ i ] . mFlags = actor - > GetClass ( ) - > IsDescendantOf ( " SkyCamCompat " ) ? 0 : PORTSF_SKYFLATONLY ;
2017-01-14 15:05:40 +00:00
level . sectorPortals [ i ] . mSkybox = actor ;
level . sectorPortals [ i ] . mDestination = actor - > Sector ;
2016-04-20 17:20:11 +00:00
return i ;
}
2017-01-14 17:04:49 +00:00
DEFINE_ACTION_FUNCTION ( FSectorPortal , GetSkyboxPortal )
{
PARAM_PROLOGUE ;
PARAM_OBJECT ( actor , AActor ) ;
ACTION_RETURN_INT ( P_GetSkyboxPortal ( actor ) ) ;
}
2016-04-20 17:20:11 +00:00
//============================================================================
//
// P_GetPortal
//
// Creates a portal struct for a linedef-based portal
//
//============================================================================
unsigned P_GetPortal ( int type , int plane , sector_t * from , sector_t * to , const DVector2 & displacement )
{
2017-01-14 15:05:40 +00:00
unsigned i = level . sectorPortals . Reserve ( 1 ) ;
level . sectorPortals [ i ] . mType = type ;
level . sectorPortals [ i ] . mPlane = plane ;
level . sectorPortals [ i ] . mOrigin = from ;
level . sectorPortals [ i ] . mDestination = to ;
level . sectorPortals [ i ] . mDisplacement = displacement ;
level . sectorPortals [ i ] . mPlaneZ = type = = PORTS_LINKEDPORTAL ? from - > GetPlaneTexZ ( plane ) : FLT_MAX ;
2016-04-20 17:20:11 +00:00
return i ;
}
//============================================================================
//
// P_GetStackPortal
//
// Creates a portal for a stacked sector thing
//
//============================================================================
unsigned P_GetStackPortal ( AActor * point , int plane )
{
2017-01-14 15:05:40 +00:00
unsigned i = level . sectorPortals . Reserve ( 1 ) ;
level . sectorPortals [ i ] . mType = PORTS_STACKEDSECTORTHING ;
level . sectorPortals [ i ] . mPlane = plane ;
level . sectorPortals [ i ] . mOrigin = point - > target - > Sector ;
level . sectorPortals [ i ] . mDestination = point - > Sector ;
level . sectorPortals [ i ] . mPlaneZ = FLT_MAX ;
level . sectorPortals [ i ] . mSkybox = point ;
2016-04-20 17:20:11 +00:00
return i ;
}
2016-03-01 15:47:10 +00:00
//============================================================================
//
// P_GetOffsetPosition
//
// Offsets a given coordinate if the trace from the origin crosses an
// interactive line-to-line portal.
//
//============================================================================
2016-03-31 14:52:25 +00:00
DVector2 P_GetOffsetPosition ( double x , double y , double dx , double dy )
2016-03-01 15:47:10 +00:00
{
2016-03-31 14:52:25 +00:00
DVector2 dest ( x + dx , y + dy ) ;
2016-03-01 15:47:10 +00:00
if ( PortalBlockmap . containsLines )
{
2016-03-31 14:52:25 +00:00
double actx = x , acty = y ;
2016-03-01 15:47:10 +00:00
// Try some easily discoverable early-out first. If we know that the trace cannot possibly find a portal, this saves us from calling the traverser completely for vast parts of the map.
2016-03-31 14:52:25 +00:00
if ( dx < 128 & & dy < 128 )
2016-03-01 15:47:10 +00:00
{
2017-03-17 13:24:21 +00:00
int blockx = level . blockmap . GetBlockX ( actx ) ;
int blocky = level . blockmap . GetBlockY ( acty ) ;
if ( blockx < 0 | | blocky < 0 | | blockx > = PortalBlockmap . dx | | blocky > = PortalBlockmap . dy | | ! PortalBlockmap ( blockx , blocky ) . neighborContainsLines ) return dest ;
2016-03-01 15:47:10 +00:00
}
bool repeat ;
do
{
2016-04-15 22:17:12 +00:00
FLinePortalTraverse it ;
2016-03-01 15:47:10 +00:00
it . init ( actx , acty , dx , dy , PT_ADDLINES | PT_DELTA ) ;
intercept_t * in ;
repeat = false ;
while ( ( in = it . Next ( ) ) )
{
// hit a portal line.
line_t * line = in - > d . line ;
FLinePortal * port = line - > getPortal ( ) ;
// Teleport portals are intentionally ignored since skipping this stuff is their entire reason for existence.
if ( port - > mFlags & PORTF_INTERACTIVE )
{
2016-03-31 14:52:25 +00:00
DVector2 hit = it . InterceptPoint ( in ) ;
2016-03-01 15:47:10 +00:00
if ( port - > mType = = PORTT_LINKED )
{
// optimized handling for linked portals where we only need to add an offset.
2016-03-31 14:52:25 +00:00
hit + = port - > mDisplacement ;
dest + = port - > mDisplacement ;
2016-03-01 15:47:10 +00:00
}
else
{
// interactive ones are more complex because the vector may be rotated.
// Note: There is no z-translation here, there's just too much code in the engine that wouldn't be able to handle interactive portals with a height difference.
2016-03-31 14:52:25 +00:00
P_TranslatePortalXY ( line , hit . X , hit . Y ) ;
P_TranslatePortalXY ( line , dest . X , dest . Y ) ;
2016-03-01 15:47:10 +00:00
}
// update the fields, end this trace and restart from the new position
2016-03-31 14:52:25 +00:00
dx = dest . X - hit . X ;
dy = dest . Y - hit . Y ;
2016-04-15 22:17:12 +00:00
actx = hit . X ;
acty = hit . Y ;
2016-03-01 15:47:10 +00:00
repeat = true ;
}
break ;
}
} while ( repeat ) ;
}
return dest ;
}
2018-12-20 10:01:24 +00:00
static void GetOffsetPosition ( double x , double y , double dx , double dy , DVector2 * result )
{
* result = P_GetOffsetPosition ( x , y , dx , dy ) ;
}
2016-03-01 15:47:10 +00:00
//============================================================================
//
// CollectSectors
//
// Collects all sectors that are connected to any sector belonging to a portal
// because they all will need the same displacement values
//
//============================================================================
static bool CollectSectors ( int groupid , sector_t * origin )
{
if ( origin - > PortalGroup ! = 0 ) return false ; // already processed
origin - > PortalGroup = groupid ;
TArray < sector_t * > list ( 16 ) ;
list . Push ( origin ) ;
for ( unsigned i = 0 ; i < list . Size ( ) ; i + + )
{
sector_t * sec = list [ i ] ;
2017-01-02 20:40:52 +00:00
for ( auto line : sec - > Lines )
2016-03-01 15:47:10 +00:00
{
sector_t * other = line - > frontsector = = sec ? line - > backsector : line - > frontsector ;
2017-02-14 22:13:29 +00:00
if ( other ! = nullptr & & other ! = sec & & other - > PortalGroup ! = groupid )
2016-03-01 15:47:10 +00:00
{
other - > PortalGroup = groupid ;
list . Push ( other ) ;
}
}
}
return true ;
}
//============================================================================
//
// AddDisplacementForPortal
//
// Adds the displacement for one portal to the displacement array
// (one version for sector to sector plane, one for line to line portals)
//
// Note: Despite the similarities to Eternity's equivalent this is
// original code!
//
//============================================================================
2016-04-20 17:20:11 +00:00
static void AddDisplacementForPortal ( FSectorPortal * portal )
2016-03-01 15:47:10 +00:00
{
2016-04-20 17:20:11 +00:00
int thisgroup = portal - > mOrigin - > PortalGroup ;
int othergroup = portal - > mDestination - > PortalGroup ;
2016-03-01 15:47:10 +00:00
if ( thisgroup = = othergroup )
{
2016-04-20 17:20:11 +00:00
Printf ( " Portal between sectors %d and %d has both sides in same group and will be disabled \n " , portal - > mOrigin - > sectornum , portal - > mDestination - > sectornum ) ;
portal - > mType = PORTS_PORTAL ;
2016-03-01 15:47:10 +00:00
return ;
}
if ( thisgroup < = 0 | | thisgroup > = Displacements . size | | othergroup < = 0 | | othergroup > = Displacements . size )
{
2016-04-20 17:20:11 +00:00
Printf ( " Portal between sectors %d and %d has invalid group and will be disabled \n " , portal - > mOrigin - > sectornum , portal - > mDestination - > sectornum ) ;
portal - > mType = PORTS_PORTAL ;
2016-03-01 15:47:10 +00:00
return ;
}
FDisplacement & disp = Displacements ( thisgroup , othergroup ) ;
if ( ! disp . isSet )
{
2016-04-20 17:20:11 +00:00
disp . pos = portal - > mDisplacement ;
2016-03-01 15:47:10 +00:00
disp . isSet = true ;
}
else
{
2016-04-20 17:20:11 +00:00
if ( disp . pos ! = portal - > mDisplacement )
2016-03-01 15:47:10 +00:00
{
2016-04-20 17:20:11 +00:00
Printf ( " Portal between sectors %d and %d has displacement mismatch and will be disabled \n " , portal - > mOrigin - > sectornum , portal - > mDestination - > sectornum ) ;
portal - > mType = PORTS_PORTAL ;
2016-03-01 15:47:10 +00:00
return ;
}
}
}
static void AddDisplacementForPortal ( FLinePortal * portal )
{
int thisgroup = portal - > mOrigin - > frontsector - > PortalGroup ;
int othergroup = portal - > mDestination - > frontsector - > PortalGroup ;
if ( thisgroup = = othergroup )
{
2017-01-08 13:39:16 +00:00
Printf ( " Portal between lines %d and %d has both sides in same group \n " , portal - > mOrigin - > Index ( ) , portal - > mDestination - > Index ( ) ) ;
2016-03-01 15:47:10 +00:00
portal - > mType = linePortals [ portal - > mDestination - > portalindex ] . mType = PORTT_TELEPORT ;
return ;
}
if ( thisgroup < = 0 | | thisgroup > = Displacements . size | | othergroup < = 0 | | othergroup > = Displacements . size )
{
2017-01-08 13:39:16 +00:00
Printf ( " Portal between lines %d and %d has invalid group \n " , portal - > mOrigin - > Index ( ) , portal - > mDestination - > Index ( ) ) ;
2016-03-01 15:47:10 +00:00
portal - > mType = linePortals [ portal - > mDestination - > portalindex ] . mType = PORTT_TELEPORT ;
return ;
}
FDisplacement & disp = Displacements ( thisgroup , othergroup ) ;
if ( ! disp . isSet )
{
2016-03-31 14:52:25 +00:00
disp . pos = portal - > mDisplacement ;
2016-03-01 15:47:10 +00:00
disp . isSet = true ;
}
else
{
2016-03-31 14:52:25 +00:00
if ( disp . pos ! = portal - > mDisplacement )
2016-03-01 15:47:10 +00:00
{
2017-01-08 13:39:16 +00:00
Printf ( " Portal between lines %d and %d has displacement mismatch \n " , portal - > mOrigin - > Index ( ) , portal - > mDestination - > Index ( ) ) ;
2016-03-01 15:47:10 +00:00
portal - > mType = linePortals [ portal - > mDestination - > portalindex ] . mType = PORTT_TELEPORT ;
return ;
}
}
}
//============================================================================
//
// ConnectGroups
//
// Do the indirect connections. This loop will run until it cannot find any new connections
//
//============================================================================
static bool ConnectGroups ( )
{
// Now
2017-03-08 14:20:00 +00:00
uint8_t indirect = 1 ;
2016-03-01 15:47:10 +00:00
bool bogus = false ;
bool changed ;
do
{
changed = false ;
for ( int x = 1 ; x < Displacements . size ; x + + )
{
for ( int y = 1 ; y < Displacements . size ; y + + )
{
FDisplacement & dispxy = Displacements ( x , y ) ;
if ( dispxy . isSet )
{
for ( int z = 1 ; z < Displacements . size ; z + + )
{
FDisplacement & dispyz = Displacements ( y , z ) ;
if ( dispyz . isSet )
{
FDisplacement & dispxz = Displacements ( x , z ) ;
if ( dispxz . isSet )
{
2016-03-31 14:52:25 +00:00
if ( dispxy . pos . X + dispyz . pos . X ! = dispxz . pos . X | | dispxy . pos . Y + dispyz . pos . Y ! = dispxz . pos . Y )
2016-03-01 15:47:10 +00:00
{
bogus = true ;
}
}
else
{
dispxz . pos = dispxy . pos + dispyz . pos ;
dispxz . isSet = true ;
dispxz . indirect = indirect ;
changed = true ;
}
}
}
}
}
}
indirect + + ;
} while ( changed ) ;
return bogus ;
}
//============================================================================
//
// P_CreateLinkedPortals
//
// Creates the data structures needed for linked portals
// Removes portals from sloped sectors (as they cannot work on them)
// Group all sectors connected to one side of the portal
// Caclculate displacements between all created groups.
//
// Portals with the same offset but different anchors will not be merged.
//
//============================================================================
void P_CreateLinkedPortals ( )
{
2016-04-20 17:20:11 +00:00
TArray < FSectorPortal * > orgs ;
2016-03-12 16:43:36 +00:00
int id = 1 ;
2016-03-01 15:47:10 +00:00
bool bogus = false ;
2017-01-14 15:05:40 +00:00
for ( auto & s : level . sectorPortals )
2016-03-01 15:47:10 +00:00
{
2016-04-20 17:20:11 +00:00
if ( s . mType = = PORTS_LINKEDPORTAL )
2016-03-01 15:47:10 +00:00
{
2016-04-20 17:20:11 +00:00
orgs . Push ( & s ) ;
2016-03-01 15:47:10 +00:00
}
}
2016-03-13 01:54:10 +00:00
id = 1 ;
2016-03-12 16:43:36 +00:00
if ( orgs . Size ( ) ! = 0 )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
for ( auto & sec : level . sectors )
2016-03-01 15:47:10 +00:00
{
2016-04-20 17:20:11 +00:00
for ( int j = 0 ; j < 2 ; j + + )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
if ( sec . GetPortalType ( j ) = = PORTS_LINKEDPORTAL )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
secplane_t & plane = j = = 0 ? sec . floorplane : sec . ceilingplane ;
2016-04-20 17:20:11 +00:00
if ( plane . isSlope ( ) )
{
// The engine cannot deal with portals on a sloped plane.
2017-01-07 18:32:24 +00:00
sec . ClearPortal ( j ) ;
Printf ( " Portal on %s of sector %d is sloped and will be disabled \n " , j = = 0 ? " floor " : " ceiling " , sec . sectornum ) ;
2016-04-20 17:20:11 +00:00
}
2016-03-01 15:47:10 +00:00
}
}
}
2016-04-20 17:20:11 +00:00
// Group all sectors, starting at each portal origin.
for ( unsigned i = 0 ; i < orgs . Size ( ) ; i + + )
{
if ( CollectSectors ( id , orgs [ i ] - > mOrigin ) ) id + + ;
if ( CollectSectors ( id , orgs [ i ] - > mDestination ) ) id + + ;
}
2016-03-01 15:47:10 +00:00
}
for ( unsigned i = 0 ; i < linePortals . Size ( ) ; i + + )
{
if ( linePortals [ i ] . mType = = PORTT_LINKED )
{
2017-05-13 15:45:24 +00:00
if ( linePortals [ i ] . mDestination = = nullptr )
{
Printf ( " Linked portal on line %d is unconnected and will be disabled \n " , linePortals [ i ] . mOrigin - > Index ( ) ) ;
linePortals [ i ] . mOrigin - > portalindex = UINT_MAX ;
linePortals [ i ] . mType = PORTT_VISUAL ;
}
else
{
if ( CollectSectors ( id , linePortals [ i ] . mOrigin - > frontsector ) ) id + + ;
if ( CollectSectors ( id , linePortals [ i ] . mDestination - > frontsector ) ) id + + ;
}
2016-03-01 15:47:10 +00:00
}
}
Displacements . Create ( id ) ;
// Check for leftover sectors that connect to a portal
2017-01-07 18:32:24 +00:00
for ( auto & sec : level . sectors )
2016-03-01 15:47:10 +00:00
{
for ( int j = 0 ; j < 2 ; j + + )
{
2017-01-07 18:32:24 +00:00
if ( sec . GetPortalType ( j ) = = PORTS_LINKEDPORTAL & & sec . PortalGroup = = 0 )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
auto p = sec . GetPortal ( j ) ;
CollectSectors ( p - > mOrigin - > PortalGroup , & sec ) ;
2016-03-01 15:47:10 +00:00
}
}
}
for ( unsigned i = 0 ; i < orgs . Size ( ) ; i + + )
{
AddDisplacementForPortal ( orgs [ i ] ) ;
}
for ( unsigned i = 0 ; i < linePortals . Size ( ) ; i + + )
{
if ( linePortals [ i ] . mType = = PORTT_LINKED )
{
AddDisplacementForPortal ( & linePortals [ i ] ) ;
}
}
for ( int x = 1 ; x < Displacements . size ; x + + )
{
for ( int y = x + 1 ; y < Displacements . size ; y + + )
{
FDisplacement & dispxy = Displacements ( x , y ) ;
FDisplacement & dispyx = Displacements ( y , x ) ;
if ( dispxy . isSet & & dispyx . isSet & &
2016-03-31 14:52:25 +00:00
( dispxy . pos . X ! = - dispyx . pos . X | | dispxy . pos . Y ! = - dispyx . pos . Y ) )
2016-03-01 15:47:10 +00:00
{
int sec1 = - 1 , sec2 = - 1 ;
2017-01-07 18:32:24 +00:00
for ( unsigned i = 0 ; i < level . sectors . Size ( ) & & ( sec1 = = - 1 | | sec2 = = - 1 ) ; i + + )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
if ( sec1 = = - 1 & & level . sectors [ i ] . PortalGroup = = x ) sec1 = i ;
if ( sec2 = = - 1 & & level . sectors [ i ] . PortalGroup = = y ) sec2 = i ;
2016-03-01 15:47:10 +00:00
}
Printf ( " Link offset mismatch between sectors %d and %d \n " , sec1 , sec2 ) ;
bogus = true ;
}
// mark everything that connects to a one-sided line
2017-01-08 13:39:16 +00:00
for ( auto & line : level . lines )
2016-03-01 15:47:10 +00:00
{
2017-01-08 13:39:16 +00:00
if ( line . backsector = = nullptr & & line . frontsector - > PortalGroup = = 0 )
2016-03-01 15:47:10 +00:00
{
2017-01-08 13:39:16 +00:00
CollectSectors ( - 1 , line . frontsector ) ;
2016-03-01 15:47:10 +00:00
}
}
// and now print a message for everything that still wasn't processed.
2017-01-07 18:32:24 +00:00
for ( auto & sec : level . sectors )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
if ( sec . PortalGroup = = 0 )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
Printf ( " Unable to assign sector %d to any group. Possibly self-referencing \n " , sec . sectornum ) ;
2016-03-01 15:47:10 +00:00
}
2017-01-07 18:32:24 +00:00
else if ( sec . PortalGroup = = - 1 ) sec . PortalGroup = 0 ;
2016-03-01 15:47:10 +00:00
}
}
}
bogus | = ConnectGroups ( ) ;
if ( bogus )
{
// todo: disable all portals whose offsets do not match the associated groups
}
// reject would just get in the way when checking sight through portals.
2017-03-17 11:49:43 +00:00
if ( Displacements . size > 1 )
2016-03-01 15:47:10 +00:00
{
2017-03-17 11:49:43 +00:00
level . rejectmatrix . Reset ( ) ;
2016-03-01 15:47:10 +00:00
}
// finally we must flag all planes which are obstructed by the sector's own ceiling or floor.
2017-01-07 18:32:24 +00:00
for ( auto & sec : level . sectors )
2016-03-01 15:47:10 +00:00
{
2017-01-07 18:32:24 +00:00
sec . CheckPortalPlane ( sector_t : : floor ) ;
sec . CheckPortalPlane ( sector_t : : ceiling ) ;
2016-03-07 20:58:34 +00:00
// set a flag on each line connecting to a plane portal sector. This is used to reduce the amount of checks in P_CheckSight.
2017-01-07 18:32:24 +00:00
if ( sec . PortalIsLinked ( sector_t : : floor ) | | sec . PortalIsLinked ( sector_t : : ceiling ) )
2016-03-07 20:58:34 +00:00
{
2017-01-07 18:32:24 +00:00
for ( auto ln : sec . Lines )
2016-03-07 20:58:34 +00:00
{
2017-01-02 20:40:52 +00:00
ln - > flags | = ML_PORTALCONNECT ;
2016-03-07 20:58:34 +00:00
}
}
2017-01-07 18:32:24 +00:00
if ( sec . PortalIsLinked ( sector_t : : ceiling ) & & sec . PortalIsLinked ( sector_t : : floor ) )
2016-03-08 09:09:02 +00:00
{
2017-01-07 18:32:24 +00:00
double cz = sec . GetPortalPlaneZ ( sector_t : : ceiling ) ;
double fz = sec . GetPortalPlaneZ ( sector_t : : floor ) ;
2016-03-08 09:09:02 +00:00
if ( cz < fz )
{
// This is a fatal condition. We have to remove one of the two portals. Choose the one that doesn't match the current plane
2017-01-07 18:32:24 +00:00
Printf ( " Error in sector %d: Ceiling portal at z=%f is below floor portal at z=%f \n " , sec . sectornum , cz , fz ) ;
double cp = - sec . ceilingplane . fD ( ) ;
double fp = sec . floorplane . fD ( ) ;
2016-03-08 09:09:02 +00:00
if ( cp < fp | | fz = = fp )
{
2017-01-07 18:32:24 +00:00
sec . ClearPortal ( sector_t : : ceiling ) ;
2016-03-08 09:09:02 +00:00
}
else
{
2017-01-07 18:32:24 +00:00
sec . ClearPortal ( sector_t : : floor ) ;
2016-03-08 09:09:02 +00:00
}
}
}
2016-04-18 11:38:56 +00:00
// mark all sector planes that check out ok for everything.
2017-01-07 18:32:24 +00:00
if ( sec . PortalIsLinked ( sector_t : : floor ) ) sec . planes [ sector_t : : floor ] . Flags | = PLANEF_LINKED ;
if ( sec . PortalIsLinked ( sector_t : : ceiling ) ) sec . planes [ sector_t : : ceiling ] . Flags | = PLANEF_LINKED ;
2016-03-01 15:47:10 +00:00
}
2016-03-08 13:41:37 +00:00
if ( linkedPortals . Size ( ) > 0 )
{
// We need to relink all actors that may touch a linked line portal
TThinkerIterator < AActor > it ;
AActor * actor ;
while ( ( actor = it . Next ( ) ) )
{
if ( ! ( actor - > flags & MF_NOBLOCKMAP ) )
{
FPortalGroupArray check ( FPortalGroupArray : : PGA_NoSectorPortals ) ;
2016-03-31 14:52:25 +00:00
P_CollectConnectedGroups ( actor - > Sector - > PortalGroup , actor - > Pos ( ) , actor - > Top ( ) , actor - > radius , check ) ;
2016-03-08 13:41:37 +00:00
if ( check . Size ( ) > 0 )
{
2016-12-25 21:40:26 +00:00
FLinkContext ctx ;
actor - > UnlinkFromWorld ( & ctx ) ;
actor - > LinkToWorld ( & ctx ) ;
2016-03-08 13:41:37 +00:00
}
}
}
}
2016-03-01 15:47:10 +00:00
}
//============================================================================
//
// Collect all portal groups this actor would occupy at the given position
// This is used to determine which parts of the map need to be checked.
//
//============================================================================
2016-03-31 14:52:25 +00:00
bool P_CollectConnectedGroups ( int startgroup , const DVector3 & position , double upperz , double checkradius , FPortalGroupArray & out )
2016-03-01 15:47:10 +00:00
{
// Keep this temporary work stuff static. This function can never be called recursively
// and this would have to be reallocated for each call otherwise.
static FPortalBits processMask ;
static TArray < FLinePortal * > foundPortals ;
static TArray < int > groupsToCheck ;
bool retval = false ;
out . inited = true ;
2016-03-02 21:26:47 +00:00
processMask . setSize ( Displacements . size ) ;
if ( Displacements . size = = 1 )
{
processMask . setBit ( startgroup ) ;
return false ;
}
2016-03-01 15:47:10 +00:00
if ( linkedPortals . Size ( ) ! = 0 )
{
processMask . clear ( ) ;
foundPortals . Clear ( ) ;
int thisgroup = startgroup ;
processMask . setBit ( thisgroup ) ;
//out.Add(thisgroup);
for ( unsigned i = 0 ; i < linkedPortals . Size ( ) ; i + + )
{
line_t * ld = linkedPortals [ i ] - > mOrigin ;
int othergroup = ld - > frontsector - > PortalGroup ;
FDisplacement & disp = Displacements ( thisgroup , othergroup ) ;
if ( ! disp . isSet ) continue ; // no connection.
2016-03-31 14:52:25 +00:00
FBoundingBox box ( position . X + disp . pos . X , position . Y + disp . pos . Y , checkradius ) ;
2016-03-01 15:47:10 +00:00
2016-03-30 23:22:49 +00:00
if ( ! box . inRange ( ld ) | | box . BoxOnLineSide ( linkedPortals [ i ] - > mOrigin ) ! = - 1 ) continue ; // not touched
2016-03-01 15:47:10 +00:00
foundPortals . Push ( linkedPortals [ i ] ) ;
}
bool foundone = true ;
while ( foundone )
{
foundone = false ;
for ( int i = foundPortals . Size ( ) - 1 ; i > = 0 ; i - - )
{
if ( processMask . getBit ( foundPortals [ i ] - > mOrigin - > frontsector - > PortalGroup ) & &
! processMask . getBit ( foundPortals [ i ] - > mDestination - > frontsector - > PortalGroup ) )
{
processMask . setBit ( foundPortals [ i ] - > mDestination - > frontsector - > PortalGroup ) ;
out . Add ( foundPortals [ i ] - > mDestination - > frontsector - > PortalGroup ) ;
foundone = true ;
retval = true ;
foundPortals . Delete ( i ) ;
}
}
}
}
if ( out . method ! = FPortalGroupArray : : PGA_NoSectorPortals )
{
2016-03-31 14:52:25 +00:00
sector_t * sec = P_PointInSector ( position ) ;
2016-03-01 15:47:10 +00:00
sector_t * wsec = sec ;
2016-04-19 09:35:28 +00:00
while ( ! wsec - > PortalBlocksMovement ( sector_t : : ceiling ) & & upperz > wsec - > GetPortalPlaneZ ( sector_t : : ceiling ) )
2016-03-01 15:47:10 +00:00
{
2016-04-19 09:35:28 +00:00
int othergroup = wsec - > GetOppositePortalGroup ( sector_t : : ceiling ) ;
DVector2 pos = Displacements . getOffset ( startgroup , othergroup ) + position ;
2017-05-14 10:24:52 +00:00
if ( processMask . getBit ( othergroup ) ) break ;
2016-04-19 09:35:28 +00:00
processMask . setBit ( othergroup ) ;
out . Add ( othergroup | FPortalGroupArray : : UPPER ) ;
2016-03-31 14:52:25 +00:00
wsec = P_PointInSector ( pos ) ; // get upper sector at the exact spot we want to check and repeat
2016-03-01 15:47:10 +00:00
retval = true ;
}
wsec = sec ;
2016-04-19 09:35:28 +00:00
while ( ! wsec - > PortalBlocksMovement ( sector_t : : floor ) & & position . Z < wsec - > GetPortalPlaneZ ( sector_t : : floor ) )
2016-03-01 15:47:10 +00:00
{
2016-04-19 09:35:28 +00:00
int othergroup = wsec - > GetOppositePortalGroup ( sector_t : : floor ) ;
DVector2 pos = Displacements . getOffset ( startgroup , othergroup ) + position ;
2017-05-14 10:24:52 +00:00
if ( processMask . getBit ( othergroup ) ) break ;
2016-12-07 23:59:04 +00:00
processMask . setBit ( othergroup ) ;
out . Add ( othergroup | FPortalGroupArray : : LOWER ) ;
2016-03-31 14:52:25 +00:00
wsec = P_PointInSector ( pos ) ; // get lower sector at the exact spot we want to check and repeat
2016-03-01 15:47:10 +00:00
retval = true ;
}
2016-03-08 13:41:37 +00:00
if ( out . method = = FPortalGroupArray : : PGA_Full3d & & PortalBlockmap . hasLinkedSectorPortals )
2016-03-01 15:47:10 +00:00
{
groupsToCheck . Clear ( ) ;
groupsToCheck . Push ( startgroup ) ;
int thisgroup = startgroup ;
for ( unsigned i = 0 ; i < groupsToCheck . Size ( ) ; i + + )
{
2016-03-31 14:52:25 +00:00
DVector2 disp = Displacements . getOffset ( startgroup , thisgroup & ~ FPortalGroupArray : : FLAT ) ;
FBoundingBox box ( position . X + disp . X , position . Y + disp . Y , checkradius ) ;
2016-03-01 15:47:10 +00:00
FBlockLinesIterator it ( box ) ;
line_t * ld ;
while ( ( ld = it . Next ( ) ) )
{
2016-03-30 23:22:49 +00:00
if ( ! box . inRange ( ld ) | | box . BoxOnLineSide ( ld ) ! = - 1 )
2016-03-01 15:47:10 +00:00
continue ;
if ( ! ( thisgroup & FPortalGroupArray : : LOWER ) )
{
for ( int s = 0 ; s < 2 ; s + + )
{
sector_t * sec = s ? ld - > backsector : ld - > frontsector ;
if ( sec & & ! ( sec - > PortalBlocksMovement ( sector_t : : ceiling ) ) )
{
2016-04-19 09:35:28 +00:00
if ( sec - > GetPortalPlaneZ ( sector_t : : ceiling ) < upperz )
2016-03-01 15:47:10 +00:00
{
2016-04-19 09:35:28 +00:00
int grp = sec - > GetOppositePortalGroup ( sector_t : : ceiling ) ;
2016-03-01 15:47:10 +00:00
if ( ! ( processMask . getBit ( grp ) ) )
{
processMask . setBit ( grp ) ;
groupsToCheck . Push ( grp | FPortalGroupArray : : UPPER ) ;
}
}
}
}
}
if ( ! ( thisgroup & FPortalGroupArray : : UPPER ) )
{
for ( int s = 0 ; s < 2 ; s + + )
{
sector_t * sec = s ? ld - > backsector : ld - > frontsector ;
if ( sec & & ! ( sec - > PortalBlocksMovement ( sector_t : : floor ) ) )
{
2016-04-19 09:35:28 +00:00
if ( sec - > GetPortalPlaneZ ( sector_t : : floor ) > position . Z )
2016-03-01 15:47:10 +00:00
{
2016-04-19 09:35:28 +00:00
int grp = sec - > GetOppositePortalGroup ( sector_t : : floor ) ;
2016-03-01 15:47:10 +00:00
if ( ! ( processMask . getBit ( grp ) ) )
{
processMask . setBit ( grp ) ;
groupsToCheck . Push ( grp | FPortalGroupArray : : LOWER ) ;
}
}
}
}
}
}
}
}
}
return retval ;
}
//============================================================================
//
// print the group link table to the console
//
//============================================================================
CCMD ( dumplinktable )
{
for ( int x = 1 ; x < Displacements . size ; x + + )
{
for ( int y = 1 ; y < Displacements . size ; y + + )
{
FDisplacement & disp = Displacements ( x , y ) ;
2016-03-31 14:52:25 +00:00
Printf ( " %c%c(%6d, %6d) " , TEXTCOLOR_ESCAPE , ' C ' + disp . indirect , int ( disp . pos . X ) , int ( disp . pos . Y ) ) ;
2016-03-01 15:47:10 +00:00
}
Printf ( " \n " ) ;
}
}