2019-11-20 16:21:32 +00:00
//-------------------------------------------------------------------------
/*
Copyright ( C ) 2010 - 2019 EDuke32 developers and contributors
Copyright ( C ) 2019 sirlemonhead , Nuke . YKT
This file is part of PCExhumed .
PCExhumed is free software ; you can redistribute it and / or
modify it under the terms of the GNU General Public License version 2
as published by the Free Software Foundation .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
See the GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*/
//-------------------------------------------------------------------------
2019-11-22 23:11:37 +00:00
# include "ns.h"
2019-08-26 03:59:14 +00:00
# include "engine.h"
2020-08-18 07:52:08 +00:00
# include "aistuff.h"
2019-08-26 03:59:14 +00:00
# include "exhumed.h"
# include "view.h"
# include "sound.h"
# include "names.h"
# include "sequence.h"
# include "player.h"
2021-01-04 19:34:27 +00:00
# include "interpolate.h"
2020-08-23 14:24:54 +00:00
# include "mapinfo.h"
2019-08-26 03:59:14 +00:00
# include <string.h>
# include <assert.h>
2019-11-22 23:11:37 +00:00
BEGIN_PS_NS
2021-11-21 19:34:15 +00:00
static const int8_t ObjectSeq [ ] = {
2019-08-31 07:47:15 +00:00
46 , - 1 , 72 , - 1
2019-08-26 03:59:14 +00:00
} ;
2021-11-21 19:48:11 +00:00
static const int16_t ObjectStatnum [ ] = {
2019-08-31 07:47:15 +00:00
kStatExplodeTrigger , kStatExplodeTarget , 98 , kStatDestructibleSprite
2019-08-26 03:59:14 +00:00
} ;
struct Trail
{
2021-11-21 19:48:11 +00:00
int16_t nPoint ;
int16_t nVal ;
int16_t nPoint2 ;
2019-08-26 03:59:14 +00:00
} ;
struct TrailPoint
{
2019-08-31 07:47:15 +00:00
int x ;
int y ;
2021-11-09 16:46:10 +00:00
uint8_t nTrailPointVal ;
2021-11-21 19:34:15 +00:00
int16_t nTrailPointPrev ;
int16_t nTrailPointNext ;
2020-11-29 23:49:25 +00:00
2019-08-26 03:59:14 +00:00
} ;
struct Bob
{
2021-11-22 22:40:53 +00:00
sectortype * pSector ;
2021-11-21 19:48:11 +00:00
int z ;
2021-11-21 20:09:27 +00:00
uint8_t nPhase ;
2021-11-09 16:46:10 +00:00
uint8_t field_3 ;
2021-11-21 19:34:15 +00:00
uint16_t sBobID ;
2019-08-26 03:59:14 +00:00
} ;
struct Drip
{
2021-12-07 17:53:02 +00:00
TObjPtr < DExhumedActor * > pActor ;
2021-11-21 19:48:11 +00:00
int16_t nCount ;
2019-08-26 03:59:14 +00:00
} ;
// 56 bytes
struct Elev
{
2021-12-07 17:53:02 +00:00
TObjPtr < DExhumedActor * > pActor ;
2021-11-22 22:45:19 +00:00
sectortype * pSector ;
2021-11-21 19:34:15 +00:00
int16_t nFlags ;
int16_t nChannel ;
int nParam1 ;
int nParam2 ;
int16_t nCountZOffsets ; // count of items in zOffsets
int16_t nCurZOffset ;
2019-08-31 07:47:15 +00:00
int zOffsets [ 8 ] ; // different Z offsets
2021-11-21 19:34:15 +00:00
int16_t nRunRec ;
2019-08-26 03:59:14 +00:00
} ;
// 16 bytes
struct MoveSect
{
2021-11-22 20:03:10 +00:00
sectortype * pSector ;
sectortype * pCurSector ;
2019-08-31 07:47:15 +00:00
int field_10 ;
2021-11-21 19:48:11 +00:00
int16_t nTrail ;
int16_t nTrailPoint ;
int16_t nFlags ;
int16_t nChannel ; // nChannel?
int16_t sMoveDir ;
2020-11-29 23:49:25 +00:00
2019-08-26 03:59:14 +00:00
} ;
struct wallFace
{
2021-11-22 17:22:38 +00:00
walltype * pWall ;
2021-11-21 18:24:46 +00:00
int16_t nChannel ;
int16_t count ;
int16_t piclist [ 8 ] ;
2019-08-26 03:59:14 +00:00
} ;
2020-11-29 23:49:25 +00:00
struct slideData
2019-08-26 03:59:14 +00:00
{
2021-12-07 17:53:02 +00:00
TObjPtr < DExhumedActor * > pActor ;
2021-11-21 19:48:11 +00:00
int16_t nChannel ;
int16_t nStart ;
int16_t nRunRec ;
int16_t nRunC ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
walltype * pStartWall ;
walltype * pWall1 ;
walltype * pWall2 ;
walltype * pWall3 ;
2020-02-04 21:45:10 +00:00
int x1 ;
int y1 ;
int x2 ;
int y2 ;
2021-10-21 10:08:22 +00:00
int x3 ;
int y3 ;
int x4 ;
int y4 ;
int x5 ;
int y5 ;
int x6 ;
int y6 ;
2019-08-26 03:59:14 +00:00
} ;
struct Point
{
2021-11-22 18:32:47 +00:00
sectortype * pSector ;
2021-11-21 20:09:27 +00:00
int16_t nNext ;
2019-08-26 03:59:14 +00:00
} ;
struct Trap
{
2021-12-07 17:53:02 +00:00
TObjPtr < DExhumedActor * > pActor ;
2021-11-22 18:45:48 +00:00
walltype * pWall1 ;
walltype * pWall2 ;
2021-10-21 11:09:29 +00:00
2021-11-21 20:09:27 +00:00
int16_t nState ;
int16_t nType ;
2021-11-22 18:45:48 +00:00
int16_t nPicnum1 ;
2021-11-21 20:09:27 +00:00
int16_t nPicnum2 ;
int16_t nTrapInterval ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
} ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
TArray < Point > PointList ;
TArray < Trap > sTrap ;
TArray < Bob > sBob ;
TArray < Trail > sTrail ;
TArray < TrailPoint > sTrailPoint ;
TArray < Elev > Elevator ;
2021-10-21 15:24:52 +00:00
TArray < DExhumedActor * > ObjectList ;
2020-11-29 23:49:25 +00:00
TArray < MoveSect > sMoveSect ;
TArray < slideData > SlideData ;
TArray < wallFace > WallFace ;
TArray < Drip > sDrip ;
2021-10-21 13:02:18 +00:00
TArray < DExhumedActor * > EnergyBlocks ;
2021-12-07 17:53:02 +00:00
TObjPtr < DExhumedActor * > pFinaleSpr ;
size_t MarkObjects ( )
{
GC : : Mark ( pFinaleSpr ) ;
for ( auto & d : sDrip ) GC : : Mark ( d . pActor ) ;
for ( auto & d : sTrap ) GC : : Mark ( d . pActor ) ;
for ( auto & d : Elevator ) GC : : Mark ( d . pActor ) ;
for ( auto & d : SlideData ) GC : : Mark ( d . pActor ) ;
GC : : MarkArray ( EnergyBlocks ) ;
return 1 + sDrip . Size ( ) + sTrap . Size ( ) + Elevator . Size ( ) + SlideData . Size ( ) ;
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int lFinaleStart ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:13:19 +00:00
int nFinaleStage ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:13:19 +00:00
int nDronePitch = 0 ;
int nSmokeSparks = 0 ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Trail & w , Trail * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-21 19:48:11 +00:00
arc ( " at0 " , w . nPoint )
( " at2 " , w . nVal )
( " at4 " , w . nPoint2 )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , TrailPoint & w , TrailPoint * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
arc ( " x " , w . x )
( " y " , w . y )
( " val " , w . nTrailPointVal )
( " next " , w . nTrailPointNext )
( " prev " , w . nTrailPointPrev )
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Bob & w , Bob * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-22 22:40:53 +00:00
arc ( " sector " , w . pSector )
2021-11-21 20:09:27 +00:00
( " at2 " , w . nPhase )
2021-10-15 20:13:05 +00:00
( " at3 " , w . field_3 )
( " z " , w . z )
( " id " , w . sBobID )
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Drip & w , Drip * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-10-21 16:28:06 +00:00
arc ( " sprite " , w . pActor )
( " at2 " , w . nCount )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Elev & w , Elev * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-21 19:34:15 +00:00
arc ( " at0 " , w . nFlags )
2021-10-15 20:13:05 +00:00
( " channel " , w . nChannel )
2021-11-22 22:45:19 +00:00
( " sector " , w . pSector )
2021-11-21 19:34:15 +00:00
( " at6 " , w . nParam1 )
( " ata " , w . nParam2 )
2021-10-15 20:13:05 +00:00
( " countz " , w . nCountZOffsets )
( " curz " , w . nCurZOffset )
. Array ( " zofs " , w . zOffsets , 8 )
2021-11-21 19:34:15 +00:00
( " at32 " , w . nRunRec )
2021-10-21 21:41:54 +00:00
( " sprite " , w . pActor )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , MoveSect & w , MoveSect * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-22 20:03:10 +00:00
arc ( " sector " , w . pSector )
2021-10-15 20:13:05 +00:00
( " trail " , w . nTrail )
( " trailpoint " , w . nTrailPoint )
2021-11-21 19:48:11 +00:00
( " at6 " , w . nFlags )
2021-11-22 20:03:10 +00:00
( " at8 " , w . pCurSector )
2021-10-15 20:13:05 +00:00
( " at10 " , w . field_10 )
2021-11-21 19:48:11 +00:00
( " at14 " , w . nChannel )
2021-10-15 20:13:05 +00:00
( " movedir " , w . sMoveDir )
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
2021-10-21 15:24:52 +00:00
2020-11-29 23:49:25 +00:00
FSerializer & Serialize ( FSerializer & arc , const char * keyname , wallFace & w , wallFace * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
arc ( " channel " , w . nChannel )
2021-11-22 17:22:38 +00:00
( " wall " , w . pWall )
2021-11-21 18:24:46 +00:00
( " at4 " , w . count )
. Array ( " at6 " , w . piclist , 8 )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , slideData & w , slideData * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-22 18:32:47 +00:00
arc ( " at0 " , w . pStartWall )
( " at4 " , w . pWall1 )
( " at8 " , w . pWall2 )
( " atc " , w . pWall3 )
2021-10-15 20:13:05 +00:00
( " x1 " , w . x1 )
( " y1 " , w . y1 )
( " x2 " , w . x2 )
( " y2 " , w . y2 )
2021-10-21 10:08:22 +00:00
( " at20 " , w . x3 )
( " at24 " , w . y3 )
( " at28 " , w . x4 )
( " at2c " , w . y4 )
( " at30 " , w . x5 )
( " at34 " , w . y5 )
( " at38 " , w . x6 )
( " at3c " , w . y6 )
2021-10-15 20:13:05 +00:00
( " channel " , w . nChannel )
2021-10-21 10:08:22 +00:00
( " at2a " , w . nStart )
2021-11-21 19:48:11 +00:00
( " at4a " , w . nRunRec )
2021-10-21 10:08:22 +00:00
( " at6a " , w . pActor )
2021-11-21 19:48:11 +00:00
( " at8a " , w . nRunC )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Point & w , Point * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-22 18:32:47 +00:00
arc ( " at0 " , w . pSector )
2021-10-21 10:08:22 +00:00
( " ate " , w . nNext )
2021-10-15 20:13:05 +00:00
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
FSerializer & Serialize ( FSerializer & arc , const char * keyname , Trap & w , Trap * def )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( keyname ) )
{
2021-11-21 20:09:27 +00:00
arc ( " at0 " , w . nState )
2021-10-21 11:09:29 +00:00
( " sprite " , w . pActor )
2021-10-15 20:13:05 +00:00
( " type " , w . nType )
2021-11-22 18:45:48 +00:00
( " wall1 " , w . pWall1 )
( " wall2 " , w . pWall2 )
( " pic1 " , w . nPicnum1 )
( " pic2 " , w . nPicnum2 )
2021-10-15 20:13:05 +00:00
( " interval " , w . nTrapInterval )
. EndObject ( ) ;
}
return arc ;
2020-11-29 23:49:25 +00:00
}
void SerializeObjects ( FSerializer & arc )
{
2021-10-15 20:13:05 +00:00
if ( arc . BeginObject ( " objects " ) )
{
arc ( " points " , PointList )
( " traps " , sTrap )
( " bob " , sBob )
( " trails " , sTrail )
( " trailpoints " , sTrailPoint )
( " elevators " , Elevator )
( " objects " , ObjectList )
( " movesect " , sMoveSect )
( " slides " , SlideData )
( " wallface " , WallFace )
( " drips " , sDrip )
( " finalestart " , lFinaleStart )
2021-10-21 13:02:18 +00:00
( " energyblocks " , EnergyBlocks )
2021-10-15 20:13:05 +00:00
( " finalestage " , nFinaleStage )
2021-10-21 11:20:00 +00:00
( " finalespr " , pFinaleSpr )
2021-10-15 20:13:05 +00:00
( " dronepitch " , nDronePitch )
( " smokesparks " , nSmokeSparks )
. EndObject ( ) ;
}
2020-11-29 23:49:25 +00:00
}
2019-08-26 03:59:14 +00:00
// done
void InitObjects ( )
{
2020-11-29 23:49:25 +00:00
sTrap . Clear ( ) ;
sBob . Clear ( ) ;
sTrail . Clear ( ) ;
sTrailPoint . Clear ( ) ;
ObjectList . Clear ( ) ;
sMoveSect . Clear ( ) ;
sDrip . Clear ( ) ;
2021-10-21 13:02:18 +00:00
EnergyBlocks . Clear ( ) ;
2019-08-31 07:47:15 +00:00
nDronePitch = 0 ;
nFinaleStage = 0 ;
lFinaleStart = 0 ;
nSmokeSparks = 0 ;
2019-08-26 03:59:14 +00:00
}
// done
void InitElev ( )
{
2020-11-29 23:49:25 +00:00
Elevator . Clear ( ) ;
2019-08-26 03:59:14 +00:00
}
// done
2021-11-22 22:45:19 +00:00
DExhumedActor * BuildWallSprite ( sectortype * pSector )
2019-08-26 03:59:14 +00:00
{
2021-11-22 22:45:19 +00:00
auto wal = pSector - > firstWall ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 22:45:19 +00:00
auto pActor = insertActor ( pSector , 401 ) ;
2019-08-26 03:59:14 +00:00
2022-08-30 21:05:03 +00:00
pActor - > spr . pos = DVector3 ( wal - > center ( ) , ( pSector - > floorz + pSector - > ceilingz ) * 0.5 ) ;
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2019-08-26 03:59:14 +00:00
2021-10-20 20:16:50 +00:00
return pActor ;
}
2019-08-26 03:59:14 +00:00
// done
2021-11-22 22:50:40 +00:00
DExhumedActor * FindWallSprites ( sectortype * pSector )
2019-08-26 03:59:14 +00:00
{
2022-08-30 19:35:00 +00:00
double min_x = DBL_MAX ;
double min_y = DBL_MAX ;
2019-08-31 07:47:15 +00:00
2022-08-30 19:35:00 +00:00
double max_x = - DBL_MAX ;
double max_y = - DBL_MAX ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:50:40 +00:00
for ( auto & wal : wallsofsector ( pSector ) )
2019-08-31 07:47:15 +00:00
{
2022-08-30 19:35:00 +00:00
if ( wal . pos . X < min_x ) {
min_x = wal . pos . X ;
2019-08-31 07:47:15 +00:00
}
2022-08-30 19:35:00 +00:00
if ( max_x < wal . pos . X ) {
max_x = wal . pos . X ;
2019-08-31 07:47:15 +00:00
}
2022-08-30 19:35:00 +00:00
if ( min_y > wal . pos . Y ) {
min_y = wal . pos . Y ;
2019-08-31 07:47:15 +00:00
}
2022-08-30 19:35:00 +00:00
if ( max_y < wal . pos . Y ) {
max_y = wal . pos . Y ;
2019-08-31 07:47:15 +00:00
}
}
2022-08-30 19:35:00 +00:00
min_y - = 5. / 16 ;
max_x + = 5. / 16 ;
max_y + = 5. / 16 ;
min_x - = 5. / 16 ;
2019-08-31 07:47:15 +00:00
2021-10-20 20:16:50 +00:00
DExhumedActor * pAct = nullptr ;
2019-08-31 07:47:15 +00:00
2021-10-20 20:16:50 +00:00
ExhumedSpriteIterator it ;
while ( auto actor = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2021-12-23 17:54:40 +00:00
if ( actor - > spr . lotag = = 0 )
2019-08-31 07:47:15 +00:00
{
2021-12-23 17:54:40 +00:00
if ( ( actor - > spr . cstat & ( CSTAT_SPRITE_ALIGNMENT_WALL | CSTAT_SPRITE_ONE_SIDE ) ) = = ( CSTAT_SPRITE_ALIGNMENT_WALL | CSTAT_SPRITE_ONE_SIDE ) )
2019-08-31 07:47:15 +00:00
{
2022-08-30 19:35:00 +00:00
double act_x = actor - > spr . pos . X ;
double act_y = actor - > spr . pos . Y ;
2019-08-31 07:47:15 +00:00
2022-08-29 17:43:25 +00:00
if ( ( act_x > = min_x ) & & ( max_x > = act_x ) & & ( act_y > = min_y ) & & ( act_y < = max_y ) )
2019-08-31 07:47:15 +00:00
{
2021-10-20 20:16:50 +00:00
actor - > pTarget = pAct ;
pAct = actor ;
2019-08-31 07:47:15 +00:00
}
}
}
}
2021-10-20 20:16:50 +00:00
if ( pAct = = nullptr )
2019-08-31 07:47:15 +00:00
{
2021-11-22 22:45:19 +00:00
pAct = insertActor ( pSector , 401 ) ;
2021-12-23 15:44:43 +00:00
2022-08-30 19:35:00 +00:00
pAct - > spr . pos = { ( min_x + max_x ) / 2 , ( min_y + max_y ) / 2 , pSector - > floorz } ;
2021-12-23 15:44:43 +00:00
pAct - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2022-05-23 22:30:41 +00:00
pAct - > spr . intowner = - 1 ;
2021-12-23 15:44:43 +00:00
pAct - > spr . lotag = 0 ;
pAct - > spr . hitag = 0 ;
2019-08-31 07:47:15 +00:00
}
2021-10-21 21:41:54 +00:00
return pAct ;
2019-08-26 03:59:14 +00:00
}
2022-08-30 19:35:00 +00:00
2021-11-22 22:45:19 +00:00
int BuildElevF ( int nChannel , sectortype * pSector , DExhumedActor * nWallSprite , int arg_4 , int arg_5 , int nCount , . . . )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto ElevCount = Elevator . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:34:15 +00:00
Elevator [ ElevCount ] . nFlags = 2 ;
Elevator [ ElevCount ] . nParam1 = arg_4 ;
Elevator [ ElevCount ] . nRunRec = - 1 ;
Elevator [ ElevCount ] . nParam2 = arg_5 ;
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nChannel = nChannel ;
2021-11-22 22:45:19 +00:00
Elevator [ ElevCount ] . pSector = pSector ;
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nCountZOffsets = 0 ;
Elevator [ ElevCount ] . nCurZOffset = 0 ;
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
if ( nWallSprite = = nullptr ) {
2021-11-22 22:45:19 +00:00
nWallSprite = BuildWallSprite ( pSector ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
Elevator [ ElevCount ] . pActor = nWallSprite ;
2019-11-20 16:21:32 +00:00
2019-08-31 10:36:26 +00:00
va_list zlist ;
va_start ( zlist , nCount ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
while ( 1 )
{
if ( Elevator [ ElevCount ] . nCountZOffsets > = nCount ) {
return ElevCount ;
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int nVal = Elevator [ ElevCount ] . nCountZOffsets ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nCountZOffsets + + ;
2019-08-26 03:59:14 +00:00
2019-08-31 10:36:26 +00:00
Elevator [ ElevCount ] . zOffsets [ nVal ] = va_arg ( zlist , int ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-31 10:36:26 +00:00
va_end ( zlist ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
return ElevCount ;
2019-08-26 03:59:14 +00:00
}
2021-11-22 22:45:19 +00:00
int BuildElevC ( int arg1 , int nChannel , sectortype * pSector , DExhumedActor * nWallSprite , int arg5 , int arg6 , int nCount , . . . )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
int edi = arg5 ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
auto ElevCount = Elevator . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:34:15 +00:00
Elevator [ ElevCount ] . nFlags = arg1 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( arg1 & 4 )
{
edi = arg5 / 2 ;
}
2019-08-26 03:59:14 +00:00
2021-11-21 19:34:15 +00:00
Elevator [ ElevCount ] . nParam1 = edi ;
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nCountZOffsets = 0 ;
Elevator [ ElevCount ] . nCurZOffset = 0 ;
2021-11-21 19:34:15 +00:00
Elevator [ ElevCount ] . nParam2 = arg6 ;
Elevator [ ElevCount ] . nRunRec = - 1 ;
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nChannel = nChannel ;
2021-11-22 22:45:19 +00:00
Elevator [ ElevCount ] . pSector = pSector ;
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
if ( nWallSprite = = nullptr ) {
2021-11-22 22:45:19 +00:00
nWallSprite = BuildWallSprite ( pSector ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
Elevator [ ElevCount ] . pActor = nWallSprite ;
2019-08-26 03:59:14 +00:00
2019-08-31 10:36:26 +00:00
va_list zlist ;
va_start ( zlist , nCount ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
while ( 1 )
{
if ( Elevator [ ElevCount ] . nCountZOffsets > = nCount ) {
return ElevCount ;
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int nVal = Elevator [ ElevCount ] . nCountZOffsets ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
Elevator [ ElevCount ] . nCountZOffsets + + ;
2019-08-26 03:59:14 +00:00
2019-08-31 10:36:26 +00:00
Elevator [ ElevCount ] . zOffsets [ nVal ] = va_arg ( zlist , int ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-31 10:36:26 +00:00
va_end ( zlist ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
return ElevCount ;
2019-08-26 03:59:14 +00:00
}
// TODO - tidy me up
// RENAME param A - not always Z
// Confirmed 100% correct with original .exe
2021-10-15 20:13:05 +00:00
int LongSeek ( int * pZVal , int a2 , int a3 , int a4 )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
int v4 ; // edx@1
int v5 ; // ebx@2
v4 = a2 - * pZVal ;
if ( v4 < 0 )
{
v5 = - a3 ;
if ( v5 > v4 )
v4 = v5 ;
( * pZVal ) + = v4 ;
}
if ( v4 > 0 )
{
if ( a4 < v4 )
v4 = a4 ;
( * pZVal ) + = v4 ;
}
return v4 ;
2019-08-26 03:59:14 +00:00
}
// done
2021-11-22 22:45:19 +00:00
int CheckSectorSprites ( sectortype * pSector , int nVal )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
int b = 0 ;
if ( nVal )
{
2022-02-02 23:46:04 +00:00
int nZDiff = pSector - > int_floorz ( ) - pSector - > int_ceilingz ( ) ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:45:19 +00:00
ExhumedSectIterator it ( pSector ) ;
2021-10-20 20:27:16 +00:00
while ( auto pActor = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2021-12-23 16:04:54 +00:00
if ( ( pActor - > spr . cstat & CSTAT_SPRITE_BLOCK_ALL ) & & ( nZDiff < GetActorHeight ( pActor ) ) )
2019-08-31 07:47:15 +00:00
{
if ( nVal ! = 1 ) {
return 1 ;
}
b = 1 ;
2021-10-21 22:08:40 +00:00
runlist_DamageEnemy ( pActor , nullptr , 5 ) ;
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . statnum = = 100 & & PlayerList [ GetPlayerFromActor ( pActor ) ] . nHealth < = 0 )
2019-08-31 07:47:15 +00:00
{
PlayFXAtXYZ ( StaticSound [ kSoundJonFDie ] ,
2022-08-20 15:52:06 +00:00
pActor - > spr . pos ,
2021-11-22 20:37:33 +00:00
CHANF_NONE , 0x4000 ) ;
2019-08-31 07:47:15 +00:00
}
}
}
}
else
{
2021-11-22 22:45:19 +00:00
ExhumedSectIterator it ( pSector ) ;
2021-10-20 20:27:16 +00:00
while ( auto pActor = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2021-12-21 22:18:23 +00:00
if ( pActor - > spr . cstat & CSTAT_SPRITE_BLOCK_ALL ) {
2019-08-31 07:47:15 +00:00
return 1 ;
}
}
b = 0 ;
}
return b ;
2019-08-26 03:59:14 +00:00
}
// done
2022-08-20 14:44:30 +00:00
void MoveSectorSprites ( sectortype * pSector , double z )
2019-08-26 03:59:14 +00:00
{
2022-08-20 14:44:30 +00:00
double newz = pSector - > floorz ;
double oldz = newz - z ;
double minz = min ( newz , oldz ) ;
double maxz = max ( newz , oldz ) ;
2021-11-22 22:40:53 +00:00
ExhumedSectIterator it ( pSector ) ;
2021-10-20 20:27:16 +00:00
while ( auto pActor = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2022-10-11 22:17:17 +00:00
double actz = pActor - > spr . pos . Z ;
2021-12-24 09:48:35 +00:00
if ( ( pActor - > spr . statnum ! = 200 & & actz > = minz & & actz < = maxz ) | | pActor - > spr . statnum > = 900 )
2021-01-12 19:57:28 +00:00
{
2022-08-20 14:44:30 +00:00
pActor - > spr . pos . Z = newz ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
}
2021-10-21 21:41:54 +00:00
void StartElevSound ( DExhumedActor * pActor , int nVal )
2019-08-26 03:59:14 +00:00
{
2021-11-21 20:09:27 +00:00
int nSound ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( nVal & 2 ) {
nSound = nElevSound ;
}
else {
nSound = nStoneSound ;
}
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
D3PlayFX ( StaticSound [ nSound ] , pActor ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AIElev : : ProcessChannel ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-15 20:13:05 +00:00
int nRun = ev - > nRun ;
2021-11-21 20:09:27 +00:00
int nElev = RunData [ nRun ] . nObjIndex ;
2020-11-29 23:49:25 +00:00
assert ( nElev > = 0 & & nElev < ( int ) Elevator . Size ( ) ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:34:15 +00:00
int nChannel = Elevator [ nElev ] . nChannel ;
int nFlags = Elevator [ nElev ] . nFlags ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
assert ( nChannel > = 0 & & nChannel < kMaxChannels ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 20:09:27 +00:00
// int ax = var_18 & 8;
int dx = sRunChannels [ nChannel ] . c ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
int edi = 999 ; // FIXME CHECKME - this isn't default set to anything in the ASM that I can see - if ax is 0 and var_24 is 0, this will never be set to a known value otherwise!
2019-08-26 03:59:14 +00:00
2021-11-21 19:34:15 +00:00
if ( nFlags & 0x8 )
2019-08-31 07:47:15 +00:00
{
2021-10-15 20:13:05 +00:00
if ( dx ) {
edi = 1 ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
else {
edi = 0 ;
}
}
else
{
// loc_20D48:
2021-11-21 19:34:15 +00:00
if ( nFlags & 0x10 ) // was var_24
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:34:15 +00:00
if ( Elevator [ nElev ] . nRunRec < 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:34:15 +00:00
Elevator [ nElev ] . nRunRec = runlist_AddRunRec ( NewRun , & RunData [ nRun ] ) ;
StartElevSound ( Elevator [ nElev ] . pActor , nFlags ) ;
2021-10-15 20:13:05 +00:00
edi = 1 ;
}
}
else
{
if ( dx < 0 ) {
edi = 0 ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-10-15 20:13:05 +00:00
if ( dx = = Elevator [ nElev ] . nCurZOffset | | dx > = Elevator [ nElev ] . nCountZOffsets )
2019-08-31 07:47:15 +00:00
{
2021-10-15 20:13:05 +00:00
edi = 1 ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-10-15 20:13:05 +00:00
Elevator [ nElev ] . nCurZOffset = sRunChannels [ nChannel ] . c ;
edi = 1 ;
2019-08-31 07:47:15 +00:00
}
}
2021-10-15 20:13:05 +00:00
}
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
assert ( edi ! = 999 ) ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
// loc_20DF9:
if ( edi )
{
2021-11-21 19:34:15 +00:00
if ( Elevator [ nElev ] . nRunRec < 0 )
2021-10-15 20:13:05 +00:00
{
2021-11-21 19:34:15 +00:00
Elevator [ nElev ] . nRunRec = runlist_AddRunRec ( NewRun , & RunData [ nRun ] ) ;
2019-08-31 07:47:15 +00:00
2021-11-21 19:34:15 +00:00
StartElevSound ( Elevator [ nElev ] . pActor , nFlags ) ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
}
else
{
//loc_20E4E:
2021-11-21 19:34:15 +00:00
if ( Elevator [ nElev ] . nRunRec > = 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:34:15 +00:00
runlist_SubRunRec ( Elevator [ nElev ] . nRunRec ) ;
Elevator [ nElev ] . nRunRec = - 1 ;
2021-10-15 20:13:05 +00:00
}
}
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
void AIElev : : Tick ( RunListEvent * ev )
{
int nRun = ev - > nRun ;
2021-11-21 20:09:27 +00:00
int nElev = RunData [ nRun ] . nObjIndex ;
2021-10-15 20:13:05 +00:00
assert ( nElev > = 0 & & nElev < ( int ) Elevator . Size ( ) ) ;
2019-08-31 07:47:15 +00:00
2021-11-21 20:09:27 +00:00
int nChannel = Elevator [ nElev ] . nChannel ;
int var_18 = Elevator [ nElev ] . nFlags ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
assert ( nChannel > = 0 & & nChannel < kMaxChannels ) ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:45:19 +00:00
auto pSector = Elevator [ nElev ] . pSector ;
2021-12-07 17:53:02 +00:00
DExhumedActor * pElevSpr = Elevator [ nElev ] . pActor ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
int ebp = 0 ; // initialise to *something*
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( var_18 & 0x2 )
{
int nZOffset = Elevator [ nElev ] . nCurZOffset ;
int nZVal = Elevator [ nElev ] . zOffsets [ nZOffset ] ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:45:19 +00:00
StartInterpolation ( pSector , Interp_Sect_Floorz ) ;
2022-02-02 23:46:04 +00:00
int fz = pSector - > int_floorz ( ) ;
int nVal = LongSeek ( & fz , nZVal , Elevator [ nElev ] . nParam1 , Elevator [ nElev ] . nParam2 ) ;
pSector - > set_int_floorz ( fz ) ;
2021-10-15 20:13:05 +00:00
ebp = nVal ;
if ( ! nVal )
{
if ( var_18 & 0x10 )
{
Elevator [ nElev ] . nCurZOffset ^ = 1 ;
2021-10-21 21:41:54 +00:00
StartElevSound ( pElevSpr , var_18 ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-10-20 20:16:50 +00:00
StopActorSound ( pElevSpr ) ;
2021-10-15 20:13:05 +00:00
runlist_SubRunRec ( nRun ) ;
2021-11-21 19:34:15 +00:00
Elevator [ nElev ] . nRunRec = - 1 ;
2021-10-15 20:13:05 +00:00
runlist_ReadyChannel ( nChannel ) ;
2019-08-31 07:47:15 +00:00
2021-10-21 21:41:54 +00:00
D3PlayFX ( StaticSound [ nStopSound ] , Elevator [ nElev ] . pActor ) ;
2021-10-15 20:13:05 +00:00
}
}
else
{
2022-08-20 14:44:30 +00:00
MoveSectorSprites ( pSector , nVal * inttoworld ) ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:45:19 +00:00
if ( nVal < 0 & & CheckSectorSprites ( pSector , 2 ) )
2021-10-15 20:13:05 +00:00
{
runlist_ChangeChannel ( nChannel , sRunChannels [ nChannel ] . c = = 0 ) ;
return ;
}
}
}
else
{
// loc_20FC3:
2022-02-02 23:46:04 +00:00
int ceilZ = pSector - > int_ceilingz ( ) ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
int nZOffset = Elevator [ nElev ] . nCurZOffset ;
int zVal = Elevator [ nElev ] . zOffsets [ nZOffset ] ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:45:19 +00:00
StartInterpolation ( pSector , Interp_Sect_Ceilingz ) ;
2021-11-21 19:34:15 +00:00
int nVal = LongSeek ( & ceilZ , zVal , Elevator [ nElev ] . nParam1 , Elevator [ nElev ] . nParam2 ) ;
2021-10-15 20:13:05 +00:00
ebp = nVal ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( ! nVal )
{
if ( var_18 & 0x10 )
{
Elevator [ nElev ] . nCurZOffset ^ = 1 ;
2019-08-26 03:59:14 +00:00
2021-10-21 21:41:54 +00:00
StartElevSound ( Elevator [ nElev ] . pActor , var_18 ) ;
2021-10-15 20:13:05 +00:00
}
else
{
runlist_SubRunRec ( nRun ) ;
2021-11-21 19:34:15 +00:00
Elevator [ nElev ] . nRunRec = - 1 ;
2021-10-21 21:41:54 +00:00
StopActorSound ( Elevator [ nElev ] . pActor ) ;
D3PlayFX ( StaticSound [ nStopSound ] , Elevator [ nElev ] . pActor ) ;
2021-10-15 20:13:05 +00:00
runlist_ReadyChannel ( nChannel ) ;
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
return ;
}
else if ( nVal > 0 )
{
if ( ceilZ = = zVal )
{
if ( var_18 & 0x4 ) {
2021-10-20 20:16:50 +00:00
SetQuake ( pElevSpr , 30 ) ;
2019-08-31 07:47:15 +00:00
}
2022-08-20 15:52:06 +00:00
PlayFXAtXYZ ( StaticSound [ kSound26 ] , pElevSpr - > spr . pos ) ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
if ( var_18 & 0x4 )
2019-08-31 07:47:15 +00:00
{
2021-11-22 22:45:19 +00:00
if ( CheckSectorSprites ( pSector , 1 ) ) {
2021-10-15 20:13:05 +00:00
return ;
}
}
else
{
2021-11-22 22:45:19 +00:00
if ( CheckSectorSprites ( pSector , 0 ) )
2021-10-15 20:13:05 +00:00
{
runlist_ChangeChannel ( nChannel , sRunChannels [ nChannel ] . c = = 0 ) ;
return ;
}
2019-08-31 07:47:15 +00:00
}
}
2021-10-15 20:13:05 +00:00
2021-11-22 22:45:19 +00:00
StartInterpolation ( pSector , Interp_Sect_Ceilingz ) ;
2022-02-02 23:35:12 +00:00
pSector - > set_int_ceilingz ( ceilZ ) ;
2021-10-15 20:13:05 +00:00
}
// maybe this doesn't go here?
2021-10-20 20:16:50 +00:00
while ( pElevSpr )
2021-10-15 20:13:05 +00:00
{
2022-01-31 19:07:15 +00:00
pElevSpr - > add_int_z ( ebp ) ;
2021-10-20 20:16:50 +00:00
pElevSpr = pElevSpr - > pTarget ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
2019-08-26 03:59:14 +00:00
// done
void InitWallFace ( )
{
2020-11-29 23:49:25 +00:00
WallFace . Clear ( ) ;
2019-08-26 03:59:14 +00:00
}
2021-11-22 17:22:38 +00:00
int BuildWallFace ( int nChannel , walltype * pWall , int nCount , . . . )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto WallFaceCount = WallFace . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
WallFace [ WallFaceCount ] . count = 0 ;
2021-11-22 17:22:38 +00:00
WallFace [ WallFaceCount ] . pWall = pWall ;
2019-08-31 07:47:15 +00:00
WallFace [ WallFaceCount ] . nChannel = nChannel ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( nCount > 8 ) {
nCount = 8 ;
}
2019-08-26 03:59:14 +00:00
2019-08-31 10:36:26 +00:00
va_list piclist ;
va_start ( piclist , nCount ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
while ( WallFace [ WallFaceCount ] . count < nCount )
2019-08-31 07:47:15 +00:00
{
2021-11-21 18:24:46 +00:00
int i = WallFace [ WallFaceCount ] . count ;
WallFace [ WallFaceCount ] . count + + ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
WallFace [ WallFaceCount ] . piclist [ i ] = ( int16_t ) va_arg ( piclist , int ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-31 10:36:26 +00:00
va_end ( piclist ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 16:37:39 +00:00
return WallFaceCount ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AIWallFace : : ProcessChannel ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-17 14:22:25 +00:00
int nWallFace = RunData [ ev - > nRun ] . nObjIndex ;
2020-11-29 23:49:25 +00:00
assert ( nWallFace > = 0 & & nWallFace < ( int ) WallFace . Size ( ) ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
int16_t nChannel = WallFace [ nWallFace ] . nChannel ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
int16_t si = sRunChannels [ nChannel ] . c ;
2019-08-26 03:59:14 +00:00
2021-11-21 18:24:46 +00:00
if ( ( si < = WallFace [ nWallFace ] . count ) & & ( si > = 0 ) )
2019-08-31 07:47:15 +00:00
{
2021-11-22 17:22:38 +00:00
WallFace [ nWallFace ] . pWall - > picnum = WallFace [ nWallFace ] . piclist [ si ] ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
}
// done
void InitPoint ( )
{
2020-11-29 23:49:25 +00:00
PointList . Clear ( ) ;
2019-08-26 03:59:14 +00:00
}
// done
int GrabPoint ( )
{
2020-11-29 23:49:25 +00:00
return PointList . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
}
// done
void InitSlide ( )
{
2020-11-29 23:49:25 +00:00
SlideData . Clear ( ) ;
2019-08-26 03:59:14 +00:00
}
2021-11-22 18:32:47 +00:00
int BuildSlide ( int nChannel , walltype * pStartWall , walltype * pWall1 , walltype * p2ndLastWall , walltype * pWall2 , walltype * pWall3 , walltype * pWall4 )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto nSlide = SlideData . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
auto pSector = pStartWall - > sectorp ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
SlideData [ nSlide ] . nRunRec = - 1 ;
2020-11-29 23:49:25 +00:00
SlideData [ nSlide ] . nChannel = nChannel ;
2021-10-21 10:08:22 +00:00
SlideData [ nSlide ] . nStart = - 1 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int nPoint = GrabPoint ( ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
SlideData [ nSlide ] . nStart = nPoint ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
PointList [ nPoint ] . nNext = - 1 ;
2021-11-22 18:32:47 +00:00
PointList [ nPoint ] . pSector = pSector ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
for ( auto & wal : wallsofsector ( pSector ) )
2019-08-31 07:47:15 +00:00
{
2021-11-21 20:09:27 +00:00
int ax = SlideData [ nSlide ] . nStart ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( ax > = 0 )
{
while ( ax > = 0 )
{
2021-11-22 18:32:47 +00:00
if ( wal . nextSector ( ) = = PointList [ ax ] . pSector ) {
2019-08-31 07:47:15 +00:00
break ;
}
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
ax = PointList [ ax ] . nNext ;
2019-08-31 07:47:15 +00:00
}
}
else
{
2021-11-22 18:32:47 +00:00
if ( wal . twoSided ( ) )
2019-08-31 07:47:15 +00:00
{
nPoint = GrabPoint ( ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
PointList [ nPoint ] . nNext = SlideData [ nSlide ] . nStart ;
2021-11-22 18:32:47 +00:00
PointList [ nPoint ] . pSector = wal . nextSector ( ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
SlideData [ nSlide ] . nStart = nPoint ;
2019-08-31 07:47:15 +00:00
}
}
}
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
SlideData [ nSlide ] . pStartWall = pStartWall ;
SlideData [ nSlide ] . pWall1 = pWall1 ;
SlideData [ nSlide ] . pWall2 = pWall2 ;
SlideData [ nSlide ] . pWall3 = pWall3 ;
2020-02-04 21:45:10 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x1 = pStartWall - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y1 = pStartWall - > wall_int_pos ( ) . Y ;
2020-02-04 21:45:10 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x2 = pWall2 - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y2 = pWall2 - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x3 = pWall1 - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y3 = pWall1 - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x4 = pWall3 - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y4 = pWall3 - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x5 = p2ndLastWall - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y5 = p2ndLastWall - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
SlideData [ nSlide ] . x6 = pWall4 - > wall_int_pos ( ) . X ;
SlideData [ nSlide ] . y6 = pWall4 - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
StartInterpolation ( pStartWall , Interp_Wall_X ) ;
StartInterpolation ( pStartWall , Interp_Wall_Y ) ;
2021-01-04 19:34:27 +00:00
2021-11-22 18:32:47 +00:00
StartInterpolation ( pWall1 , Interp_Wall_X ) ;
StartInterpolation ( pWall1 , Interp_Wall_Y ) ;
2021-01-04 19:34:27 +00:00
2021-11-22 18:32:47 +00:00
StartInterpolation ( pWall2 , Interp_Wall_X ) ;
StartInterpolation ( pWall2 , Interp_Wall_Y ) ;
2021-01-04 19:34:27 +00:00
2021-11-22 18:32:47 +00:00
StartInterpolation ( pWall3 , Interp_Wall_X ) ;
StartInterpolation ( pWall3 , Interp_Wall_Y ) ;
2021-01-04 19:34:27 +00:00
2021-11-22 18:32:47 +00:00
auto pActor = insertActor ( pSector , 899 ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
SlideData [ nSlide ] . pActor = pActor ;
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2022-10-05 15:38:44 +00:00
pActor - > spr . pos = DVector3 ( pStartWall - > pos , pSector - > floorz ) ;
2021-12-30 16:10:08 +00:00
pActor - > backuppos ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
SlideData [ nSlide ] . nRunC = 0 ;
2019-08-26 03:59:14 +00:00
2021-10-15 16:37:39 +00:00
return nSlide ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AISlide : : ProcessChannel ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-15 20:13:05 +00:00
int nRun = ev - > nRun ;
2021-10-17 14:22:25 +00:00
int nSlide = RunData [ nRun ] . nObjIndex ;
2020-11-29 23:49:25 +00:00
assert ( nSlide > = 0 & & nSlide < ( int ) SlideData . Size ( ) ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 20:09:27 +00:00
int nChannel = SlideData [ nSlide ] . nChannel ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
if ( SlideData [ nSlide ] . nRunRec > = 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:48:11 +00:00
runlist_SubRunRec ( SlideData [ nSlide ] . nRunRec ) ;
SlideData [ nSlide ] . nRunRec = - 1 ;
2021-10-15 20:13:05 +00:00
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( sRunChannels [ nChannel ] . c & & sRunChannels [ nChannel ] . c ! = 1 ) {
return ;
}
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
SlideData [ nSlide ] . nRunRec = runlist_AddRunRec ( NewRun , & RunData [ nRun ] ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
if ( SlideData [ nSlide ] . nRunC ! = sRunChannels [ nChannel ] . c )
2021-10-15 20:13:05 +00:00
{
2021-10-21 10:08:22 +00:00
D3PlayFX ( StaticSound [ kSound23 ] , SlideData [ nSlide ] . pActor ) ;
2021-11-21 19:48:11 +00:00
SlideData [ nSlide ] . nRunC = sRunChannels [ nChannel ] . c ;
2021-10-15 20:13:05 +00:00
}
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
void AISlide : : Tick ( RunListEvent * ev )
{
int nRun = ev - > nRun ;
2021-10-17 14:22:25 +00:00
int nSlide = RunData [ nRun ] . nObjIndex ;
2021-10-15 20:13:05 +00:00
assert ( nSlide > = 0 & & nSlide < ( int ) SlideData . Size ( ) ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 20:09:27 +00:00
int nChannel = SlideData [ nSlide ] . nChannel ;
2021-10-15 20:13:05 +00:00
int ebp = 0 ;
2019-08-26 03:59:14 +00:00
2021-11-21 20:09:27 +00:00
int cx = sRunChannels [ nChannel ] . c ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
int clipmask = ebp + 1 ; // RENAME
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( cx = = 1 )
{
2021-11-22 18:32:47 +00:00
auto pWall = SlideData [ nSlide ] . pWall1 ;
2022-01-27 16:41:10 +00:00
int x = pWall - > wall_int_pos ( ) . X ;
int y = pWall - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
int nSeekA = LongSeek ( & x , SlideData [ nSlide ] . x5 , 20 , 20 ) ;
2021-10-15 20:13:05 +00:00
int var_34 = nSeekA ;
int var_20 = nSeekA ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
int nSeekB = LongSeek ( & y , SlideData [ nSlide ] . y5 , 20 , 20 ) ;
2021-10-15 20:13:05 +00:00
int var_2C = nSeekB ;
int var_24 = nSeekB ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall1 , x , y ) ;
2021-10-21 10:08:22 +00:00
movesprite ( SlideData [ nSlide ] . pActor , var_34 < < 14 , var_2C < < 14 , 0 , 0 , 0 , CLIPMASK1 ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( var_34 = = 0 )
{
if ( var_2C = = 0 )
{
ebp = clipmask ;
}
}
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pStartWall ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
y = pWall - > wall_int_pos ( ) . Y + var_24 ;
x = pWall - > wall_int_pos ( ) . X + var_20 ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pStartWall , x , y ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pWall3 ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
x = pWall - > wall_int_pos ( ) . X ;
y = pWall - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
int nSeekC = LongSeek ( & x , SlideData [ nSlide ] . x6 , 20 , 20 ) ;
2021-10-15 20:13:05 +00:00
int var_30 = nSeekC ;
var_20 = nSeekC ;
2019-08-26 03:59:14 +00:00
2021-10-21 10:08:22 +00:00
int nSeekD = LongSeek ( & y , SlideData [ nSlide ] . y6 , 20 , 20 ) ;
2021-10-15 20:13:05 +00:00
int edi = nSeekD ;
var_24 = nSeekD ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall3 , x , y ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( var_30 = = 0 & & edi = = 0 ) {
ebp + + ;
}
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pWall2 ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
x = pWall - > wall_int_pos ( ) . X + var_20 ;
y = pWall - > wall_int_pos ( ) . Y + var_24 ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall2 , x , y ) ;
2021-10-15 20:13:05 +00:00
}
else if ( cx = = 0 ) // right branch
{
2021-11-22 18:32:47 +00:00
auto pWall = SlideData [ nSlide ] . pStartWall ;
2022-01-27 16:41:10 +00:00
int x = pWall - > wall_int_pos ( ) . X ;
int y = pWall - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
int nSeekA = LongSeek ( & x , SlideData [ nSlide ] . x1 , 20 , 20 ) ;
int edi = nSeekA ;
int var_1C = nSeekA ;
2019-11-20 16:21:32 +00:00
2021-10-15 20:13:05 +00:00
int nSeekB = LongSeek ( & y , SlideData [ nSlide ] . y1 , 20 , 20 ) ;
int ecx = nSeekB ;
int var_28 = nSeekB ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pStartWall , x , y ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( edi = = 0 & & ecx = = 0 ) {
ebp = clipmask ;
}
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pWall1 ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
y = pWall - > wall_int_pos ( ) . Y + var_28 ;
x = pWall - > wall_int_pos ( ) . X + var_1C ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall1 , x , y ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pWall2 ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
x = pWall - > wall_int_pos ( ) . X ;
y = pWall - > wall_int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
int nSeekC = LongSeek ( & x , SlideData [ nSlide ] . x2 , 20 , 20 ) ;
edi = nSeekC ;
var_1C = nSeekC ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
int nSeekD = LongSeek ( & y , SlideData [ nSlide ] . y2 , 20 , 20 ) ;
ecx = nSeekD ;
var_28 = nSeekD ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall2 , x , y ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( edi = = 0 & & ecx = = 0 ) {
ebp + + ;
}
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
pWall = SlideData [ nSlide ] . pWall3 ;
2019-08-26 03:59:14 +00:00
2022-01-27 16:41:10 +00:00
y = pWall - > wall_int_pos ( ) . Y + var_28 ;
x = pWall - > wall_int_pos ( ) . X + var_1C ;
2019-08-26 03:59:14 +00:00
2021-11-22 18:32:47 +00:00
dragpoint ( SlideData [ nSlide ] . pWall3 , x , y ) ;
2021-10-15 20:13:05 +00:00
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// loc_21A51:
if ( ebp > = 2 )
{
2021-11-21 19:48:11 +00:00
runlist_SubRunRec ( SlideData [ nSlide ] . nRunRec ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
SlideData [ nSlide ] . nRunRec = - 1 ;
2021-10-21 10:08:22 +00:00
D3PlayFX ( StaticSound [ nStopSound ] , SlideData [ nSlide ] . pActor ) ;
2021-10-15 20:13:05 +00:00
runlist_ReadyChannel ( nChannel ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
}
2021-10-21 11:09:29 +00:00
int BuildTrap ( DExhumedActor * pActor , int edx , int ebx , int ecx )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
int var_14 = edx ;
int var_18 = ebx ;
int var_10 = ecx ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
int nTrap = sTrap . Reserve ( 1 ) ;
2021-12-21 13:33:25 +00:00
sTrap [ nTrap ] = { } ;
2019-08-26 03:59:14 +00:00
2021-10-21 11:09:29 +00:00
ChangeActorStat ( pActor , 0 ) ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2021-12-23 16:04:54 +00:00
pActor - > spr . yvel = 0 ;
2022-08-31 22:50:13 +00:00
pActor - > clear_zvel ( ) ;
2021-12-23 16:04:54 +00:00
pActor - > spr . extra = - 1 ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . lotag = runlist_HeadRun ( ) + 1 ;
pActor - > spr . hitag = runlist_AddRunRec ( NewRun , nTrap , 0x1F0000 ) ;
2022-05-23 22:30:41 +00:00
pActor - > spr . intowner = runlist_AddRunRec ( pActor - > spr . lotag - 1 , nTrap , 0x1F0000 ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// GrabTimeSlot(3);
2019-08-26 03:59:14 +00:00
2021-10-21 11:09:29 +00:00
sTrap [ nTrap ] . pActor = pActor ;
2019-08-31 07:47:15 +00:00
sTrap [ nTrap ] . nType = ( var_14 = = 0 ) + 14 ;
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nState = - 1 ;
2019-08-31 07:47:15 +00:00
2020-11-29 23:49:25 +00:00
sTrap [ nTrap ] . nTrapInterval = 64 - ( 2 * var_10 ) ;
if ( sTrap [ nTrap ] . nTrapInterval < 5 ) {
sTrap [ nTrap ] . nTrapInterval = 5 ;
2019-08-31 07:47:15 +00:00
}
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nPicnum2 = 0 ;
2021-11-22 18:45:48 +00:00
sTrap [ nTrap ] . nPicnum1 = 0 ;
2019-08-31 07:47:15 +00:00
if ( var_18 = = - 1 ) {
2021-10-22 06:06:24 +00:00
return nTrap ;
2019-08-31 07:47:15 +00:00
}
2021-12-30 15:51:56 +00:00
auto pSector = pActor - > sector ( ) ;
2019-08-31 07:47:15 +00:00
2021-12-21 13:33:25 +00:00
for ( auto & wal : wallsofsector ( pSector ) )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
if ( var_18 = = wal . hitag )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
if ( sTrap [ nTrap ] . pWall1 ! = nullptr )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
sTrap [ nTrap ] . pWall2 = & wal ;
sTrap [ nTrap ] . nPicnum2 = wal . picnum ;
2021-10-21 11:09:29 +00:00
break ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-11-22 18:45:48 +00:00
sTrap [ nTrap ] . pWall1 = & wal ;
sTrap [ nTrap ] . nPicnum1 = wal . picnum ;
2019-08-31 07:47:15 +00:00
}
}
}
2021-12-30 16:10:08 +00:00
pActor - > backuppos ( ) ;
2021-10-21 11:09:29 +00:00
return nTrap ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AITrap : : ProcessChannel ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-11-21 20:09:27 +00:00
int nChannel = ev - > nParam & 0x3FFF ;
int nTrap = RunData [ ev - > nRun ] . nObjIndex ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( sRunChannels [ nChannel ] . c > 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nState = 12 ;
2021-10-15 20:13:05 +00:00
}
else
{
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nState = - 1 ;
2021-10-15 20:13:05 +00:00
}
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
void AITrap : : Tick ( RunListEvent * ev )
{
2021-11-21 20:09:27 +00:00
int nTrap = RunData [ ev - > nRun ] . nObjIndex ;
2021-12-07 17:53:02 +00:00
DExhumedActor * pActor = sTrap [ nTrap ] . pActor ;
if ( ! pActor ) return ;
2019-08-31 07:47:15 +00:00
2021-11-21 20:09:27 +00:00
if ( sTrap [ nTrap ] . nState > = 0 )
2021-10-15 20:13:05 +00:00
{
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nState - - ;
if ( sTrap [ nTrap ] . nState > 10 ) {
2019-08-31 07:47:15 +00:00
return ;
}
2021-11-21 20:09:27 +00:00
int nType = sTrap [ nTrap ] . nType ;
2021-10-15 20:13:05 +00:00
2021-11-21 20:09:27 +00:00
if ( sTrap [ nTrap ] . nState = = 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 20:09:27 +00:00
sTrap [ nTrap ] . nState = sTrap [ nTrap ] . nTrapInterval ;
2021-10-15 20:13:05 +00:00
if ( nType = = 14 )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
auto pWall = sTrap [ nTrap ] . pWall1 ;
if ( pWall )
2021-10-15 20:13:05 +00:00
{
2021-11-22 18:45:48 +00:00
pWall - > picnum = sTrap [ nTrap ] . nPicnum1 ;
2019-08-31 07:47:15 +00:00
}
2019-11-20 16:21:32 +00:00
2021-11-22 18:45:48 +00:00
pWall = sTrap [ nTrap ] . pWall1 ;
if ( pWall )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
pWall - > picnum = sTrap [ nTrap ] . nPicnum2 ;
2021-10-15 20:13:05 +00:00
}
}
}
else
{
// loc_21D92:
2021-11-21 20:09:27 +00:00
if ( sTrap [ nTrap ] . nState ! = 5 ) {
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-31 07:47:15 +00:00
2022-08-16 21:17:01 +00:00
auto pBullet = BuildBullet ( pActor , nType , 0 , pActor - > int_ang ( ) , nullptr , 1 ) ;
2021-10-19 22:26:26 +00:00
if ( pBullet )
2021-10-15 20:13:05 +00:00
{
if ( nType = = 15 )
{
2022-08-16 21:21:10 +00:00
pBullet - > set_int_ang ( ( pBullet - > int_ang ( ) - 512 ) & kAngleMask ) ;
2021-10-19 22:26:26 +00:00
D3PlayFX ( StaticSound [ kSound32 ] , pBullet ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-12-21 22:18:23 +00:00
pBullet - > spr . clipdist = 50 ;
2019-08-31 07:47:15 +00:00
2021-11-22 18:45:48 +00:00
auto pWall = sTrap [ nTrap ] . pWall1 ;
if ( pWall )
2019-08-31 07:47:15 +00:00
{
2021-11-22 18:45:48 +00:00
pWall - > picnum = sTrap [ nTrap ] . nPicnum1 + 1 ;
2021-10-15 20:13:05 +00:00
}
2019-11-22 00:10:56 +00:00
2021-11-22 18:45:48 +00:00
pWall = sTrap [ nTrap ] . pWall2 ;
if ( pWall )
2021-10-15 20:13:05 +00:00
{
2021-11-22 18:45:48 +00:00
pWall - > picnum = sTrap [ nTrap ] . nPicnum2 ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
2021-10-19 22:26:26 +00:00
D3PlayFX ( StaticSound [ kSound36 ] , pBullet ) ;
2019-08-31 07:47:15 +00:00
}
}
}
}
2019-08-26 03:59:14 +00:00
}
2021-10-21 11:09:29 +00:00
int BuildArrow ( DExhumedActor * nSprite , int nVal )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
return BuildTrap ( nSprite , 0 , - 1 , nVal ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-21 11:09:29 +00:00
int BuildFireBall ( DExhumedActor * nSprite , int a , int b )
2019-08-26 03:59:14 +00:00
{
2019-08-31 07:47:15 +00:00
return BuildTrap ( nSprite , 1 , a , b ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-21 11:15:47 +00:00
DExhumedActor * BuildSpark ( DExhumedActor * pActor , int nVal )
2019-08-26 03:59:14 +00:00
{
2021-12-30 15:51:56 +00:00
auto pSpark = insertActor ( pActor - > sector ( ) , 0 ) ;
2019-08-31 07:47:15 +00:00
2022-08-29 17:27:52 +00:00
pSpark - > spr . pos . XY ( ) = pActor - > spr . pos . XY ( ) ;
2021-12-23 17:54:40 +00:00
pSpark - > spr . cstat = 0 ;
pSpark - > spr . shade = - 127 ;
pSpark - > spr . pal = 1 ;
pSpark - > spr . xoffset = 0 ;
pSpark - > spr . yoffset = 0 ;
pSpark - > spr . xrepeat = 50 ;
pSpark - > spr . yrepeat = 50 ;
2019-08-31 07:47:15 +00:00
if ( nVal > = 2 )
{
2021-12-23 17:54:40 +00:00
pSpark - > spr . picnum = kEnergy2 ;
2019-08-31 07:47:15 +00:00
nSmokeSparks + + ;
if ( nVal = = 3 )
{
2021-12-23 17:54:40 +00:00
pSpark - > spr . xrepeat = 120 ;
pSpark - > spr . yrepeat = 120 ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-12-23 17:54:40 +00:00
pSpark - > spr . xrepeat = pActor - > spr . xrepeat + 15 ;
pSpark - > spr . yrepeat = pActor - > spr . xrepeat + 15 ;
2019-08-31 07:47:15 +00:00
}
}
else
{
2022-08-16 21:15:49 +00:00
int nAngle = ( pActor - > int_ang ( ) + 256 ) - RandomSize ( 9 ) ;
2019-08-31 07:47:15 +00:00
if ( nVal )
{
2021-12-23 17:54:40 +00:00
pSpark - > spr . xvel = bcos ( nAngle , - 5 ) ;
pSpark - > spr . yvel = bsin ( nAngle , - 5 ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-12-23 17:54:40 +00:00
pSpark - > spr . xvel = bcos ( nAngle , - 6 ) ;
pSpark - > spr . yvel = bsin ( nAngle , - 6 ) ;
2019-08-31 07:47:15 +00:00
}
2022-08-31 22:28:40 +00:00
pSpark - > set_int_zvel ( - ( RandomSize ( 4 ) < < 7 ) ) ;
2021-12-23 17:54:40 +00:00
pSpark - > spr . picnum = kTile985 + nVal ;
2019-08-31 07:47:15 +00:00
}
2022-08-20 14:44:30 +00:00
pSpark - > spr . pos . Z = pActor - > spr . pos . Z ;
2021-12-23 17:54:40 +00:00
pSpark - > spr . lotag = runlist_HeadRun ( ) + 1 ;
pSpark - > spr . clipdist = 1 ;
pSpark - > spr . hitag = 0 ;
2021-12-30 16:10:08 +00:00
pSpark - > backuppos ( ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// GrabTimeSlot(3);
2019-08-26 03:59:14 +00:00
2021-12-23 17:54:40 +00:00
pSpark - > spr . extra = - 1 ;
2022-05-23 22:30:41 +00:00
pSpark - > spr . intowner = runlist_AddRunRec ( pSpark - > spr . lotag - 1 , pSpark , 0x260000 ) ;
2021-12-23 17:54:40 +00:00
pSpark - > spr . hitag = runlist_AddRunRec ( NewRun , pSpark , 0x260000 ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 11:15:47 +00:00
return pSpark ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AISpark : : Tick ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-21 11:15:47 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . shade + = 3 ;
pActor - > spr . xrepeat - = 2 ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . xrepeat > = 4 & & pActor - > spr . shade < = 100 )
2019-08-31 07:47:15 +00:00
{
2021-12-23 16:04:54 +00:00
pActor - > spr . yrepeat - = 2 ;
2019-08-26 03:59:14 +00:00
2020-02-04 21:45:10 +00:00
// calling BuildSpark() with 2nd parameter as '1' will set kTile986
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . picnum = = kTile986 & & ( pActor - > spr . xrepeat & 2 ) )
2019-08-31 07:47:15 +00:00
{
2021-10-21 11:15:47 +00:00
BuildSpark ( pActor , 2 ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . picnum > = kTile3000 ) {
2019-08-31 07:47:15 +00:00
return ;
}
2019-08-26 03:59:14 +00:00
2022-08-31 22:30:16 +00:00
pActor - > add_int_zvel ( 128 ) ;
2019-08-26 03:59:14 +00:00
2022-09-01 14:54:28 +00:00
auto nMov = movesprite ( pActor , pActor - > int_xvel ( ) < < 12 , pActor - > spr . yvel < < 12 , pActor - > int_zvel ( ) , 2560 , - 2560 , CLIPMASK1 ) ;
2021-10-21 11:15:47 +00:00
if ( ! nMov . type & & ! nMov . exbits ) {
2019-08-31 07:47:15 +00:00
return ;
}
2019-08-26 03:59:14 +00:00
2022-08-31 23:10:51 +00:00
if ( pActor - > float_zvel ( ) < = 0 ) {
2019-08-31 07:47:15 +00:00
return ;
}
}
2019-08-26 03:59:14 +00:00
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2021-12-23 16:04:54 +00:00
pActor - > spr . yvel = 0 ;
2022-08-31 22:50:13 +00:00
pActor - > clear_zvel ( ) ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . picnum > kTile3000 ) {
2019-08-31 07:47:15 +00:00
nSmokeSparks - - ;
}
2019-08-26 03:59:14 +00:00
2022-05-23 22:30:41 +00:00
runlist_DoSubRunRec ( pActor - > spr . intowner ) ;
2021-12-23 16:04:54 +00:00
runlist_FreeRun ( pActor - > spr . lotag - 1 ) ;
runlist_SubRunRec ( pActor - > spr . hitag ) ;
2021-10-21 11:15:47 +00:00
DeleteActor ( pActor ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
2019-08-26 03:59:14 +00:00
void DimLights ( )
{
2021-11-21 20:09:27 +00:00
static int word_96786 = 0 ;
2019-08-31 07:47:15 +00:00
word_96786 = word_96786 = = 0 ;
if ( word_96786 = = 0 )
return ;
2021-12-21 08:23:39 +00:00
for ( auto & sect : sector )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
if ( sect . ceilingshade < 100 )
sect . ceilingshade + + ;
2019-08-31 07:47:15 +00:00
2021-11-22 19:18:14 +00:00
if ( sect . floorshade < 100 )
sect . floorshade + + ;
2019-08-31 07:47:15 +00:00
2021-11-22 19:18:14 +00:00
for ( auto & wal : wallsofsector ( & sect ) )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
if ( wal . shade < 100 )
wal . shade + + ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
}
void DoFinale ( )
{
2019-08-31 07:47:15 +00:00
static int dword_96788 = 0 ;
2020-09-02 20:55:57 +00:00
static int nextstage = 0 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( ! lFinaleStart )
return ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
dword_96788 + + ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( dword_96788 < 90 )
{
if ( ! ( dword_96788 & 2 ) )
{
int nAng = RandomSize ( 11 ) ;
2022-08-16 21:21:10 +00:00
pFinaleSpr - > set_int_ang ( nAng ) ;
2021-10-21 11:20:00 +00:00
BuildSpark ( pFinaleSpr , 1 ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( ! RandomSize ( 2 ) )
{
2021-10-21 11:20:00 +00:00
PlayFX2 ( StaticSound [ kSound78 ] | 0x2000 , pFinaleSpr ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
for ( int i = 0 ; i < nTotalPlayers ; i + + ) {
nQuake [ i ] = 1280 ;
}
}
}
else
{
DimLights ( ) ;
if ( nDronePitch < = - 2400 )
{
if ( nFinaleStage < 2 )
{
if ( nFinaleStage = = 1 )
{
StopLocalSound ( ) ;
PlayLocalSound ( StaticSound [ kSound76 ] , 0 ) ;
2021-02-18 10:46:36 +00:00
nextstage = PlayClock + 120 ;
2020-02-04 21:45:10 +00:00
nFinaleStage + + ;
2019-08-31 07:47:15 +00:00
}
}
else if ( nFinaleStage < = 2 )
{
2021-02-18 10:46:36 +00:00
if ( PlayClock > = nextstage )
2019-08-31 07:47:15 +00:00
{
PlayLocalSound ( StaticSound [ kSound77 ] , 0 ) ;
nFinaleStage + + ;
2021-02-18 10:46:36 +00:00
nextstage = PlayClock + 360 ;
2019-08-31 07:47:15 +00:00
}
}
2021-02-18 10:46:36 +00:00
else if ( nFinaleStage = = 3 & & PlayClock > = nextstage )
2019-08-31 07:47:15 +00:00
{
2020-09-11 18:56:46 +00:00
LevelFinished ( ) ;
2019-08-31 07:47:15 +00:00
}
}
else
{
nDronePitch - = 128 ;
BendAmbientSound ( ) ;
nFinaleStage = 1 ;
}
}
2019-08-26 03:59:14 +00:00
}
2021-11-22 22:56:36 +00:00
DExhumedActor * BuildEnergyBlock ( sectortype * pSector )
2019-08-26 03:59:14 +00:00
{
2022-08-29 17:40:20 +00:00
DVector2 apos ( 0 , 0 ) ;
2021-12-30 09:30:21 +00:00
2021-11-21 08:42:43 +00:00
for ( auto & wal : wallsofsector ( pSector ) )
2019-08-31 07:47:15 +00:00
{
2022-08-29 17:40:20 +00:00
apos + = wal . pos ;
2021-11-21 08:42:43 +00:00
wal . picnum = kClockSymbol16 ;
wal . pal = 0 ;
wal . shade = 50 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-22 22:56:36 +00:00
auto pActor = insertActor ( pSector , 406 ) ;
2020-10-14 22:10:37 +00:00
2022-08-29 17:40:20 +00:00
pActor - > spr . pos . XY ( ) = apos / pSector - > wallnum ;
2019-08-26 03:59:14 +00:00
2021-11-22 22:56:36 +00:00
pSector - > extra = ( int16_t ) EnergyBlocks . Push ( pActor ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// GrabTimeSlot(3);
2019-08-26 03:59:14 +00:00
2022-08-20 14:44:30 +00:00
pActor - > spr . pos . Z = pSector - > firstWall ( ) - > nextSector ( ) - > floorz ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
// CHECKME - name of this variable?
2022-02-02 23:46:04 +00:00
int nRepeat = ( pActor - > int_pos ( ) . Z - pSector - > int_floorz ( ) ) > > 8 ;
2019-08-31 07:47:15 +00:00
if ( nRepeat > 255 ) {
nRepeat = 255 ;
}
2019-08-26 03:59:14 +00:00
2021-12-23 17:54:40 +00:00
pActor - > spr . xrepeat = nRepeat ;
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2021-12-23 17:54:40 +00:00
pActor - > spr . yvel = 0 ;
2022-08-31 22:50:13 +00:00
pActor - > clear_zvel ( ) ;
2021-12-23 17:54:40 +00:00
pActor - > spr . extra = - 1 ;
pActor - > spr . lotag = runlist_HeadRun ( ) + 1 ;
pActor - > spr . hitag = 0 ;
2022-05-23 22:30:41 +00:00
pActor - > spr . intowner = runlist_AddRunRec ( pActor - > spr . lotag - 1 , pActor , 0x250000 ) ;
2021-12-30 16:10:08 +00:00
pActor - > backuppos ( ) ;
2019-08-26 03:59:14 +00:00
2021-10-21 13:02:18 +00:00
return pActor ;
2019-08-26 03:59:14 +00:00
}
// TODO - tidy
void KillCreatures ( )
{
2019-08-31 07:47:15 +00:00
signed int v0 ;
signed int v1 ;
v0 = 99 ;
v1 = 99 ;
while ( 1 )
{
if ( v0 ! = 100 )
{
2021-10-20 20:27:16 +00:00
ExhumedStatIterator it ( v1 ) ;
while ( auto i = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2021-10-21 13:02:18 +00:00
runlist_DamageEnemy ( i , nullptr , 1600 ) ;
2019-08-31 07:47:15 +00:00
}
}
+ + v0 ;
+ + v1 ;
if ( v0 > 107 ) {
return ;
}
}
2019-08-26 03:59:14 +00:00
}
2021-10-21 13:02:18 +00:00
void ExplodeEnergyBlock ( DExhumedActor * pActor )
2019-08-26 03:59:14 +00:00
{
2021-12-30 15:51:56 +00:00
auto pSector = pActor - > sector ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
for ( auto & wal : wallsofsector ( pSector ) )
2019-08-31 07:47:15 +00:00
{
2021-11-21 08:20:44 +00:00
if ( ! wal . twoSided ( ) ) continue ;
2021-11-26 19:55:13 +00:00
auto nextwal = wal . nextWall ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-26 19:55:13 +00:00
if ( nextwal - > pal > = 4 ) {
nextwal - > pal = 7 ;
2019-08-31 07:47:15 +00:00
}
else {
2021-11-26 19:55:13 +00:00
nextwal - > pal = 0 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-26 19:55:13 +00:00
nextwal - > shade = 50 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
if ( pSector - > floorpal > = 4 ) {
pSector - > floorpal = 7 ;
2019-08-31 07:47:15 +00:00
}
else {
2021-11-21 08:20:44 +00:00
pSector - > floorpal = 0 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
pSector - > floorshade = 50 ;
pSector - > extra = - 1 ;
2022-02-02 23:35:12 +00:00
pSector - > set_int_floorz ( pActor - > int_pos ( ) . Z ) ;
2019-11-20 16:21:32 +00:00
2022-08-20 14:44:30 +00:00
pActor - > spr . pos . Z = ( pActor - > spr . pos . Z + pSector - > floorz ) * 0.5 ;
2019-08-26 03:59:14 +00:00
2021-10-21 11:15:47 +00:00
BuildSpark ( pActor , 3 ) ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = 0 ;
pActor - > spr . xrepeat = 100 ;
2019-08-26 03:59:14 +00:00
2021-10-21 13:02:18 +00:00
PlayFX2 ( StaticSound [ kSound78 ] , pActor ) ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . xrepeat = 0 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
nEnergyTowers - - ;
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
for ( int i = 0 ; i < 20 ; i + + )
2019-08-31 07:47:15 +00:00
{
2022-08-16 21:21:10 +00:00
pActor - > set_int_ang ( RandomSize ( 11 ) ) ;
2021-10-21 11:15:47 +00:00
BuildSpark ( pActor , 1 ) ; // shoot out blue orbs
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2019-10-12 21:09:55 +00:00
TintPalette ( 64 , 64 , 64 ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( nEnergyTowers = = 1 )
{
runlist_ChangeChannel ( nEnergyChan , nEnergyTowers ) ;
2022-08-15 14:07:45 +00:00
StatusMessage ( 1000 , GStrings ( " TXT_EX_TAKEOUT " ) ) ;
2019-08-31 07:47:15 +00:00
}
else if ( nEnergyTowers ! = 0 )
{
2022-08-15 14:07:45 +00:00
FString msg = GStrings ( " TXT_EX_TOWERSREMAIN " ) ;
msg . Substitute ( " %d " , std : : to_string ( nEnergyTowers ) . c_str ( ) ) ;
StatusMessage ( 500 , msg . GetChars ( ) ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-10-21 13:02:18 +00:00
pFinaleSpr = pActor ;
2021-02-18 10:46:36 +00:00
lFinaleStart = PlayClock ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( ! lFinaleStart ) {
lFinaleStart = lFinaleStart + 1 ;
}
2019-08-26 03:59:14 +00:00
2021-12-21 08:23:39 +00:00
for ( auto & sect : sector )
2019-08-31 07:47:15 +00:00
{
2021-11-21 08:20:44 +00:00
if ( sect . ceilingpal = = 1 ) {
sect . ceilingpal = 0 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
if ( sect . floorpal = = 1 ) {
sect . floorpal = 0 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-11-21 08:20:44 +00:00
for ( auto & wal : wallsofsector ( & sect ) )
{
if ( wal . pal = = 1 ) {
wal . pal = 0 ;
2019-08-31 07:47:15 +00:00
}
}
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
KillCreatures ( ) ;
}
2019-08-26 03:59:14 +00:00
2021-10-21 13:02:18 +00:00
ChangeActorStat ( pActor , 0 ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AIEnergyBlock : : Damage ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-21 13:02:18 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
ev - > nDamage > > = 2 ;
if ( ev - > nDamage < = 0 ) {
return ;
}
2021-12-23 17:54:40 +00:00
if ( ev - > nDamage < pActor - > spr . xrepeat )
2019-08-31 07:47:15 +00:00
{
2021-12-23 17:54:40 +00:00
pActor - > spr . xrepeat - = ev - > nDamage ;
2019-08-26 03:59:14 +00:00
2021-10-21 11:15:47 +00:00
auto pActor2 = insertActor ( lasthitsect , 0 ) ;
2019-08-26 03:59:14 +00:00
2022-08-16 21:21:10 +00:00
pActor2 - > set_int_ang ( ev - > nParam ) ;
2022-08-20 15:17:50 +00:00
pActor2 - > spr . pos = lasthit ;
2019-08-26 03:59:14 +00:00
2021-10-21 11:15:47 +00:00
BuildSpark ( pActor2 , 0 ) ; // shoot out blue orb when damaged
DeleteActor ( pActor2 ) ;
2021-10-15 20:13:05 +00:00
}
else
{
2021-12-23 17:54:40 +00:00
pActor - > spr . xrepeat = 0 ; // using xrepeat to store health
2021-10-21 13:02:18 +00:00
ExplodeEnergyBlock ( pActor ) ;
2021-10-15 20:13:05 +00:00
}
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
void AIEnergyBlock : : RadialDamage ( RunListEvent * ev )
{
2021-10-21 13:02:18 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2019-08-26 03:59:14 +00:00
2021-12-30 15:51:56 +00:00
auto pSector = pActor - > sector ( ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 23:51:32 +00:00
if ( pSector - > extra = = - 1 ) {
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-26 03:59:14 +00:00
2022-08-20 18:25:38 +00:00
double nFloorZ = pSector - > floorz ;
2019-08-26 03:59:14 +00:00
2022-08-20 18:25:38 +00:00
pSector - > setfloorz ( pActor - > spr . pos . Z ) ;
2022-08-20 14:47:27 +00:00
pActor - > spr . pos . Z - - ;
2019-08-26 03:59:14 +00:00
2021-10-21 13:02:18 +00:00
ev - > nDamage = runlist_CheckRadialDamage ( pActor ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// restore previous values
2022-02-02 23:35:12 +00:00
pSector - > set_int_floorz ( nFloorZ ) ;
2022-08-20 14:47:27 +00:00
pActor - > spr . pos . Z + + ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
if ( ev - > nDamage < = 0 ) {
return ;
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// fall through to case 0x80000
Damage ( ev ) ;
}
2019-08-26 03:59:14 +00:00
2021-10-21 15:24:52 +00:00
DExhumedActor * BuildObject ( DExhumedActor * pActor , int nOjectType , int nHitag )
2019-08-26 03:59:14 +00:00
{
2021-10-21 13:33:41 +00:00
ChangeActorStat ( pActor , ObjectStatnum [ nOjectType ] ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
// 0x7FFD to ensure set as blocking ('B' and 'H') sprite and also disable translucency and set not invisible
2021-12-23 17:54:40 +00:00
pActor - > spr . cstat = ( pActor - > spr . cstat | CSTAT_SPRITE_BLOCK_ALL ) & ~ ( CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_INVISIBLE ) ;
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2021-12-23 17:54:40 +00:00
pActor - > spr . yvel = 0 ;
2022-08-31 22:50:13 +00:00
pActor - > clear_zvel ( ) ;
2021-12-23 17:54:40 +00:00
pActor - > spr . extra = - 1 ;
pActor - > spr . lotag = runlist_HeadRun ( ) + 1 ;
pActor - > spr . hitag = 0 ;
2022-05-23 22:30:41 +00:00
pActor - > spr . intowner = runlist_AddRunRec ( pActor - > spr . lotag - 1 , pActor , 0x170000 ) ;
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// GrabTimeSlot(3);
2021-10-21 15:24:52 +00:00
pActor - > nPhase = ObjectList . Push ( pActor ) ;
2021-12-23 17:54:40 +00:00
if ( pActor - > spr . statnum = = kStatDestructibleSprite ) {
2021-10-21 15:24:52 +00:00
pActor - > nHealth = 4 ;
2019-08-31 07:47:15 +00:00
}
else {
2021-10-21 15:24:52 +00:00
pActor - > nHealth = 120 ;
2019-08-31 07:47:15 +00:00
}
2021-10-21 15:24:52 +00:00
pActor - > nRun = runlist_AddRunRec ( NewRun , pActor , 0x170000 ) ;
2019-08-31 07:47:15 +00:00
2021-11-21 20:09:27 +00:00
int nSeq = ObjectSeq [ nOjectType ] ;
2019-08-31 07:47:15 +00:00
2019-12-02 21:19:27 +00:00
if ( nSeq > - 1 )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
pActor - > nIndex = SeqOffsets [ nSeq ] ;
2019-08-31 07:47:15 +00:00
if ( ! nOjectType ) // if not Explosion Trigger (e.g. Exploding Fire Cauldron)
{
2021-10-21 15:24:52 +00:00
pActor - > nFrame = RandomSize ( 4 ) % ( SeqSize [ pActor - > nIndex ] - 1 ) ;
2019-08-31 07:47:15 +00:00
}
2021-12-30 15:51:56 +00:00
auto pActor2 = insertActor ( pActor - > sector ( ) , 0 ) ;
2021-10-21 15:24:52 +00:00
pActor - > pTarget = pActor2 ;
pActor - > nIndex2 = - 1 ;
2019-08-31 07:47:15 +00:00
2021-12-23 17:54:40 +00:00
pActor2 - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2022-10-05 15:38:44 +00:00
pActor2 - > spr . pos = pActor - > spr . pos ;
}
2019-12-02 21:19:27 +00:00
else
{
2021-10-21 15:24:52 +00:00
pActor - > nFrame = 0 ;
pActor - > nIndex = - 1 ;
2019-12-02 21:19:27 +00:00
2021-12-23 17:54:40 +00:00
if ( pActor - > spr . statnum = = kStatDestructibleSprite ) {
2021-10-21 15:24:52 +00:00
pActor - > nIndex2 = - 1 ;
2019-12-02 21:19:27 +00:00
}
else {
2021-10-21 15:24:52 +00:00
pActor - > nIndex2 = - nHitag ;
2019-12-02 21:19:27 +00:00
}
}
2021-12-30 16:10:08 +00:00
pActor - > backuppos ( ) ;
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
return pActor ;
2019-08-26 03:59:14 +00:00
}
2020-02-04 21:45:10 +00:00
// in-game destructable wall mounted screen
2021-10-21 15:24:52 +00:00
void ExplodeScreen ( DExhumedActor * pActor )
2019-08-26 03:59:14 +00:00
{
2022-08-20 14:38:47 +00:00
pActor - > spr . pos . Z - = GetActorHeightF ( pActor ) * 0.5 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
for ( int i = 0 ; i < 30 ; i + + ) {
2021-10-21 15:24:52 +00:00
BuildSpark ( pActor , 0 ) ; // shoot out blue orbs
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2021-10-21 15:24:52 +00:00
PlayFX2 ( StaticSound [ kSound78 ] , pActor ) ;
2019-08-26 03:59:14 +00:00
}
2021-10-15 20:13:05 +00:00
void AIObject : : Tick ( RunListEvent * ev )
2019-08-26 03:59:14 +00:00
{
2021-10-21 15:24:52 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2021-12-23 16:04:54 +00:00
int nStat = pActor - > spr . statnum ;
2021-11-16 17:45:07 +00:00
int bx = pActor - > nIndex ;
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( nStat = = 97 | | ( ! ( pActor - > spr . cstat & CSTAT_SPRITE_BLOCK_ALL ) ) ) {
2021-10-15 20:13:05 +00:00
return ;
}
if ( nStat ! = kStatExplodeTarget ) {
2021-10-20 22:00:26 +00:00
Gravity ( pActor ) ;
2021-10-15 20:13:05 +00:00
}
// do animation
if ( bx ! = - 1 )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
pActor - > nFrame + + ;
if ( pActor - > nFrame > = SeqSize [ bx ] ) {
pActor - > nFrame = 0 ;
2019-08-31 07:47:15 +00:00
}
2021-12-23 16:04:54 +00:00
pActor - > spr . picnum = seq_GetSeqPicnum2 ( bx , pActor - > nFrame ) ;
2021-10-15 20:13:05 +00:00
}
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
if ( pActor - > nHealth > = 0 ) {
2021-10-15 20:13:05 +00:00
goto FUNCOBJECT_GOTO ;
}
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
pActor - > nHealth + + ;
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
if ( pActor - > nHealth )
2021-10-15 20:13:05 +00:00
{
FUNCOBJECT_GOTO :
if ( nStat ! = kStatExplodeTarget )
{
2022-09-01 14:54:28 +00:00
auto nMov = movesprite ( pActor , pActor - > int_xvel ( ) < < 6 , pActor - > spr . yvel < < 6 , pActor - > int_zvel ( ) , 0 , 0 , CLIPMASK0 ) ;
2021-10-15 20:13:05 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . statnum = = kStatExplodeTrigger ) {
pActor - > spr . pal = 1 ;
2019-08-31 07:47:15 +00:00
}
2021-10-21 15:24:52 +00:00
if ( nMov . exbits & kHitAux2 )
2019-08-31 07:47:15 +00:00
{
2021-12-23 16:04:54 +00:00
pActor - > spr . xvel - = pActor - > spr . xvel > > 3 ;
pActor - > spr . yvel - = pActor - > spr . yvel > > 3 ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
2021-10-21 15:24:52 +00:00
if ( nMov . type = = kHitSprite )
2019-08-31 07:47:15 +00:00
{
2021-12-23 16:04:54 +00:00
pActor - > spr . yvel = 0 ;
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2019-08-31 07:47:15 +00:00
}
}
2021-10-15 20:13:05 +00:00
return ;
}
else
{
int var_18 ;
// red branch
2022-08-20 18:25:38 +00:00
if ( ( nStat = = kStatExplodeTarget ) | | ( pActor - > spr . pos . Z < pActor - > sector ( ) - > floorz ) )
2019-08-31 07:47:15 +00:00
{
2021-10-15 20:13:05 +00:00
var_18 = 36 ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
else
2019-08-31 07:47:15 +00:00
{
2021-10-15 20:13:05 +00:00
var_18 = 34 ;
}
2019-08-31 07:47:15 +00:00
2022-08-20 14:19:07 +00:00
AddFlash ( pActor - > sector ( ) , pActor - > spr . pos , 128 ) ;
2022-08-20 13:42:53 +00:00
BuildAnim ( nullptr , var_18 , 0 , DVector3 ( pActor - > spr . pos . XY ( ) , pActor - > sector ( ) - > floorz ) , pActor - > sector ( ) , 240 , 4 ) ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
// int edi = nSprite | 0x4000;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( nStat = = kStatExplodeTrigger )
{
for ( int i = 4 ; i < 8 ; i + + ) {
2021-10-21 15:24:52 +00:00
BuildCreatureChunk ( pActor , seq_GetSeqPicnum ( kSeqFirePot , ( i > > 2 ) + 1 , 0 ) , true ) ;
2019-08-31 07:47:15 +00:00
}
2021-10-21 15:24:52 +00:00
runlist_RadialDamageEnemy ( pActor , 200 , 20 ) ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
else if ( nStat = = kStatExplodeTarget )
2019-08-31 07:47:15 +00:00
{
2021-10-15 20:13:05 +00:00
for ( int i = 0 ; i < 8 ; i + + ) {
2021-10-21 15:24:52 +00:00
BuildCreatureChunk ( pActor , seq_GetSeqPicnum ( kSeqFirePot , ( i > > 1 ) + 3 , 0 ) , true ) ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( ! ( currentLevel - > gameflags & LEVEL_EX_MULTI ) | | nStat ! = kStatExplodeTrigger )
{
2022-05-23 22:30:41 +00:00
runlist_SubRunRec ( pActor - > spr . intowner ) ;
2021-10-21 15:24:52 +00:00
runlist_SubRunRec ( pActor - > nRun ) ;
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
DeleteActor ( pActor ) ;
2021-10-15 20:13:05 +00:00
return ;
}
else
{
2021-10-21 08:20:08 +00:00
StartRegenerate ( pActor ) ;
2021-10-21 15:24:52 +00:00
pActor - > nHealth = 120 ;
2019-08-31 07:47:15 +00:00
2022-10-05 15:38:44 +00:00
pActor - > spr . pos = pActor - > pTarget - > spr . pos ;
2021-12-30 15:51:56 +00:00
ChangeActorSect ( pActor , pActor - > pTarget - > sector ( ) ) ;
2021-10-15 20:13:05 +00:00
return ;
}
}
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
void AIObject : : Damage ( RunListEvent * ev )
{
2021-10-21 15:24:52 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2021-12-23 16:04:54 +00:00
int nStat = pActor - > spr . statnum ;
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
if ( nStat > = 150 | | pActor - > nHealth < = 0 ) {
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( nStat = = 98 )
{
2021-10-21 15:24:52 +00:00
D3PlayFX ( ( StaticSound [ kSound47 ] | 0x2000 ) | ( RandomSize ( 2 ) < < 9 ) , pActor ) ;
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-31 07:47:15 +00:00
2021-11-21 18:24:46 +00:00
pActor - > nHealth - = ( int16_t ) ev - > nDamage ;
2021-10-21 15:24:52 +00:00
if ( pActor - > nHealth > 0 ) {
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( nStat = = kStatDestructibleSprite )
{
2021-10-21 15:24:52 +00:00
ExplodeScreen ( pActor ) ;
2021-10-15 20:13:05 +00:00
}
else
{
2021-10-21 15:24:52 +00:00
pActor - > nHealth = - ( RandomSize ( 3 ) + 1 ) ;
2021-10-15 20:13:05 +00:00
}
}
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
void AIObject : : Draw ( RunListEvent * ev )
{
2021-10-21 15:24:52 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2021-11-16 17:45:07 +00:00
int bx = pActor - > nIndex ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
if ( bx > - 1 )
{
2021-10-21 15:24:52 +00:00
seq_PlotSequence ( ev - > nParam , bx , pActor - > nFrame , 1 ) ;
2021-10-15 20:13:05 +00:00
}
return ;
}
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
void AIObject : : RadialDamage ( RunListEvent * ev )
{
2021-10-21 15:24:52 +00:00
auto pActor = ev - > pObjActor ;
if ( ! pActor ) return ;
2019-08-26 03:59:14 +00:00
2021-12-23 16:04:54 +00:00
int nStat = pActor - > spr . statnum ;
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > nHealth > 0 & & pActor - > spr . cstat & CSTAT_SPRITE_BLOCK_ALL
2021-10-15 20:13:05 +00:00
& & ( nStat ! = kStatExplodeTarget
2021-12-21 22:18:23 +00:00
| | ev - > pRadialActor - > spr . statnum = = 201
2021-10-15 20:13:05 +00:00
| | ( nRadialBullet ! = 3 & & nRadialBullet > - 1 )
2021-12-21 22:18:23 +00:00
| | ev - > pRadialActor - > spr . statnum = = kStatExplodeTrigger ) )
2021-10-15 20:13:05 +00:00
{
2021-10-21 15:24:52 +00:00
int nDamage = runlist_CheckRadialDamage ( pActor ) ;
2021-10-15 20:13:05 +00:00
if ( nDamage < = 0 ) {
return ;
}
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . statnum ! = kStatAnubisDrum ) {
2021-10-21 15:24:52 +00:00
pActor - > nHealth - = nDamage ;
2021-10-15 20:13:05 +00:00
}
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . statnum = = kStatExplodeTarget )
2021-10-15 20:13:05 +00:00
{
2022-09-01 14:35:47 +00:00
pActor - > clear_xvel ( ) ;
2021-12-23 16:04:54 +00:00
pActor - > spr . yvel = 0 ;
2022-08-31 22:50:13 +00:00
pActor - > clear_zvel ( ) ;
2021-10-15 20:13:05 +00:00
}
2021-12-23 16:04:54 +00:00
else if ( pActor - > spr . statnum ! = kStatAnubisDrum )
2021-10-15 20:13:05 +00:00
{
2021-12-23 16:04:54 +00:00
pActor - > spr . xvel > > = 1 ;
pActor - > spr . yvel > > = 1 ;
2022-08-31 22:42:06 +00:00
pActor - > mul_int_zvel ( 0.5 ) ;
2021-10-15 20:13:05 +00:00
}
2019-08-31 07:47:15 +00:00
2021-10-21 15:24:52 +00:00
if ( pActor - > nHealth > 0 ) {
2021-10-15 20:13:05 +00:00
return ;
}
2019-08-31 07:47:15 +00:00
2021-12-23 16:04:54 +00:00
if ( pActor - > spr . statnum = = kStatExplodeTarget )
2021-10-15 20:13:05 +00:00
{
2021-10-21 15:24:52 +00:00
pActor - > nHealth = - 1 ;
2021-11-21 20:09:27 +00:00
int ax = pActor - > nIndex2 ;
2021-10-15 20:13:05 +00:00
2021-10-21 15:24:52 +00:00
if ( ax < 0 | | ObjectList [ ax ] - > nHealth < = 0 ) {
2021-10-15 20:13:05 +00:00
return ;
2019-08-31 07:47:15 +00:00
}
2021-10-15 20:13:05 +00:00
2021-10-21 15:24:52 +00:00
ObjectList [ ax ] - > nHealth = - 1 ;
2021-10-15 20:13:05 +00:00
}
2021-12-23 16:04:54 +00:00
else if ( pActor - > spr . statnum = = kStatDestructibleSprite )
2021-10-15 20:13:05 +00:00
{
2021-10-21 15:24:52 +00:00
pActor - > nHealth = 0 ;
2021-10-15 20:13:05 +00:00
2021-10-21 15:24:52 +00:00
ExplodeScreen ( pActor ) ;
2021-10-15 20:13:05 +00:00
}
else
{
2021-10-21 15:24:52 +00:00
pActor - > nHealth = - ( RandomSize ( 4 ) + 1 ) ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
}
2021-10-21 16:28:06 +00:00
void BuildDrip ( DExhumedActor * pActor )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto nDrips = sDrip . Reserve ( 1 ) ;
2021-10-21 16:28:06 +00:00
sDrip [ nDrips ] . pActor = pActor ;
sDrip [ nDrips ] . nCount = RandomSize ( 8 ) + 90 ;
2021-12-23 16:04:54 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2019-08-26 03:59:14 +00:00
}
void DoDrips ( )
{
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sDrip . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-10-21 16:28:06 +00:00
sDrip [ i ] . nCount - - ;
if ( sDrip [ i ] . nCount < = 0 )
2019-08-31 07:47:15 +00:00
{
2021-12-07 17:53:02 +00:00
DExhumedActor * pActor = sDrip [ i ] . pActor ;
if ( ! pActor ) continue ;
2021-11-21 20:09:27 +00:00
int nSeqOffset = SeqOffsets [ kSeqDrips ] ;
2019-08-31 07:47:15 +00:00
2021-12-30 15:51:56 +00:00
if ( ! ( pActor - > sector ( ) - > Flag & kSectLava ) ) {
2019-08-31 07:47:15 +00:00
nSeqOffset + + ;
}
2021-10-21 16:28:06 +00:00
seq_MoveSequence ( pActor , nSeqOffset , RandomSize ( 2 ) % SeqSize [ nSeqOffset ] ) ;
2019-08-31 07:47:15 +00:00
2021-10-21 16:28:06 +00:00
sDrip [ i ] . nCount = RandomSize ( 8 ) + 90 ;
2019-08-31 07:47:15 +00:00
}
}
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sBob . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-11-21 20:09:27 +00:00
sBob [ i ] . nPhase + = 4 ;
2019-08-31 07:47:15 +00:00
2021-11-21 20:09:27 +00:00
int edx = bsin ( sBob [ i ] . nPhase < < 3 , - 4 ) ;
2021-11-22 22:40:53 +00:00
auto pSector = sBob [ i ] . pSector ;
2019-08-31 07:47:15 +00:00
if ( sBob [ i ] . field_3 )
{
2022-02-02 23:35:12 +00:00
pSector - > set_int_ceilingz ( edx + sBob [ i ] . z ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2022-08-20 14:44:30 +00:00
double nFloorZ = pSector - > floorz ;
2022-02-02 23:35:12 +00:00
pSector - > set_int_floorz ( edx + sBob [ i ] . z ) ;
2022-08-20 14:44:30 +00:00
MoveSectorSprites ( pSector , pSector - > floorz - nFloorZ ) ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
}
2021-11-22 22:40:53 +00:00
void SnapBobs ( sectortype * pSectorA , sectortype * pSectorB )
2019-08-26 03:59:14 +00:00
{
2021-11-22 22:40:53 +00:00
int select1 = - 1 ;
int select2 = select1 ;
int esi ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sBob . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-11-22 22:40:53 +00:00
auto pSector = sBob [ i ] . pSector ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:40:53 +00:00
if ( pSector ! = pSectorA )
2019-08-31 07:47:15 +00:00
{
2021-11-22 22:40:53 +00:00
if ( pSectorB ! = pSector )
2019-08-31 07:47:15 +00:00
continue ;
2021-11-22 22:40:53 +00:00
esi = select2 ;
select1 = i ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-11-22 22:40:53 +00:00
esi = select1 ;
select2 = i ;
2019-08-31 07:47:15 +00:00
}
if ( esi ! = - 1 ) {
break ;
}
}
2021-11-22 22:40:53 +00:00
if ( select1 < = - 1 ) {
2019-08-31 07:47:15 +00:00
return ;
}
2021-11-22 22:40:53 +00:00
if ( select2 < = - 1 ) {
2019-08-31 07:47:15 +00:00
return ;
}
2021-11-22 22:40:53 +00:00
sBob [ select1 ] . nPhase = sBob [ select2 ] . nPhase ;
2019-08-26 03:59:14 +00:00
}
2021-11-22 22:40:53 +00:00
void AddSectorBob ( sectortype * pSector , int nHitag , int bx )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto nBobs = sBob . Reserve ( 1 ) ;
2019-08-31 07:47:15 +00:00
sBob [ nBobs ] . field_3 = bx ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int z ;
2019-11-20 16:21:32 +00:00
2019-08-31 07:47:15 +00:00
if ( bx = = 0 ) {
2022-02-02 23:46:04 +00:00
z = pSector - > int_floorz ( ) ;
2019-08-31 07:47:15 +00:00
}
else {
2022-02-02 23:46:04 +00:00
z = pSector - > int_ceilingz ( ) ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
sBob [ nBobs ] . z = z ;
2021-11-21 20:09:27 +00:00
sBob [ nBobs ] . nPhase = nHitag < < 4 ;
2020-11-29 23:49:25 +00:00
sBob [ nBobs ] . sBobID = nHitag ;
2019-08-26 03:59:14 +00:00
2021-11-22 22:40:53 +00:00
sBob [ nBobs ] . pSector = pSector ;
StartInterpolation ( pSector , bx = = 0 ? Interp_Sect_Floorz : Interp_Sect_Ceilingz ) ;
2019-08-26 03:59:14 +00:00
2021-11-22 22:40:53 +00:00
pSector - > Flag | = 0x0010 ;
2019-08-26 03:59:14 +00:00
}
int FindTrail ( int nVal )
{
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sTrail . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:48:11 +00:00
if ( sTrail [ i ] . nVal = = nVal )
2019-08-31 07:47:15 +00:00
return i ;
}
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
auto nTrails = sTrail . Reserve ( 1 ) ;
2021-11-21 19:48:11 +00:00
sTrail [ nTrails ] . nVal = nVal ;
sTrail [ nTrails ] . nPoint = - 1 ;
sTrail [ nTrails ] . nPoint2 = - 1 ;
2020-11-29 23:49:25 +00:00
return nTrails ;
2019-08-26 03:59:14 +00:00
}
// ok ?
2021-10-21 16:28:06 +00:00
void ProcessTrailSprite ( DExhumedActor * pActor , int nLotag , int nHitag )
2019-08-26 03:59:14 +00:00
{
2020-11-29 23:49:25 +00:00
auto nPoint = sTrailPoint . Reserve ( 1 ) ;
2019-08-26 03:59:14 +00:00
2022-01-31 22:33:44 +00:00
sTrailPoint [ nPoint ] . x = pActor - > int_pos ( ) . X ;
sTrailPoint [ nPoint ] . y = pActor - > int_pos ( ) . Y ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int nTrail = FindTrail ( nHitag ) ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
int var_14 = nLotag - 900 ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
sTrailPoint [ nPoint ] . nTrailPointVal = var_14 ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
int field0 = sTrail [ nTrail ] . nPoint ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( field0 = = - 1 )
{
2021-11-21 19:48:11 +00:00
sTrail [ nTrail ] . nPoint = nPoint ;
sTrail [ nTrail ] . nPoint2 = nPoint ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
sTrailPoint [ nPoint ] . nTrailPointNext = - 1 ;
sTrailPoint [ nPoint ] . nTrailPointPrev = - 1 ;
2019-08-31 07:47:15 +00:00
}
else
{
int ecx = - 1 ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
while ( field0 ! = - 1 )
{
2020-11-29 23:49:25 +00:00
if ( sTrailPoint [ field0 ] . nTrailPointVal > var_14 )
2019-08-31 07:47:15 +00:00
{
2020-11-29 23:49:25 +00:00
sTrailPoint [ nPoint ] . nTrailPointPrev = sTrailPoint [ field0 ] . nTrailPointPrev ;
sTrailPoint [ field0 ] . nTrailPointPrev = nPoint ;
sTrailPoint [ nPoint ] . nTrailPointNext = field0 ;
2019-08-26 03:59:14 +00:00
2021-11-21 19:48:11 +00:00
if ( field0 = = sTrail [ nTrail ] . nPoint ) {
sTrail [ nTrail ] . nPoint = nPoint ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
break ;
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
ecx = field0 ;
2020-11-29 23:49:25 +00:00
field0 = sTrailPoint [ field0 ] . nTrailPointNext ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( field0 = = - 1 )
{
2020-11-29 23:49:25 +00:00
sTrailPoint [ ecx ] . nTrailPointNext = nPoint ;
sTrailPoint [ nPoint ] . nTrailPointPrev = ecx ;
sTrailPoint [ nPoint ] . nTrailPointNext = - 1 ;
2021-11-21 19:48:11 +00:00
sTrail [ nTrail ] . nPoint2 = nPoint ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
2021-10-21 16:28:06 +00:00
DeleteActor ( pActor ) ;
2019-08-26 03:59:14 +00:00
}
// ok?
2021-11-22 23:03:18 +00:00
void AddMovingSector ( sectortype * pSector , int edx , int ebx , int ecx )
2019-08-26 03:59:14 +00:00
{
2021-11-22 23:03:18 +00:00
CreatePushBlock ( pSector ) ;
2021-11-22 20:03:10 +00:00
setsectinterpolate ( pSector ) ;
2019-08-26 03:59:14 +00:00
2021-11-21 20:09:27 +00:00
int nTrail = FindTrail ( ebx ) ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
auto nMoveSects = sMoveSect . Reserve ( 1 ) ;
2021-10-15 20:13:05 +00:00
MoveSect * pMoveSect = & sMoveSect [ nMoveSects ] ;
2019-08-26 03:59:14 +00:00
2020-11-29 23:49:25 +00:00
pMoveSect - > sMoveDir = 1 ;
2019-08-31 07:47:15 +00:00
pMoveSect - > nTrail = nTrail ;
pMoveSect - > nTrailPoint = - 1 ;
2021-11-22 20:03:10 +00:00
pMoveSect - > pCurSector = nullptr ;
2021-11-21 19:48:11 +00:00
pMoveSect - > nFlags = ecx ;
2019-08-31 07:47:15 +00:00
pMoveSect - > field_10 = ( edx / 1000 ) + 1 ;
2021-11-22 20:03:10 +00:00
pMoveSect - > pSector = pSector ;
2019-08-26 03:59:14 +00:00
2019-08-31 07:47:15 +00:00
if ( ecx & 8 )
{
2021-11-21 19:48:11 +00:00
pMoveSect - > nChannel = runlist_AllocChannel ( ebx % 1000 ) ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-11-21 19:48:11 +00:00
pMoveSect - > nChannel = - 1 ;
2019-08-31 07:47:15 +00:00
}
2019-08-26 03:59:14 +00:00
2021-12-18 12:23:08 +00:00
pSector - > floorstat | = CSTAT_SECTOR_ALIGN ;
2019-08-26 03:59:14 +00:00
}
void DoMovingSects ( )
{
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sMoveSect . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-11-22 20:03:10 +00:00
if ( sMoveSect [ i ] . pSector = = nullptr ) {
2019-08-31 07:47:15 +00:00
continue ;
}
2021-11-21 19:48:11 +00:00
if ( sMoveSect [ i ] . nChannel ! = - 1 & & ! sRunChannels [ sMoveSect [ i ] . nChannel ] . c ) {
2019-08-31 07:47:15 +00:00
continue ;
}
2021-11-22 20:03:10 +00:00
auto pSector = sMoveSect [ i ] . pSector ;
int nBlock = pSector - > extra ;
2019-08-31 07:47:15 +00:00
2021-10-15 20:13:05 +00:00
BlockInfo * pBlockInfo = & sBlockInfo [ nBlock ] ;
2019-08-31 07:47:15 +00:00
if ( sMoveSect [ i ] . nTrailPoint = = - 1 )
{
2021-11-21 19:48:11 +00:00
if ( sMoveSect [ i ] . nFlags & 0x20 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:48:11 +00:00
runlist_ChangeChannel ( sMoveSect [ i ] . nChannel , 0 ) ;
2019-08-31 07:47:15 +00:00
}
2021-11-21 20:09:27 +00:00
int ax ;
2019-08-31 07:47:15 +00:00
2021-11-21 19:48:11 +00:00
if ( sMoveSect [ i ] . nFlags & 0x10 )
2019-08-31 07:47:15 +00:00
{
2020-11-29 23:49:25 +00:00
sMoveSect [ i ] . sMoveDir = - sMoveSect [ i ] . sMoveDir ;
if ( sMoveSect [ i ] . sMoveDir > 0 )
2019-08-31 07:47:15 +00:00
{
2021-11-21 19:48:11 +00:00
ax = sTrail [ sMoveSect [ i ] . nTrail ] . nPoint ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-11-21 19:48:11 +00:00
ax = sTrail [ sMoveSect [ i ] . nTrail ] . nPoint2 ;
2019-08-31 07:47:15 +00:00
}
}
else
{
2021-11-21 19:48:11 +00:00
ax = sTrail [ sMoveSect [ i ] . nTrail ] . nPoint ;
2019-08-31 07:47:15 +00:00
}
sMoveSect [ i ] . nTrailPoint = ax ;
}
2021-11-21 20:09:27 +00:00
int nTrail = sMoveSect [ i ] . nTrailPoint ;
2021-10-15 20:13:05 +00:00
// TrailPoint *pTrail = &sTrailPoint[nTrail];
2019-08-26 03:59:14 +00:00
2021-10-15 20:13:05 +00:00
// loc_23872:
2022-08-20 14:55:48 +00:00
int nAngle = getangle ( sTrailPoint [ nTrail ] . x - pBlockInfo - > x , sTrailPoint [ nTrail ] . y - pBlockInfo - > y ) ;
2019-08-31 07:47:15 +00:00
2020-11-14 08:45:08 +00:00
int nXVel = bcos ( nAngle , 4 ) * sMoveSect [ i ] . field_10 ;
int nYVel = bsin ( nAngle , 4 ) * sMoveSect [ i ] . field_10 ;
2019-08-31 07:47:15 +00:00
int ebx = ( sTrailPoint [ nTrail ] . x - pBlockInfo - > x ) < < 14 ;
int eax = nXVel ;
if ( eax < 0 ) {
eax = - eax ;
}
int edx = eax ;
eax = ebx ;
int ecx = ( sTrailPoint [ nTrail ] . y - pBlockInfo - > y ) < < 14 ;
if ( eax < 0 ) {
eax = - eax ;
}
// loc_238EC:
if ( edx < = eax )
{
eax = nYVel ;
if ( eax < 0 ) {
eax = - eax ;
}
edx = eax ;
eax = ecx ;
if ( eax < 0 ) {
eax = - eax ;
}
if ( edx > eax )
{
// loc_23908:
nYVel = ecx ;
nXVel = ebx ;
2020-11-29 23:49:25 +00:00
if ( sMoveSect [ i ] . sMoveDir > 0 )
2019-08-31 07:47:15 +00:00
{
2020-11-29 23:49:25 +00:00
sMoveSect [ i ] . nTrailPoint = sTrailPoint [ sMoveSect [ i ] . nTrailPoint ] . nTrailPointNext ;
2019-08-31 07:47:15 +00:00
}
else
{
2020-11-29 23:49:25 +00:00
sMoveSect [ i ] . nTrailPoint = sTrailPoint [ sMoveSect [ i ] . nTrailPoint ] . nTrailPointPrev ;
2019-08-31 07:47:15 +00:00
}
}
}
2019-11-20 16:21:32 +00:00
else
2019-08-31 07:47:15 +00:00
{
// repeat of code from loc_23908
nYVel = ecx ;
nXVel = ebx ;
2020-11-29 23:49:25 +00:00
if ( sMoveSect [ i ] . sMoveDir > 0 )
2019-08-31 07:47:15 +00:00
{
2020-11-29 23:49:25 +00:00
sMoveSect [ i ] . nTrailPoint = sTrailPoint [ sMoveSect [ i ] . nTrailPoint ] . nTrailPointNext ;
2019-08-31 07:47:15 +00:00
}
else
{
2020-11-29 23:49:25 +00:00
sMoveSect [ i ] . nTrailPoint = sTrailPoint [ sMoveSect [ i ] . nTrailPoint ] . nTrailPointPrev ;
2019-08-31 07:47:15 +00:00
}
}
// loc_2393A:
2021-11-22 20:03:10 +00:00
if ( sMoveSect [ i ] . pCurSector ! = nullptr )
2019-08-31 07:47:15 +00:00
{
2021-11-22 20:03:10 +00:00
MoveSector ( sMoveSect [ i ] . pCurSector , - 1 , & nXVel , & nYVel ) ;
2019-08-31 07:47:15 +00:00
}
int var_2C = nXVel ;
int var_30 = nYVel ;
2021-11-22 20:03:10 +00:00
MoveSector ( pSector , - 1 , & nXVel , & nYVel ) ;
2019-08-31 07:47:15 +00:00
if ( nXVel ! = var_2C | | nYVel ! = var_30 )
{
2021-11-22 20:03:10 +00:00
MoveSector ( sMoveSect [ i ] . pCurSector , - 1 , & var_2C , & var_30 ) ;
MoveSector ( sMoveSect [ i ] . pCurSector , - 1 , & nXVel , & nYVel ) ;
2019-08-31 07:47:15 +00:00
}
}
2019-08-26 03:59:14 +00:00
}
void PostProcess ( )
{
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sMoveSect . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
int nTrail = sMoveSect [ i ] . nTrail ;
2021-11-21 19:48:11 +00:00
sMoveSect [ i ] . nTrailPoint = sTrail [ nTrail ] . nPoint ;
2019-08-31 07:47:15 +00:00
2021-11-21 19:48:11 +00:00
if ( sMoveSect [ i ] . nFlags & 0x40 ) {
runlist_ChangeChannel ( sMoveSect [ i ] . nChannel , 1 ) ;
2019-08-31 07:47:15 +00:00
}
2021-11-22 20:03:10 +00:00
auto pSector = sMoveSect [ i ] . pSector ;
2019-08-31 07:47:15 +00:00
2021-11-22 20:03:10 +00:00
if ( pSector - > Flag & kSectUnderwater )
2019-08-31 07:47:15 +00:00
{
2021-12-18 12:23:08 +00:00
pSector - > ceilingstat | = CSTAT_SECTOR_ALIGN ;
pSector - > floorstat & = ~ ( CSTAT_SECTOR_EXHUMED_BIT1 | CSTAT_SECTOR_EXHUMED_BIT2 ) ;
2019-08-31 07:47:15 +00:00
2020-11-29 23:49:25 +00:00
for ( unsigned j = 0 ; j < sMoveSect . Size ( ) ; j + + )
2019-08-31 07:47:15 +00:00
{
if ( j ! = i & & sMoveSect [ i ] . nTrail = = sMoveSect [ j ] . nTrail )
{
2021-11-22 20:03:10 +00:00
sMoveSect [ j ] . pCurSector = sMoveSect [ i ] . pSector ;
2019-08-31 07:47:15 +00:00
2021-11-22 22:40:53 +00:00
SnapSectors ( sMoveSect [ j ] . pSector , sMoveSect [ i ] . pSector , 0 ) ;
2021-11-22 20:03:10 +00:00
sMoveSect [ i ] . pSector = nullptr ;
2019-08-31 07:47:15 +00:00
}
}
}
}
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < sBob . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
if ( sBob [ i ] . field_3 = = 0 )
{
2020-11-29 23:49:25 +00:00
int bobID = sBob [ i ] . sBobID ;
2019-08-31 07:47:15 +00:00
2020-11-29 23:49:25 +00:00
for ( unsigned j = 0 ; j < sBob . Size ( ) ; j + + )
2019-08-31 07:47:15 +00:00
{
if ( j ! = i )
{
2020-11-29 23:49:25 +00:00
if ( sBob [ i ] . field_3 ! = 0 & & sBob [ j ] . sBobID = = bobID ) {
2021-11-22 22:40:53 +00:00
SnapSectors ( sBob [ i ] . pSector , sBob [ j ] . pSector , 0 ) ;
2019-08-31 07:47:15 +00:00
}
}
}
}
}
2021-05-02 13:54:19 +00:00
if ( ! ( currentLevel - > gameflags & LEVEL_EX_COUNTDOWN ) )
2019-08-31 07:47:15 +00:00
{
2021-12-21 08:23:39 +00:00
for ( auto & sect : sector )
2019-08-31 07:47:15 +00:00
{
int var_20 = 30000 ;
2021-11-22 19:18:14 +00:00
if ( sect . Speed & & sect . Depth & & ! ( sect . Flag & kSectLava ) )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
sect . pSoundSect = & sect ;
sect . Sound = StaticSound [ kSound43 ] ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-12-21 08:23:39 +00:00
for ( auto & sectj : sector )
2019-08-31 07:47:15 +00:00
{
// loc_23CA6:
2021-11-22 19:18:14 +00:00
if ( & sect ! = & sectj & & sectj . Speed & & ! ( sect . Flag & kSectLava ) )
2019-08-31 07:47:15 +00:00
{
2022-01-27 16:41:10 +00:00
int xVal = abs ( sect . firstWall ( ) - > wall_int_pos ( ) . X - sectj . firstWall ( ) - > wall_int_pos ( ) . X ) ;
int yVal = abs ( sect . firstWall ( ) - > wall_int_pos ( ) . Y - sectj . firstWall ( ) - > wall_int_pos ( ) . Y ) ;
2019-08-31 07:47:15 +00:00
if ( xVal < 15000 & & yVal < 15000 & & ( xVal + yVal < var_20 ) )
{
var_20 = xVal + yVal ;
2021-11-22 19:18:14 +00:00
sect . pSoundSect = & sectj ;
sect . Sound = StaticSound [ kSound43 ] ;
2019-08-31 07:47:15 +00:00
}
}
}
}
}
}
else // nMap == kMap20)
{
2021-12-21 08:23:39 +00:00
for ( auto & sect : sector )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
sect . pSoundSect = & sect ;
sect . Sound = StaticSound [ kSound62 ] ;
2019-08-31 07:47:15 +00:00
2021-11-22 19:18:14 +00:00
for ( auto & wal : wallsofsector ( & sect ) )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
if ( wal . picnum = = kTile3603 )
2019-08-31 07:47:15 +00:00
{
2021-11-22 19:18:14 +00:00
wal . pal = 1 ;
auto pActor = insertActor ( & sect , 407 ) ;
2021-12-21 22:18:23 +00:00
pActor - > spr . cstat = CSTAT_SPRITE_INVISIBLE ;
2019-08-31 07:47:15 +00:00
}
}
}
2021-12-04 11:31:54 +00:00
ExhumedSpriteIterator it ;
2021-10-21 16:42:07 +00:00
while ( auto act = it . Next ( ) )
2019-08-31 07:47:15 +00:00
{
2021-12-23 17:54:40 +00:00
if ( act - > spr . statnum < kMaxStatus & & act - > spr . picnum = = kTile3603 )
2019-08-31 07:47:15 +00:00
{
2021-10-21 16:42:07 +00:00
ChangeActorStat ( act , 407 ) ;
2021-12-23 17:54:40 +00:00
act - > spr . pal = 1 ;
2019-08-31 07:47:15 +00:00
}
}
}
2020-11-29 23:49:25 +00:00
for ( unsigned i = 0 ; i < ObjectList . Size ( ) ; i + + )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
auto pObjectActor = ObjectList [ i ] ;
2019-08-31 07:47:15 +00:00
2021-12-21 22:18:23 +00:00
if ( pObjectActor - > spr . statnum = = kStatExplodeTarget )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
if ( ! pObjectActor - > nIndex2 ) {
pObjectActor - > nIndex2 = - 1 ;
2019-08-31 07:47:15 +00:00
}
else
{
2021-10-21 15:24:52 +00:00
int edi = pObjectActor - > nIndex2 ;
pObjectActor - > nIndex2 = - 1 ;
2019-08-31 07:47:15 +00:00
2020-11-29 23:49:25 +00:00
for ( unsigned j = 0 ; j < ObjectList . Size ( ) ; j + + )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
2021-12-21 22:18:23 +00:00
if ( i ! = j & & ObjectList [ j ] - > spr . statnum = = kStatExplodeTarget & & edi = = ObjectList [ j ] - > nIndex2 )
2019-08-31 07:47:15 +00:00
{
2021-10-21 15:24:52 +00:00
pObjectActor - > nIndex2 = j ;
ObjectList [ j ] - > nIndex2 = i ;
2019-08-31 07:47:15 +00:00
}
}
}
}
}
2019-08-26 03:59:14 +00:00
}
2019-12-26 21:00:04 +00:00
2019-11-22 23:11:37 +00:00
END_PS_NS
2020-09-07 21:03:18 +00:00