mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-24 01:11:21 +00:00
18091 lines
536 KiB
C++
18091 lines
536 KiB
C++
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
Shadow Warrior is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
#include "ns.h"
|
|
#include "build.h"
|
|
|
|
#include "names2.h"
|
|
#include "panel.h"
|
|
#include "misc.h"
|
|
#include "interpso.h"
|
|
#include "tags.h"
|
|
#include "break.h"
|
|
#include "network.h"
|
|
#include "pal.h"
|
|
|
|
#include "ai.h"
|
|
#include "weapon.h"
|
|
#include "sector.h"
|
|
#include "sprite.h"
|
|
#include "misc.h"
|
|
#include "player.h"
|
|
#include "gstrings.h"
|
|
|
|
BEGIN_SW_NS
|
|
|
|
struct MISSILE_PLACEMENT
|
|
{
|
|
int dist_over, dist_out;
|
|
short ang;
|
|
};
|
|
|
|
|
|
void SpawnZombie2(DSWActor*);
|
|
Collision move_ground_missile(DSWActor* actor, const DVector2& change, double ceildist, double flordist, uint32_t cliptype, int numtics);
|
|
void DoPlayerBeginDie(PLAYER*);
|
|
void VehicleSetSmoke(SECTOR_OBJECT* sop, ANIMATOR* animator);
|
|
|
|
void ScaleSpriteVector(DSWActor* actor, int scalex, int scaley, int scalez);
|
|
void ScaleSpriteVector(DSWActor* actor, int scale);
|
|
|
|
ANIMATOR DoBettyBeginDeath;
|
|
ANIMATOR DoSkullBeginDeath;
|
|
ANIMATOR DoRipperGrow;
|
|
|
|
//
|
|
// Damage Amounts defined in damage.h
|
|
//
|
|
#define DAMAGE_TABLE
|
|
DAMAGE_DATA DamageData[] =
|
|
{
|
|
#include "damage.h"
|
|
};
|
|
#undef DAMAGE_TABLE
|
|
|
|
FOOT_TYPE FootMode=WATER_FOOT;
|
|
bool left_foot = false;
|
|
int FinishTimer = 0;
|
|
|
|
// This is how many bullet shells have been spawned since the beginning of the game.
|
|
int ShellCount = 0;
|
|
|
|
//int Zombies = 0;
|
|
int StarQueueHead=0;
|
|
TObjPtr<DSWActor*> StarQueue[MAX_STAR_QUEUE];
|
|
int HoleQueueHead=0;
|
|
TObjPtr<DSWActor*> HoleQueue[MAX_HOLE_QUEUE];
|
|
int WallBloodQueueHead=0;
|
|
TObjPtr<DSWActor*> WallBloodQueue[MAX_WALLBLOOD_QUEUE];
|
|
int FloorBloodQueueHead=0;
|
|
TObjPtr<DSWActor*> FloorBloodQueue[MAX_FLOORBLOOD_QUEUE];
|
|
int GenericQueueHead=0;
|
|
TObjPtr<DSWActor*> GenericQueue[MAX_GENERIC_QUEUE];
|
|
int LoWangsQueueHead=0;
|
|
TObjPtr<DSWActor*> LoWangsQueue[MAX_LOWANGS_QUEUE];
|
|
void SpawnBreakStaticFlames(DSWActor* actor);
|
|
|
|
bool GlobalSkipZrange = false;
|
|
|
|
int WeaponIsAmmo = BIT(WPN_STAR) | BIT(WPN_SWORD) | BIT(WPN_MINE) | BIT(WPN_FIST);
|
|
|
|
short target_ang;
|
|
|
|
ANIMATOR DoStar;
|
|
ANIMATOR DoCrossBolt;
|
|
ANIMATOR DoSuicide, DoUziSmoke;
|
|
ANIMATOR DoShrapJumpFall;
|
|
ANIMATOR DoFastShrapJumpFall;
|
|
|
|
int SpawnSmokePuff(DSWActor* actor);
|
|
|
|
bool WarpToSurface(DVector3& pos, sectortype** sect);
|
|
bool WarpToUnderwater(DVector3& pos, sectortype** sect);
|
|
|
|
bool TestDontStickSector(sectortype* hit_sect);
|
|
ANIMATOR SpawnShrapX;
|
|
bool WeaponMoveHit(DSWActor* actor);
|
|
void SpawnMidSplash(DSWActor* actor);
|
|
|
|
int SopDamage(SECTOR_OBJECT* sop,short amt);
|
|
int SopCheckKill(SECTOR_OBJECT* sop);
|
|
int QueueStar(DSWActor*);
|
|
int DoBlurExtend(DSWActor* actor, int16_t interval, int16_t blur_num);
|
|
int SpawnDemonFist(DSWActor*);
|
|
void SpawnTankShellExp(DSWActor*);
|
|
void SpawnMicroExp(DSWActor*);
|
|
void SpawnExpZadjust(DSWActor* actor, DSWActor* expActor, double upper_zsize, double lower_zsize);
|
|
int BulletHitSprite(DSWActor* actor, DSWActor* hitActor, const DVector3& pos, short ID);
|
|
int SpawnSplashXY(int hit_x,int hit_y,int hit_z,sectortype*);
|
|
DSWActor* SpawnBoatSparks(PLAYER* pp, sectortype* hit_sect, walltype* hit_wall, int hit_x, int hit_y, int hit_z, short hit_ang);
|
|
|
|
short StatDamageList[STAT_DAMAGE_LIST_SIZE] =
|
|
{
|
|
STAT_SO_SP_CHILD,
|
|
STAT_ENEMY,
|
|
STAT_ENEMY_SKIP4,
|
|
STAT_PLAYER0,
|
|
STAT_PLAYER1,
|
|
STAT_PLAYER2,
|
|
STAT_PLAYER3,
|
|
STAT_PLAYER4,
|
|
STAT_PLAYER5,
|
|
STAT_PLAYER6,
|
|
STAT_PLAYER7,
|
|
STAT_PLAYER_UNDER0,
|
|
STAT_PLAYER_UNDER1,
|
|
STAT_PLAYER_UNDER2,
|
|
STAT_PLAYER_UNDER3,
|
|
STAT_PLAYER_UNDER4,
|
|
STAT_PLAYER_UNDER5,
|
|
STAT_PLAYER_UNDER6,
|
|
STAT_PLAYER_UNDER7,
|
|
// MINE MUST BE LAST
|
|
STAT_MINE_STUCK
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// SPECIAL STATES
|
|
//
|
|
//////////////////////
|
|
|
|
// state for sprites that are not restored
|
|
STATE s_NotRestored[] =
|
|
{
|
|
{2323, 100, NullAnimator, &s_NotRestored[0]}
|
|
};
|
|
|
|
STATE s_Suicide[] =
|
|
{
|
|
{1, 100, DoSuicide, &s_Suicide[0]}
|
|
};
|
|
|
|
STATE s_DeadLoWang[] =
|
|
{
|
|
{1160, 100, NullAnimator, &s_DeadLoWang[0]},
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// BREAKABLE STATES
|
|
//
|
|
//////////////////////
|
|
|
|
ANIMATOR DoDefaultStat;
|
|
|
|
#define BREAK_LIGHT_RATE 18
|
|
STATE s_BreakLight[] =
|
|
{
|
|
{BREAK_LIGHT_ANIM + 0, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[1]},
|
|
{BREAK_LIGHT_ANIM + 1, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[2]},
|
|
{BREAK_LIGHT_ANIM + 2, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[3]},
|
|
{BREAK_LIGHT_ANIM + 3, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[4]},
|
|
{BREAK_LIGHT_ANIM + 4, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[5]},
|
|
{BREAK_LIGHT_ANIM + 5, BREAK_LIGHT_RATE, NullAnimator, &s_BreakLight[5]}
|
|
};
|
|
|
|
#define BREAK_BARREL_RATE 18
|
|
STATE s_BreakBarrel[] =
|
|
{
|
|
{BREAK_BARREL + 4, BREAK_BARREL_RATE, NullAnimator, &s_BreakBarrel[1]},
|
|
{BREAK_BARREL + 5, BREAK_BARREL_RATE, NullAnimator, &s_BreakBarrel[2]},
|
|
{BREAK_BARREL + 6, BREAK_BARREL_RATE, NullAnimator, &s_BreakBarrel[3]},
|
|
{BREAK_BARREL + 7, BREAK_BARREL_RATE, NullAnimator, &s_BreakBarrel[4]},
|
|
{BREAK_BARREL + 8, BREAK_BARREL_RATE, NullAnimator, &s_BreakBarrel[5]},
|
|
{BREAK_BARREL + 9, BREAK_BARREL_RATE, DoDefaultStat, &s_BreakBarrel[5]},
|
|
};
|
|
|
|
#define BREAK_PEDISTAL_RATE 28
|
|
STATE s_BreakPedistal[] =
|
|
{
|
|
{BREAK_PEDISTAL + 1, BREAK_PEDISTAL_RATE, NullAnimator, &s_BreakPedistal[1]},
|
|
{BREAK_PEDISTAL + 2, BREAK_PEDISTAL_RATE, NullAnimator, &s_BreakPedistal[2]},
|
|
{BREAK_PEDISTAL + 3, BREAK_PEDISTAL_RATE, NullAnimator, &s_BreakPedistal[3]},
|
|
{BREAK_PEDISTAL + 4, BREAK_PEDISTAL_RATE, NullAnimator, &s_BreakPedistal[3]},
|
|
};
|
|
|
|
#define BREAK_BOTTLE1_RATE 18
|
|
STATE s_BreakBottle1[] =
|
|
{
|
|
{BREAK_BOTTLE1 + 1, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[1]},
|
|
{BREAK_BOTTLE1 + 2, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[2]},
|
|
{BREAK_BOTTLE1 + 3, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[3]},
|
|
{BREAK_BOTTLE1 + 4, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[4]},
|
|
{BREAK_BOTTLE1 + 5, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[5]},
|
|
{BREAK_BOTTLE1 + 6, BREAK_BOTTLE1_RATE, NullAnimator, &s_BreakBottle1[5]},
|
|
};
|
|
|
|
#define BREAK_BOTTLE2_RATE 18
|
|
STATE s_BreakBottle2[] =
|
|
{
|
|
{BREAK_BOTTLE2 + 1, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[1]},
|
|
{BREAK_BOTTLE2 + 2, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[2]},
|
|
{BREAK_BOTTLE2 + 3, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[3]},
|
|
{BREAK_BOTTLE2 + 4, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[4]},
|
|
{BREAK_BOTTLE2 + 5, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[5]},
|
|
{BREAK_BOTTLE2 + 6, BREAK_BOTTLE2_RATE, NullAnimator, &s_BreakBottle2[5]},
|
|
};
|
|
|
|
#define PUFF_RATE 8
|
|
ANIMATOR DoPuff;
|
|
STATE s_Puff[] =
|
|
{
|
|
{PUFF + 0, PUFF_RATE, DoPuff, &s_Puff[1]},
|
|
{PUFF + 1, PUFF_RATE, DoPuff, &s_Puff[2]},
|
|
{PUFF + 2, PUFF_RATE, DoPuff, &s_Puff[3]},
|
|
{PUFF + 3, PUFF_RATE, DoPuff, &s_Puff[4]},
|
|
{PUFF + 4, PUFF_RATE, DoPuff, &s_Puff[5]},
|
|
{PUFF + 5, PUFF_RATE, DoPuff, &s_Puff[6]},
|
|
{PUFF + 5, 100, DoSuicide, &s_Puff[0]}
|
|
};
|
|
|
|
#define RAIL_PUFF_R0 3969
|
|
#define RAIL_PUFF_R1 3985
|
|
#define RAIL_PUFF_R2 4001
|
|
#define RAIL_PUFF_RATE 6
|
|
|
|
ANIMATOR DoRailPuff;
|
|
|
|
STATE s_RailPuff[3][17] =
|
|
{
|
|
{
|
|
{RAIL_PUFF_R0 + 0, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][1]},
|
|
{RAIL_PUFF_R0 + 1, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][2]},
|
|
{RAIL_PUFF_R0 + 2, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][3]},
|
|
{RAIL_PUFF_R0 + 3, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][4]},
|
|
{RAIL_PUFF_R0 + 4, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][5]},
|
|
{RAIL_PUFF_R0 + 5, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][6]},
|
|
{RAIL_PUFF_R0 + 6, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][7]},
|
|
{RAIL_PUFF_R0 + 7, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][8]},
|
|
{RAIL_PUFF_R0 + 8, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][9]},
|
|
{RAIL_PUFF_R0 + 9, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][10]},
|
|
{RAIL_PUFF_R0 + 10, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][11]},
|
|
{RAIL_PUFF_R0 + 11, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][12]},
|
|
{RAIL_PUFF_R0 + 12, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][13]},
|
|
{RAIL_PUFF_R0 + 13, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][14]},
|
|
{RAIL_PUFF_R0 + 14, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][15]},
|
|
{RAIL_PUFF_R0 + 15, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[0][16]},
|
|
{RAIL_PUFF_R0 + 15, 100, DoSuicide, &s_RailPuff[0][0]},
|
|
},
|
|
{
|
|
{RAIL_PUFF_R1 + 0, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][1]},
|
|
{RAIL_PUFF_R1 + 1, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][2]},
|
|
{RAIL_PUFF_R1 + 2, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][3]},
|
|
{RAIL_PUFF_R1 + 3, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][4]},
|
|
{RAIL_PUFF_R1 + 4, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][5]},
|
|
{RAIL_PUFF_R1 + 5, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][6]},
|
|
{RAIL_PUFF_R1 + 6, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][7]},
|
|
{RAIL_PUFF_R1 + 7, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][8]},
|
|
{RAIL_PUFF_R1 + 8, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][9]},
|
|
{RAIL_PUFF_R1 + 9, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][10]},
|
|
{RAIL_PUFF_R1 + 10, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][11]},
|
|
{RAIL_PUFF_R1 + 11, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][12]},
|
|
{RAIL_PUFF_R1 + 12, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][13]},
|
|
{RAIL_PUFF_R1 + 13, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][14]},
|
|
{RAIL_PUFF_R1 + 14, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][15]},
|
|
{RAIL_PUFF_R1 + 15, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[1][16]},
|
|
{RAIL_PUFF_R1 + 15, 100, DoSuicide, &s_RailPuff[1][0]},
|
|
},
|
|
{
|
|
{RAIL_PUFF_R2 + 0, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][1]},
|
|
{RAIL_PUFF_R2 + 1, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][2]},
|
|
{RAIL_PUFF_R2 + 2, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][3]},
|
|
{RAIL_PUFF_R2 + 3, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][4]},
|
|
{RAIL_PUFF_R2 + 4, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][5]},
|
|
{RAIL_PUFF_R2 + 5, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][6]},
|
|
{RAIL_PUFF_R2 + 6, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][7]},
|
|
{RAIL_PUFF_R2 + 7, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][8]},
|
|
{RAIL_PUFF_R2 + 8, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][9]},
|
|
{RAIL_PUFF_R2 + 9, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][10]},
|
|
{RAIL_PUFF_R2 + 10, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][11]},
|
|
{RAIL_PUFF_R2 + 11, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][12]},
|
|
{RAIL_PUFF_R2 + 12, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][13]},
|
|
{RAIL_PUFF_R2 + 13, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][14]},
|
|
{RAIL_PUFF_R2 + 14, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][15]},
|
|
{RAIL_PUFF_R2 + 15, RAIL_PUFF_RATE, DoRailPuff, &s_RailPuff[2][16]},
|
|
{RAIL_PUFF_R2 + 15, 100, DoSuicide, &s_RailPuff[2][0]},
|
|
}
|
|
};
|
|
|
|
|
|
STATE* sg_RailPuff[] =
|
|
{
|
|
&s_RailPuff[0][0],
|
|
&s_RailPuff[1][0],
|
|
&s_RailPuff[2][0],
|
|
&s_RailPuff[1][0],
|
|
&s_RailPuff[0][0]
|
|
};
|
|
|
|
#define LASER_PUFF 3201
|
|
#define LASER_PUFF_RATE 8
|
|
STATE s_LaserPuff[] =
|
|
{
|
|
{LASER_PUFF + 0, LASER_PUFF_RATE, nullptr, &s_LaserPuff[1]},
|
|
//{LASER_PUFF + 1, LASER_PUFF_RATE, nullptr, &s_LaserPuff[2]},
|
|
//{LASER_PUFF + 2, LASER_PUFF_RATE, nullptr, &s_LaserPuff[3]},
|
|
{LASER_PUFF + 0, 100, DoSuicide, &s_LaserPuff[0]}
|
|
};
|
|
|
|
#define TRACER 3201
|
|
#define TRACER_RATE 6
|
|
ANIMATOR DoTracer;
|
|
STATE s_Tracer[] =
|
|
{
|
|
{TRACER + 0, TRACER_RATE, DoTracer, &s_Tracer[1]},
|
|
{TRACER + 1, TRACER_RATE, DoTracer, &s_Tracer[2]},
|
|
{TRACER + 2, TRACER_RATE, DoTracer, &s_Tracer[3]},
|
|
{TRACER + 3, TRACER_RATE, DoTracer, &s_Tracer[4]},
|
|
{TRACER + 4, TRACER_RATE, DoTracer, &s_Tracer[5]},
|
|
{TRACER + 5, TRACER_RATE, DoTracer, &s_Tracer[0]}
|
|
};
|
|
|
|
#define EMP 2058
|
|
#define EMP_RATE 6
|
|
ANIMATOR DoEMP;
|
|
STATE s_EMP[] =
|
|
{
|
|
{EMP + 0, EMP_RATE, DoEMP, &s_EMP[1]},
|
|
{EMP + 1, EMP_RATE, DoEMP, &s_EMP[2]},
|
|
{EMP + 2, EMP_RATE, DoEMP, &s_EMP[0]}
|
|
};
|
|
|
|
ANIMATOR DoEMPBurst;
|
|
STATE s_EMPBurst[] =
|
|
{
|
|
{EMP + 0, EMP_RATE, DoEMPBurst, &s_EMPBurst[1]},
|
|
{EMP + 1, EMP_RATE, DoEMPBurst, &s_EMPBurst[2]},
|
|
{EMP + 2, EMP_RATE, DoEMPBurst, &s_EMPBurst[0]}
|
|
};
|
|
|
|
STATE s_EMPShrap[] =
|
|
{
|
|
{EMP + 0, EMP_RATE, DoFastShrapJumpFall, &s_EMPShrap[1]},
|
|
{EMP + 1, EMP_RATE, DoFastShrapJumpFall, &s_EMPShrap[2]},
|
|
{EMP + 2, EMP_RATE, DoFastShrapJumpFall, &s_EMPShrap[0]},
|
|
};
|
|
|
|
|
|
#define TANK_SHELL 3201
|
|
#define TANK_SHELL_RATE 6
|
|
ANIMATOR DoTankShell;
|
|
STATE s_TankShell[] =
|
|
{
|
|
{TRACER + 0, 200, DoTankShell, &s_TankShell[0]}
|
|
};
|
|
|
|
ANIMATOR DoVehicleSmoke;
|
|
#define VEHICLE_SMOKE_RATE 18
|
|
STATE s_VehicleSmoke[] =
|
|
{
|
|
{PUFF + 0, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[1]},
|
|
{PUFF + 1, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[2]},
|
|
{PUFF + 2, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[3]},
|
|
{PUFF + 3, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[4]},
|
|
{PUFF + 4, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[5]},
|
|
{PUFF + 5, VEHICLE_SMOKE_RATE, DoVehicleSmoke, &s_VehicleSmoke[6]},
|
|
{PUFF + 5, 100, DoSuicide, &s_VehicleSmoke[6]}
|
|
};
|
|
|
|
ANIMATOR DoWaterSmoke, SpawnWaterSmoke;
|
|
#define WATER_SMOKE_RATE 18
|
|
STATE s_WaterSmoke[] =
|
|
{
|
|
{PUFF + 0, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[1]},
|
|
{PUFF + 1, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[2]},
|
|
{PUFF + 2, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[3]},
|
|
{PUFF + 3, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[4]},
|
|
{PUFF + 4, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[5]},
|
|
{PUFF + 5, WATER_SMOKE_RATE, DoWaterSmoke, &s_WaterSmoke[6]},
|
|
{PUFF + 5, 100, DoSuicide, &s_WaterSmoke[6]}
|
|
};
|
|
|
|
#define UZI_SPARK_REPEAT 24
|
|
#define UZI_SMOKE_REPEAT 24 // Was 32
|
|
#define UZI_SMOKE_RATE 16 // Was 9
|
|
STATE s_UziSmoke[] =
|
|
{
|
|
{UZI_SMOKE + 0, UZI_SMOKE_RATE, DoUziSmoke, &s_UziSmoke[1]},
|
|
{UZI_SMOKE + 1, UZI_SMOKE_RATE, DoUziSmoke, &s_UziSmoke[2]},
|
|
{UZI_SMOKE + 2, UZI_SMOKE_RATE, DoUziSmoke, &s_UziSmoke[3]},
|
|
{UZI_SMOKE + 3, UZI_SMOKE_RATE, DoUziSmoke, &s_UziSmoke[4]},
|
|
{UZI_SMOKE + 3, 100, DoSuicide, &s_UziSmoke[0]},
|
|
};
|
|
|
|
#define SHOTGUN_SMOKE_RATE 16
|
|
#define SHOTGUN_SMOKE_REPEAT 18 // Was 32
|
|
#define SHOTGUN_SMOKE UZI_SMOKE+1
|
|
ANIMATOR DoShotgunSmoke;
|
|
STATE s_ShotgunSmoke[] =
|
|
{
|
|
{UZI_SMOKE + 0, SHOTGUN_SMOKE_RATE, DoShotgunSmoke, &s_ShotgunSmoke[1]},
|
|
{UZI_SMOKE + 1, SHOTGUN_SMOKE_RATE, DoShotgunSmoke, &s_ShotgunSmoke[2]},
|
|
{UZI_SMOKE + 2, SHOTGUN_SMOKE_RATE, DoShotgunSmoke, &s_ShotgunSmoke[3]},
|
|
{UZI_SMOKE + 3, SHOTGUN_SMOKE_RATE, DoShotgunSmoke, &s_ShotgunSmoke[4]},
|
|
{UZI_SMOKE + 3, 100, DoSuicide, &s_ShotgunSmoke[0]},
|
|
};
|
|
|
|
#define UZI_BULLET_RATE 100
|
|
#define UZI_BULLET 717 // actually a bubble
|
|
ANIMATOR DoUziBullet;
|
|
|
|
STATE s_UziBullet[] =
|
|
{
|
|
{UZI_BULLET + 0, UZI_BULLET_RATE, DoUziBullet, &s_UziBullet[0]},
|
|
};
|
|
|
|
#define UZI_SPARK_RATE 8
|
|
|
|
STATE s_UziSpark[] =
|
|
{
|
|
{UZI_SPARK + 0, UZI_SPARK_RATE, NullAnimator, &s_UziSpark[1]},
|
|
{UZI_SPARK + 1, UZI_SPARK_RATE, NullAnimator, &s_UziSpark[2]},
|
|
{UZI_SPARK + 2, UZI_SPARK_RATE, NullAnimator, &s_UziSpark[3]},
|
|
{UZI_SPARK + 3, UZI_SPARK_RATE, NullAnimator, &s_UziSpark[4]},
|
|
{UZI_SPARK + 4, UZI_SPARK_RATE, NullAnimator, &s_UziSpark[5]},
|
|
{UZI_SPARK + 4, 100, DoSuicide, &s_UziSpark[0]},
|
|
};
|
|
|
|
STATE s_UziPowerSpark[] =
|
|
{
|
|
{UZI_SPARK + 0, UZI_SPARK_RATE, DoUziSmoke, &s_UziSpark[1]},
|
|
{UZI_SPARK + 1, UZI_SPARK_RATE, DoUziSmoke, &s_UziSpark[2]},
|
|
{UZI_SPARK + 2, UZI_SPARK_RATE, DoUziSmoke, &s_UziSpark[3]},
|
|
{UZI_SPARK + 3, UZI_SPARK_RATE, DoUziSmoke, &s_UziSpark[4]},
|
|
{UZI_SPARK + 4, UZI_SPARK_RATE, DoUziSmoke, &s_UziSpark[5]},
|
|
{UZI_SPARK + 4, 100, DoSuicide, &s_UziSpark[0]},
|
|
};
|
|
|
|
#define BUBBLE 716
|
|
#define BUBBLE_RATE 100
|
|
ANIMATOR DoBubble;
|
|
|
|
STATE s_Bubble[] =
|
|
{
|
|
{BUBBLE + 0, BUBBLE_RATE, DoBubble, &s_Bubble[0]}
|
|
};
|
|
|
|
|
|
//#define SPLASH 2190
|
|
#define SPLASH 772
|
|
#define SPLASH_RATE 10
|
|
|
|
STATE s_Splash[] =
|
|
{
|
|
{SPLASH + 0, SPLASH_RATE, NullAnimator, &s_Splash[1]},
|
|
{SPLASH + 1, SPLASH_RATE, NullAnimator, &s_Splash[2]},
|
|
{SPLASH + 2, SPLASH_RATE, NullAnimator, &s_Splash[3]},
|
|
{SPLASH + 3, SPLASH_RATE, NullAnimator, &s_Splash[4]},
|
|
{SPLASH + 4, SPLASH_RATE, NullAnimator, &s_Splash[5]},
|
|
{SPLASH + 4, 100, DoSuicide, &s_Splash[0]}
|
|
};
|
|
|
|
#define CROSSBOLT 2230
|
|
#define CROSSBOLT_RATE 100
|
|
STATE s_CrossBolt[5][1] =
|
|
{
|
|
{
|
|
{CROSSBOLT + 0, CROSSBOLT_RATE, DoCrossBolt, &s_CrossBolt[0][0]},
|
|
},
|
|
{
|
|
{CROSSBOLT + 2, CROSSBOLT_RATE, DoCrossBolt, &s_CrossBolt[1][0]},
|
|
},
|
|
{
|
|
{CROSSBOLT + 3, CROSSBOLT_RATE, DoCrossBolt, &s_CrossBolt[2][0]},
|
|
},
|
|
{
|
|
{CROSSBOLT + 4, CROSSBOLT_RATE, DoCrossBolt, &s_CrossBolt[3][0]},
|
|
},
|
|
{
|
|
{CROSSBOLT + 1, CROSSBOLT_RATE, DoCrossBolt, &s_CrossBolt[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_CrossBolt[] =
|
|
{
|
|
&s_CrossBolt[0][0],
|
|
&s_CrossBolt[1][0],
|
|
&s_CrossBolt[2][0],
|
|
&s_CrossBolt[3][0],
|
|
&s_CrossBolt[4][0]
|
|
};
|
|
|
|
#undef STAR
|
|
#define STAR 2102
|
|
#define STAR_RATE 6
|
|
STATE s_Star[] =
|
|
{
|
|
{STAR + 0, STAR_RATE, DoStar, &s_Star[1]},
|
|
{STAR + 1, STAR_RATE, DoStar, &s_Star[2]},
|
|
{STAR + 2, STAR_RATE, DoStar, &s_Star[3]},
|
|
{STAR + 3, STAR_RATE, DoStar, &s_Star[0]}
|
|
};
|
|
|
|
STATE s_StarStuck[] =
|
|
{
|
|
{STAR + 0, STAR_RATE, NullAnimator, &s_StarStuck[0]},
|
|
};
|
|
|
|
#define STAR_DOWN 2066
|
|
STATE s_StarDown[] =
|
|
{
|
|
{STAR_DOWN + 0, STAR_RATE, DoStar, &s_StarDown[1]},
|
|
{STAR_DOWN + 1, STAR_RATE, DoStar, &s_StarDown[2]},
|
|
{STAR_DOWN + 2, STAR_RATE, DoStar, &s_StarDown[3]},
|
|
{STAR_DOWN + 3, STAR_RATE, DoStar, &s_StarDown[0]}
|
|
};
|
|
|
|
STATE s_StarDownStuck[] =
|
|
{
|
|
{STAR + 0, STAR_RATE, NullAnimator, &s_StarDownStuck[0]},
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// LAVA BOSS
|
|
//
|
|
//////////////////////
|
|
|
|
#define LAVA_BOULDER_RATE 6
|
|
ANIMATOR DoLavaBoulder,DoShrapDamage,DoVulcanBoulder;
|
|
|
|
STATE s_LavaBoulder[] =
|
|
{
|
|
{LAVA_BOULDER + 1, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[1]},
|
|
{LAVA_BOULDER + 2, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[2]},
|
|
{LAVA_BOULDER + 3, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[3]},
|
|
{LAVA_BOULDER + 4, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[4]},
|
|
{LAVA_BOULDER + 5, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[5]},
|
|
{LAVA_BOULDER + 6, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[6]},
|
|
{LAVA_BOULDER + 7, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[7]},
|
|
{LAVA_BOULDER + 8, LAVA_BOULDER_RATE, DoLavaBoulder, &s_LavaBoulder[0]},
|
|
};
|
|
|
|
#define LAVA_SHARD (LAVA_BOULDER+1)
|
|
|
|
STATE s_LavaShard[] =
|
|
{
|
|
{LAVA_BOULDER + 1, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[1]},
|
|
{LAVA_BOULDER + 2, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[2]},
|
|
{LAVA_BOULDER + 3, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[3]},
|
|
{LAVA_BOULDER + 4, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[4]},
|
|
{LAVA_BOULDER + 5, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[5]},
|
|
{LAVA_BOULDER + 6, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[6]},
|
|
{LAVA_BOULDER + 7, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[7]},
|
|
{LAVA_BOULDER + 8, LAVA_BOULDER_RATE, DoShrapDamage, &s_LavaShard[0]},
|
|
};
|
|
|
|
STATE s_VulcanBoulder[] =
|
|
{
|
|
{LAVA_BOULDER + 1, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[1]},
|
|
{LAVA_BOULDER + 2, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[2]},
|
|
{LAVA_BOULDER + 3, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[3]},
|
|
{LAVA_BOULDER + 4, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[4]},
|
|
{LAVA_BOULDER + 5, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[5]},
|
|
{LAVA_BOULDER + 6, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[6]},
|
|
{LAVA_BOULDER + 7, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[7]},
|
|
{LAVA_BOULDER + 8, LAVA_BOULDER_RATE, DoVulcanBoulder, &s_VulcanBoulder[0]},
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// GRENADE
|
|
//
|
|
//////////////////////
|
|
|
|
#if 0
|
|
ANIMATOR DoGrenade;
|
|
#undef GRENADE
|
|
#define GRENADE 5000
|
|
#define GRENADE_RATE 8
|
|
|
|
STATE s_Grenade[] =
|
|
{
|
|
{FIREBALL + 0, GRENADE_RATE, DoGrenade, &s_Grenade[1]},
|
|
{FIREBALL + 1, GRENADE_RATE, DoGrenade, &s_Grenade[2]},
|
|
{FIREBALL + 2, GRENADE_RATE, DoGrenade, &s_Grenade[3]},
|
|
{FIREBALL + 3, GRENADE_RATE, DoGrenade, &s_Grenade[0]}
|
|
};
|
|
#else
|
|
#define GRENADE_FRAMES 1
|
|
#define GRENADE_R0 2110
|
|
#define GRENADE_R1 GRENADE_R0 + (GRENADE_FRAMES * 1)
|
|
#define GRENADE_R2 GRENADE_R0 + (GRENADE_FRAMES * 2)
|
|
#define GRENADE_R3 GRENADE_R0 + (GRENADE_FRAMES * 3)
|
|
#define GRENADE_R4 GRENADE_R0 + (GRENADE_FRAMES * 4)
|
|
|
|
#undef GRENADE
|
|
#define GRENADE GRENADE_R0
|
|
#define GRENADE_RATE 8
|
|
ANIMATOR DoGrenade;
|
|
|
|
STATE s_Grenade[5][1] =
|
|
{
|
|
{
|
|
{GRENADE_R0 + 0, GRENADE_RATE, DoGrenade, &s_Grenade[0][0]},
|
|
},
|
|
{
|
|
{GRENADE_R1 + 0, GRENADE_RATE, DoGrenade, &s_Grenade[1][0]},
|
|
},
|
|
{
|
|
{GRENADE_R2 + 0, GRENADE_RATE, DoGrenade, &s_Grenade[2][0]},
|
|
},
|
|
{
|
|
{GRENADE_R3 + 0, GRENADE_RATE, DoGrenade, &s_Grenade[3][0]},
|
|
},
|
|
{
|
|
{GRENADE_R4 + 0, GRENADE_RATE, DoGrenade, &s_Grenade[4][0]},
|
|
}
|
|
};
|
|
|
|
|
|
STATE* sg_Grenade[] =
|
|
{
|
|
&s_Grenade[0][0],
|
|
&s_Grenade[1][0],
|
|
&s_Grenade[2][0],
|
|
&s_Grenade[3][0],
|
|
&s_Grenade[4][0]
|
|
};
|
|
#endif
|
|
|
|
//////////////////////
|
|
//
|
|
// MINE
|
|
//
|
|
//////////////////////
|
|
|
|
ANIMATOR DoMine,DoMineStuck;
|
|
#undef MINE
|
|
#define MINE 2223
|
|
#define MINE_SHRAP 5011
|
|
#define MINE_RATE 16
|
|
|
|
STATE s_MineStuck[] =
|
|
{
|
|
{MINE + 0, MINE_RATE, DoMineStuck, &s_MineStuck[0]},
|
|
};
|
|
|
|
STATE s_Mine[] =
|
|
{
|
|
{MINE + 0, MINE_RATE, DoMine, &s_Mine[1]},
|
|
{MINE + 1, MINE_RATE, DoMine, &s_Mine[0]},
|
|
};
|
|
|
|
ANIMATOR DoMineSpark;
|
|
STATE s_MineSpark[] =
|
|
{
|
|
{UZI_SPARK + 0, UZI_SPARK_RATE, DoMineSpark, &s_MineSpark[1]},
|
|
{UZI_SPARK + 1, UZI_SPARK_RATE, DoMineSpark, &s_MineSpark[2]},
|
|
{UZI_SPARK + 2, UZI_SPARK_RATE, DoMineSpark, &s_MineSpark[3]},
|
|
{UZI_SPARK + 3, UZI_SPARK_RATE, DoMineSpark, &s_MineSpark[4]},
|
|
{UZI_SPARK + 4, UZI_SPARK_RATE, DoMineSpark, &s_MineSpark[5]},
|
|
{UZI_SPARK + 4, 100, DoSuicide, &s_MineSpark[0]},
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// METEOR
|
|
//
|
|
//////////////////////
|
|
|
|
#define METEOR_R0 2098
|
|
#define METEOR_R1 2090
|
|
#define METEOR_R2 2094
|
|
#define METEOR_R3 2090
|
|
#define METEOR_R4 2098
|
|
|
|
|
|
#define METEOR STAR
|
|
#define METEOR_RATE 8
|
|
ANIMATOR DoMeteor;
|
|
|
|
STATE s_Meteor[5][4] =
|
|
{
|
|
{
|
|
{METEOR_R0 + 0, METEOR_RATE, DoMeteor, &s_Meteor[0][1]},
|
|
{METEOR_R0 + 1, METEOR_RATE, DoMeteor, &s_Meteor[0][2]},
|
|
{METEOR_R0 + 2, METEOR_RATE, DoMeteor, &s_Meteor[0][3]},
|
|
{METEOR_R0 + 3, METEOR_RATE, DoMeteor, &s_Meteor[0][0]},
|
|
},
|
|
{
|
|
{METEOR_R1 + 0, METEOR_RATE, DoMeteor, &s_Meteor[1][1]},
|
|
{METEOR_R1 + 1, METEOR_RATE, DoMeteor, &s_Meteor[1][2]},
|
|
{METEOR_R1 + 2, METEOR_RATE, DoMeteor, &s_Meteor[1][3]},
|
|
{METEOR_R1 + 3, METEOR_RATE, DoMeteor, &s_Meteor[1][0]},
|
|
},
|
|
{
|
|
{METEOR_R2 + 0, METEOR_RATE, DoMeteor, &s_Meteor[2][1]},
|
|
{METEOR_R2 + 1, METEOR_RATE, DoMeteor, &s_Meteor[2][2]},
|
|
{METEOR_R2 + 2, METEOR_RATE, DoMeteor, &s_Meteor[2][3]},
|
|
{METEOR_R2 + 3, METEOR_RATE, DoMeteor, &s_Meteor[2][0]},
|
|
},
|
|
{
|
|
{METEOR_R3 + 0, METEOR_RATE, DoMeteor, &s_Meteor[3][1]},
|
|
{METEOR_R3 + 1, METEOR_RATE, DoMeteor, &s_Meteor[3][2]},
|
|
{METEOR_R3 + 2, METEOR_RATE, DoMeteor, &s_Meteor[3][3]},
|
|
{METEOR_R3 + 3, METEOR_RATE, DoMeteor, &s_Meteor[3][0]},
|
|
},
|
|
{
|
|
{METEOR_R4 + 0, METEOR_RATE, DoMeteor, &s_Meteor[4][1]},
|
|
{METEOR_R4 + 1, METEOR_RATE, DoMeteor, &s_Meteor[4][2]},
|
|
{METEOR_R4 + 2, METEOR_RATE, DoMeteor, &s_Meteor[4][3]},
|
|
{METEOR_R4 + 3, METEOR_RATE, DoMeteor, &s_Meteor[4][0]},
|
|
}
|
|
};
|
|
|
|
|
|
STATE* sg_Meteor[] =
|
|
{
|
|
&s_Meteor[0][0],
|
|
&s_Meteor[1][0],
|
|
&s_Meteor[2][0],
|
|
&s_Meteor[3][0],
|
|
&s_Meteor[4][0]
|
|
};
|
|
|
|
#define METEOR_EXP 2115
|
|
#define METEOR_EXP_RATE 7
|
|
|
|
STATE s_MeteorExp[] =
|
|
{
|
|
{METEOR_EXP + 0, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[1]},
|
|
{METEOR_EXP + 1, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[2]},
|
|
{METEOR_EXP + 2, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[3]},
|
|
{METEOR_EXP + 3, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[4]},
|
|
{METEOR_EXP + 4, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[5]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, NullAnimator, &s_MeteorExp[6]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, DoSuicide, &s_MeteorExp[6]}
|
|
};
|
|
|
|
#define MIRV_METEOR METEOR_R0
|
|
ANIMATOR DoMirvMissile;
|
|
STATE s_MirvMeteor[5][4] =
|
|
{
|
|
{
|
|
{METEOR_R0 + 0, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[0][1]},
|
|
{METEOR_R0 + 1, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[0][2]},
|
|
{METEOR_R0 + 2, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[0][3]},
|
|
{METEOR_R0 + 3, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[0][0]},
|
|
},
|
|
{
|
|
{METEOR_R1 + 0, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[1][1]},
|
|
{METEOR_R1 + 1, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[1][2]},
|
|
{METEOR_R1 + 2, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[1][3]},
|
|
{METEOR_R1 + 3, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[1][0]},
|
|
},
|
|
{
|
|
{METEOR_R2 + 0, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[2][1]},
|
|
{METEOR_R2 + 1, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[2][2]},
|
|
{METEOR_R2 + 2, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[2][3]},
|
|
{METEOR_R2 + 3, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[2][0]},
|
|
},
|
|
{
|
|
{METEOR_R3 + 0, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[3][1]},
|
|
{METEOR_R3 + 1, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[3][2]},
|
|
{METEOR_R3 + 2, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[3][3]},
|
|
{METEOR_R3 + 3, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[3][0]},
|
|
},
|
|
{
|
|
{METEOR_R4 + 0, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[4][1]},
|
|
{METEOR_R4 + 1, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[4][2]},
|
|
{METEOR_R4 + 2, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[4][3]},
|
|
{METEOR_R4 + 3, METEOR_RATE, DoMirvMissile, &s_MirvMeteor[4][0]},
|
|
}
|
|
};
|
|
|
|
|
|
STATE* sg_MirvMeteor[] =
|
|
{
|
|
&s_MirvMeteor[0][0],
|
|
&s_MirvMeteor[1][0],
|
|
&s_MirvMeteor[2][0],
|
|
&s_MirvMeteor[3][0],
|
|
&s_MirvMeteor[4][0]
|
|
};
|
|
|
|
STATE s_MirvMeteorExp[] =
|
|
{
|
|
{METEOR_EXP + 0, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[1]},
|
|
{METEOR_EXP + 1, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[2]},
|
|
{METEOR_EXP + 2, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[3]},
|
|
{METEOR_EXP + 3, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[4]},
|
|
{METEOR_EXP + 4, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[5]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, NullAnimator, &s_MirvMeteorExp[6]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, DoSuicide, &s_MirvMeteorExp[6]}
|
|
};
|
|
|
|
#define SERP_METEOR METEOR_R0+1
|
|
ANIMATOR DoSerpMeteor;
|
|
STATE s_SerpMeteor[5][4] =
|
|
{
|
|
{
|
|
{2031 + 0, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[0][1]},
|
|
{2031 + 1, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[0][2]},
|
|
{2031 + 2, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[0][3]},
|
|
{2031 + 3, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[0][0]},
|
|
},
|
|
{
|
|
{2031 + 0, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[1][1]},
|
|
{2031 + 1, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[1][2]},
|
|
{2031 + 2, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[1][3]},
|
|
{2031 + 3, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[1][0]},
|
|
},
|
|
{
|
|
{2031 + 0, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[2][1]},
|
|
{2031 + 1, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[2][2]},
|
|
{2031 + 2, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[2][3]},
|
|
{2031 + 3, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[2][0]},
|
|
},
|
|
{
|
|
{2031 + 0, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[3][1]},
|
|
{2031 + 1, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[3][2]},
|
|
{2031 + 2, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[3][3]},
|
|
{2031 + 3, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[3][0]},
|
|
},
|
|
{
|
|
{2031 + 0, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[4][1]},
|
|
{2031 + 1, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[4][2]},
|
|
{2031 + 2, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[4][3]},
|
|
{2031 + 3, METEOR_RATE, DoSerpMeteor, &s_SerpMeteor[4][0]},
|
|
}
|
|
};
|
|
|
|
|
|
STATE* sg_SerpMeteor[] =
|
|
{
|
|
&s_SerpMeteor[0][0],
|
|
&s_SerpMeteor[1][0],
|
|
&s_SerpMeteor[2][0],
|
|
&s_SerpMeteor[3][0],
|
|
&s_SerpMeteor[4][0]
|
|
};
|
|
|
|
|
|
STATE s_SerpMeteorExp[] =
|
|
{
|
|
{METEOR_EXP + 0, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[1]},
|
|
{METEOR_EXP + 1, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[2]},
|
|
{METEOR_EXP + 2, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[3]},
|
|
{METEOR_EXP + 3, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[4]},
|
|
{METEOR_EXP + 4, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[5]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, NullAnimator, &s_SerpMeteorExp[6]},
|
|
{METEOR_EXP + 5, METEOR_EXP_RATE, DoSuicide, &s_SerpMeteorExp[6]}
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// SPEAR
|
|
//
|
|
//////////////////////
|
|
|
|
#define SPEAR_RATE 8
|
|
ANIMATOR DoSpear;
|
|
|
|
STATE s_Spear[5][1] =
|
|
{
|
|
{
|
|
{SPEAR_R0 + 0, SPEAR_RATE, DoSpear, s_Spear[0]},
|
|
},
|
|
{
|
|
{SPEAR_R1 + 0, SPEAR_RATE, DoSpear, s_Spear[1]},
|
|
},
|
|
{
|
|
{SPEAR_R2 + 0, SPEAR_RATE, DoSpear, s_Spear[2]},
|
|
},
|
|
{
|
|
{SPEAR_R3 + 0, SPEAR_RATE, DoSpear, s_Spear[3]},
|
|
},
|
|
{
|
|
{SPEAR_R4 + 0, SPEAR_RATE, DoSpear, s_Spear[4]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Spear[] =
|
|
{
|
|
s_Spear[0],
|
|
s_Spear[1],
|
|
s_Spear[2],
|
|
s_Spear[3],
|
|
s_Spear[4]
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// ROCKET
|
|
//
|
|
//////////////////////
|
|
|
|
#define ROCKET_FRAMES 1
|
|
#define ROCKET_R0 2206
|
|
#define ROCKET_R1 ROCKET_R0 + (ROCKET_FRAMES * 2)
|
|
#define ROCKET_R2 ROCKET_R0 + (ROCKET_FRAMES * 3)
|
|
#define ROCKET_R3 ROCKET_R0 + (ROCKET_FRAMES * 4)
|
|
#define ROCKET_R4 ROCKET_R0 + (ROCKET_FRAMES * 1)
|
|
|
|
ANIMATOR DoRocket;
|
|
#define ROCKET_RATE 8
|
|
|
|
STATE s_Rocket[5][1] =
|
|
{
|
|
{
|
|
{ROCKET_R0 + 0, ROCKET_RATE, DoRocket, &s_Rocket[0][0]},
|
|
},
|
|
{
|
|
{ROCKET_R1 + 0, ROCKET_RATE, DoRocket, &s_Rocket[1][0]},
|
|
},
|
|
{
|
|
{ROCKET_R2 + 0, ROCKET_RATE, DoRocket, &s_Rocket[2][0]},
|
|
},
|
|
{
|
|
{ROCKET_R3 + 0, ROCKET_RATE, DoRocket, &s_Rocket[3][0]},
|
|
},
|
|
{
|
|
{ROCKET_R4 + 0, ROCKET_RATE, DoRocket, &s_Rocket[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Rocket[] =
|
|
{
|
|
&s_Rocket[0][0],
|
|
&s_Rocket[1][0],
|
|
&s_Rocket[2][0],
|
|
&s_Rocket[3][0],
|
|
&s_Rocket[4][0]
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// BUNNY ROCKET
|
|
//
|
|
//////////////////////
|
|
|
|
#define BUNNYROCKET_FRAMES 5
|
|
#define BUNNYROCKET_R0 4550
|
|
#define BUNNYROCKET_R1 BUNNYROCKET_R0 + (BUNNYROCKET_FRAMES * 1)
|
|
#define BUNNYROCKET_R2 BUNNYROCKET_R0 + (BUNNYROCKET_FRAMES * 2)
|
|
#define BUNNYROCKET_R3 BUNNYROCKET_R0 + (BUNNYROCKET_FRAMES * 3)
|
|
#define BUNNYROCKET_R4 BUNNYROCKET_R0 + (BUNNYROCKET_FRAMES * 4)
|
|
|
|
ANIMATOR DoRocket;
|
|
#define BUNNYROCKET_RATE 8
|
|
|
|
STATE s_BunnyRocket[5][1] =
|
|
{
|
|
{
|
|
{BUNNYROCKET_R0 + 2, BUNNYROCKET_RATE, DoRocket, &s_BunnyRocket[0][0]},
|
|
},
|
|
{
|
|
{BUNNYROCKET_R1 + 2, BUNNYROCKET_RATE, DoRocket, &s_BunnyRocket[1][0]},
|
|
},
|
|
{
|
|
{BUNNYROCKET_R2 + 2, BUNNYROCKET_RATE, DoRocket, &s_BunnyRocket[2][0]},
|
|
},
|
|
{
|
|
{BUNNYROCKET_R3 + 2, BUNNYROCKET_RATE, DoRocket, &s_BunnyRocket[3][0]},
|
|
},
|
|
{
|
|
{BUNNYROCKET_R4 + 2, BUNNYROCKET_RATE, DoRocket, &s_BunnyRocket[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_BunnyRocket[] =
|
|
{
|
|
&s_BunnyRocket[0][0],
|
|
&s_BunnyRocket[1][0],
|
|
&s_BunnyRocket[2][0],
|
|
&s_BunnyRocket[3][0],
|
|
&s_BunnyRocket[4][0]
|
|
};
|
|
|
|
ANIMATOR DoRail;
|
|
#define RAIL_RATE 8
|
|
|
|
STATE s_Rail[5][1] =
|
|
{
|
|
{
|
|
{ROCKET_R0 + 0, RAIL_RATE, DoRail, &s_Rail[0][0]},
|
|
},
|
|
{
|
|
{ROCKET_R1 + 0, RAIL_RATE, DoRail, &s_Rail[1][0]},
|
|
},
|
|
{
|
|
{ROCKET_R2 + 0, RAIL_RATE, DoRail, &s_Rail[2][0]},
|
|
},
|
|
{
|
|
{ROCKET_R3 + 0, RAIL_RATE, DoRail, &s_Rail[3][0]},
|
|
},
|
|
{
|
|
{ROCKET_R4 + 0, RAIL_RATE, DoRail, &s_Rail[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Rail[] =
|
|
{
|
|
&s_Rail[0][0],
|
|
&s_Rail[1][0],
|
|
&s_Rail[2][0],
|
|
&s_Rail[3][0],
|
|
&s_Rail[4][0]
|
|
};
|
|
|
|
ANIMATOR DoLaser;
|
|
#define LASER_RATE 8
|
|
|
|
STATE s_Laser[] =
|
|
{
|
|
{ROCKET_R0 + 0, LASER_RATE, DoLaser, &s_Laser[0]}
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// MICRO
|
|
//
|
|
//////////////////////
|
|
|
|
#define MICRO_FRAMES 1
|
|
#define MICRO_R0 2206
|
|
#define MICRO_R1 MICRO_R0 + (MICRO_FRAMES * 2)
|
|
#define MICRO_R2 MICRO_R0 + (MICRO_FRAMES * 3)
|
|
#define MICRO_R3 MICRO_R0 + (MICRO_FRAMES * 4)
|
|
#define MICRO_R4 MICRO_R0 + (MICRO_FRAMES * 1)
|
|
|
|
ANIMATOR DoMicro;
|
|
#define MICRO_RATE 8
|
|
|
|
#if 0
|
|
#define PUFF 1748
|
|
#define PUFF_RATE 8
|
|
ANIMATOR DoMicroPuffSuicide;
|
|
STATE s_MicroPuff[] =
|
|
{
|
|
{PUFF + 0, PUFF_RATE, DoPuff, &s_MicroPuff[1]},
|
|
{PUFF + 1, PUFF_RATE, DoPuff, &s_MicroPuff[2]},
|
|
{PUFF + 2, PUFF_RATE, DoPuff, &s_MicroPuff[3]},
|
|
{PUFF + 3, PUFF_RATE, DoPuff, &s_MicroPuff[4]},
|
|
{PUFF + 4, PUFF_RATE, DoPuff, &s_MicroPuff[5]},
|
|
{PUFF + 5, PUFF_RATE, DoPuff, &s_MicroPuff[6]},
|
|
{PUFF + 5, 100, DoMicroPuffSuicide, &s_MicroPuff[6]}
|
|
};
|
|
#endif
|
|
|
|
STATE s_Micro[5][1] =
|
|
{
|
|
{
|
|
{MICRO_R0 + 0, MICRO_RATE, DoMicro, &s_Micro[0][0]},
|
|
},
|
|
{
|
|
{MICRO_R1 + 0, MICRO_RATE, DoMicro, &s_Micro[1][0]},
|
|
},
|
|
{
|
|
{MICRO_R2 + 0, MICRO_RATE, DoMicro, &s_Micro[2][0]},
|
|
},
|
|
{
|
|
{MICRO_R3 + 0, MICRO_RATE, DoMicro, &s_Micro[3][0]},
|
|
},
|
|
{
|
|
{MICRO_R4 + 0, MICRO_RATE, DoMicro, &s_Micro[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Micro[] =
|
|
{
|
|
&s_Micro[0][0],
|
|
&s_Micro[1][0],
|
|
&s_Micro[2][0],
|
|
&s_Micro[3][0],
|
|
&s_Micro[4][0]
|
|
};
|
|
|
|
ANIMATOR DoMicroMini;
|
|
STATE s_MicroMini[5][1] =
|
|
{
|
|
{
|
|
{MICRO_R0 + 0, MICRO_RATE, DoMicroMini, &s_MicroMini[0][0]},
|
|
},
|
|
{
|
|
{MICRO_R1 + 0, MICRO_RATE, DoMicroMini, &s_MicroMini[1][0]},
|
|
},
|
|
{
|
|
{MICRO_R2 + 0, MICRO_RATE, DoMicroMini, &s_MicroMini[2][0]},
|
|
},
|
|
{
|
|
{MICRO_R3 + 0, MICRO_RATE, DoMicroMini, &s_MicroMini[3][0]},
|
|
},
|
|
{
|
|
{MICRO_R4 + 0, MICRO_RATE, DoMicroMini, &s_MicroMini[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_MicroMini[] =
|
|
{
|
|
&s_MicroMini[0][0],
|
|
&s_MicroMini[1][0],
|
|
&s_MicroMini[2][0],
|
|
&s_MicroMini[3][0],
|
|
&s_MicroMini[4][0]
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// BOLT THINMAN
|
|
//
|
|
//////////////////////
|
|
|
|
#define BOLT_THINMAN_RATE 8
|
|
ANIMATOR DoBoltThinMan;
|
|
|
|
STATE s_BoltThinMan[5][1] =
|
|
{
|
|
{
|
|
{BOLT_THINMAN_R0 + 0, BOLT_THINMAN_RATE, DoBoltThinMan, &s_BoltThinMan[0][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R1 + 0, BOLT_THINMAN_RATE, DoBoltThinMan, &s_BoltThinMan[1][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R2 + 0, BOLT_THINMAN_RATE, DoBoltThinMan, &s_BoltThinMan[2][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R3 + 0, BOLT_THINMAN_RATE, DoBoltThinMan, &s_BoltThinMan[3][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R4 + 0, BOLT_THINMAN_RATE, DoBoltThinMan, &s_BoltThinMan[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_BoltThinMan[] =
|
|
{
|
|
&s_BoltThinMan[0][0],
|
|
&s_BoltThinMan[1][0],
|
|
&s_BoltThinMan[2][0],
|
|
&s_BoltThinMan[3][0],
|
|
&s_BoltThinMan[4][0]
|
|
};
|
|
|
|
#define BOLT_SEEKER_RATE 8
|
|
ANIMATOR DoBoltSeeker;
|
|
|
|
STATE s_BoltSeeker[5][1] =
|
|
{
|
|
{
|
|
{BOLT_THINMAN_R0 + 0, BOLT_SEEKER_RATE, DoBoltSeeker, &s_BoltSeeker[0][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R1 + 0, BOLT_SEEKER_RATE, DoBoltSeeker, &s_BoltSeeker[1][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R2 + 0, BOLT_SEEKER_RATE, DoBoltSeeker, &s_BoltSeeker[2][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R3 + 0, BOLT_SEEKER_RATE, DoBoltSeeker, &s_BoltSeeker[3][0]},
|
|
},
|
|
{
|
|
{BOLT_THINMAN_R4 + 0, BOLT_SEEKER_RATE, DoBoltSeeker, &s_BoltSeeker[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_BoltSeeker[] =
|
|
{
|
|
&s_BoltSeeker[0][0],
|
|
&s_BoltSeeker[1][0],
|
|
&s_BoltSeeker[2][0],
|
|
&s_BoltSeeker[3][0],
|
|
&s_BoltSeeker[4][0]
|
|
};
|
|
|
|
|
|
#define BOLT_FATMAN STAR
|
|
#define BOLT_FATMAN_RATE 8
|
|
ANIMATOR DoBoltFatMan;
|
|
|
|
STATE s_BoltFatMan[] =
|
|
{
|
|
{BOLT_FATMAN + 0, BOLT_FATMAN_RATE, DoBoltFatMan, &s_BoltFatMan[1]},
|
|
{BOLT_FATMAN + 1, BOLT_FATMAN_RATE, DoBoltFatMan, &s_BoltFatMan[2]},
|
|
{BOLT_FATMAN + 2, BOLT_FATMAN_RATE, DoBoltFatMan, &s_BoltFatMan[3]},
|
|
{BOLT_FATMAN + 3, BOLT_FATMAN_RATE, DoBoltFatMan, &s_BoltFatMan[0]}
|
|
};
|
|
|
|
|
|
#define BOLT_SHRAPNEL STAR
|
|
#define BOLT_SHRAPNEL_RATE 8
|
|
ANIMATOR DoBoltShrapnel;
|
|
|
|
STATE s_BoltShrapnel[] =
|
|
{
|
|
{BOLT_SHRAPNEL + 0, BOLT_SHRAPNEL_RATE, DoBoltShrapnel, &s_BoltShrapnel[1]},
|
|
{BOLT_SHRAPNEL + 1, BOLT_SHRAPNEL_RATE, DoBoltShrapnel, &s_BoltShrapnel[2]},
|
|
{BOLT_SHRAPNEL + 2, BOLT_SHRAPNEL_RATE, DoBoltShrapnel, &s_BoltShrapnel[3]},
|
|
{BOLT_SHRAPNEL + 3, BOLT_SHRAPNEL_RATE, DoBoltShrapnel, &s_BoltShrapnel[0]}
|
|
};
|
|
|
|
#define COOLG_FIRE 2430
|
|
//#define COOLG_FIRE 1465
|
|
#define COOLG_W_FIRE_RATE 8
|
|
ANIMATOR DoCoolgFire;
|
|
|
|
STATE s_CoolgFire[] =
|
|
{
|
|
{2031 + 0, COOLG_W_FIRE_RATE, DoCoolgFire, &s_CoolgFire[1]},
|
|
{2031 + 1, COOLG_W_FIRE_RATE, DoCoolgFire, &s_CoolgFire[2]},
|
|
{2031 + 2, COOLG_W_FIRE_RATE, DoCoolgFire, &s_CoolgFire[3]},
|
|
{2031 + 3, COOLG_W_FIRE_RATE, DoCoolgFire, &s_CoolgFire[0]}
|
|
};
|
|
|
|
#define COOLG_FIRE_DONE 2410
|
|
//#define COOLG_FIRE_DONE 1466
|
|
//#define COOLG_FIRE_DONE_RATE 8
|
|
#define COOLG_FIRE_DONE_RATE 3
|
|
|
|
STATE s_CoolgFireDone[] =
|
|
{
|
|
{COOLG_FIRE_DONE + 0, COOLG_FIRE_DONE_RATE, NullAnimator, &s_CoolgFireDone[1]},
|
|
{COOLG_FIRE_DONE + 1, COOLG_FIRE_DONE_RATE, NullAnimator, &s_CoolgFireDone[2]},
|
|
{COOLG_FIRE_DONE + 2, COOLG_FIRE_DONE_RATE, NullAnimator, &s_CoolgFireDone[3]},
|
|
{COOLG_FIRE_DONE + 3, COOLG_FIRE_DONE_RATE, NullAnimator, &s_CoolgFireDone[4]},
|
|
{COOLG_FIRE_DONE + 4, COOLG_FIRE_DONE_RATE, NullAnimator, &s_CoolgFireDone[5]},
|
|
{COOLG_FIRE_DONE + 4, COOLG_FIRE_DONE_RATE, DoSuicide, &s_CoolgFireDone[5]}
|
|
};
|
|
|
|
ANIMATOR DoCoolgDrip;
|
|
#define COOLG_DRIP 1720
|
|
STATE s_CoolgDrip[] =
|
|
{
|
|
{COOLG_DRIP + 0, 100, DoCoolgDrip, &s_CoolgDrip[0]}
|
|
};
|
|
|
|
#define GORE_FLOOR_SPLASH_RATE 8
|
|
#define GORE_FLOOR_SPLASH 1710
|
|
STATE s_GoreFloorSplash[] =
|
|
{
|
|
{GORE_FLOOR_SPLASH + 0, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[1]},
|
|
{GORE_FLOOR_SPLASH + 1, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[2]},
|
|
{GORE_FLOOR_SPLASH + 2, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[3]},
|
|
{GORE_FLOOR_SPLASH + 3, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[4]},
|
|
{GORE_FLOOR_SPLASH + 4, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[5]},
|
|
{GORE_FLOOR_SPLASH + 5, GORE_FLOOR_SPLASH_RATE, NullAnimator, &s_GoreFloorSplash[6]},
|
|
{GORE_FLOOR_SPLASH + 5, GORE_FLOOR_SPLASH_RATE, DoSuicide, &s_GoreFloorSplash[6]}
|
|
};
|
|
|
|
#define GORE_SPLASH_RATE 8
|
|
#define GORE_SPLASH 2410
|
|
STATE s_GoreSplash[] =
|
|
{
|
|
{GORE_SPLASH + 0, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[1]},
|
|
{GORE_SPLASH + 1, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[2]},
|
|
{GORE_SPLASH + 2, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[3]},
|
|
{GORE_SPLASH + 3, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[4]},
|
|
{GORE_SPLASH + 4, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[5]},
|
|
{GORE_SPLASH + 5, GORE_SPLASH_RATE, NullAnimator, &s_GoreSplash[6]},
|
|
{GORE_SPLASH + 5, GORE_SPLASH_RATE, DoSuicide, &s_GoreSplash[6]}
|
|
};
|
|
|
|
//////////////////////////////////////////////
|
|
//
|
|
// HEART ATTACK & PLASMA
|
|
//
|
|
//////////////////////////////////////////////
|
|
|
|
#define PLASMA 1562 //2058
|
|
#define PLASMA_FOUNTAIN 2058+1
|
|
#define PLASMA_RATE 8
|
|
#define PLASMA_FOUNTAIN_TIME (3*120);
|
|
|
|
ANIMATOR DoPlasma;
|
|
|
|
// regular bolt from heart
|
|
STATE s_Plasma[] =
|
|
{
|
|
{PLASMA + 0, PLASMA_RATE, DoPlasma, &s_Plasma[1]},
|
|
{PLASMA + 1, PLASMA_RATE, DoPlasma, &s_Plasma[2]},
|
|
{PLASMA + 2, PLASMA_RATE, DoPlasma, &s_Plasma[0]}
|
|
};
|
|
|
|
ANIMATOR DoPlasmaFountain;
|
|
|
|
// follows actor spewing blood
|
|
#define PLASMA_Drip 1562 //2420
|
|
STATE s_PlasmaFountain[] =
|
|
{
|
|
{PLASMA_Drip + 0, PLASMA_RATE, DoPlasmaFountain, &s_PlasmaFountain[1]},
|
|
{PLASMA_Drip + 1, PLASMA_RATE, DoPlasmaFountain, &s_PlasmaFountain[2]},
|
|
{PLASMA_Drip + 2, PLASMA_RATE, DoPlasmaFountain, &s_PlasmaFountain[3]},
|
|
{PLASMA_Drip + 3, PLASMA_RATE, DoPlasmaFountain, &s_PlasmaFountain[0]},
|
|
};
|
|
|
|
#define PLASMA_Drip_RATE 12
|
|
STATE s_PlasmaDrip[] =
|
|
{
|
|
{PLASMA_Drip + 0, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[1]},
|
|
{PLASMA_Drip + 1, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[2]},
|
|
{PLASMA_Drip + 2, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[3]},
|
|
{PLASMA_Drip + 3, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[4]},
|
|
{PLASMA_Drip + 4, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[5]},
|
|
{PLASMA_Drip + 5, PLASMA_Drip_RATE, DoShrapJumpFall, &s_PlasmaDrip[6]},
|
|
{PLASMA_Drip + 7, PLASMA_Drip_RATE, DoSuicide, &s_PlasmaDrip[6]}
|
|
};
|
|
|
|
#define PLASMA_DONE 2061
|
|
#define PLASMA_DONE_RATE 15
|
|
|
|
ANIMATOR DoPlasmaDone;
|
|
|
|
STATE s_PlasmaDone[] =
|
|
{
|
|
{PLASMA + 0, PLASMA_DONE_RATE, DoPlasmaDone, &s_PlasmaDone[1]},
|
|
{PLASMA + 2, PLASMA_DONE_RATE, DoPlasmaDone, &s_PlasmaDone[2]},
|
|
{PLASMA + 1, PLASMA_DONE_RATE, DoPlasmaDone, &s_PlasmaDone[0]}
|
|
};
|
|
|
|
#define TELEPORT_EFFECT 3240
|
|
#define TELEPORT_EFFECT_RATE 6
|
|
ANIMATOR DoTeleportEffect;
|
|
STATE s_TeleportEffect[] =
|
|
{
|
|
{TELEPORT_EFFECT + 0, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[1]},
|
|
{TELEPORT_EFFECT + 1, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[2]},
|
|
{TELEPORT_EFFECT + 2, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[3]},
|
|
{TELEPORT_EFFECT + 3, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[4]},
|
|
{TELEPORT_EFFECT + 4, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[5]},
|
|
{TELEPORT_EFFECT + 5, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[6]},
|
|
{TELEPORT_EFFECT + 6, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[7]},
|
|
{TELEPORT_EFFECT + 7, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[8]},
|
|
{TELEPORT_EFFECT + 8, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[9]},
|
|
{TELEPORT_EFFECT + 9, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[10]},
|
|
{TELEPORT_EFFECT + 10, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[11]},
|
|
{TELEPORT_EFFECT + 11, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[12]},
|
|
{TELEPORT_EFFECT + 12, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[13]},
|
|
{TELEPORT_EFFECT + 13, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[14]},
|
|
{TELEPORT_EFFECT + 14, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[15]},
|
|
{TELEPORT_EFFECT + 15, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[16]},
|
|
{TELEPORT_EFFECT + 16, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[17]},
|
|
{TELEPORT_EFFECT + 17, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect[18]},
|
|
{TELEPORT_EFFECT + 17, TELEPORT_EFFECT_RATE, DoSuicide, &s_TeleportEffect[18]},
|
|
};
|
|
|
|
// Spawn a RIPPER teleport effect
|
|
ANIMATOR DoTeleRipper;
|
|
STATE s_TeleportEffect2[] =
|
|
{
|
|
{TELEPORT_EFFECT + 0, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[1]},
|
|
{TELEPORT_EFFECT + 1, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[2]},
|
|
{TELEPORT_EFFECT + 2, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[3]},
|
|
{TELEPORT_EFFECT + 3, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[4]},
|
|
{TELEPORT_EFFECT + 4, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[5]},
|
|
{TELEPORT_EFFECT + 5, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[6]},
|
|
{TELEPORT_EFFECT + 6, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[7]},
|
|
{TELEPORT_EFFECT + 7, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[8]},
|
|
{TELEPORT_EFFECT + 8, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[9]},
|
|
{TELEPORT_EFFECT + 9, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[10]},
|
|
{TELEPORT_EFFECT + 10, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[11]},
|
|
{TELEPORT_EFFECT + 11, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[12]},
|
|
{TELEPORT_EFFECT + 12, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[13]},
|
|
{TELEPORT_EFFECT + 13, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[14]},
|
|
{TELEPORT_EFFECT + 14, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[15]},
|
|
{TELEPORT_EFFECT + 15, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[16]},
|
|
{TELEPORT_EFFECT + 16, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[17]},
|
|
{TELEPORT_EFFECT + 17, TELEPORT_EFFECT_RATE, NullAnimator, &s_TeleportEffect2[18]},
|
|
{TELEPORT_EFFECT + 17, SF_QUICK_CALL, DoTeleRipper, &s_TeleportEffect2[19]},
|
|
{TELEPORT_EFFECT + 17, TELEPORT_EFFECT_RATE, DoSuicide, &s_TeleportEffect2[19]},
|
|
};
|
|
|
|
|
|
ANIMATOR DoElectro;
|
|
#define ELECTRO_SNAKE 2073
|
|
#define ELECTRO_PLAYER (ELECTRO)
|
|
#define ELECTRO_ENEMY (ELECTRO + 1)
|
|
#define ELECTRO_SHARD (ELECTRO + 2)
|
|
|
|
STATE s_Electro[] =
|
|
{
|
|
{ELECTRO + 0, 12, DoElectro, &s_Electro[1]},
|
|
{ELECTRO + 1, 12, DoElectro, &s_Electro[2]},
|
|
{ELECTRO + 2, 12, DoElectro, &s_Electro[3]},
|
|
{ELECTRO + 3, 12, DoElectro, &s_Electro[0]}
|
|
};
|
|
|
|
STATE s_ElectroShrap[] =
|
|
{
|
|
{ELECTRO + 0, 12, DoShrapDamage, &s_ElectroShrap[1]},
|
|
{ELECTRO + 1, 12, DoShrapDamage, &s_ElectroShrap[2]},
|
|
{ELECTRO + 2, 12, DoShrapDamage, &s_ElectroShrap[3]},
|
|
{ELECTRO + 3, 12, DoShrapDamage, &s_ElectroShrap[0]}
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// EXPS
|
|
//
|
|
//////////////////////
|
|
#define GRENADE_EXP 3121
|
|
#define MINE_EXP GRENADE_EXP+1
|
|
#define GRENADE_EXP_RATE 6
|
|
|
|
#if 0
|
|
STATE s_GrenadeExp[] =
|
|
{
|
|
{GRENADE_EXP + 0, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[1]},
|
|
{GRENADE_EXP + 1, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[2]},
|
|
{GRENADE_EXP + 2, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[3]},
|
|
{GRENADE_EXP + 3, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[4]},
|
|
{GRENADE_EXP + 4, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[5]},
|
|
{GRENADE_EXP + 5, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[6]},
|
|
{GRENADE_EXP + 6, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[7]},
|
|
{GRENADE_EXP + 7, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[8]},
|
|
{GRENADE_EXP + 8, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[9]},
|
|
{GRENADE_EXP + 9, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[10]},
|
|
{GRENADE_EXP + 10, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[11]},
|
|
{GRENADE_EXP + 11, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[12]},
|
|
{GRENADE_EXP + 12, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[13]},
|
|
{GRENADE_EXP + 13, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[14]},
|
|
{GRENADE_EXP + 14, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[15]},
|
|
{GRENADE_EXP + 15, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[16]},
|
|
{GRENADE_EXP + 16, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[17]},
|
|
{GRENADE_EXP + 17, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[18]},
|
|
{GRENADE_EXP + 18, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[19]},
|
|
{GRENADE_EXP + 19, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[20]},
|
|
{GRENADE_EXP + 20, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[21]},
|
|
{GRENADE_EXP + 21, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[22]},
|
|
{GRENADE_EXP + 21, 100, DoSuicide, &s_GrenadeExp[22]}
|
|
};
|
|
#else
|
|
STATE s_GrenadeSmallExp[] =
|
|
{
|
|
{GRENADE_EXP + 0, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[1]},
|
|
{GRENADE_EXP + 1, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[2]},
|
|
{GRENADE_EXP + 2, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[3]},
|
|
{GRENADE_EXP + 3, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[4]},
|
|
{GRENADE_EXP + 4, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[5]},
|
|
{GRENADE_EXP + 5, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[6]},
|
|
{GRENADE_EXP + 6, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[7]},
|
|
{GRENADE_EXP + 7, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[8]},
|
|
{GRENADE_EXP + 8, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[9]},
|
|
{GRENADE_EXP + 9, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[10]},
|
|
{GRENADE_EXP + 10, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[11]},
|
|
{GRENADE_EXP + 11, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[12]},
|
|
{GRENADE_EXP + 12, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[13]},
|
|
{GRENADE_EXP + 13, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[14]},
|
|
{GRENADE_EXP + 14, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[15]},
|
|
{GRENADE_EXP + 15, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[16]},
|
|
{GRENADE_EXP + 16, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[17]},
|
|
{GRENADE_EXP + 17, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[18]},
|
|
{GRENADE_EXP + 18, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[19]},
|
|
{GRENADE_EXP + 19, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[20]},
|
|
{GRENADE_EXP + 20, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[21]},
|
|
{GRENADE_EXP + 21, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeSmallExp[22]},
|
|
{GRENADE_EXP + 21, 100, DoSuicide, &s_GrenadeSmallExp[22]}
|
|
};
|
|
|
|
ANIMATOR SpawnGrenadeSmallExp;
|
|
STATE s_GrenadeExp[] =
|
|
{
|
|
{GRENADE_EXP + 0, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[1]},
|
|
{GRENADE_EXP + 1, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[2]},
|
|
{GRENADE_EXP + 2, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[3]},
|
|
{GRENADE_EXP + 3, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[4]},
|
|
{GRENADE_EXP + 4, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[5]},
|
|
{GRENADE_EXP + 5, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[6]},
|
|
{GRENADE_EXP + 6, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[7]},
|
|
{GRENADE_EXP + 6, SF_QUICK_CALL, SpawnGrenadeSmallExp, &s_GrenadeExp[8]},
|
|
{GRENADE_EXP + 7, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[9]},
|
|
{GRENADE_EXP + 8, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[10]},
|
|
{GRENADE_EXP + 9, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[11]},
|
|
{GRENADE_EXP + 10, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[12]},
|
|
{GRENADE_EXP + 11, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[13]},
|
|
{GRENADE_EXP + 12, SF_QUICK_CALL, SpawnGrenadeSmallExp, &s_GrenadeExp[14]},
|
|
{GRENADE_EXP + 12, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[15]},
|
|
{GRENADE_EXP + 13, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[16]},
|
|
{GRENADE_EXP + 14, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[17]},
|
|
{GRENADE_EXP + 15, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[18]},
|
|
{GRENADE_EXP + 16, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[19]},
|
|
{GRENADE_EXP + 17, SF_QUICK_CALL, SpawnGrenadeSmallExp, &s_GrenadeExp[20]},
|
|
{GRENADE_EXP + 17, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[21]},
|
|
{GRENADE_EXP + 18, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[22]},
|
|
{GRENADE_EXP + 19, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[23]},
|
|
{GRENADE_EXP + 20, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[24]},
|
|
{GRENADE_EXP + 21, GRENADE_EXP_RATE, NullAnimator, &s_GrenadeExp[25]},
|
|
{GRENADE_EXP + 21, 100, DoSuicide, &s_GrenadeExp[25]}
|
|
};
|
|
#endif
|
|
|
|
#define MINE_EXP GRENADE_EXP+1
|
|
ANIMATOR DoMineExp, DoMineExpMine;
|
|
STATE s_MineExp[] =
|
|
{
|
|
{GRENADE_EXP + 0, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[1]},
|
|
{GRENADE_EXP + 1, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[2]},
|
|
{GRENADE_EXP + 2, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[3]},
|
|
{GRENADE_EXP + 3, 0|SF_QUICK_CALL, DoMineExp, &s_MineExp[4]},
|
|
{GRENADE_EXP + 3, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[5]},
|
|
{GRENADE_EXP + 4, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[6]},
|
|
{GRENADE_EXP + 5, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[7]},
|
|
{GRENADE_EXP + 6, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[8]},
|
|
{GRENADE_EXP + 7, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[9]},
|
|
{GRENADE_EXP + 8, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[10]},
|
|
{GRENADE_EXP + 9, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[11]},
|
|
{GRENADE_EXP + 10, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[12]},
|
|
{GRENADE_EXP + 11, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[13]},
|
|
{GRENADE_EXP + 12, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[14]},
|
|
{GRENADE_EXP + 13, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[15]},
|
|
{GRENADE_EXP + 14, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[16]},
|
|
{GRENADE_EXP + 15, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[17]},
|
|
{GRENADE_EXP + 16, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[18]},
|
|
{GRENADE_EXP + 17, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[19]},
|
|
{GRENADE_EXP + 17, 0|SF_QUICK_CALL, DoMineExpMine, &s_MineExp[20]},
|
|
{GRENADE_EXP + 18, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[21]},
|
|
{GRENADE_EXP + 19, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[22]},
|
|
{GRENADE_EXP + 20, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[23]},
|
|
{GRENADE_EXP + 21, GRENADE_EXP_RATE, NullAnimator, &s_MineExp[24]},
|
|
{GRENADE_EXP + 21, 100, DoSuicide, &s_MineExp[24]}
|
|
};
|
|
|
|
#define EXP_RATE_W 7
|
|
#define BOLT_EXP EXP
|
|
#define FIREBALL_EXP EXP+1
|
|
#define BASIC_EXP EXP+2
|
|
#define SECTOR_EXP EXP+3
|
|
#define MICRO_EXP EXP+5
|
|
#define TRACER_EXP EXP+6
|
|
#define TANK_SHELL_EXP EXP+7
|
|
|
|
STATE s_BasicExp[] =
|
|
{
|
|
{EXP + 0, EXP_RATE_W, NullAnimator, &s_BasicExp[1]},
|
|
{EXP + 1, EXP_RATE_W, NullAnimator, &s_BasicExp[2]},
|
|
{EXP + 2, EXP_RATE_W, NullAnimator, &s_BasicExp[3]},
|
|
{EXP + 3, EXP_RATE_W, NullAnimator, &s_BasicExp[4]},
|
|
{EXP + 4, EXP_RATE_W, NullAnimator, &s_BasicExp[5]},
|
|
{EXP + 5, EXP_RATE_W, NullAnimator, &s_BasicExp[6]},
|
|
{EXP + 6, EXP_RATE_W, NullAnimator, &s_BasicExp[7]},
|
|
{EXP + 7, EXP_RATE_W, NullAnimator, &s_BasicExp[8]},
|
|
{EXP + 8, EXP_RATE_W, NullAnimator, &s_BasicExp[9]},
|
|
{EXP + 9, EXP_RATE_W, NullAnimator, &s_BasicExp[10]},
|
|
{EXP + 10, EXP_RATE_W, NullAnimator, &s_BasicExp[11]},
|
|
{EXP + 11, EXP_RATE_W, NullAnimator, &s_BasicExp[12]},
|
|
{EXP + 12, EXP_RATE_W, NullAnimator, &s_BasicExp[13]},
|
|
{EXP + 13, EXP_RATE_W, NullAnimator, &s_BasicExp[14]},
|
|
{EXP + 14, EXP_RATE_W, NullAnimator, &s_BasicExp[15]},
|
|
{EXP + 15, EXP_RATE_W, NullAnimator, &s_BasicExp[16]},
|
|
{EXP + 16, EXP_RATE_W, NullAnimator, &s_BasicExp[17]},
|
|
{EXP + 17, EXP_RATE_W, NullAnimator, &s_BasicExp[18]},
|
|
{EXP + 18, EXP_RATE_W, NullAnimator, &s_BasicExp[19]},
|
|
{EXP + 19, EXP_RATE_W, NullAnimator, &s_BasicExp[20]},
|
|
{EXP + 20, 100, DoSuicide, &s_BasicExp[0]}
|
|
};
|
|
|
|
#define MICRO_EXP_RATE 3
|
|
ANIMATOR DoExpDamageTest;
|
|
|
|
STATE s_MicroExp[] =
|
|
{
|
|
{EXP + 0, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[1]},
|
|
{EXP + 0, SF_QUICK_CALL, DoExpDamageTest, &s_MicroExp[2]},
|
|
{EXP + 1, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[3]},
|
|
{EXP + 2, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[4]},
|
|
{EXP + 3, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[5]},
|
|
{EXP + 4, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[6]},
|
|
{EXP + 5, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[7]},
|
|
{EXP + 6, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[8]},
|
|
{EXP + 7, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[9]},
|
|
{EXP + 8, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[10]},
|
|
{EXP + 9, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[11]},
|
|
{EXP + 10, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[12]},
|
|
{EXP + 11, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[13]},
|
|
{EXP + 12, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[14]},
|
|
{EXP + 13, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[15]},
|
|
{EXP + 14, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[16]},
|
|
{EXP + 15, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[17]},
|
|
{EXP + 16, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[18]},
|
|
{EXP + 17, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[19]},
|
|
{EXP + 18, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[20]},
|
|
{EXP + 19, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[21]},
|
|
{EXP + 20, MICRO_EXP_RATE, NullAnimator, &s_MicroExp[22]},
|
|
{EXP + 20, 100, DoSuicide, &s_MicroExp[22]}
|
|
};
|
|
|
|
|
|
#define BIG_GUN_FLAME_RATE 15
|
|
STATE s_BigGunFlame[] =
|
|
{
|
|
// first 3 frames
|
|
{EXP + 0, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[1]},
|
|
{EXP + 1, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[2]},
|
|
{EXP + 2, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[3]},
|
|
// last 4 frames frames
|
|
{EXP + 17, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[4]},
|
|
{EXP + 18, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[5]},
|
|
{EXP + 19, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[6]},
|
|
{EXP + 20, BIG_GUN_FLAME_RATE, NullAnimator, &s_BigGunFlame[7]},
|
|
{EXP + 20, 100, DoSuicide, &s_BigGunFlame[0]}
|
|
};
|
|
|
|
STATE s_BoltExp[] =
|
|
{
|
|
{EXP + 0, EXP_RATE_W, NullAnimator, &s_BoltExp[1]},
|
|
{EXP + 0, SF_QUICK_CALL, NullAnimator, &s_BoltExp[2]},
|
|
{EXP + 0, SF_QUICK_CALL, SpawnShrapX, &s_BoltExp[3]},
|
|
{EXP + 1, EXP_RATE_W, NullAnimator, &s_BoltExp[4]},
|
|
{EXP + 2, EXP_RATE_W, NullAnimator, &s_BoltExp[5]},
|
|
{EXP + 3, EXP_RATE_W, NullAnimator, &s_BoltExp[6]},
|
|
{EXP + 4, EXP_RATE_W, NullAnimator, &s_BoltExp[7]},
|
|
{EXP + 5, EXP_RATE_W, NullAnimator, &s_BoltExp[8]},
|
|
{EXP + 6, EXP_RATE_W, NullAnimator, &s_BoltExp[9]},
|
|
{EXP + 7, EXP_RATE_W, NullAnimator, &s_BoltExp[10]},
|
|
{EXP + 7, SF_QUICK_CALL, SpawnShrapX, &s_BoltExp[11]},
|
|
{EXP + 8, EXP_RATE_W, NullAnimator, &s_BoltExp[12]},
|
|
{EXP + 9, EXP_RATE_W, NullAnimator, &s_BoltExp[13]},
|
|
{EXP + 10, EXP_RATE_W, NullAnimator, &s_BoltExp[14]},
|
|
{EXP + 11, EXP_RATE_W, NullAnimator, &s_BoltExp[15]},
|
|
{EXP + 12, EXP_RATE_W, NullAnimator, &s_BoltExp[16]},
|
|
{EXP + 13, EXP_RATE_W, NullAnimator, &s_BoltExp[17]},
|
|
{EXP + 14, EXP_RATE_W, NullAnimator, &s_BoltExp[18]},
|
|
{EXP + 15, EXP_RATE_W, NullAnimator, &s_BoltExp[19]},
|
|
{EXP + 16, EXP_RATE_W, NullAnimator, &s_BoltExp[20]},
|
|
{EXP + 17, EXP_RATE_W, NullAnimator, &s_BoltExp[21]},
|
|
{EXP + 18, EXP_RATE_W, NullAnimator, &s_BoltExp[22]},
|
|
{EXP + 19, EXP_RATE_W, NullAnimator, &s_BoltExp[23]},
|
|
{EXP + 20, EXP_RATE_W, NullAnimator, &s_BoltExp[24]},
|
|
{EXP + 20, 100, DoSuicide, &s_BoltExp[0]}
|
|
};
|
|
|
|
STATE s_TankShellExp[] =
|
|
{
|
|
{EXP + 0, EXP_RATE_W, NullAnimator, &s_TankShellExp[1]},
|
|
{EXP + 0, SF_QUICK_CALL, NullAnimator, &s_TankShellExp[2]},
|
|
{EXP + 0, SF_QUICK_CALL, SpawnShrapX, &s_TankShellExp[3]},
|
|
{EXP + 1, EXP_RATE_W, NullAnimator, &s_TankShellExp[4]},
|
|
{EXP + 2, EXP_RATE_W, NullAnimator, &s_TankShellExp[5]},
|
|
{EXP + 3, EXP_RATE_W, NullAnimator, &s_TankShellExp[6]},
|
|
{EXP + 4, EXP_RATE_W, NullAnimator, &s_TankShellExp[7]},
|
|
{EXP + 5, EXP_RATE_W, NullAnimator, &s_TankShellExp[8]},
|
|
{EXP + 6, EXP_RATE_W, NullAnimator, &s_TankShellExp[9]},
|
|
{EXP + 7, EXP_RATE_W, NullAnimator, &s_TankShellExp[10]},
|
|
{EXP + 7, SF_QUICK_CALL, SpawnShrapX, &s_TankShellExp[11]},
|
|
{EXP + 8, EXP_RATE_W, NullAnimator, &s_TankShellExp[12]},
|
|
{EXP + 9, EXP_RATE_W, NullAnimator, &s_TankShellExp[13]},
|
|
{EXP + 10, EXP_RATE_W, NullAnimator, &s_TankShellExp[14]},
|
|
{EXP + 11, EXP_RATE_W, NullAnimator, &s_TankShellExp[15]},
|
|
{EXP + 12, EXP_RATE_W, NullAnimator, &s_TankShellExp[16]},
|
|
{EXP + 13, EXP_RATE_W, NullAnimator, &s_TankShellExp[17]},
|
|
{EXP + 14, EXP_RATE_W, NullAnimator, &s_TankShellExp[18]},
|
|
{EXP + 15, EXP_RATE_W, NullAnimator, &s_TankShellExp[19]},
|
|
{EXP + 16, EXP_RATE_W, NullAnimator, &s_TankShellExp[20]},
|
|
{EXP + 17, EXP_RATE_W, NullAnimator, &s_TankShellExp[21]},
|
|
{EXP + 18, EXP_RATE_W, NullAnimator, &s_TankShellExp[22]},
|
|
{EXP + 19, EXP_RATE_W, NullAnimator, &s_TankShellExp[23]},
|
|
{EXP + 20, EXP_RATE_W, NullAnimator, &s_TankShellExp[24]},
|
|
{EXP + 20, 100, DoSuicide, &s_TankShellExp[0]}
|
|
};
|
|
|
|
#define TRACER_EXP_RATE 4
|
|
STATE s_TracerExp[] =
|
|
{
|
|
{EXP + 0, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[1]},
|
|
{EXP + 0, SF_QUICK_CALL, NullAnimator, &s_TracerExp[2]},
|
|
{EXP + 0, SF_QUICK_CALL, NullAnimator, &s_TracerExp[3]},
|
|
{EXP + 1, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[4]},
|
|
{EXP + 2, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[5]},
|
|
{EXP + 3, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[6]},
|
|
{EXP + 4, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[7]},
|
|
{EXP + 5, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[8]},
|
|
{EXP + 6, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[9]},
|
|
{EXP + 7, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[10]},
|
|
{EXP + 7, SF_QUICK_CALL, NullAnimator, &s_TracerExp[11]},
|
|
{EXP + 8, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[12]},
|
|
{EXP + 9, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[13]},
|
|
{EXP + 10, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[14]},
|
|
{EXP + 11, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[15]},
|
|
{EXP + 12, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[16]},
|
|
{EXP + 13, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[17]},
|
|
{EXP + 14, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[18]},
|
|
{EXP + 15, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[19]},
|
|
{EXP + 16, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[20]},
|
|
{EXP + 17, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[21]},
|
|
{EXP + 18, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[22]},
|
|
{EXP + 19, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[23]},
|
|
{EXP + 20, TRACER_EXP_RATE, NullAnimator, &s_TracerExp[24]},
|
|
{EXP + 20, 100, DoSuicide, &s_TracerExp[0]}
|
|
};
|
|
|
|
#define EXP_RATE_W 7
|
|
ANIMATOR DoSectorExp;
|
|
|
|
STATE s_SectorExp[] =
|
|
{
|
|
{EXP + 0, EXP_RATE_W, DoSectorExp, &s_SectorExp[1]},
|
|
{EXP + 0, SF_QUICK_CALL, SpawnShrapX, &s_SectorExp[2]},
|
|
{EXP + 0, SF_QUICK_CALL, DoSectorExp, &s_SectorExp[3]},
|
|
{EXP + 1, EXP_RATE_W, DoSectorExp, &s_SectorExp[4]},
|
|
{EXP + 2, EXP_RATE_W, DoSectorExp, &s_SectorExp[5]},
|
|
{EXP + 3, EXP_RATE_W, DoSectorExp, &s_SectorExp[6]},
|
|
{EXP + 4, EXP_RATE_W, DoSectorExp, &s_SectorExp[7]},
|
|
{EXP + 5, EXP_RATE_W, DoSectorExp, &s_SectorExp[8]},
|
|
{EXP + 6, EXP_RATE_W, DoSectorExp, &s_SectorExp[9]},
|
|
{EXP + 7, EXP_RATE_W, DoSectorExp, &s_SectorExp[10]},
|
|
{EXP + 7, SF_QUICK_CALL, DoSectorExp, &s_SectorExp[11]},
|
|
{EXP + 8, EXP_RATE_W, DoSectorExp, &s_SectorExp[12]},
|
|
{EXP + 9, EXP_RATE_W, DoSectorExp, &s_SectorExp[13]},
|
|
{EXP + 10, EXP_RATE_W, DoSectorExp, &s_SectorExp[14]},
|
|
{EXP + 11, EXP_RATE_W, DoSectorExp, &s_SectorExp[15]},
|
|
{EXP + 12, EXP_RATE_W, DoSectorExp, &s_SectorExp[16]},
|
|
{EXP + 13, EXP_RATE_W, DoSectorExp, &s_SectorExp[17]},
|
|
{EXP + 14, EXP_RATE_W, DoSectorExp, &s_SectorExp[18]},
|
|
{EXP + 15, EXP_RATE_W, DoSectorExp, &s_SectorExp[19]},
|
|
{EXP + 16, EXP_RATE_W, DoSectorExp, &s_SectorExp[20]},
|
|
{EXP + 17, EXP_RATE_W, DoSectorExp, &s_SectorExp[21]},
|
|
{EXP + 18, EXP_RATE_W, DoSectorExp, &s_SectorExp[22]},
|
|
{EXP + 19, EXP_RATE_W, DoSectorExp, &s_SectorExp[23]},
|
|
{EXP + 20, EXP_RATE_W, DoSectorExp, &s_SectorExp[24]},
|
|
{EXP + 20, 100, DoSuicide, &s_SectorExp[0]}
|
|
};
|
|
|
|
|
|
#define FIREBALL_DISS 3196
|
|
#define FIREBALL_DISS_RATE 8
|
|
STATE s_FireballExp[] =
|
|
{
|
|
{FIREBALL_DISS + 0, FIREBALL_DISS_RATE, NullAnimator, &s_FireballExp[1]},
|
|
{FIREBALL_DISS + 1, FIREBALL_DISS_RATE, NullAnimator, &s_FireballExp[2]},
|
|
{FIREBALL_DISS + 2, FIREBALL_DISS_RATE, NullAnimator, &s_FireballExp[3]},
|
|
{FIREBALL_DISS + 3, FIREBALL_DISS_RATE, NullAnimator, &s_FireballExp[4]},
|
|
{FIREBALL_DISS + 4, FIREBALL_DISS_RATE, NullAnimator, &s_FireballExp[5]},
|
|
{FIREBALL_DISS + 4, 100, DoSuicide, &s_FireballExp[0]}
|
|
};
|
|
|
|
#define NAP_EXP (3072)
|
|
#define NAP_EXP_RATE 6
|
|
|
|
STATE s_NapExp[] =
|
|
{
|
|
{NAP_EXP + 0, NAP_EXP_RATE, NullAnimator, &s_NapExp[1]},
|
|
{NAP_EXP + 0, 0 | SF_QUICK_CALL, DoDamageTest, &s_NapExp[2]},
|
|
{NAP_EXP + 1, NAP_EXP_RATE, NullAnimator, &s_NapExp[3]},
|
|
{NAP_EXP + 2, NAP_EXP_RATE, NullAnimator, &s_NapExp[4]},
|
|
{NAP_EXP + 3, NAP_EXP_RATE, NullAnimator, &s_NapExp[5]},
|
|
{NAP_EXP + 4, NAP_EXP_RATE, NullAnimator, &s_NapExp[6]},
|
|
{NAP_EXP + 5, NAP_EXP_RATE, NullAnimator, &s_NapExp[7]},
|
|
{NAP_EXP + 6, NAP_EXP_RATE, NullAnimator, &s_NapExp[8]},
|
|
{NAP_EXP + 7, NAP_EXP_RATE, NullAnimator, &s_NapExp[9]},
|
|
{NAP_EXP + 8, NAP_EXP_RATE, NullAnimator, &s_NapExp[10]},
|
|
{NAP_EXP + 9, NAP_EXP_RATE, NullAnimator, &s_NapExp[11]},
|
|
{NAP_EXP + 10, NAP_EXP_RATE, NullAnimator, &s_NapExp[12]},
|
|
{NAP_EXP + 11, NAP_EXP_RATE, NullAnimator, &s_NapExp[13]},
|
|
{NAP_EXP + 12, NAP_EXP_RATE, NullAnimator, &s_NapExp[14]},
|
|
{NAP_EXP + 13, NAP_EXP_RATE, NullAnimator, &s_NapExp[15]},
|
|
{NAP_EXP + 14, NAP_EXP_RATE, NullAnimator, &s_NapExp[16]},
|
|
{NAP_EXP + 15, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[17]},
|
|
{NAP_EXP + 16, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[18]},
|
|
{NAP_EXP + 17, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[19]},
|
|
{NAP_EXP + 18, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[20]},
|
|
{NAP_EXP + 19, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[21]},
|
|
{NAP_EXP + 20, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[22]},
|
|
{NAP_EXP + 21, NAP_EXP_RATE-2, NullAnimator, &s_NapExp[23]},
|
|
{NAP_EXP + 21, NAP_EXP_RATE-2, DoSuicide, &s_NapExp[23]}
|
|
};
|
|
|
|
ANIMATOR DoFireballFlames;
|
|
#define FLAME_RATE 6
|
|
|
|
STATE s_FireballFlames[] =
|
|
{
|
|
{FIREBALL_FLAMES + 0, FLAME_RATE, DoFireballFlames, &s_FireballFlames[1]},
|
|
{FIREBALL_FLAMES + 1, FLAME_RATE, DoFireballFlames, &s_FireballFlames[2]},
|
|
{FIREBALL_FLAMES + 2, FLAME_RATE, DoFireballFlames, &s_FireballFlames[3]},
|
|
{FIREBALL_FLAMES + 3, FLAME_RATE, DoFireballFlames, &s_FireballFlames[4]},
|
|
{FIREBALL_FLAMES + 4, FLAME_RATE, DoFireballFlames, &s_FireballFlames[5]},
|
|
{FIREBALL_FLAMES + 5, FLAME_RATE, DoFireballFlames, &s_FireballFlames[6]},
|
|
{FIREBALL_FLAMES + 6, FLAME_RATE, DoFireballFlames, &s_FireballFlames[7]},
|
|
{FIREBALL_FLAMES + 7, FLAME_RATE, DoFireballFlames, &s_FireballFlames[8]},
|
|
{FIREBALL_FLAMES + 8, FLAME_RATE, DoFireballFlames, &s_FireballFlames[9]},
|
|
{FIREBALL_FLAMES + 9, FLAME_RATE, DoFireballFlames, &s_FireballFlames[10]},
|
|
{FIREBALL_FLAMES +10, FLAME_RATE, DoFireballFlames, &s_FireballFlames[11]},
|
|
{FIREBALL_FLAMES +11, FLAME_RATE, DoFireballFlames, &s_FireballFlames[12]},
|
|
{FIREBALL_FLAMES +12, FLAME_RATE, DoFireballFlames, &s_FireballFlames[13]},
|
|
{FIREBALL_FLAMES +13, FLAME_RATE, DoFireballFlames, &s_FireballFlames[0]},
|
|
};
|
|
|
|
ANIMATOR DoBreakFlames;
|
|
#define FLAME_RATE 6
|
|
|
|
STATE s_BreakFlames[] =
|
|
{
|
|
{FIREBALL_FLAMES + 0, FLAME_RATE, DoBreakFlames, &s_BreakFlames[1]},
|
|
{FIREBALL_FLAMES + 1, FLAME_RATE, DoBreakFlames, &s_BreakFlames[2]},
|
|
{FIREBALL_FLAMES + 2, FLAME_RATE, DoBreakFlames, &s_BreakFlames[3]},
|
|
{FIREBALL_FLAMES + 3, FLAME_RATE, DoBreakFlames, &s_BreakFlames[4]},
|
|
{FIREBALL_FLAMES + 4, FLAME_RATE, DoBreakFlames, &s_BreakFlames[5]},
|
|
{FIREBALL_FLAMES + 5, FLAME_RATE, DoBreakFlames, &s_BreakFlames[6]},
|
|
{FIREBALL_FLAMES + 6, FLAME_RATE, DoBreakFlames, &s_BreakFlames[7]},
|
|
{FIREBALL_FLAMES + 7, FLAME_RATE, DoBreakFlames, &s_BreakFlames[8]},
|
|
{FIREBALL_FLAMES + 8, FLAME_RATE, DoBreakFlames, &s_BreakFlames[9]},
|
|
{FIREBALL_FLAMES + 9, FLAME_RATE, DoBreakFlames, &s_BreakFlames[10]},
|
|
{FIREBALL_FLAMES +10, FLAME_RATE, DoBreakFlames, &s_BreakFlames[11]},
|
|
{FIREBALL_FLAMES +11, FLAME_RATE, DoBreakFlames, &s_BreakFlames[12]},
|
|
{FIREBALL_FLAMES +12, FLAME_RATE, DoBreakFlames, &s_BreakFlames[13]},
|
|
{FIREBALL_FLAMES +13, FLAME_RATE, DoBreakFlames, &s_BreakFlames[0]},
|
|
};
|
|
|
|
//////////////////////
|
|
//
|
|
// FIREBALL
|
|
//
|
|
//////////////////////
|
|
|
|
#if 1
|
|
ANIMATOR DoFireball;
|
|
#define FIREBALL_RATE 8
|
|
#define GORO_FIREBALL FIREBALL+1
|
|
|
|
STATE s_Fireball[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoFireball, &s_Fireball[1]},
|
|
{FIREBALL + 1, 12, DoFireball, &s_Fireball[2]},
|
|
{FIREBALL + 2, 12, DoFireball, &s_Fireball[3]},
|
|
{FIREBALL + 3, 12, DoFireball, &s_Fireball[0]}
|
|
};
|
|
|
|
#else
|
|
#define GORO_FIREBALL FIREBALL_R0
|
|
#define FIREBALL_RATE 6
|
|
ANIMATOR DoFireball;
|
|
|
|
STATE s_Fireball[5][4] =
|
|
{
|
|
{
|
|
{FIREBALL_R0 + 0, FIREBALL_RATE, DoFireball, &s_Fireball[0][1]},
|
|
{FIREBALL_R0 + 1, FIREBALL_RATE, DoFireball, &s_Fireball[0][2]},
|
|
{FIREBALL_R0 + 2, FIREBALL_RATE, DoFireball, &s_Fireball[0][3]},
|
|
{FIREBALL_R0 + 3, FIREBALL_RATE, DoFireball, &s_Fireball[0][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R1 + 0, FIREBALL_RATE, DoFireball, &s_Fireball[1][1]},
|
|
{FIREBALL_R1 + 1, FIREBALL_RATE, DoFireball, &s_Fireball[1][2]},
|
|
{FIREBALL_R1 + 2, FIREBALL_RATE, DoFireball, &s_Fireball[1][3]},
|
|
{FIREBALL_R1 + 3, FIREBALL_RATE, DoFireball, &s_Fireball[1][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R2 + 0, FIREBALL_RATE, DoFireball, &s_Fireball[2][1]},
|
|
{FIREBALL_R2 + 1, FIREBALL_RATE, DoFireball, &s_Fireball[2][2]},
|
|
{FIREBALL_R2 + 2, FIREBALL_RATE, DoFireball, &s_Fireball[2][3]},
|
|
{FIREBALL_R2 + 3, FIREBALL_RATE, DoFireball, &s_Fireball[2][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R3 + 0, FIREBALL_RATE, DoFireball, &s_Fireball[3][1]},
|
|
{FIREBALL_R3 + 1, FIREBALL_RATE, DoFireball, &s_Fireball[3][2]},
|
|
{FIREBALL_R3 + 2, FIREBALL_RATE, DoFireball, &s_Fireball[3][3]},
|
|
{FIREBALL_R3 + 3, FIREBALL_RATE, DoFireball, &s_Fireball[3][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R4 + 0, FIREBALL_RATE, DoFireball, &s_Fireball[4][1]},
|
|
{FIREBALL_R4 + 1, FIREBALL_RATE, DoFireball, &s_Fireball[4][2]},
|
|
{FIREBALL_R4 + 2, FIREBALL_RATE, DoFireball, &s_Fireball[4][3]},
|
|
{FIREBALL_R4 + 3, FIREBALL_RATE, DoFireball, &s_Fireball[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Fireball[] =
|
|
{
|
|
s_Fireball[0],
|
|
s_Fireball[1],
|
|
s_Fireball[2],
|
|
s_Fireball[3],
|
|
s_Fireball[4]
|
|
};
|
|
#endif
|
|
|
|
#if 0
|
|
ANIMATOR DoRing;
|
|
|
|
STATE s_Ring[5][4] =
|
|
{
|
|
{
|
|
{FIREBALL_R0 + 0, FIREBALL_RATE, DoRing, &s_Ring[0][1]},
|
|
{FIREBALL_R0 + 1, FIREBALL_RATE, DoRing, &s_Ring[0][2]},
|
|
{FIREBALL_R0 + 2, FIREBALL_RATE, DoRing, &s_Ring[0][3]},
|
|
{FIREBALL_R0 + 3, FIREBALL_RATE, DoRing, &s_Ring[0][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R1 + 0, FIREBALL_RATE, DoRing, &s_Ring[1][1]},
|
|
{FIREBALL_R1 + 1, FIREBALL_RATE, DoRing, &s_Ring[1][2]},
|
|
{FIREBALL_R1 + 2, FIREBALL_RATE, DoRing, &s_Ring[1][3]},
|
|
{FIREBALL_R1 + 3, FIREBALL_RATE, DoRing, &s_Ring[1][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R2 + 0, FIREBALL_RATE, DoRing, &s_Ring[2][1]},
|
|
{FIREBALL_R2 + 1, FIREBALL_RATE, DoRing, &s_Ring[2][2]},
|
|
{FIREBALL_R2 + 2, FIREBALL_RATE, DoRing, &s_Ring[2][3]},
|
|
{FIREBALL_R2 + 3, FIREBALL_RATE, DoRing, &s_Ring[2][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R3 + 0, FIREBALL_RATE, DoRing, &s_Ring[3][1]},
|
|
{FIREBALL_R3 + 1, FIREBALL_RATE, DoRing, &s_Ring[3][2]},
|
|
{FIREBALL_R3 + 2, FIREBALL_RATE, DoRing, &s_Ring[3][3]},
|
|
{FIREBALL_R3 + 3, FIREBALL_RATE, DoRing, &s_Ring[3][0]},
|
|
},
|
|
{
|
|
{FIREBALL_R4 + 0, FIREBALL_RATE, DoRing, &s_Ring[4][1]},
|
|
{FIREBALL_R4 + 1, FIREBALL_RATE, DoRing, &s_Ring[4][2]},
|
|
{FIREBALL_R4 + 2, FIREBALL_RATE, DoRing, &s_Ring[4][3]},
|
|
{FIREBALL_R4 + 3, FIREBALL_RATE, DoRing, &s_Ring[4][0]},
|
|
}
|
|
};
|
|
|
|
STATE* sg_Ring[] =
|
|
{
|
|
s_Ring[0],
|
|
s_Ring[1],
|
|
s_Ring[2],
|
|
s_Ring[3],
|
|
s_Ring[4]
|
|
};
|
|
#else
|
|
ANIMATOR DoRing;
|
|
|
|
STATE s_Ring[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoRing, &s_Ring[1]},
|
|
{FIREBALL + 1, 12, DoRing, &s_Ring[2]},
|
|
{FIREBALL + 2, 12, DoRing, &s_Ring[3]},
|
|
{FIREBALL + 3, 12, DoRing, &s_Ring[0]}
|
|
};
|
|
#endif
|
|
|
|
STATE s_Ring2[] =
|
|
{
|
|
{2031 + 0, 12, DoRing, &s_Ring2[1]},
|
|
{2031 + 1, 12, DoRing, &s_Ring2[2]},
|
|
{2031 + 2, 12, DoRing, &s_Ring2[3]},
|
|
{2031 + 3, 12, DoRing, &s_Ring2[0]}
|
|
};
|
|
|
|
ANIMATOR DoNapalm;
|
|
|
|
STATE s_Napalm[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoNapalm, &s_Napalm[1]},
|
|
{FIREBALL + 1, 12, DoNapalm, &s_Napalm[2]},
|
|
{FIREBALL + 2, 12, DoNapalm, &s_Napalm[3]},
|
|
{FIREBALL + 3, 12, DoNapalm, &s_Napalm[0]}
|
|
};
|
|
|
|
|
|
ANIMATOR DoBloodWorm;
|
|
|
|
#if 1
|
|
#define BLOOD_WORM 2106
|
|
STATE s_BloodWorm[] =
|
|
{
|
|
{BLOOD_WORM + 0, 12, DoBloodWorm, &s_BloodWorm[1]},
|
|
{BLOOD_WORM + 1, 12, DoBloodWorm, &s_BloodWorm[2]},
|
|
{BLOOD_WORM + 2, 12, DoBloodWorm, &s_BloodWorm[3]},
|
|
{BLOOD_WORM + 3, 12, DoBloodWorm, &s_BloodWorm[4]},
|
|
{BLOOD_WORM + 2, 12, DoBloodWorm, &s_BloodWorm[5]},
|
|
{BLOOD_WORM + 1, 12, DoBloodWorm, &s_BloodWorm[0]}
|
|
};
|
|
#else
|
|
#define BLOOD_WORM FIREBALL+5
|
|
STATE s_BloodWorm[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoBloodWorm, &s_BloodWorm[1]},
|
|
{FIREBALL + 1, 12, DoBloodWorm, &s_BloodWorm[2]},
|
|
{FIREBALL + 2, 12, DoBloodWorm, &s_BloodWorm[3]},
|
|
{FIREBALL + 3, 12, DoBloodWorm, &s_BloodWorm[0]}
|
|
};
|
|
#endif
|
|
|
|
#if 1
|
|
#define PLASMA_EXP BLOOD_WORM+1
|
|
#define PLASMA_EXP_RATE 8
|
|
|
|
STATE s_PlasmaExp[] =
|
|
{
|
|
{BLOOD_WORM + 0, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[1]},
|
|
{BLOOD_WORM + 1, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[2]},
|
|
{BLOOD_WORM + 2, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[3]},
|
|
{BLOOD_WORM + 3, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[4]},
|
|
{BLOOD_WORM + 2, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[5]},
|
|
{BLOOD_WORM + 1, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[6]},
|
|
{BLOOD_WORM + 0, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[7]},
|
|
{BLOOD_WORM + 1, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[8]},
|
|
{BLOOD_WORM + 2, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[9]},
|
|
{BLOOD_WORM + 3, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[10]},
|
|
{BLOOD_WORM + 2, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[11]},
|
|
{BLOOD_WORM + 1, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[12]},
|
|
{BLOOD_WORM + 0, PLASMA_EXP_RATE, DoSuicide, &s_PlasmaExp[12]},
|
|
};
|
|
|
|
#else
|
|
#define PLASMA_EXP (NAP_EXP+1)
|
|
#define PLASMA_EXP_RATE 4
|
|
|
|
STATE s_PlasmaExp[] =
|
|
{
|
|
{PLASMA_EXP + 0, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[1]},
|
|
{PLASMA_EXP + 0, 0 | SF_QUICK_CALL, DoDamageTest, &s_PlasmaExp[2]},
|
|
{PLASMA_EXP + 1, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[3]},
|
|
{PLASMA_EXP + 2, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[4]},
|
|
{PLASMA_EXP + 3, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[5]},
|
|
{PLASMA_EXP + 4, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[6]},
|
|
{PLASMA_EXP + 5, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[7]},
|
|
{PLASMA_EXP + 6, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[8]},
|
|
{PLASMA_EXP + 7, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[9]},
|
|
{PLASMA_EXP + 8, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[10]},
|
|
{PLASMA_EXP + 9, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[11]},
|
|
{PLASMA_EXP + 10, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[12]},
|
|
{PLASMA_EXP + 11, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[13]},
|
|
{PLASMA_EXP + 9, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[14]},
|
|
{PLASMA_EXP + 8, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[15]},
|
|
{PLASMA_EXP + 7, PLASMA_EXP_RATE, NullAnimator, &s_PlasmaExp[16]},
|
|
{PLASMA_EXP + 6, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[17]},
|
|
{PLASMA_EXP + 5, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[18]},
|
|
{PLASMA_EXP + 4, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[19]},
|
|
{PLASMA_EXP + 3, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[20]},
|
|
{PLASMA_EXP + 2, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[21]},
|
|
{PLASMA_EXP + 1, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[22]},
|
|
{PLASMA_EXP + 1, PLASMA_EXP_RATE-2, NullAnimator, &s_PlasmaExp[23]},
|
|
{PLASMA_EXP + 1, PLASMA_EXP_RATE-2, DoSuicide, &s_PlasmaExp[23]}
|
|
};
|
|
#endif
|
|
|
|
|
|
ANIMATOR DoMirv;
|
|
|
|
STATE s_Mirv[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoMirv, &s_Mirv[1]},
|
|
{FIREBALL + 1, 12, DoMirv, &s_Mirv[2]},
|
|
{FIREBALL + 2, 12, DoMirv, &s_Mirv[3]},
|
|
{FIREBALL + 3, 12, DoMirv, &s_Mirv[0]}
|
|
};
|
|
|
|
ANIMATOR DoMirvMissile;
|
|
|
|
STATE s_MirvMissile[] =
|
|
{
|
|
{FIREBALL + 0, 12, DoMirvMissile, &s_MirvMissile[1]},
|
|
{FIREBALL + 1, 12, DoMirvMissile, &s_MirvMissile[2]},
|
|
{FIREBALL + 2, 12, DoMirvMissile, &s_MirvMissile[3]},
|
|
{FIREBALL + 3, 12, DoMirvMissile, &s_MirvMissile[0]}
|
|
};
|
|
|
|
//#define Vomit1 1740
|
|
//#define Vomit2 1741
|
|
#define Vomit1 1719
|
|
#define Vomit2 1721
|
|
//#define VomitSplash 1742
|
|
#define VomitSplash 1711
|
|
#define Vomit_RATE 16
|
|
ANIMATOR DoVomit,DoVomitSplash;
|
|
|
|
STATE s_Vomit1[] =
|
|
{
|
|
{Vomit1 + 0, Vomit_RATE, DoVomit, &s_Vomit1[0]}
|
|
};
|
|
|
|
STATE s_Vomit2[] =
|
|
{
|
|
{Vomit2 + 0, Vomit_RATE, DoVomit, &s_Vomit2[0]}
|
|
};
|
|
|
|
STATE s_VomitSplash[] =
|
|
{
|
|
{VomitSplash + 0, Vomit_RATE, DoVomitSplash, &s_VomitSplash[0]}
|
|
};
|
|
|
|
#define GORE_Head 1670
|
|
#define GORE_Head_RATE 16
|
|
|
|
STATE s_GoreHead[] =
|
|
{
|
|
{GORE_Head + 0, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[1]},
|
|
{GORE_Head + 1, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[2]},
|
|
{GORE_Head + 2, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[3]},
|
|
{GORE_Head + 3, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[4]},
|
|
{GORE_Head + 4, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[5]},
|
|
{GORE_Head + 5, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[6]},
|
|
{GORE_Head + 6, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[7]},
|
|
{GORE_Head + 7, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[8]},
|
|
{GORE_Head + 8, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[9]},
|
|
{GORE_Head + 9, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[10]},
|
|
{GORE_Head + 10, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[11]},
|
|
{GORE_Head + 11, GORE_Head_RATE, DoShrapJumpFall, &s_GoreHead[0]},
|
|
};
|
|
|
|
#define GORE_Leg 1689
|
|
#define GORE_Leg_RATE 16
|
|
|
|
STATE s_GoreLeg[] =
|
|
{
|
|
{GORE_Leg + 0, GORE_Leg_RATE, DoShrapJumpFall, &s_GoreLeg[1]},
|
|
{GORE_Leg + 1, GORE_Leg_RATE, DoShrapJumpFall, &s_GoreLeg[2]},
|
|
{GORE_Leg + 2, GORE_Leg_RATE, DoShrapJumpFall, &s_GoreLeg[0]},
|
|
};
|
|
|
|
#define GORE_Eye 1692
|
|
#define GORE_Eye_RATE 16
|
|
|
|
STATE s_GoreEye[] =
|
|
{
|
|
{GORE_Eye + 0, GORE_Eye_RATE, DoShrapJumpFall, &s_GoreEye[1]},
|
|
{GORE_Eye + 1, GORE_Eye_RATE, DoShrapJumpFall, &s_GoreEye[2]},
|
|
{GORE_Eye + 2, GORE_Eye_RATE, DoShrapJumpFall, &s_GoreEye[3]},
|
|
{GORE_Eye + 3, GORE_Eye_RATE, DoShrapJumpFall, &s_GoreEye[0]},
|
|
};
|
|
|
|
#define GORE_Torso 1696
|
|
#define GORE_Torso_RATE 16
|
|
|
|
STATE s_GoreTorso[] =
|
|
{
|
|
{GORE_Torso + 0, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[1]},
|
|
{GORE_Torso + 1, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[2]},
|
|
{GORE_Torso + 2, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[3]},
|
|
{GORE_Torso + 3, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[4]},
|
|
{GORE_Torso + 4, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[5]},
|
|
{GORE_Torso + 5, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[6]},
|
|
{GORE_Torso + 6, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[7]},
|
|
{GORE_Torso + 7, GORE_Torso_RATE, DoShrapJumpFall, &s_GoreTorso[0]},
|
|
};
|
|
|
|
|
|
#define GORE_Arm 1550
|
|
#define GORE_Arm_RATE 16
|
|
|
|
STATE s_GoreArm[] =
|
|
{
|
|
{GORE_Arm + 0, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[1]},
|
|
{GORE_Arm + 1, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[2]},
|
|
{GORE_Arm + 2, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[3]},
|
|
{GORE_Arm + 3, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[4]},
|
|
{GORE_Arm + 4, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[5]},
|
|
{GORE_Arm + 5, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[6]},
|
|
{GORE_Arm + 6, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[7]},
|
|
{GORE_Arm + 7, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[8]},
|
|
{GORE_Arm + 8, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[9]},
|
|
{GORE_Arm + 9, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[10]},
|
|
{GORE_Arm + 10, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[11]},
|
|
{GORE_Arm + 11, GORE_Arm_RATE, DoShrapJumpFall, &s_GoreArm[0]},
|
|
};
|
|
|
|
#define GORE_Lung 903
|
|
#define GORE_Lung_RATE 16
|
|
|
|
STATE s_GoreLung[] =
|
|
{
|
|
{GORE_Lung + 0, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[1]},
|
|
{GORE_Lung + 1, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[2]},
|
|
{GORE_Lung + 2, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[3]},
|
|
{GORE_Lung + 3, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[4]},
|
|
{GORE_Lung + 4, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[5]},
|
|
{GORE_Lung + 5, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[6]},
|
|
{GORE_Lung + 6, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[7]},
|
|
{GORE_Lung + 7, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[8]},
|
|
{GORE_Lung + 8, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[9]},
|
|
{GORE_Lung + 9, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[10]},
|
|
{GORE_Lung + 10, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[11]},
|
|
{GORE_Lung + 11, GORE_Lung_RATE, DoShrapJumpFall, &s_GoreLung[0]},
|
|
};
|
|
|
|
#define GORE_Liver 918
|
|
#define GORE_Liver_RATE 16
|
|
|
|
STATE s_GoreLiver[] =
|
|
{
|
|
{GORE_Liver + 0, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[1]},
|
|
{GORE_Liver + 1, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[2]},
|
|
{GORE_Liver + 2, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[3]},
|
|
{GORE_Liver + 3, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[4]},
|
|
{GORE_Liver + 4, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[5]},
|
|
{GORE_Liver + 5, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[6]},
|
|
{GORE_Liver + 6, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[7]},
|
|
{GORE_Liver + 7, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[8]},
|
|
{GORE_Liver + 8, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[9]},
|
|
{GORE_Liver + 9, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[10]},
|
|
{GORE_Liver + 10, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[11]},
|
|
{GORE_Liver + 11, GORE_Liver_RATE, DoShrapJumpFall, &s_GoreLiver[0]},
|
|
};
|
|
|
|
#define GORE_SkullCap 933
|
|
#define GORE_SkullCap_RATE 16
|
|
|
|
STATE s_GoreSkullCap[] =
|
|
{
|
|
{GORE_SkullCap + 0, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[1]},
|
|
{GORE_SkullCap + 1, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[2]},
|
|
{GORE_SkullCap + 2, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[3]},
|
|
{GORE_SkullCap + 3, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[4]},
|
|
{GORE_SkullCap + 4, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[5]},
|
|
{GORE_SkullCap + 5, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[6]},
|
|
{GORE_SkullCap + 6, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[7]},
|
|
{GORE_SkullCap + 7, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[8]},
|
|
{GORE_SkullCap + 8, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[9]},
|
|
{GORE_SkullCap + 9, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[10]},
|
|
{GORE_SkullCap + 10, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[11]},
|
|
{GORE_SkullCap + 11, GORE_SkullCap_RATE, DoShrapJumpFall, &s_GoreSkullCap[0]},
|
|
};
|
|
|
|
|
|
#define GORE_ChunkS 2430
|
|
#define GORE_ChunkS_RATE 16
|
|
|
|
STATE s_GoreChunkS[] =
|
|
{
|
|
{GORE_ChunkS + 0, GORE_ChunkS_RATE, DoShrapJumpFall, &s_GoreChunkS[1]},
|
|
{GORE_ChunkS + 1, GORE_ChunkS_RATE, DoShrapJumpFall, &s_GoreChunkS[2]},
|
|
{GORE_ChunkS + 2, GORE_ChunkS_RATE, DoShrapJumpFall, &s_GoreChunkS[3]},
|
|
{GORE_ChunkS + 3, GORE_ChunkS_RATE, DoShrapJumpFall, &s_GoreChunkS[0]},
|
|
};
|
|
|
|
#define GORE_Drip 1562 //2430
|
|
#define GORE_Drip_RATE 16
|
|
|
|
STATE s_GoreDrip[] =
|
|
{
|
|
{GORE_Drip + 0, GORE_Drip_RATE, DoShrapJumpFall, &s_GoreDrip[1]},
|
|
{GORE_Drip + 1, GORE_Drip_RATE, DoShrapJumpFall, &s_GoreDrip[2]},
|
|
{GORE_Drip + 2, GORE_Drip_RATE, DoShrapJumpFall, &s_GoreDrip[3]},
|
|
{GORE_Drip + 3, GORE_Drip_RATE, DoShrapJumpFall, &s_GoreDrip[0]},
|
|
};
|
|
|
|
STATE s_FastGoreDrip[] =
|
|
{
|
|
{GORE_Drip + 0, GORE_Drip_RATE, DoFastShrapJumpFall, &s_FastGoreDrip[1]},
|
|
{GORE_Drip + 1, GORE_Drip_RATE, DoFastShrapJumpFall, &s_FastGoreDrip[2]},
|
|
{GORE_Drip + 2, GORE_Drip_RATE, DoFastShrapJumpFall, &s_FastGoreDrip[3]},
|
|
{GORE_Drip + 3, GORE_Drip_RATE, DoFastShrapJumpFall, &s_FastGoreDrip[0]},
|
|
};
|
|
|
|
///////////////////////////////////////////////
|
|
//
|
|
// This GORE mostly for the Accursed Heads
|
|
//
|
|
///////////////////////////////////////////////
|
|
|
|
#define GORE_Flame 847
|
|
#define GORE_Flame_RATE 8
|
|
|
|
STATE s_GoreFlame[] =
|
|
{
|
|
{GORE_Flame + 0, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[1]},
|
|
{GORE_Flame + 1, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[2]},
|
|
{GORE_Flame + 2, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[3]},
|
|
{GORE_Flame + 3, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[4]},
|
|
{GORE_Flame + 4, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[5]},
|
|
{GORE_Flame + 5, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[6]},
|
|
{GORE_Flame + 6, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[7]},
|
|
{GORE_Flame + 7, GORE_Flame_RATE, DoFastShrapJumpFall, &s_GoreFlame[0]},
|
|
};
|
|
|
|
ANIMATOR DoTracerShrap;
|
|
STATE s_TracerShrap[] =
|
|
{
|
|
{GORE_Flame + 0, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[1]},
|
|
{GORE_Flame + 1, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[2]},
|
|
{GORE_Flame + 2, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[3]},
|
|
{GORE_Flame + 3, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[4]},
|
|
{GORE_Flame + 4, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[5]},
|
|
{GORE_Flame + 5, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[6]},
|
|
{GORE_Flame + 6, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[7]},
|
|
{GORE_Flame + 7, GORE_Flame_RATE, DoTracerShrap, &s_TracerShrap[0]},
|
|
};
|
|
|
|
#define UZI_SHELL 2152
|
|
#define UZISHELL_RATE 8
|
|
//ANIMATOR DoShellShrap;
|
|
STATE s_UziShellShrap[] =
|
|
{
|
|
{UZI_SHELL + 0, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[1]},
|
|
{UZI_SHELL + 1, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[2]},
|
|
{UZI_SHELL + 2, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[3]},
|
|
{UZI_SHELL + 3, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[4]},
|
|
{UZI_SHELL + 4, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[5]},
|
|
{UZI_SHELL + 5, UZISHELL_RATE, DoShrapJumpFall, &s_UziShellShrap[0]},
|
|
};
|
|
|
|
STATE s_UziShellShrapStill1[] =
|
|
{
|
|
{UZI_SHELL + 0, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill1[0]}
|
|
};
|
|
STATE s_UziShellShrapStill2[] =
|
|
{
|
|
{UZI_SHELL + 1, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill2[0]}
|
|
};
|
|
STATE s_UziShellShrapStill3[] =
|
|
{
|
|
{UZI_SHELL + 2, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill3[0]}
|
|
};
|
|
STATE s_UziShellShrapStill4[] =
|
|
{
|
|
{UZI_SHELL + 3, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill4[0]}
|
|
};
|
|
STATE s_UziShellShrapStill5[] =
|
|
{
|
|
{UZI_SHELL + 4, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill5[0]}
|
|
};
|
|
STATE s_UziShellShrapStill6[] =
|
|
{
|
|
{UZI_SHELL + 5, UZISHELL_RATE, NullAnimator, &s_UziShellShrapStill6[0]}
|
|
};
|
|
|
|
#define SHOT_SHELL 2180
|
|
#define SHOTSHELL_RATE 8
|
|
STATE s_ShotgunShellShrap[] =
|
|
{
|
|
{SHOT_SHELL + 0, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[1]},
|
|
{SHOT_SHELL + 1, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[2]},
|
|
{SHOT_SHELL + 2, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[3]},
|
|
{SHOT_SHELL + 3, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[4]},
|
|
{SHOT_SHELL + 4, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[5]},
|
|
{SHOT_SHELL + 5, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[6]},
|
|
{SHOT_SHELL + 6, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[7]},
|
|
{SHOT_SHELL + 7, SHOTSHELL_RATE, DoShrapJumpFall, &s_ShotgunShellShrap[0]},
|
|
};
|
|
|
|
STATE s_ShotgunShellShrapStill1[] =
|
|
{
|
|
{SHOT_SHELL + 1, SHOTSHELL_RATE, NullAnimator, &s_ShotgunShellShrapStill1[0]}
|
|
};
|
|
STATE s_ShotgunShellShrapStill2[] =
|
|
{
|
|
{SHOT_SHELL + 3, SHOTSHELL_RATE, NullAnimator, &s_ShotgunShellShrapStill2[0]}
|
|
};
|
|
STATE s_ShotgunShellShrapStill3[] =
|
|
{
|
|
{SHOT_SHELL + 7, SHOTSHELL_RATE, NullAnimator, &s_ShotgunShellShrapStill3[0]}
|
|
};
|
|
|
|
#define GORE_FlameChunkA 839
|
|
#define GORE_FlameChunkA_RATE 8
|
|
|
|
STATE s_GoreFlameChunkA[] =
|
|
{
|
|
{GORE_FlameChunkA + 0, GORE_FlameChunkA_RATE, DoShrapJumpFall, &s_GoreFlameChunkA[1]},
|
|
{GORE_FlameChunkA + 1, GORE_FlameChunkA_RATE, DoShrapJumpFall, &s_GoreFlameChunkA[2]},
|
|
{GORE_FlameChunkA + 2, GORE_FlameChunkA_RATE, DoShrapJumpFall, &s_GoreFlameChunkA[3]},
|
|
{GORE_FlameChunkA + 3, GORE_FlameChunkA_RATE, DoShrapJumpFall, &s_GoreFlameChunkA[0]},
|
|
};
|
|
|
|
#define GORE_FlameChunkB 843
|
|
#define GORE_FlameChunkB_RATE 8
|
|
|
|
STATE s_GoreFlameChunkB[] =
|
|
{
|
|
{GORE_FlameChunkB + 0, GORE_FlameChunkB_RATE, DoShrapJumpFall, &s_GoreFlameChunkB[1]},
|
|
{GORE_FlameChunkB + 1, GORE_FlameChunkB_RATE, DoShrapJumpFall, &s_GoreFlameChunkB[2]},
|
|
{GORE_FlameChunkB + 2, GORE_FlameChunkB_RATE, DoShrapJumpFall, &s_GoreFlameChunkB[3]},
|
|
{GORE_FlameChunkB + 3, GORE_FlameChunkB_RATE, DoShrapJumpFall, &s_GoreFlameChunkB[0]},
|
|
};
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
//
|
|
// General Breaking Shrapnel
|
|
//
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
#define COIN_SHRAP 2530
|
|
#define CoinShrap_RATE 12
|
|
|
|
STATE s_CoinShrap[] =
|
|
{
|
|
{COIN_SHRAP + 0, CoinShrap_RATE, DoShrapJumpFall, &s_CoinShrap[1]},
|
|
{COIN_SHRAP + 1, CoinShrap_RATE, DoShrapJumpFall, &s_CoinShrap[2]},
|
|
{COIN_SHRAP + 2, CoinShrap_RATE, DoShrapJumpFall, &s_CoinShrap[3]},
|
|
{COIN_SHRAP + 3, CoinShrap_RATE, DoShrapJumpFall, &s_CoinShrap[0]},
|
|
};
|
|
|
|
#define MARBEL 5096
|
|
#define Marbel_RATE 12
|
|
|
|
STATE s_Marbel[] =
|
|
{
|
|
{MARBEL, Marbel_RATE, DoShrapJumpFall, &s_Marbel[0]},
|
|
};
|
|
|
|
//
|
|
// Glass
|
|
//
|
|
|
|
#define GLASS_SHRAP_A 3864
|
|
#define GlassShrapA_RATE 12
|
|
|
|
STATE s_GlassShrapA[] =
|
|
{
|
|
{GLASS_SHRAP_A + 0, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[1]},
|
|
{GLASS_SHRAP_A + 1, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[2]},
|
|
{GLASS_SHRAP_A + 2, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[3]},
|
|
{GLASS_SHRAP_A + 3, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[4]},
|
|
{GLASS_SHRAP_A + 4, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[5]},
|
|
{GLASS_SHRAP_A + 5, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[6]},
|
|
{GLASS_SHRAP_A + 6, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[7]},
|
|
{GLASS_SHRAP_A + 7, GlassShrapA_RATE, DoShrapJumpFall, &s_GlassShrapA[0]},
|
|
};
|
|
|
|
#define GLASS_SHRAP_B 3872
|
|
#define GlassShrapB_RATE 12
|
|
|
|
STATE s_GlassShrapB[] =
|
|
{
|
|
{GLASS_SHRAP_B + 0, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[1]},
|
|
{GLASS_SHRAP_B + 1, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[2]},
|
|
{GLASS_SHRAP_B + 2, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[3]},
|
|
{GLASS_SHRAP_B + 3, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[4]},
|
|
{GLASS_SHRAP_B + 4, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[5]},
|
|
{GLASS_SHRAP_B + 5, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[6]},
|
|
{GLASS_SHRAP_B + 6, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[7]},
|
|
{GLASS_SHRAP_B + 7, GlassShrapB_RATE, DoShrapJumpFall, &s_GlassShrapB[0]},
|
|
};
|
|
|
|
#define GLASS_SHRAP_C 3880
|
|
#define GlassShrapC_RATE 12
|
|
|
|
STATE s_GlassShrapC[] =
|
|
{
|
|
{GLASS_SHRAP_C + 0, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[1]},
|
|
{GLASS_SHRAP_C + 1, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[2]},
|
|
{GLASS_SHRAP_C + 2, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[3]},
|
|
{GLASS_SHRAP_C + 3, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[4]},
|
|
{GLASS_SHRAP_C + 4, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[5]},
|
|
{GLASS_SHRAP_C + 5, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[6]},
|
|
{GLASS_SHRAP_C + 6, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[7]},
|
|
{GLASS_SHRAP_C + 7, GlassShrapC_RATE, DoShrapJumpFall, &s_GlassShrapC[0]},
|
|
};
|
|
|
|
//
|
|
// Wood
|
|
//
|
|
|
|
#define WOOD_SHRAP_A 3924
|
|
#define WoodShrapA_RATE 12
|
|
|
|
STATE s_WoodShrapA[] =
|
|
{
|
|
{WOOD_SHRAP_A + 0, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[1]},
|
|
{WOOD_SHRAP_A + 1, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[2]},
|
|
{WOOD_SHRAP_A + 2, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[3]},
|
|
{WOOD_SHRAP_A + 3, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[4]},
|
|
{WOOD_SHRAP_A + 4, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[5]},
|
|
{WOOD_SHRAP_A + 5, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[6]},
|
|
{WOOD_SHRAP_A + 6, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[7]},
|
|
{WOOD_SHRAP_A + 7, WoodShrapA_RATE, DoShrapJumpFall, &s_WoodShrapA[0]},
|
|
};
|
|
|
|
#define WOOD_SHRAP_B 3932
|
|
#define WoodShrapB_RATE 12
|
|
|
|
STATE s_WoodShrapB[] =
|
|
{
|
|
{WOOD_SHRAP_B + 0, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[1]},
|
|
{WOOD_SHRAP_B + 1, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[2]},
|
|
{WOOD_SHRAP_B + 2, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[3]},
|
|
{WOOD_SHRAP_B + 3, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[4]},
|
|
{WOOD_SHRAP_B + 4, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[5]},
|
|
{WOOD_SHRAP_B + 5, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[6]},
|
|
{WOOD_SHRAP_B + 6, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[7]},
|
|
{WOOD_SHRAP_B + 7, WoodShrapB_RATE, DoShrapJumpFall, &s_WoodShrapB[0]},
|
|
};
|
|
|
|
#define WOOD_SHRAP_C 3941
|
|
#define WoodShrapC_RATE 12
|
|
|
|
STATE s_WoodShrapC[] =
|
|
{
|
|
{WOOD_SHRAP_C + 0, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[1]},
|
|
{WOOD_SHRAP_C + 1, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[2]},
|
|
{WOOD_SHRAP_C + 2, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[3]},
|
|
{WOOD_SHRAP_C + 3, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[4]},
|
|
{WOOD_SHRAP_C + 4, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[5]},
|
|
{WOOD_SHRAP_C + 5, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[6]},
|
|
{WOOD_SHRAP_C + 6, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[7]},
|
|
{WOOD_SHRAP_C + 7, WoodShrapC_RATE, DoShrapJumpFall, &s_WoodShrapC[0]},
|
|
};
|
|
|
|
//
|
|
// Stone
|
|
//
|
|
|
|
#define STONE_SHRAP_A 3840
|
|
#define StoneShrapA_RATE 12
|
|
|
|
STATE s_StoneShrapA[] =
|
|
{
|
|
{STONE_SHRAP_A + 0, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[1]},
|
|
{STONE_SHRAP_A + 1, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[2]},
|
|
{STONE_SHRAP_A + 2, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[3]},
|
|
{STONE_SHRAP_A + 3, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[4]},
|
|
{STONE_SHRAP_A + 4, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[5]},
|
|
{STONE_SHRAP_A + 5, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[6]},
|
|
{STONE_SHRAP_A + 6, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[7]},
|
|
{STONE_SHRAP_A + 7, StoneShrapA_RATE, DoShrapJumpFall, &s_StoneShrapA[0]},
|
|
};
|
|
|
|
#define STONE_SHRAP_B 3848
|
|
#define StoneShrapB_RATE 12
|
|
|
|
STATE s_StoneShrapB[] =
|
|
{
|
|
{STONE_SHRAP_B + 0, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[1]},
|
|
{STONE_SHRAP_B + 1, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[2]},
|
|
{STONE_SHRAP_B + 2, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[3]},
|
|
{STONE_SHRAP_B + 3, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[4]},
|
|
{STONE_SHRAP_B + 4, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[5]},
|
|
{STONE_SHRAP_B + 5, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[6]},
|
|
{STONE_SHRAP_B + 6, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[7]},
|
|
{STONE_SHRAP_B + 7, StoneShrapB_RATE, DoShrapJumpFall, &s_StoneShrapB[0]},
|
|
};
|
|
|
|
#define STONE_SHRAP_C 3856
|
|
#define StoneShrapC_RATE 12
|
|
|
|
STATE s_StoneShrapC[] =
|
|
{
|
|
{STONE_SHRAP_C + 0, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[1]},
|
|
{STONE_SHRAP_C + 1, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[2]},
|
|
{STONE_SHRAP_C + 2, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[3]},
|
|
{STONE_SHRAP_C + 3, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[4]},
|
|
{STONE_SHRAP_C + 4, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[5]},
|
|
{STONE_SHRAP_C + 5, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[6]},
|
|
{STONE_SHRAP_C + 6, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[7]},
|
|
{STONE_SHRAP_C + 7, StoneShrapC_RATE, DoShrapJumpFall, &s_StoneShrapC[0]},
|
|
};
|
|
|
|
//
|
|
// Metal
|
|
//
|
|
|
|
#define METAL_SHRAP_A 3888
|
|
#define MetalShrapA_RATE 12
|
|
|
|
STATE s_MetalShrapA[] =
|
|
{
|
|
{METAL_SHRAP_A + 0, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[1]},
|
|
{METAL_SHRAP_A + 1, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[2]},
|
|
{METAL_SHRAP_A + 2, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[3]},
|
|
{METAL_SHRAP_A + 3, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[4]},
|
|
{METAL_SHRAP_A + 4, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[5]},
|
|
{METAL_SHRAP_A + 5, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[6]},
|
|
{METAL_SHRAP_A + 6, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[7]},
|
|
{METAL_SHRAP_A + 7, MetalShrapA_RATE, DoShrapJumpFall, &s_MetalShrapA[0]},
|
|
};
|
|
|
|
#define METAL_SHRAP_B 3896
|
|
#define MetalShrapB_RATE 12
|
|
|
|
STATE s_MetalShrapB[] =
|
|
{
|
|
{METAL_SHRAP_B + 0, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[1]},
|
|
{METAL_SHRAP_B + 1, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[2]},
|
|
{METAL_SHRAP_B + 2, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[3]},
|
|
{METAL_SHRAP_B + 3, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[4]},
|
|
{METAL_SHRAP_B + 4, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[5]},
|
|
{METAL_SHRAP_B + 5, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[6]},
|
|
{METAL_SHRAP_B + 6, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[7]},
|
|
{METAL_SHRAP_B + 7, MetalShrapB_RATE, DoShrapJumpFall, &s_MetalShrapB[0]},
|
|
};
|
|
|
|
#define METAL_SHRAP_C 3904
|
|
#define MetalShrapC_RATE 12
|
|
|
|
STATE s_MetalShrapC[] =
|
|
{
|
|
{METAL_SHRAP_C + 0, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[1]},
|
|
{METAL_SHRAP_C + 1, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[2]},
|
|
{METAL_SHRAP_C + 2, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[3]},
|
|
{METAL_SHRAP_C + 3, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[4]},
|
|
{METAL_SHRAP_C + 4, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[5]},
|
|
{METAL_SHRAP_C + 5, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[6]},
|
|
{METAL_SHRAP_C + 6, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[7]},
|
|
{METAL_SHRAP_C + 7, MetalShrapC_RATE, DoShrapJumpFall, &s_MetalShrapC[0]},
|
|
};
|
|
|
|
//
|
|
// Paper
|
|
//
|
|
|
|
#define PAPER_SHRAP_A 3924
|
|
#define PaperShrapA_RATE 12
|
|
|
|
STATE s_PaperShrapA[] =
|
|
{
|
|
{PAPER_SHRAP_A + 0, PaperShrapA_RATE, DoShrapJumpFall, &s_PaperShrapA[1]},
|
|
{PAPER_SHRAP_A + 1, PaperShrapA_RATE, DoShrapJumpFall, &s_PaperShrapA[2]},
|
|
{PAPER_SHRAP_A + 2, PaperShrapA_RATE, DoShrapJumpFall, &s_PaperShrapA[3]},
|
|
{PAPER_SHRAP_A + 3, PaperShrapA_RATE, DoShrapJumpFall, &s_PaperShrapA[0]},
|
|
};
|
|
|
|
#define PAPER_SHRAP_B 3932
|
|
#define PaperShrapB_RATE 12
|
|
|
|
STATE s_PaperShrapB[] =
|
|
{
|
|
{PAPER_SHRAP_B + 0, PaperShrapB_RATE, DoShrapJumpFall, &s_PaperShrapB[1]},
|
|
{PAPER_SHRAP_B + 1, PaperShrapB_RATE, DoShrapJumpFall, &s_PaperShrapB[2]},
|
|
{PAPER_SHRAP_B + 2, PaperShrapB_RATE, DoShrapJumpFall, &s_PaperShrapB[3]},
|
|
{PAPER_SHRAP_B + 3, PaperShrapB_RATE, DoShrapJumpFall, &s_PaperShrapB[0]},
|
|
};
|
|
|
|
#define PAPER_SHRAP_C 3941
|
|
#define PaperShrapC_RATE 12
|
|
|
|
STATE s_PaperShrapC[] =
|
|
{
|
|
{PAPER_SHRAP_C + 0, PaperShrapC_RATE, DoShrapJumpFall, &s_PaperShrapC[1]},
|
|
{PAPER_SHRAP_C + 1, PaperShrapC_RATE, DoShrapJumpFall, &s_PaperShrapC[2]},
|
|
{PAPER_SHRAP_C + 2, PaperShrapC_RATE, DoShrapJumpFall, &s_PaperShrapC[3]},
|
|
{PAPER_SHRAP_C + 3, PaperShrapC_RATE, DoShrapJumpFall, &s_PaperShrapC[0]},
|
|
};
|
|
|
|
bool MissileHitMatch(DSWActor* weapActor, int WeaponNum, DSWActor* hitActor)
|
|
{
|
|
if (WeaponNum <= -1)
|
|
{
|
|
ASSERT(weapActor != nullptr);
|
|
WeaponNum = weapActor->user.WeaponNum;
|
|
|
|
// can be hit by SO only
|
|
if (SP_TAG7(hitActor) == 4)
|
|
{
|
|
if ((weapActor->user.Flags2 & SPR2_SO_MISSILE))
|
|
{
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (SP_TAG7(hitActor) == 0)
|
|
{
|
|
switch (WeaponNum)
|
|
{
|
|
case WPN_RAIL:
|
|
case WPN_MICRO:
|
|
case WPN_NAPALM:
|
|
case WPN_ROCKET:
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
return true;
|
|
}
|
|
}
|
|
else if (SP_TAG7(hitActor) == 1)
|
|
{
|
|
switch (WeaponNum)
|
|
{
|
|
case WPN_MICRO:
|
|
case WPN_RAIL:
|
|
case WPN_HOTHEAD:
|
|
case WPN_NAPALM:
|
|
case WPN_ROCKET:
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
return true;
|
|
}
|
|
}
|
|
else if (SP_TAG7(hitActor) == 2)
|
|
{
|
|
switch (WeaponNum)
|
|
{
|
|
case WPN_MICRO:
|
|
case WPN_RAIL:
|
|
case WPN_HOTHEAD:
|
|
case WPN_NAPALM:
|
|
case WPN_ROCKET:
|
|
case WPN_UZI:
|
|
case WPN_SHOTGUN:
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
return true;
|
|
}
|
|
}
|
|
else if (SP_TAG7(hitActor) == 3)
|
|
{
|
|
switch (WeaponNum)
|
|
{
|
|
case WPN_STAR:
|
|
case WPN_SWORD:
|
|
case WPN_FIST:
|
|
case WPN_MICRO:
|
|
case WPN_RAIL:
|
|
case WPN_HOTHEAD:
|
|
case WPN_NAPALM:
|
|
case WPN_ROCKET:
|
|
case WPN_UZI:
|
|
case WPN_SHOTGUN:
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
|
|
#if 0
|
|
WPN_STAR
|
|
WPN_UZI
|
|
WPN_SHOTGUN
|
|
WPN_MICRO
|
|
WPN_GRENADE
|
|
WPN_MINE
|
|
WPN_RAIL
|
|
WPN_HEART
|
|
WPN_HOTHEAD
|
|
WPN_NAPALM
|
|
WPN_RING
|
|
WPN_ROCKET
|
|
WPN_SWORD
|
|
WPN_FIST
|
|
#endif
|
|
}
|
|
|
|
int SpawnShrapX(DSWActor* actor)
|
|
{
|
|
//For shrap that has no Weapon to send over
|
|
SpawnShrap(actor, nullptr);
|
|
return 0;
|
|
}
|
|
|
|
int DoLavaErupt(DSWActor* actor)
|
|
{
|
|
short i,pnum;
|
|
PLAYER* pp;
|
|
bool found = false;
|
|
|
|
if (TEST_BOOL1(actor))
|
|
{
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
pp = Player + pnum;
|
|
if (pp->insector() && (pp->cursector->extra & SECTFX_TRIGGER))
|
|
{
|
|
SWSectIterator it(pp->cursector);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor->spr.statnum == STAT_TRIGGER && SP_TAG7(itActor) == 0 && SP_TAG5(itActor) == 1)
|
|
{
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (found)
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
return 0;
|
|
}
|
|
|
|
if (!(actor->user.Flags & SPR_ACTIVE))
|
|
{
|
|
// inactive
|
|
if ((actor->user.WaitTics -= synctics) <= 0)
|
|
{
|
|
actor->user.Flags |= (SPR_ACTIVE);
|
|
actor->user.Counter = 0;
|
|
actor->user.WaitTics = SP_TAG9(actor) * 120L;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// active
|
|
if ((actor->user.WaitTics -= synctics) <= 0)
|
|
{
|
|
// Stop for this long
|
|
actor->user.Flags &= ~(SPR_ACTIVE);
|
|
actor->user.Counter = 0;
|
|
actor->user.WaitTics = SP_TAG10(actor) * 120L;
|
|
}
|
|
|
|
// Counter controls the volume of lava erupting
|
|
// starts out slow and increases to a max
|
|
actor->user.Counter += synctics;
|
|
if (actor->user.Counter > SP_TAG2(actor))
|
|
actor->user.Counter = SP_TAG2(actor);
|
|
|
|
if ((RANDOM_P2(1024<<6)>>6) < actor->user.Counter)
|
|
{
|
|
switch (SP_TAG3(actor))
|
|
{
|
|
case 0:
|
|
SpawnShrapX(actor);
|
|
break;
|
|
case 1:
|
|
InitVulcanBoulder(actor);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int SpawnShrap(DSWActor* parentActor, DSWActor* secondaryActor, int means, BREAK_INFO* breakinfo)
|
|
{
|
|
/////////////////////////////////////////
|
|
//
|
|
// BREAK shrap types
|
|
//
|
|
/////////////////////////////////////////
|
|
|
|
// Individual shraps can be copied to this and then values can be changed
|
|
static SHRAP CustomShrap[20];
|
|
|
|
static SHRAP CoinShrap[] =
|
|
{
|
|
{s_CoinShrap, COIN_SHRAP, 5, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapA, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapB, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapC, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP GlassShrap[] =
|
|
{
|
|
{s_GlassShrapA, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapB, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapC, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP WoodShrap[] =
|
|
{
|
|
{s_WoodShrapA, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapB, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapC, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP StoneShrap[] =
|
|
{
|
|
{s_StoneShrapA, STONE_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_StoneShrapB, STONE_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_StoneShrapC, STONE_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP PaperShrap[] =
|
|
{
|
|
{s_PaperShrapA, PAPER_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_PaperShrapB, PAPER_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_PaperShrapC, PAPER_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP MetalShrap[] =
|
|
{
|
|
{s_MetalShrapA, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapB, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapC, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP MetalMix[] =
|
|
{
|
|
{s_GlassShrapA, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapB, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapC, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapA, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapB, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapC, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP WoodMix[] =
|
|
{
|
|
{s_WoodShrapA, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapB, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapC, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapA, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapB, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_MetalShrapC, METAL_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP PaperMix[] =
|
|
{
|
|
{s_WoodShrapA, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapB, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_WoodShrapC, WOOD_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_PaperShrapA, PAPER_SHRAP_A, 2, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_PaperShrapB, PAPER_SHRAP_A, 2, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_PaperShrapC, PAPER_SHRAP_A, 2, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP Marbels[] =
|
|
{
|
|
{s_Marbel, MARBEL, 5, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapA, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapB, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{s_GlassShrapC, GLASS_SHRAP_A, 1, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP BloodShrap[] =
|
|
{
|
|
{s_BloodShrap, BLOOD_SHRAP, 8, Z_MID, 200, 600, 100, 500, true, 2048},
|
|
{nullptr},
|
|
};
|
|
#endif
|
|
|
|
////
|
|
// END - BREAK shrap types
|
|
////
|
|
|
|
static SHRAP EMPShrap[] =
|
|
{
|
|
{s_EMPShrap, EMP, 1, Z_MID, 500, 1100, 300, 600, false, 128},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP StdShrap[] =
|
|
{
|
|
{s_GoreHead, GORE_Head, 1, Z_TOP, 400, 700, 20, 40, true, 2048},
|
|
{s_GoreLung, GORE_Lung, 2, Z_TOP, 500, 800, 100, 300, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 300, 500, 100, 150, true, 2048},
|
|
{s_GoreArm, GORE_Arm, 1, Z_MID, 300, 500, 250, 500, true, 2048},
|
|
{s_GoreSkullCap, GORE_SkullCap, 1, Z_TOP, 300, 500, 250, 500, true, 2048},
|
|
{s_FastGoreDrip, GORE_Drip, 8, Z_BOT, 600, 800, 50, 70, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP HeartAttackShrap[] = // fewer gibs because of the plasma fountain sprites
|
|
{
|
|
{s_GoreLung, GORE_Lung, 2, Z_TOP, 500, 1100, 300, 600, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 500, 1100, 300, 500, true, 2048},
|
|
{s_GoreArm, GORE_Arm, 2, Z_MID, 500, 1100, 350, 600, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP SkelGore[] =
|
|
{
|
|
{s_GoreHead, GORE_Head, 1, Z_TOP, 400, 700, 20, 40, true, 2048},
|
|
{s_GoreLung, GORE_Lung, 2, Z_TOP, 500, 800, 100, 300, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 300, 500, 100, 150, true, 2048},
|
|
{s_GoreSkullCap, GORE_SkullCap, 1, Z_TOP, 300, 500, 100, 150, true, 2048},
|
|
{s_GoreArm, GORE_Arm, 1, Z_MID, 300, 500, 250, 500, true, 2048},
|
|
{s_GoreLeg, GORE_Leg, 2, Z_BOT, 200, 400, 250, 500, true, 2048},
|
|
{s_GoreChunkS, GORE_ChunkS, 4, Z_BOT, 200, 400, 250, 400, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP UpperGore[] =
|
|
{
|
|
{s_GoreHead, GORE_Head, 1, Z_TOP, 400, 700, 20, 40, true, 2048},
|
|
{s_GoreLung, GORE_Lung, 2, Z_TOP, 500, 800, 100, 300, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 300, 500, 100, 150, true, 2048},
|
|
{s_GoreSkullCap, GORE_SkullCap, 1, Z_TOP, 300, 500, 100, 150, true, 2048},
|
|
{s_GoreArm, GORE_Arm, 1, Z_MID, 300, 500, 250, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP LowerGore[] =
|
|
{
|
|
{s_GoreLeg, GORE_Leg, 4, Z_BOT, 300, 500, 100, 200, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
#endif
|
|
|
|
static SHRAP SmallGore[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 3, Z_TOP, 600, 800, 50, 70, false, 2048},
|
|
{s_FastGoreDrip, GORE_Drip, 3, Z_BOT, 600, 800, 70, 100, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP FlamingGore[] =
|
|
{
|
|
{s_GoreFlame, GORE_Drip, 2, Z_TOP, 600, 800, 100, 200, false, 2048},
|
|
{s_GoreFlameChunkB, GORE_Drip, 4, Z_MID, 300, 500, 100, 200, false, 2048},
|
|
{s_GoreFlame, GORE_Drip, 2, Z_BOT, 100, 200, 100, 200, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP BoltExpShrap[] =
|
|
{
|
|
{s_GoreFlame, GORE_Drip, 4, Z_MID, 300, 700, 300, 600, true, 2048},
|
|
{s_GoreFlame, GORE_Drip, 4, Z_BOT, 300, 700, 300, 600, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP TracerExpShrap[] =
|
|
{
|
|
{s_TracerShrap, GORE_Drip, 3, Z_MID, 300, 700, 300, 600, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP FireballExpShrap1[] =
|
|
{
|
|
{s_GoreFlame, GORE_Drip, 1, Z_MID, 100, 300, 100, 200, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP FireballExpShrap2[] =
|
|
{
|
|
{s_GoreFlame, GORE_Drip, 2, Z_MID, 100, 300, 100, 200, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP* FireballExpShrap[] =
|
|
{
|
|
FireballExpShrap1,
|
|
FireballExpShrap2
|
|
};
|
|
#endif
|
|
|
|
// state, id, num, zlevel, min_jspeed, max_jspeed, min_vel, max_vel,
|
|
// random_disperse, ang_range;
|
|
static SHRAP ElectroShrap[] =
|
|
{
|
|
{s_ElectroShrap, ELECTRO_SHARD, 12, Z_TOP, 200, 600, 100, 500, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
// state, id, num, zlevel, min_jspeed, max_jspeed, min_vel, max_vel,
|
|
// random_disperse, ang_range;
|
|
static SHRAP LavaShrap1[] =
|
|
{
|
|
{s_GoreFlame, GORE_Drip, 1, Z_TOP, 400, 1400, 100, 400, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP LavaShrap2[] =
|
|
{
|
|
{s_GoreFlameChunkB, GORE_Drip, 1, Z_TOP, 400, 1400, 100, 400, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP *LavaShrapTable[] =
|
|
{
|
|
LavaShrap1,
|
|
LavaShrap2
|
|
};
|
|
|
|
static SHRAP LavaBoulderShrap[] =
|
|
{
|
|
{s_LavaShard, LAVA_SHARD, 16, Z_MID, 400, 900, 200, 600, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP SectorSquishGore[] =
|
|
{
|
|
{s_FastGoreDrip, GORE_Drip, 24, Z_MID, -400, -200, 600, 800, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
#endif
|
|
|
|
|
|
|
|
//
|
|
// PLAYER SHRAP
|
|
//
|
|
|
|
// state, id, num, zlevel, min_jspeed, max_jspeed, min_vel, max_vel,
|
|
// random_disperse, ang_range;
|
|
static SHRAP PlayerGoreFall[] =
|
|
{
|
|
{s_GoreSkullCap,GORE_SkullCap, 1, Z_TOP, 200, 300, 100, 200, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 200, 300, 100, 200, true, 2048},
|
|
{s_GoreLung, GORE_Lung, 1, Z_MID, 200, 300, 100, 200, true, 2048},
|
|
{s_GoreDrip, GORE_Drip, 10, Z_MID, 200, 300, 100, 200, false, 2048},
|
|
{s_GoreArm, GORE_Arm, 1, Z_MID, 200, 300, 100, 200, true, 2048},
|
|
{s_FastGoreDrip, GORE_Drip, 10, Z_BOT, 200, 300, 100, 200, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP PlayerGoreFly[] =
|
|
{
|
|
{s_GoreSkullCap,GORE_SkullCap, 1, Z_TOP, 500, 1100, 300, 600, true, 2048},
|
|
{s_GoreTorso, GORE_Torso, 1, Z_MID, 500, 1100, 300, 500, true, 2048},
|
|
{s_GoreLiver, GORE_Liver, 1, Z_MID, 200, 300, 100, 200, true, 2048},
|
|
{s_GoreArm, GORE_Arm, 1, Z_MID, 500, 1100, 350, 600, true, 2048},
|
|
{s_FastGoreDrip, GORE_Drip, 16, Z_MID, 500, 1100, 350, 600, false, 2048},
|
|
{s_FastGoreDrip, GORE_Drip, 16, Z_BOT, 500, 1100, 350, 600, false, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP PlayerDeadHead[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 2, Z_TOP, 150, 400, 40, 80, true, 2048},
|
|
{s_GoreDrip, GORE_Drip, 2, Z_MID, 150, 400, 40, 80, true, 2048},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
// state, id, num, zlevel, min_jspeed, max_jspeed, min_vel, max_vel,
|
|
// random_disperse, ang_range;
|
|
static SHRAP PlayerHeadHurl1[] =
|
|
{
|
|
{s_Vomit1, Vomit1, 1, Z_BOT, 250, 400, 100, 200, true, 256},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#define WALL_FLOOR_SHRAP 4097
|
|
ANIMATOR DoShrapWallFloor;
|
|
#if 0
|
|
static SHRAP SectorExpShrap[] =
|
|
{
|
|
{nullptr, WALL_FLOOR_SHRAP, 1, Z_BOT, 550, 800, 200, 400, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
#endif
|
|
|
|
double hZ[3];
|
|
DAngle dangl = nullAngle;
|
|
|
|
SHRAP* p = SmallGore;
|
|
short shrap_shade = -15;
|
|
short shrap_xsize = 48, shrap_ysize = 48;
|
|
short retval = true;
|
|
short shrap_pal = PALETTE_DEFAULT;
|
|
short jump_grav = ACTOR_GRAVITY;
|
|
DAngle start_angl = nullAngle;
|
|
DSWActor* ShrapOwner = nullptr;
|
|
int shrap_bounce = false;
|
|
short WaitTics = 64; // for FastShrap
|
|
short shrap_type;
|
|
int shrap_rand_zamt = 0;
|
|
DAngle shrap_angl = parentActor->spr.angle;
|
|
short shrap_delta_size = 0;
|
|
short shrap_amt = 0;
|
|
|
|
if (Prediction)
|
|
return 0;
|
|
|
|
// Don't spawn shrapnel in invalid sectors gosh dern it!
|
|
if (!parentActor->insector())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (breakinfo)
|
|
{
|
|
shrap_type = breakinfo->shrap_type;
|
|
shrap_amt = breakinfo->shrap_amt;
|
|
goto AutoShrap;
|
|
}
|
|
else if ((parentActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
// if no user
|
|
if (!parentActor->hasU())
|
|
{
|
|
// Jump to shrap type
|
|
shrap_type = SP_TAG8(parentActor);
|
|
goto UserShrap;
|
|
}
|
|
else
|
|
{
|
|
// has a user - is programmed
|
|
change_actor_stat(parentActor, STAT_MISC);
|
|
parentActor->spr.extra &= ~(SPRX_BREAKABLE);
|
|
parentActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
}
|
|
}
|
|
|
|
ASSERT(parentActor->hasU());
|
|
|
|
switch (parentActor->user.ID)
|
|
{
|
|
case ST1:
|
|
switch (parentActor->spr.hitag)
|
|
{
|
|
case SPAWN_SPOT:
|
|
{
|
|
if (parentActor->user.LastDamage)
|
|
shrap_type = SP_TAG3(parentActor);
|
|
else
|
|
shrap_type = SP_TAG6(parentActor);
|
|
|
|
UserShrap:
|
|
|
|
shrap_delta_size = (int8_t)SP_TAG10(parentActor);
|
|
shrap_rand_zamt = SP_TAG9(parentActor);
|
|
// Hey, better limit this in case mappers go crazy, like I did. :)
|
|
// Kills frame rate!
|
|
shrap_amt = SP_TAG8(parentActor);
|
|
if (shrap_amt > 5)
|
|
shrap_amt = 5;
|
|
|
|
AutoShrap:
|
|
|
|
switch (shrap_type)
|
|
{
|
|
case SHRAP_NONE:
|
|
return false;
|
|
|
|
case SHRAP_GLASS:
|
|
PlaySound(DIGI_BREAKGLASS, parentActor, v3df_dontpan | v3df_doppler);
|
|
p = GlassShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, GlassShrap, sizeof(GlassShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_GENERIC:
|
|
case SHRAP_STONE:
|
|
PlaySound(DIGI_BREAKSTONES,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = StoneShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, StoneShrap, sizeof(StoneShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 8 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_WOOD:
|
|
PlaySound(DIGI_BREAKINGWOOD,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = WoodShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, WoodShrap, sizeof(WoodShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_BLOOD:
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
break;
|
|
|
|
case SHRAP_GIBS:
|
|
PlaySound(DIGI_GIBS1,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = SmallGore;
|
|
shrap_xsize = shrap_ysize = 34;
|
|
shrap_bounce = false;
|
|
break;
|
|
|
|
case SHRAP_TREE_BARK:
|
|
PlaySound(DIGI_BREAKINGWOOD,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = WoodShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, WoodShrap, sizeof(WoodShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_PAPER:
|
|
p = PaperShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, PaperShrap, sizeof(PaperShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
break;
|
|
|
|
case SHRAP_METAL:
|
|
PlaySound(DIGI_BREAKMETAL,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = MetalShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, MetalShrap, sizeof(MetalShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
|
|
case SHRAP_COIN:
|
|
PlaySound(DIGI_COINS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = CoinShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, CoinShrap, sizeof(CoinShrap));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_METALMIX:
|
|
PlaySound(DIGI_BREAKMETAL,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = MetalMix;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, MetalMix, sizeof(MetalMix));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_MARBELS:
|
|
{
|
|
p = Marbels;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, Marbels, sizeof(Marbels));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 10 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
}
|
|
break;
|
|
|
|
case SHRAP_WOODMIX:
|
|
PlaySound(DIGI_BREAKINGWOOD,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = WoodMix;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, WoodMix, sizeof(WoodMix));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = true;
|
|
break;
|
|
|
|
case SHRAP_PAPERMIX:
|
|
PlaySound(DIGI_BREAKINGWOOD,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = PaperMix;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, PaperMix, sizeof(PaperMix));
|
|
CustomShrap->num = shrap_amt;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 16 + shrap_delta_size;
|
|
shrap_bounce = false;
|
|
break;
|
|
|
|
case SHRAP_SO_SMOKE:
|
|
return false;
|
|
|
|
case SHRAP_EXPLOSION:
|
|
{
|
|
auto spnum = SpawnLargeExp(parentActor);
|
|
short size = spnum->spr.xrepeat;
|
|
spnum->spr.xrepeat = spnum->spr.yrepeat = size + shrap_delta_size;
|
|
|
|
return false;
|
|
}
|
|
|
|
case SHRAP_LARGE_EXPLOSION:
|
|
{
|
|
auto spnum = SpawnLargeExp(parentActor);
|
|
short size = spnum->spr.xrepeat;
|
|
spnum->spr.xrepeat = spnum->spr.yrepeat = size + shrap_delta_size;
|
|
|
|
InitPhosphorus(spnum);
|
|
|
|
return false;
|
|
}
|
|
|
|
default:
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
p = LavaShrapTable[RANDOM_P2(2<<8)>>8];
|
|
}
|
|
break;
|
|
|
|
case BREAK_BARREL:
|
|
PlaySound(DIGI_BREAKDEBRIS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = WoodShrap;
|
|
shrap_xsize = shrap_ysize = 24;
|
|
shrap_bounce = true;
|
|
ChangeState(parentActor, s_BreakBarrel);
|
|
break;
|
|
case BREAK_LIGHT:
|
|
PlaySound(DIGI_BREAKGLASS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = GlassShrap;
|
|
shrap_xsize = shrap_ysize = 24;
|
|
shrap_bounce = true;
|
|
ChangeState(parentActor, s_BreakLight);
|
|
break;
|
|
case BREAK_PEDISTAL:
|
|
PlaySound(DIGI_BREAKSTONES,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = StoneShrap;
|
|
shrap_xsize = shrap_ysize = 24;
|
|
shrap_bounce = true;
|
|
ChangeState(parentActor, s_BreakPedistal);
|
|
break;
|
|
case BREAK_BOTTLE1:
|
|
PlaySound(DIGI_BREAKGLASS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = GlassShrap;
|
|
shrap_xsize = shrap_ysize = 8;
|
|
shrap_bounce = true;
|
|
ChangeState(parentActor, s_BreakBottle1);
|
|
break;
|
|
case BREAK_BOTTLE2:
|
|
PlaySound(DIGI_BREAKGLASS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = GlassShrap;
|
|
shrap_xsize = shrap_ysize = 8;
|
|
shrap_bounce = true;
|
|
ChangeState(parentActor, s_BreakBottle2);
|
|
break;
|
|
case BREAK_MUSHROOM:
|
|
PlaySound(DIGI_BREAKDEBRIS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = StoneShrap;
|
|
shrap_xsize = shrap_ysize = 4;
|
|
shrap_bounce = true;
|
|
SetSuicide(parentActor); // kill next iteration
|
|
break;
|
|
case BOLT_EXP:
|
|
return false;
|
|
// p = BoltExpShrap;
|
|
// break;
|
|
case TANK_SHELL_EXP:
|
|
return false;
|
|
// p = BoltExpShrap;
|
|
// break;
|
|
case TRACER_EXP:
|
|
return false;
|
|
// p = TracerExpShrap;
|
|
// shrap_xsize = shrap_ysize = 20;
|
|
// WaitTics = 10;
|
|
// break;
|
|
case BOLT_THINMAN_R1:
|
|
p = MetalShrap;
|
|
if (shrap_amt)
|
|
{
|
|
memcpy(CustomShrap, MetalShrap, sizeof(MetalShrap));
|
|
CustomShrap->num = 1;
|
|
p = CustomShrap;
|
|
}
|
|
|
|
shrap_xsize = shrap_ysize = 10;
|
|
break;
|
|
case LAVA_BOULDER:
|
|
PlaySound(DIGI_BREAKSTONES,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = LavaBoulderShrap;
|
|
ShrapOwner = GetOwner(parentActor);
|
|
shrap_xsize = shrap_ysize = 24;
|
|
shrap_bounce = true;
|
|
break;
|
|
case SECTOR_EXP:
|
|
//p = SectorExpShrap;
|
|
//break;
|
|
return false;
|
|
case GRENADE_EXP:
|
|
//p = SectorExpShrap;
|
|
//break;
|
|
return false;
|
|
case FIREBALL_EXP:
|
|
return false;
|
|
// p = FireballExpShrap[RANDOM_P2(2<<8)>>8];
|
|
// shrap_pal = parentActor->user.spal;
|
|
break;
|
|
case ELECTRO_PLAYER:
|
|
case ELECTRO_ENEMY:
|
|
ShrapOwner = GetOwner(parentActor);
|
|
p = ElectroShrap;
|
|
shrap_xsize = shrap_ysize = 20;
|
|
break;
|
|
case COOLIE_RUN_R0:
|
|
if (means == WPN_NM_SECTOR_SQUISH)
|
|
break;
|
|
// return (false);
|
|
break;
|
|
case NINJA_DEAD:
|
|
return false;
|
|
break;
|
|
case NINJA_Head_R0:
|
|
{
|
|
extern STATE* sg_PlayerHeadHurl[];
|
|
|
|
if (parentActor->user.Rot == sg_PlayerHeadHurl)
|
|
{
|
|
p = PlayerHeadHurl1;
|
|
}
|
|
else
|
|
{
|
|
p = PlayerDeadHead;
|
|
shrap_xsize = shrap_ysize = 16+8;
|
|
shrap_bounce = true;
|
|
}
|
|
break;
|
|
}
|
|
case GIRLNINJA_RUN_R0:
|
|
p = StdShrap;
|
|
break;
|
|
case NINJA_RUN_R0:
|
|
{
|
|
p = StdShrap;
|
|
if (parentActor->user.PlayerP)
|
|
{
|
|
PLAYER* pp = parentActor->user.PlayerP;
|
|
|
|
if (pp->DeathType == PLAYER_DEATH_CRUMBLE)
|
|
p = PlayerGoreFall;
|
|
else
|
|
p = PlayerGoreFly;
|
|
}
|
|
break;
|
|
}
|
|
case GORO_RUN_R0:
|
|
p = StdShrap;
|
|
shrap_xsize = shrap_ysize = 64;
|
|
break;
|
|
case COOLG_RUN_R0:
|
|
p = UpperGore;
|
|
break;
|
|
case RIPPER_RUN_R0:
|
|
p = StdShrap;
|
|
if (parentActor->user.spal != 0)
|
|
shrap_xsize = shrap_ysize = 64;
|
|
else
|
|
shrap_xsize = shrap_ysize = 32;
|
|
break;
|
|
case RIPPER2_RUN_R0:
|
|
p = StdShrap;
|
|
if (parentActor->user.spal != 0)
|
|
shrap_xsize = shrap_ysize = 64;
|
|
else
|
|
shrap_xsize = shrap_ysize = 32;
|
|
break;
|
|
case SERP_RUN_R0:
|
|
p = StdShrap;
|
|
//return (false);
|
|
break;
|
|
case SUMO_RUN_R0:
|
|
p = StdShrap;
|
|
break;
|
|
case SKEL_RUN_R0:
|
|
p = SkelGore;
|
|
shrap_pal = PALETTE_SKEL_GORE;
|
|
break;
|
|
case HORNET_RUN_R0:
|
|
p = SmallGore;
|
|
shrap_pal = PALETTE_SKEL_GORE;
|
|
break;
|
|
case SKULL_R0:
|
|
case SKULL_R0 + 1:
|
|
p = FlamingGore;
|
|
break;
|
|
case SKULL_SERP:
|
|
return false;
|
|
case BETTY_R0:
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
PlaySound(DIGI_BREAKGLASS,parentActor,v3df_dontpan|v3df_doppler);
|
|
p = MetalShrap;
|
|
shrap_xsize = shrap_ysize = 10;
|
|
break;
|
|
case ZILLA_RUN_R0:
|
|
p = MetalShrap;
|
|
shrap_xsize = shrap_ysize = 10;
|
|
break;
|
|
case EMP:
|
|
p = EMPShrap;
|
|
shrap_xsize = shrap_ysize = 8;
|
|
shrap_bounce = false;
|
|
break;
|
|
}
|
|
|
|
// second sprite involved
|
|
// most of the time is is the weapon
|
|
if (secondaryActor != nullptr)
|
|
{
|
|
if (secondaryActor->user.PlayerP && secondaryActor->user.PlayerP->sop_control)
|
|
{
|
|
p = StdShrap;
|
|
}
|
|
else
|
|
switch (secondaryActor->user.ID)
|
|
{
|
|
case PLASMA_FOUNTAIN:
|
|
p = HeartAttackShrap;
|
|
break;
|
|
}
|
|
}
|
|
|
|
hZ[Z_TOP] = ActorZOfTop(parentActor); // top
|
|
hZ[Z_BOT] = ActorZOfBottom(parentActor); // bottom
|
|
hZ[Z_MID] = (hZ[0] + hZ[2]) * 0.5; // mid
|
|
|
|
for (; p->id; p++)
|
|
{
|
|
auto ang_range = DAngle::fromBuild(p->ang_range);
|
|
if (!p->random_disperse)
|
|
{
|
|
//dang = (2048 / p->num);
|
|
start_angl = shrap_angl - (ang_range * 0.5);
|
|
dangl = (ang_range / p->num);
|
|
}
|
|
|
|
for (int i = 0; i < p->num; i++)
|
|
{
|
|
auto actor = SpawnActor(STAT_SKIP4, p->id, p->state, parentActor->sector(),
|
|
DVector3(parentActor->spr.pos.XY(), hZ[p->zlevel]), shrap_angl, 512);
|
|
|
|
if (p->random_disperse)
|
|
{
|
|
actor->spr.angle = shrap_angl + DAngle::fromBuild(RANDOM_P2(p->ang_range << 5) >> 5) - (ang_range * 0.5);
|
|
}
|
|
else
|
|
{
|
|
actor->spr.angle = start_angl + (dangl * i);
|
|
}
|
|
|
|
// for FastShrap
|
|
int vz = abs(actor->user.jump_speed * 4) - RandomRange(abs(actor->user.jump_speed) * 8) * 2;
|
|
actor->user.change.Z = vz * zmaptoworld;
|
|
actor->user.WaitTics = WaitTics + RandomRange(WaitTics/2);
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case GORE_Drip:
|
|
shrap_bounce = false;
|
|
break;
|
|
case GORE_Lung:
|
|
shrap_xsize = 20;
|
|
shrap_ysize = 20;
|
|
shrap_bounce = false;
|
|
break;
|
|
case GORE_Liver:
|
|
shrap_xsize = 20;
|
|
shrap_ysize = 20;
|
|
shrap_bounce = false;
|
|
break;
|
|
case GORE_SkullCap:
|
|
shrap_xsize = 24;
|
|
shrap_ysize = 24;
|
|
shrap_bounce = true;
|
|
break;
|
|
case GORE_Arm:
|
|
shrap_xsize = 21;
|
|
shrap_ysize = 21;
|
|
shrap_bounce = false;
|
|
break;
|
|
case GORE_Head:
|
|
shrap_xsize = 26;
|
|
shrap_ysize = 30;
|
|
shrap_bounce = true;
|
|
break;
|
|
case Vomit1:
|
|
shrap_bounce = false;
|
|
actor->spr.pos.Z -= 4;
|
|
shrap_xsize = 12 + (RANDOM_P2(32<<8)>>8);
|
|
shrap_ysize = 12 + (RANDOM_P2(32<<8)>>8);
|
|
actor->user.pos.X = shrap_xsize; // notreallypos
|
|
actor->user.pos.Y = shrap_ysize;
|
|
actor->user.Counter = (RANDOM_P2(2048<<5)>>5);
|
|
|
|
move_missile(actor, DVector3(actor->spr.angle.ToVector() * 16, 0), 8, 8, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (RANDOM_P2(1024)<700)
|
|
actor->user.ID = 0;
|
|
|
|
break;
|
|
case EMP:
|
|
shrap_bounce = false;
|
|
actor->spr.pos.Z -= 4;
|
|
shrap_xsize = 5 + (RANDOM_P2(4<<8)>>8);
|
|
shrap_ysize = 5 + (RANDOM_P2(4<<8)>>8);
|
|
actor->user.pos.X = shrap_xsize; // notreallypos
|
|
actor->user.pos.Y = shrap_ysize;
|
|
break;
|
|
}
|
|
|
|
actor->spr.shade = int8_t(shrap_shade);
|
|
actor->spr.xrepeat = uint8_t(shrap_xsize);
|
|
actor->spr.yrepeat = uint8_t(shrap_ysize);
|
|
actor->spr.clipdist = 16 >> 2;
|
|
|
|
if (ShrapOwner != nullptr)
|
|
{
|
|
SetOwner(ShrapOwner, actor);
|
|
}
|
|
|
|
if (shrap_rand_zamt)
|
|
{
|
|
actor->spr.pos.Z += RandomRange(shrap_rand_zamt) - (shrap_rand_zamt/2);
|
|
}
|
|
|
|
actor->spr.pal = actor->user.spal = uint8_t(shrap_pal);
|
|
|
|
actor->set_int_xvel(p->min_vel*2);
|
|
actor->add_int_xvel( RandomRange(p->max_vel - p->min_vel));
|
|
|
|
actor->user.floor_dist = 2;
|
|
actor->user.ceiling_dist = 2;
|
|
actor->user.jump_speed = p->min_jspeed;
|
|
actor->user.jump_speed += RandomRange(p->max_jspeed - p->min_jspeed);
|
|
actor->user.jump_speed = -actor->user.jump_speed;
|
|
|
|
DoBeginJump(actor);
|
|
actor->user.jump_grav = jump_grav;
|
|
|
|
UpdateChangeXY(actor);
|
|
|
|
if (!shrap_bounce)
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
}
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
void DoShrapMove(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, DVector3(actor->user.change.XY(), 0), actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS*2);
|
|
}
|
|
|
|
int DoVomit(DSWActor* actor)
|
|
{
|
|
actor->user.Counter = NORM_ANGLE(actor->user.Counter + (30*MISSILEMOVETICS));
|
|
// notreallypos
|
|
actor->spr.xrepeat = int(actor->user.pos.X) + MulScale(12, bcos(actor->user.Counter), 14);
|
|
actor->spr.yrepeat = int(actor->user.pos.Y) + MulScale(12, bsin(actor->user.Counter), 14);
|
|
if (actor->user.Flags & (SPR_JUMPING))
|
|
{
|
|
DoJump(actor);
|
|
DoJump(actor);
|
|
DoShrapMove(actor);
|
|
}
|
|
else if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
DoFall(actor);
|
|
DoFall(actor);
|
|
DoShrapMove(actor);
|
|
}
|
|
else
|
|
{
|
|
ChangeState(actor, s_VomitSplash);
|
|
DoFindGroundPoint(actor);
|
|
MissileWaterAdjust(actor);
|
|
actor->spr.pos.Z = actor->user.loz;
|
|
actor->user.WaitTics = 60;
|
|
// notreallypos
|
|
actor->user.pos.X = actor->spr.xrepeat;
|
|
actor->user.pos.Y = actor->spr.yrepeat;
|
|
return 0;
|
|
}
|
|
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
if ((actor->user.coll.actor()->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
DoDamage(actor->user.coll.actor(), actor);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int DoVomitSplash(DSWActor* actor)
|
|
{
|
|
if ((actor->user.WaitTics-=MISSILEMOVETICS) < 0)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DoFastShrapJumpFall(DSWActor* actor)
|
|
{
|
|
actor->spr.pos += actor->user.change * 2;
|
|
actor->user.WaitTics -= MISSILEMOVETICS;
|
|
if (actor->user.WaitTics <= 0)
|
|
KillActor(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoTracerShrap(DSWActor* actor)
|
|
{
|
|
actor->spr.pos += actor->user.change;
|
|
|
|
actor->user.WaitTics -= MISSILEMOVETICS;
|
|
if (actor->user.WaitTics <= 0)
|
|
KillActor(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoShrapJumpFall(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_JUMPING))
|
|
{
|
|
DoShrapVelocity(actor);
|
|
}
|
|
else if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
DoShrapVelocity(actor);
|
|
}
|
|
else
|
|
{
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
DoShrapVelocity(actor);
|
|
return 0;
|
|
}
|
|
|
|
if (actor->user.ID == GORE_Drip)
|
|
ChangeState(actor, s_GoreFloorSplash);
|
|
else
|
|
ShrapKillSprite(actor);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DoShrapDamage(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_JUMPING))
|
|
{
|
|
DoJump(actor);
|
|
DoShrapMove(actor);
|
|
}
|
|
else if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
DoFall(actor);
|
|
DoShrapMove(actor);
|
|
}
|
|
else
|
|
{
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.jump_speed = -300;
|
|
actor->vel.X *= 0.25;
|
|
DoBeginJump(actor);
|
|
return 0;
|
|
}
|
|
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
WeaponMoveHit(actor);
|
|
KillActor(actor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int SpawnBlood(DSWActor* actor, DSWActor* weapActor, DAngle hit_angle, const DVector3* hit_pos)
|
|
{
|
|
DVector3 hitpos;
|
|
int i;
|
|
|
|
if (hit_pos) hitpos = *hit_pos;
|
|
else hitpos = {};
|
|
|
|
// state, id, num, zlevel, min_jspeed, max_jspeed, min_vel, max_vel,
|
|
// random_disperse, ang_range;
|
|
|
|
static SHRAP UziBlood[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 1, Z_MID, 100, 250, 10, 20, true, 512}, // 70,200 vels
|
|
//{s_GoreSplash, PLASMA_Drip, 1, Z_BOT, 0, 0, 0, 0, false, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP SmallBlood[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 1, Z_TOP, 100, 250, 10, 20, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP PlasmaFountainBlood[] =
|
|
{
|
|
{s_PlasmaDrip, PLASMA_Drip, 1, Z_TOP, 200, 500, 100, 300, true, 16},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP SomeBlood[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 1, Z_TOP, 100, 250, 10, 20, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP MoreBlood[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 2, Z_TOP, 100, 250, 10, 20, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
#endif
|
|
|
|
static SHRAP ExtraBlood[] =
|
|
{
|
|
{s_GoreDrip, GORE_Drip, 4, Z_TOP, 100, 250, 10, 20, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
static SHRAP HariKariBlood[] =
|
|
{
|
|
{s_FastGoreDrip, GORE_Drip, 32, Z_TOP, 200, 650, 70, 100, true, 1024},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
|
|
#if 0
|
|
static SHRAP SwordPowerup[] =
|
|
{
|
|
{s_ElectroShrap, ELECTRO_SHARD, 16, Z_TOP, 75, 200, 70, 150, true, 512},
|
|
{nullptr,0,0,0,0,0,0,0,0,0},
|
|
};
|
|
#endif
|
|
|
|
DAngle dangl = nullAngle;
|
|
|
|
SHRAP* p = UziBlood;
|
|
short shrap_shade = -15;
|
|
short shrap_xsize = 20, shrap_ysize = 20;
|
|
short retval = true;
|
|
short shrap_pal = PALETTE_DEFAULT;
|
|
DAngle start_angle = nullAngle;
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
case ZILLA_RUN_R0:
|
|
return 0; // Don't put blood on trashcan
|
|
}
|
|
|
|
// second sprite involved
|
|
// most of the time is the weapon
|
|
if (weapActor != nullptr)
|
|
{
|
|
switch (weapActor->user.ID)
|
|
{
|
|
case NINJA_RUN_R0: //sword
|
|
// if sprite and weapon are the same it must be HARIII-KARIII
|
|
if (actor == weapActor)
|
|
{
|
|
p = HariKariBlood;
|
|
hit_angle = actor->spr.angle;
|
|
hitpos = ActorVectOfTop(actor).plusZ(ActorSizeZ(actor) * (1./16.));
|
|
}
|
|
else
|
|
{
|
|
p = ExtraBlood;
|
|
hit_angle = weapActor->spr.angle + DAngle180;
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = ActorZOfTop(weapActor) + (ActorSizeZ(weapActor) * 0.25);
|
|
}
|
|
break;
|
|
case SERP_RUN_R0:
|
|
p = ExtraBlood;
|
|
hit_angle = weapActor->spr.angle + DAngle180;
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25);
|
|
break;
|
|
case BLADE1:
|
|
case BLADE2:
|
|
case BLADE3:
|
|
case 5011:
|
|
p = SmallBlood;
|
|
hit_angle = DAngle::fromBuild(AngToSprite(actor, weapActor) + 1024);
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = weapActor->spr.pos.Z + (ActorSizeZ(weapActor) * 0.5);
|
|
break;
|
|
case STAR1:
|
|
case CROSSBOLT:
|
|
p = SomeBlood;
|
|
hit_angle = weapActor->spr.angle + DAngle180;
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = weapActor->spr.pos.Z;
|
|
break;
|
|
case PLASMA_FOUNTAIN:
|
|
p = PlasmaFountainBlood;
|
|
hit_angle = weapActor->spr.angle;
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25);
|
|
break;
|
|
default:
|
|
p = SomeBlood;
|
|
hit_angle = weapActor->spr.angle + DAngle180;
|
|
hitpos.XY() = actor->spr.pos.XY();
|
|
hitpos.Z = ActorZOfTop(weapActor) + (ActorSizeZ(weapActor) * 0.25);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hit_angle += DAngle180;
|
|
}
|
|
|
|
for (; p->state; p++)
|
|
{
|
|
auto ang_range = DAngle::fromBuild(p->ang_range);
|
|
if (!p->random_disperse)
|
|
{
|
|
start_angle = hit_angle - (ang_range * 0.5) + DAngle180;
|
|
dangl = (ang_range / p->num);
|
|
}
|
|
|
|
for (i = 0; i < p->num; i++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_SKIP4, p->id, p->state, actor->sector(), hitpos, hit_angle, 0);
|
|
|
|
switch (actorNew->user.ID)
|
|
{
|
|
case ELECTRO_SHARD:
|
|
shrap_xsize = 7;
|
|
shrap_ysize = 7;
|
|
break;
|
|
|
|
case PLASMA_Drip:
|
|
// Don't do central blood splats for every hitscan
|
|
if (RANDOM_P2(1024) < 950)
|
|
{
|
|
actorNew->spr.xrepeat = actorNew->spr.yrepeat = 0;
|
|
}
|
|
if (RANDOM_P2(1024) < 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
//shrap_xsize = 96;
|
|
//shrap_ysize = 75;
|
|
//shrap_xsize = 10;
|
|
//shrap_ysize = 10;
|
|
break;
|
|
}
|
|
|
|
if (p->random_disperse)
|
|
{
|
|
actorNew->spr.angle = hit_angle + DAngle::fromBuild((RANDOM_P2(p->ang_range<<5)>>5) - (p->ang_range >> 1));
|
|
}
|
|
else
|
|
{
|
|
actorNew->spr.angle = start_angle + (dangl * i);
|
|
}
|
|
|
|
actorNew->user.Flags |= (SPR_BOUNCE);
|
|
|
|
actorNew->spr.shade = int8_t(shrap_shade);
|
|
actorNew->spr.xrepeat = uint8_t(shrap_xsize);
|
|
actorNew->spr.yrepeat = uint8_t(shrap_ysize);
|
|
actorNew->spr.clipdist = 16 >> 2;
|
|
|
|
actorNew->spr.pal = actorNew->user.spal = uint8_t(shrap_pal);
|
|
|
|
actorNew->set_int_xvel(p->min_vel);
|
|
actorNew->add_int_xvel( RandomRange(p->max_vel - p->min_vel));
|
|
|
|
// special case
|
|
// blood coming off of actors should have the acceleration of the actor
|
|
// so add it in
|
|
actorNew->add_int_xvel( actor->int_xvel());
|
|
|
|
actorNew->user.ceiling_dist = actorNew->user.floor_dist = 2;
|
|
actorNew->user.jump_speed = p->min_jspeed;
|
|
actorNew->user.jump_speed += RandomRange(p->max_jspeed - p->min_jspeed);
|
|
actorNew->user.jump_speed = -actorNew->user.jump_speed;
|
|
|
|
UpdateChangeXY(actorNew);
|
|
|
|
// for FastShrap
|
|
actorNew->user.set_int_change_z(abs(actorNew->user.jump_speed*4) - RandomRange(abs(actorNew->user.jump_speed)*8));
|
|
actorNew->user.WaitTics = 64 + RANDOM_P2(32);
|
|
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
|
|
DoBeginJump(actorNew);
|
|
}
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
|
|
bool VehicleMoveHit(DSWActor* actor)
|
|
{
|
|
SECTOR_OBJECT* sop;
|
|
SECTOR_OBJECT* hsop;
|
|
bool TestKillSectorObject(SECTOR_OBJECT*);
|
|
|
|
if (actor->user.coll.type == kHitNone)
|
|
return false;
|
|
|
|
sop = actor->user.sop_parent;
|
|
|
|
// sprite controlling sop
|
|
DSWActor* ctrlr = sop->controller;
|
|
if (!ctrlr) return false;
|
|
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitSector:
|
|
{
|
|
sectortype* sectp = actor->user.coll.hitSector;
|
|
|
|
if ((sectp->extra & SECTFX_SECTOR_OBJECT))
|
|
{
|
|
// shouldn't ever really happen
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
case kHitSprite:
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hitActor, actor->user.ID);
|
|
return true;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
if (hitActor != GetOwner(ctrlr))
|
|
{
|
|
DoDamage(hitActor, ctrlr);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (hitActor->spr.statnum == STAT_MINE_STUCK)
|
|
{
|
|
DoDamage(hitActor, actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
auto wph = actor->user.coll.hitWall;
|
|
|
|
if ((wph->extra & WALLFX_SECTOR_OBJECT))
|
|
{
|
|
// sector object collision
|
|
if ((hsop = DetectSectorObjectByWall(wph)))
|
|
{
|
|
SopDamage(hsop, sop->ram_damage);
|
|
if (hsop->max_damage <= 0)
|
|
SopCheckKill(hsop);
|
|
else
|
|
DoSpawnSpotsForDamage(hsop->match_event);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WeaponMoveHit(DSWActor* actor)
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
default:
|
|
break;
|
|
|
|
case kHitVoid:
|
|
SetSuicide(actor);
|
|
return true;
|
|
|
|
case kHitSector:
|
|
{
|
|
sectortype* sectp;
|
|
SECTOR_OBJECT* sop;
|
|
|
|
sectp = actor->user.coll.hitSector;
|
|
|
|
ASSERT(sectp->extra != -1);
|
|
|
|
// hit floor - closer to floor than ceiling
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
// hit a floor sprite
|
|
if (actor->user.lowActor)
|
|
{
|
|
|
|
if (actor->user.lowActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(actor, -1, actor->user.lowActor))
|
|
return true;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if (sectp->hasU() && FixedToInt(sectp->depth_fixed) > 0)
|
|
{
|
|
SpawnSplash(actor);
|
|
return true;
|
|
}
|
|
|
|
}
|
|
// hit ceiling
|
|
else
|
|
{
|
|
// hit a floor sprite
|
|
if (actor->user.highActor)
|
|
{
|
|
if (actor->user.highActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(actor, -1, actor->user.highActor))
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if ((sectp->extra & SECTFX_SECTOR_OBJECT))
|
|
{
|
|
if ((sop = DetectSectorObject(sectp)))
|
|
{
|
|
DoDamage(sop->sp_child, actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if ((sectp->ceilingstat & CSTAT_SECTOR_SKY) && sectp->ceilingpicnum != FAF_MIRROR_PIC)
|
|
{
|
|
if (abs(actor->spr.pos.Z - sectp->ceilingz) < ActorSizeZ(actor))
|
|
{
|
|
SetSuicide(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
case kHitSprite:
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
ASSERT(hitActor->spr.extra != -1);
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hitActor, actor->user.ID);
|
|
return true;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
// make sure you didn't hit the Owner of the missile
|
|
if (hitActor != GetOwner(actor))
|
|
{
|
|
if (actor->user.ID == STAR1)
|
|
{
|
|
extern STATE s_TrashCanPain[];
|
|
switch (hitActor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
PlaySound(DIGI_TRASHLID, actor, v3df_none);
|
|
PlaySound(DIGI_STARCLINK, actor, v3df_none);
|
|
if (hitActor->user.WaitTics <= 0)
|
|
{
|
|
hitActor->user.WaitTics = SEC(2);
|
|
ChangeState(hitActor,s_TrashCanPain);
|
|
}
|
|
break;
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case ZILLA_RUN_R0:
|
|
case 623:
|
|
{
|
|
PlaySound(DIGI_STARCLINK, actor, v3df_none);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
DoDamage(hitActor, actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (hitActor->spr.statnum == STAT_MINE_STUCK)
|
|
{
|
|
DoDamage(hitActor, actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(actor, -1, hitActor))
|
|
return true;
|
|
}
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
if (hitActor->spr.lotag || hitActor->spr.hitag)
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
auto wph = actor->user.coll.hitWall;
|
|
SECTOR_OBJECT* sop;
|
|
|
|
ASSERT(wph->extra != -1);
|
|
|
|
if ((wph->extra & WALLFX_SECTOR_OBJECT))
|
|
{
|
|
if ((sop = DetectSectorObjectByWall(wph)))
|
|
{
|
|
if (sop->max_damage != -999)
|
|
DoDamage(sop->sp_child, actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (wph->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(wph, actor->spr.pos, actor->spr.angle, actor->user.ID);
|
|
actor->user.coll.setNone();
|
|
return true;
|
|
}
|
|
|
|
// clipmove does not correctly return the sprite for WALL sprites
|
|
// on walls, so look with hitscan
|
|
|
|
HitInfo hit{};
|
|
hitscan(actor->int_pos(), actor->sector(), { bcos(actor->int_ang()), bsin(actor->int_ang()), actor->int_zvel()}, hit, CLIPMASK_MISSILE);
|
|
|
|
if (!hit.hitSector)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (hit.actor())
|
|
{
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(actor, -1, hitActor))
|
|
return true;
|
|
}
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
if (hitActor->spr.lotag || hitActor->spr.hitag)
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoUziSmoke(DSWActor* actor)
|
|
{
|
|
actor->spr.pos.Z -= 0.78125; // !JIM! Make them float up
|
|
return 0;
|
|
}
|
|
|
|
int DoShotgunSmoke(DSWActor* actor)
|
|
{
|
|
actor->spr.pos.Z -= 0.78125; // !JIM! Make them float up
|
|
return 0;
|
|
}
|
|
|
|
int DoMineSpark(DSWActor* actor)
|
|
{
|
|
if (actor->spr.picnum != 0)
|
|
{
|
|
DoDamageTest(actor);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DoFireballFlames(DSWActor* actor)
|
|
{
|
|
bool jumping = false;
|
|
|
|
// if no Owner then stay where you are
|
|
DSWActor* attach = actor->user.attachActor;
|
|
if (attach != nullptr)
|
|
{
|
|
actor->spr.pos = DVector3(attach->spr.pos.XY(), ActorZOfMiddle(attach));
|
|
|
|
if ((attach->spr.extra & SPRX_BURNABLE))
|
|
{
|
|
if ((actor->user.Counter2 & 1) == 0)
|
|
{
|
|
attach->spr.shade++;
|
|
if (attach->spr.shade > 10)
|
|
attach->spr.shade = 10;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.Flags & (SPR_JUMPING))
|
|
{
|
|
DoJump(actor);
|
|
jumping = true;
|
|
}
|
|
else if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
DoFall(actor);
|
|
jumping = true;
|
|
}
|
|
else
|
|
{
|
|
if (actor->sector()->hasU() && FixedToInt(actor->sector()->depth_fixed) > 0)
|
|
{
|
|
if (abs(actor->sector()->floorz - actor->spr.pos.Z) <= 4)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!jumping)
|
|
{
|
|
if ((actor->user.WaitTics += MISSILEMOVETICS) > 4 * 120)
|
|
{
|
|
// shrink and go away
|
|
actor->spr.xrepeat--;
|
|
actor->spr.yrepeat--;
|
|
|
|
if (((int8_t)actor->spr.xrepeat) == 0)
|
|
{
|
|
if (actor->user.attachActor != nullptr)
|
|
{
|
|
actor->user.attachActor->user.flameActor = nullptr;
|
|
actor->user.attachActor->user.Flags2 &= ~SPR2_FLAMEDIE;
|
|
}
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// grow until the right size
|
|
if (actor->spr.xrepeat <= actor->user.Counter)
|
|
{
|
|
actor->spr.xrepeat += 3;
|
|
actor->spr.yrepeat += 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->user.Counter2++;
|
|
if (actor->user.Counter2 > 9)
|
|
{
|
|
actor->user.Counter2 = 0;
|
|
DoFlamesDamageTest(actor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoBreakFlames(DSWActor* actor)
|
|
{
|
|
bool jumping = false;
|
|
|
|
if (actor->user.Flags & (SPR_JUMPING))
|
|
{
|
|
DoJump(actor);
|
|
jumping = true;
|
|
}
|
|
else if (actor->user.Flags & (SPR_FALLING))
|
|
{
|
|
DoFall(actor);
|
|
jumping = true;
|
|
}
|
|
else
|
|
{
|
|
if (actor->sector()->hasU() && FixedToInt(actor->sector()->depth_fixed) > 0)
|
|
{
|
|
if (abs(actor->sector()->floorz - actor->spr.pos.Z) <= 4)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (!jumping)
|
|
{
|
|
if ((actor->user.WaitTics += MISSILEMOVETICS) > 4 * 120)
|
|
{
|
|
// shrink and go away
|
|
actor->spr.xrepeat--;
|
|
actor->spr.yrepeat--;
|
|
|
|
if (((int8_t)actor->spr.xrepeat) == 0)
|
|
{
|
|
if (actor->user.attachActor != nullptr)
|
|
{
|
|
actor->user.attachActor->user.flameActor = nullptr;
|
|
actor->user.attachActor->user.Flags2 &= ~SPR2_FLAMEDIE;
|
|
}
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// grow until the right size
|
|
if (actor->spr.xrepeat <= actor->user.Counter)
|
|
{
|
|
actor->spr.xrepeat += 3;
|
|
actor->spr.yrepeat += 3;
|
|
}
|
|
|
|
if (actor->user.WaitTics + MISSILEMOVETICS > 4 * 120)
|
|
{
|
|
SpawnBreakStaticFlames(actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->user.Counter2++;
|
|
if (actor->user.Counter2 > 9)
|
|
{
|
|
actor->user.Counter2 = 0;
|
|
DoFlamesDamageTest(actor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int SetSuicide(DSWActor* actor)
|
|
{
|
|
if (actor->hasU())
|
|
{
|
|
actor->user.Flags |= (SPR_SUICIDE);
|
|
actor->user.RotNum = 0;
|
|
}
|
|
ChangeState(actor, s_Suicide);
|
|
return 0;
|
|
}
|
|
|
|
int DoActorScale(DSWActor* actor)
|
|
{
|
|
actor->user.scale_speed = 70;
|
|
actor->user.scale_value = actor->spr.xrepeat << 8;
|
|
actor->user.scale_tgt = actor->spr.xrepeat + 25;
|
|
|
|
if (actor->user.scale_tgt > 256)
|
|
{
|
|
actor->user.scale_speed = 0;
|
|
actor->user.scale_tgt = 256;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoRipperGrow(DSWActor* actor)
|
|
{
|
|
actor->user.scale_speed = 70;
|
|
actor->user.scale_value = actor->spr.xrepeat << 8;
|
|
actor->user.scale_tgt = actor->spr.xrepeat + 20;
|
|
|
|
if (actor->user.scale_tgt > 128)
|
|
{
|
|
actor->user.scale_speed = 0;
|
|
actor->user.scale_tgt = 128;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void UpdateSinglePlayKills(DSWActor* actor)
|
|
{
|
|
// single play and coop kill count
|
|
if (gNet.MultiGameType != MULTI_GAME_COMMBAT && actor->hasU())
|
|
{
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
{
|
|
return;
|
|
}
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case COOLIE_RUN_R0:
|
|
case NINJA_RUN_R0:
|
|
case NINJA_CRAWL_R0:
|
|
case GORO_RUN_R0:
|
|
case 1441:
|
|
case COOLG_RUN_R0:
|
|
case EEL_RUN_R0:
|
|
case SUMO_RUN_R0:
|
|
case ZILLA_RUN_R0:
|
|
//case TOILETGIRL_R0:
|
|
//case WASHGIRL_R0:
|
|
//case BUNNY_RUN_R0:
|
|
case RIPPER_RUN_R0:
|
|
case RIPPER2_RUN_R0:
|
|
case SERP_RUN_R0:
|
|
case LAVA_RUN_R0:
|
|
case SKEL_RUN_R0:
|
|
case HORNET_RUN_R0:
|
|
case GIRLNINJA_RUN_R0:
|
|
case SKULL_R0:
|
|
case BETTY_R0:
|
|
// always give kills to the first player
|
|
Player->Kills++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
int ActorChooseDeath(DSWActor* actor, DSWActor* weapActor)
|
|
{
|
|
if (!actor->hasU()) return false;
|
|
|
|
if (actor->user.Health > 0)
|
|
return false;
|
|
|
|
UpdateSinglePlayKills(actor);
|
|
|
|
if (actor->user.Attrib)
|
|
PlaySpriteSound(actor,attr_die,v3df_follow);
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
case TOILETGIRL_R0:
|
|
case WASHGIRL_R0:
|
|
case CARGIRL_R0:
|
|
case MECHANICGIRL_R0:
|
|
case SAILORGIRL_R0:
|
|
case PRUNEGIRL_R0:
|
|
case TRASHCAN:
|
|
case GORO_RUN_R0:
|
|
case RIPPER2_RUN_R0:
|
|
break;
|
|
case BUNNY_RUN_R0:
|
|
{
|
|
Bunny_Count--; // Bunny died, decrease the population
|
|
}
|
|
break;
|
|
default:
|
|
ActorCoughItem(actor);
|
|
break;
|
|
}
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case BETTY_R0:
|
|
{
|
|
DoBettyBeginDeath(actor);
|
|
break;
|
|
}
|
|
case SKULL_R0:
|
|
{
|
|
DoSkullBeginDeath(actor);
|
|
break;
|
|
}
|
|
case TOILETGIRL_R0:
|
|
case WASHGIRL_R0:
|
|
case CARGIRL_R0:
|
|
case MECHANICGIRL_R0:
|
|
case SAILORGIRL_R0:
|
|
case PRUNEGIRL_R0:
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
{
|
|
if ((actor->user.ID == TOILETGIRL_R0 ||
|
|
actor->user.ID == CARGIRL_R0 || actor->user.ID == MECHANICGIRL_R0 || actor->user.ID == SAILORGIRL_R0 || actor->user.ID == PRUNEGIRL_R0 ||
|
|
actor->user.ID == WASHGIRL_R0) && weapActor->hasU() && weapActor->user.ID == NINJA_RUN_R0 && weapActor->user.PlayerP)
|
|
{
|
|
PLAYER* pp = weapActor->user.PlayerP;
|
|
if (pp && !(pp->Flags & PF_DIVING)) // JBF: added null test
|
|
pp->Bloody = true;
|
|
PlaySound(DIGI_TOILETGIRLSCREAM, actor, v3df_none);
|
|
}
|
|
if (SpawnShrap(actor, weapActor))
|
|
SetSuicide(actor);
|
|
break;
|
|
}
|
|
|
|
// These are player zombies
|
|
case ZOMBIE_RUN_R0:
|
|
InitBloodSpray(actor,true,105);
|
|
InitBloodSpray(actor,true,105);
|
|
if (SpawnShrap(actor, weapActor))
|
|
SetSuicide(actor);
|
|
break;
|
|
|
|
default:
|
|
|
|
switch (weapActor->user.ID)
|
|
{
|
|
case NINJA_RUN_R0: //sword
|
|
if (weapActor->user.PlayerP)
|
|
{
|
|
PLAYER* pp = weapActor->user.PlayerP;
|
|
|
|
if (weapActor->user.WeaponNum == WPN_FIST && StdRandomRange(1000)>500 && pp == Player+myconnectindex)
|
|
{
|
|
int choosesnd = StdRandomRange(6);
|
|
|
|
if (choosesnd == 0)
|
|
PlayerSound(DIGI_KUNGFU, v3df_follow|v3df_dontpan,pp);
|
|
else if (choosesnd == 1)
|
|
PlayerSound(DIGI_PAYINGATTENTION, v3df_follow|v3df_dontpan,pp);
|
|
else if (choosesnd == 2)
|
|
PlayerSound(DIGI_EATTHIS, v3df_follow|v3df_dontpan,pp);
|
|
else if (choosesnd == 3)
|
|
PlayerSound(DIGI_TAUNTAI4, v3df_follow|v3df_dontpan,pp);
|
|
else if (choosesnd == 4)
|
|
PlayerSound(DIGI_TAUNTAI5, v3df_follow|v3df_dontpan,pp);
|
|
else if (choosesnd == 5)
|
|
PlayerSound(DIGI_HOWYOULIKEMOVE, v3df_follow|v3df_dontpan,pp);
|
|
//PlayerSound(TauntAIVocs[choosesnd],&pp->posx,
|
|
// &pp->posy,&pp->posy,v3df_dontpan|v3df_follow,pp);
|
|
}
|
|
else if (weapActor->user.WeaponNum == WPN_SWORD && StdRandomRange(1000)>500 && pp == Player+myconnectindex)
|
|
{
|
|
short choose_snd;
|
|
|
|
choose_snd = StdRandomRange(1000);
|
|
if (choose_snd > 750)
|
|
PlayerSound(DIGI_SWORDGOTU1, v3df_follow|v3df_dontpan,pp);
|
|
else if (choose_snd > 575)
|
|
PlayerSound(DIGI_SWORDGOTU2, v3df_follow|v3df_dontpan,pp);
|
|
else if (choose_snd > 250)
|
|
PlayerSound(DIGI_SWORDGOTU3, v3df_follow|v3df_dontpan,pp);
|
|
else
|
|
PlayerSound(DIGI_CANBEONLYONE, v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
if (!(pp->Flags & PF_DIVING))
|
|
pp->Bloody = true;
|
|
}
|
|
|
|
if (actor->user.WeaponNum == WPN_FIST)
|
|
DoActorDie(actor, weapActor, 0);
|
|
else if (actor->user.ID == NINJA_RUN_R0 || RandomRange(1000) < 500)
|
|
DoActorDie(actor, weapActor, 0);
|
|
else
|
|
{
|
|
// Can't gib bosses!
|
|
if (actor->user.ID == SERP_RUN_R0 || actor->user.ID == SUMO_RUN_R0 || actor->user.ID == ZILLA_RUN_R0)
|
|
{
|
|
DoActorDie(actor, weapActor, 0);
|
|
break;
|
|
}
|
|
|
|
// Gib out the ones you can't cut in half
|
|
// Blood fountains
|
|
InitBloodSpray(actor,true,-1);
|
|
|
|
if (SpawnShrap(actor, weapActor))
|
|
{
|
|
SetSuicide(actor);
|
|
}
|
|
else
|
|
DoActorDie(actor, weapActor, 0);
|
|
|
|
}
|
|
break;
|
|
case BOLT_THINMAN_R0:
|
|
case BOLT_THINMAN_R1:
|
|
case BOLT_THINMAN_R2:
|
|
case NAP_EXP:
|
|
case BOLT_EXP:
|
|
case TANK_SHELL_EXP:
|
|
case SECTOR_EXP:
|
|
case FIREBALL_EXP:
|
|
case GRENADE_EXP:
|
|
case MINE_EXP:
|
|
case SKULL_R0:
|
|
case BETTY_R0:
|
|
#if 0
|
|
if (RANDOM_P2(1024) < 256 && weapActor->user.Radius != NUKE_RADIUS)
|
|
{
|
|
if (weapActor->user.ID == BOLT_THINMAN_R1 && weapActor->user.Radius == RAIL_RADIUS)
|
|
{
|
|
SpawnShrapX(weapActor); // Do rail gun shrap
|
|
}
|
|
DoActorDie(actor, weapActor, 0);
|
|
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
int choosesnd = 0;
|
|
|
|
// For the Nuke, do residual radiation if he gibs
|
|
if (weapActor->user.Radius == NUKE_RADIUS)
|
|
SpawnFireballFlames(actor, nullptr);
|
|
|
|
// Random chance of taunting the AI's here
|
|
if (RandomRange(1000) > 400)
|
|
{
|
|
PLAYER* pp;
|
|
|
|
auto own = GetOwner(weapActor);
|
|
if (own && own->hasU())
|
|
{
|
|
pp = own->user.PlayerP;
|
|
if (pp)
|
|
{
|
|
choosesnd=StdRandomRange(MAX_TAUNTAI<<8)>>8;
|
|
|
|
if (pp == Player+myconnectindex)
|
|
PlayerSound(TauntAIVocs[choosesnd],v3df_dontpan|v3df_follow,pp);
|
|
}
|
|
}
|
|
}
|
|
|
|
// These guys cough items only if gibbed
|
|
if (actor->user.ID == GORO_RUN_R0 || actor->user.ID == RIPPER2_RUN_R0)
|
|
ActorCoughItem(actor);
|
|
|
|
// Blood fountains
|
|
InitBloodSpray(actor,true,-1);
|
|
|
|
// Bosses do not gib
|
|
if (actor->user.ID == SERP_RUN_R0 || actor->user.ID == SUMO_RUN_R0 || actor->user.ID == ZILLA_RUN_R0)
|
|
{
|
|
DoActorDie(actor, weapActor, 0);
|
|
return true;
|
|
}
|
|
|
|
if (SpawnShrap(actor, weapActor))
|
|
{
|
|
SetSuicide(actor);
|
|
}
|
|
else
|
|
DoActorDie(actor, weapActor, 0);
|
|
|
|
}
|
|
|
|
break;
|
|
default:
|
|
DoActorDie(actor, weapActor, 0);
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
return true;
|
|
|
|
}
|
|
|
|
int ActorHealth(DSWActor* actor, short amt)
|
|
{
|
|
extern int FinishAnim;
|
|
|
|
if (actor->user.ID == TRASHCAN && amt > -75)
|
|
{
|
|
actor->user.LastDamage = 100;
|
|
return true;
|
|
}
|
|
|
|
actor->user.Flags |= (SPR_ATTACKED);
|
|
|
|
actor->user.Health += amt;
|
|
|
|
if (actor->user.ID == SERP_RUN_R0 && actor->spr.pal != 16 && (currentLevel->gameflags & LEVEL_SW_DEATHEXIT_SERPENT))
|
|
{
|
|
if (actor->user.Health < actor->user.MaxHealth/2)
|
|
{
|
|
FinishAnim = ANIM_SERP;
|
|
// Hack for Last Warrior which apparently needs to continue with the next level here.
|
|
if (!(currentLevel->gameflags & LEVEL_SW_DEATHEXIT_SERPENT_NEXT))
|
|
{
|
|
ChangeLevel(nullptr, g_nextskill, true);
|
|
return true;
|
|
}
|
|
else FinishTimer = 1;
|
|
}
|
|
}
|
|
|
|
if (actor->user.ID == SUMO_RUN_R0 && (currentLevel->gameflags & LEVEL_SW_DEATHEXIT_SUMO))
|
|
{
|
|
if (actor->user.Health <= 0)
|
|
{
|
|
FinishTimer = 7*120;
|
|
FinishAnim = ANIM_SUMO;
|
|
}
|
|
}
|
|
|
|
if (actor->user.ID == ZILLA_RUN_R0 && (currentLevel->gameflags & LEVEL_SW_DEATHEXIT_ZILLA))
|
|
{
|
|
if (actor->user.Health <= 0)
|
|
//if (actor->user.Health < actor->user.MaxHealth)
|
|
{
|
|
FinishTimer = 15*120;
|
|
FinishAnim = ANIM_ZILLA;
|
|
}
|
|
}
|
|
|
|
if (actor->user.Attrib && RANDOM_P2(1024) > 850)
|
|
PlaySpriteSound(actor,attr_pain,v3df_follow|v3df_dontpan);
|
|
|
|
// keep track of the last damage
|
|
if (amt < 0)
|
|
actor->user.LastDamage = -amt;
|
|
|
|
// Do alternate Death2 if it exists
|
|
if (actor->user.ActorActionSet && actor->user.ActorActionSet->Death2) // JBF: added null check
|
|
{
|
|
#define DEATH2_HEALTH_VALUE 15
|
|
|
|
if (actor->user.Health <= DEATH2_HEALTH_VALUE)
|
|
{
|
|
// If he's dead, possibly choose a special death type
|
|
#if 1 // Problematic code, REMOVED.
|
|
switch (actor->user.ID)
|
|
{
|
|
case NINJA_RUN_R0:
|
|
{
|
|
extern STATE* sg_NinjaGrabThroat[];
|
|
extern STATE* sg_NinjaHariKari[];
|
|
|
|
if (actor->user.Flags2 & (SPR2_DYING)) return true;
|
|
if (actor->user.Flags & (SPR_FALLING | SPR_JUMPING | SPR_CLIMBING)) return true;
|
|
|
|
if (!(actor->user.Flags2 & SPR2_DYING))
|
|
{
|
|
short rnd;
|
|
|
|
rnd = RANDOM_P2(1024<<4)>>4;
|
|
if (rnd < 950)
|
|
return true;
|
|
actor->user.Flags2 |= (SPR2_DYING); // Only let it check this once!
|
|
actor->user.WaitTics = SEC(1) + SEC(RandomRange(2));
|
|
actor->user.Health = 60;
|
|
PlaySound(DIGI_NINJACHOKE, actor, v3df_follow);
|
|
InitPlasmaFountain(nullptr, actor);
|
|
InitBloodSpray(actor,false,105);
|
|
actor->set_int_ang(NORM_ANGLE(getangle(actor->user.targetActor->int_pos().X - actor->int_pos().X, actor->user.targetActor->int_pos().Y - actor->int_pos().Y) + 1024));
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_YFLIP);
|
|
if (sw_ninjahack)
|
|
NewStateGroup(actor, sg_NinjaHariKari);
|
|
else
|
|
NewStateGroup(actor, sg_NinjaGrabThroat);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
int SopDamage(SECTOR_OBJECT* sop, short amt)
|
|
{
|
|
auto actor = sop->sp_child;
|
|
|
|
// does not have damage
|
|
if (sop->max_damage == -9999)
|
|
return false;
|
|
|
|
sop->max_damage += amt;
|
|
|
|
// keep track of the last damage
|
|
if (amt < 0)
|
|
actor->user.LastDamage = -amt;
|
|
|
|
return true;
|
|
}
|
|
|
|
int SopCheckKill(SECTOR_OBJECT* sop)
|
|
{
|
|
bool killed = false;
|
|
|
|
if ((sop->flags & SOBJ_BROKEN))
|
|
return false;
|
|
|
|
// does not have damage
|
|
if (sop->max_damage == -9999)
|
|
return killed;
|
|
|
|
if (sop->max_damage <= 0)
|
|
{
|
|
// returns whether so was collapsed(killed)
|
|
killed = TestKillSectorObject(sop);
|
|
if (!killed)
|
|
{
|
|
VehicleSetSmoke(sop, SpawnVehicleSmoke);
|
|
sop->flags |= (SOBJ_BROKEN);
|
|
}
|
|
}
|
|
|
|
return killed;
|
|
}
|
|
|
|
int ActorPain(DSWActor* actor)
|
|
{
|
|
// uzi/shotgun damages
|
|
switch (actor->user.ID)
|
|
{
|
|
case TOILETGIRL_R0:
|
|
case WASHGIRL_R0:
|
|
case CARGIRL_R0:
|
|
case MECHANICGIRL_R0:
|
|
case SAILORGIRL_R0:
|
|
case PRUNEGIRL_R0:
|
|
actor->user.FlagOwner = 1;
|
|
break;
|
|
}
|
|
|
|
if (RandomRange(1000) < 875 || actor->user.WaitTics > 0)
|
|
return false;
|
|
|
|
if (!(actor->user.Flags & (SPR_JUMPING | SPR_FALLING)))
|
|
{
|
|
if (actor->user.ActorActionSet && actor->user.ActorActionSet->Pain)
|
|
{
|
|
ActorLeaveTrack(actor);
|
|
actor->user.WaitTics = 60;
|
|
NewStateGroup(actor, actor->user.ActorActionSet->Pain);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int ActorPainPlasma(DSWActor* actor)
|
|
{
|
|
if (!(actor->user.Flags & (SPR_JUMPING | SPR_FALLING | SPR_ELECTRO_TOLERANT)))
|
|
{
|
|
if (actor->user.ActorActionSet && actor->user.ActorActionSet->Pain)
|
|
{
|
|
actor->user.WaitTics = PLASMA_FOUNTAIN_TIME;
|
|
NewStateGroup(actor, actor->user.ActorActionSet->Pain);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
actor->user.Vis = PLASMA_FOUNTAIN_TIME;
|
|
InitActorPause(actor);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int ActorStdMissile(DSWActor* actor, DSWActor* weapActor)
|
|
{
|
|
assert(weapActor != nullptr);
|
|
|
|
// Attack the player that is attacking you
|
|
// Only if hes still alive
|
|
auto own = GetOwner(weapActor);
|
|
if (own && own->hasU())
|
|
{
|
|
if (own->user.PlayerP && GetOwner(actor) != own)
|
|
{
|
|
actor->user.targetActor = own;
|
|
}
|
|
}
|
|
|
|
// Reset the weapons target before dying
|
|
DSWActor* goal = weapActor->user.WpnGoalActor;
|
|
if (goal != nullptr)
|
|
{
|
|
// attempt to see if it was killed
|
|
ASSERT(goal->insector());
|
|
if (goal->hasU())
|
|
goal->user.Flags &= ~(SPR_TARGETED);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ActorDamageSlide(DSWActor* actor, int damage, int ang)
|
|
{
|
|
auto angle = DAngle::fromBuild(ang);
|
|
|
|
if (actor->user.Flags & (SPR_CLIMBING))
|
|
return false;
|
|
|
|
damage = abs(damage);
|
|
|
|
if (!damage)
|
|
return false;
|
|
|
|
if (damage <= 10)
|
|
{
|
|
DoActorBeginSlide(actor, angle, 4);
|
|
return true;
|
|
}
|
|
else if (damage <= 20)
|
|
{
|
|
DoActorBeginSlide(actor, angle, 8);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
int slide_vel = (damage * 6) - (actor->user.MaxHealth);
|
|
|
|
if (slide_vel < -1000) slide_vel = -1000;
|
|
|
|
DoActorBeginSlide(actor, angle, slide_vel / 16.);
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
int PlayerDamageSlide(PLAYER* pp, int damage, short ang)
|
|
{
|
|
DAngle angle = DAngle::fromBuild(ang);
|
|
|
|
damage = abs(damage);
|
|
|
|
if (!damage)
|
|
return false;
|
|
|
|
if (damage <= 5)
|
|
{
|
|
//nudge
|
|
//pp->slide_xvect = angle.ToVector() * 0.5;
|
|
//return(true);
|
|
return false;
|
|
}
|
|
else if (damage <= 10)
|
|
{
|
|
//nudge
|
|
pp->slide_vect = angle.ToVector() * 2;
|
|
return true;
|
|
}
|
|
else if (damage <= 20)
|
|
{
|
|
//bigger nudge
|
|
pp->slide_vect = angle.ToVector() * 8;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
pp->slide_vect = angle.ToVector() * damage * 0.75;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
int GetDamage(DSWActor* actor, DSWActor* weapActor, int DamageNdx)
|
|
{
|
|
auto d = &DamageData[DamageNdx];
|
|
|
|
// if ndx does radius
|
|
if (d->radius > 0 && weapActor)
|
|
{
|
|
int dist,a,b,c;
|
|
int damage_per_pixel, damage_force, damage_amt;
|
|
|
|
|
|
DISTANCE(weapActor->spr.pos, actor->spr.pos,dist,a,b,c);
|
|
|
|
// take off the box around the player or else you'll never get
|
|
// the max_damage;
|
|
dist -= ((int)actor->spr.clipdist)<<(2);
|
|
|
|
if (dist < 0) dist = 0;
|
|
|
|
if ((unsigned)dist < d->radius)
|
|
{
|
|
damage_per_pixel = IntToFixed(d->damage_hi)/d->radius;
|
|
|
|
//the closer your distance is to 0 the more damage
|
|
damage_force = (d->radius - dist);
|
|
damage_amt = -FixedToInt(damage_force * damage_per_pixel);
|
|
|
|
//return(damage_amt);
|
|
// formula: damage_amt = 75% + random(25%)
|
|
return (damage_amt >> 1) + (damage_amt >> 2) + RandomRange(damage_amt >> 2);
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return -(d->damage_lo + RandomRange(d->damage_hi - d->damage_lo));
|
|
}
|
|
|
|
|
|
int PlayerCheckDeath(PLAYER* pp, DSWActor* weapActor)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
|
|
// Store off what player was struck by
|
|
pp->HitBy = weapActor;
|
|
|
|
if (actor->user.Health <= 0 && !(pp->Flags & PF_DEAD))
|
|
{
|
|
// pick a death type
|
|
if (actor->user.LastDamage >= PLAYER_DEATH_EXPLODE_DAMMAGE_AMT)
|
|
pp->DeathType = PLAYER_DEATH_EXPLODE;
|
|
else if (actor->user.LastDamage >= PLAYER_DEATH_CRUMBLE_DAMMAGE_AMT)
|
|
pp->DeathType = PLAYER_DEATH_CRUMBLE;
|
|
else
|
|
pp->DeathType = PLAYER_DEATH_FLIP;
|
|
|
|
if (weapActor == nullptr)
|
|
{
|
|
pp->KillerActor = nullptr;
|
|
DoPlayerBeginDie(pp);
|
|
return true;
|
|
}
|
|
|
|
if (weapActor != nullptr && (weapActor->user.ID == RIPPER_RUN_R0 || weapActor->user.ID == RIPPER2_RUN_R0))
|
|
pp->DeathType = PLAYER_DEATH_RIPPER;
|
|
|
|
if (weapActor != nullptr && weapActor->user.ID == CALTROPS)
|
|
pp->DeathType = PLAYER_DEATH_FLIP;
|
|
|
|
if (weapActor != nullptr && weapActor->user.ID == NINJA_RUN_R0 && weapActor->user.PlayerP)
|
|
{
|
|
pp->DeathType = PLAYER_DEATH_FLIP;
|
|
weapActor->user.PlayerP->Bloody = true;
|
|
}
|
|
|
|
// keep track of who killed you for death purposes
|
|
// need to check all Killer variables when an enemy dies
|
|
if (pp->KillerActor == nullptr)
|
|
{
|
|
auto own = GetOwner(weapActor);
|
|
if (own)
|
|
pp->KillerActor = own;
|
|
else if ((weapActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
pp->KillerActor = weapActor;
|
|
}
|
|
|
|
// start the death process
|
|
DoPlayerBeginDie(pp);
|
|
|
|
// for death direction
|
|
actor->user.slide_ang = VecToAngle(actor->spr.pos - weapActor->spr.pos);
|
|
// for death velocity
|
|
actor->user.slide_vel = actor->user.LastDamage * (5 / 16.);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool PlayerTakeDamage(PLAYER* pp, DSWActor* weapActor)
|
|
{
|
|
if (weapActor == nullptr)
|
|
return true;
|
|
|
|
DSWActor* actor = pp->actor;
|
|
|
|
auto weapOwner = GetOwner(weapActor);
|
|
|
|
if (gNet.MultiGameType == MULTI_GAME_NONE)
|
|
{
|
|
// ZOMBIE special case for single play
|
|
if (weapActor->user.ID == ZOMBIE_RUN_R0)
|
|
{
|
|
// if weapons Owner the player
|
|
if (weapOwner == pp->actor)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if (gNet.MultiGameType == MULTI_GAME_COOPERATIVE)
|
|
{
|
|
// everything hurts you
|
|
if (gNet.HurtTeammate)
|
|
return true;
|
|
|
|
// if weapon IS the YOURSELF take damage
|
|
if (weapActor->user.PlayerP == pp)
|
|
return true;
|
|
|
|
// if the weapons Owner is YOURSELF take damage
|
|
if (weapOwner && weapOwner->hasU() && weapOwner->user.PlayerP == pp)
|
|
return true;
|
|
|
|
// if weapon IS the player no damage
|
|
if (weapActor->user.PlayerP)
|
|
return false;
|
|
|
|
// if the weapons Owner is a player
|
|
if (weapOwner && weapOwner->hasU() && weapOwner->user.PlayerP)
|
|
return false;
|
|
}
|
|
else if (gNet.MultiGameType == MULTI_GAME_COMMBAT && gNet.TeamPlay)
|
|
{
|
|
// everything hurts you
|
|
if (gNet.HurtTeammate)
|
|
return true;
|
|
|
|
// if weapon IS the YOURSELF take damage
|
|
if (weapActor->user.PlayerP == pp)
|
|
return true;
|
|
|
|
// if the weapons Owner is YOURSELF take damage
|
|
if (weapOwner && weapOwner->hasU() && weapOwner->user.PlayerP == pp)
|
|
return true;
|
|
|
|
if (weapActor->user.PlayerP)
|
|
{
|
|
// if both on the same team then no damage
|
|
if (weapActor->user.spal == actor->user.spal)
|
|
return false;
|
|
}
|
|
|
|
// if the weapons Owner is a player
|
|
if (weapOwner && weapOwner->hasU() && weapOwner->user.PlayerP)
|
|
{
|
|
// if both on the same team then no damage
|
|
if (weapOwner->user.spal == actor->user.spal)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
int StarBlood(DSWActor* actor, DSWActor* weapActor)
|
|
{
|
|
short blood_num = 1;
|
|
short i;
|
|
|
|
if (actor->user.Health <= 0)
|
|
blood_num = 4;
|
|
|
|
for (i = 0; i < blood_num; i++)
|
|
SpawnBlood(actor, weapActor);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
!AIC KEY - This is where damage is assesed when missiles hit actors and other
|
|
objects.
|
|
|
|
*/
|
|
|
|
// this was done wrong multiple times below, resulting in spurious crashes.
|
|
bool OwnerIs(DSWActor* actor, int pic)
|
|
{
|
|
auto Own = GetOwner(actor);
|
|
if (Own == nullptr || !Own->hasU()) return false;
|
|
return Own->user.ID == pic;
|
|
}
|
|
|
|
|
|
|
|
int DoDamage(DSWActor* actor, DSWActor* weapActor)
|
|
{
|
|
int damage=0;
|
|
|
|
ASSERT(actor->hasU());
|
|
|
|
// don't hit a dead player
|
|
if (actor->user.PlayerP && (actor->user.PlayerP->Flags & PF_DEAD))
|
|
{
|
|
SpawnBlood(actor, weapActor);
|
|
return 0;
|
|
}
|
|
|
|
if (!weapActor || !weapActor->hasU() || (actor->user.Flags & SPR_SUICIDE))
|
|
return 0;
|
|
|
|
if ((weapActor->user.Flags & SPR_SUICIDE))
|
|
return 0;
|
|
|
|
if (actor->user.Attrib && RANDOM_P2(1024) > 850)
|
|
PlaySpriteSound(actor,attr_pain,v3df_follow);
|
|
|
|
if (actor->user.Flags & (SPR_DEAD))
|
|
{
|
|
SpawnBlood(actor, weapActor);
|
|
return 0;
|
|
}
|
|
|
|
|
|
// special case for shooting mines
|
|
if (actor->spr.statnum == STAT_MINE_STUCK)
|
|
{
|
|
SpawnMineExp(actor);
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
// weapon is drivable object manned by player
|
|
if (weapActor->user.PlayerP && weapActor->user.PlayerP->sop)
|
|
{
|
|
switch (weapActor->user.PlayerP->sop->track)
|
|
{
|
|
case SO_VEHICLE:
|
|
damage = -200;
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PLAYER* pp = Player + screenpeek;
|
|
|
|
ActorHealth(actor, damage);
|
|
if (actor->user.Health <= 0)
|
|
{
|
|
int choosesnd=0;
|
|
// Random chance of taunting the AI's here
|
|
if (StdRandomRange(1024) > 512 && pp == Player+myconnectindex)
|
|
{
|
|
choosesnd=RandomRange(MAX_TAUNTAI);
|
|
PlayerSound(TauntAIVocs[choosesnd],v3df_dontpan|v3df_follow,pp);
|
|
}
|
|
SpawnShrap(actor, weapActor);
|
|
SetSuicide(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
// weapon is the actor - no missile used - example swords, axes, etc
|
|
switch (weapActor->user.ID)
|
|
{
|
|
case NINJA_RUN_R0:
|
|
|
|
ASSERT(weapActor->user.PlayerP);
|
|
|
|
if (weapActor->user.WeaponNum == WPN_SWORD)
|
|
damage = GetDamage(actor, weapActor, WPN_SWORD);
|
|
else
|
|
{
|
|
damage = GetDamage(actor, weapActor, WPN_FIST);
|
|
// Add damage for stronger attacks!
|
|
switch (weapActor->user.PlayerP->WpnKungFuMove)
|
|
{
|
|
case 1:
|
|
damage -= 2 - RandomRange(7);
|
|
break;
|
|
case 2:
|
|
damage -= 5 - RandomRange(10);
|
|
break;
|
|
}
|
|
PlaySound(DIGI_CGTHIGHBONE,weapActor,v3df_follow|v3df_dontpan);
|
|
}
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case SKEL_RUN_R0:
|
|
case COOLG_RUN_R0:
|
|
case GORO_RUN_R0:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_SKEL_SLASH);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case HORNET_RUN_R0:
|
|
PlaySound(DIGI_HORNETSTING, actor, v3df_follow|v3df_dontpan);
|
|
damage = GetDamage(actor, weapActor, DMG_HORNET_STING);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case EEL_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_RIPPER_SLASH);
|
|
damage /= 3;
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case RIPPER_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_RIPPER_SLASH);
|
|
damage /= 3; // Little rippers aren't as tough.
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
if (PlayerCheckDeath(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlaySound(DIGI_RIPPERHEARTOUT,actor->user.PlayerP,v3df_dontpan|v3df_doppler);
|
|
|
|
DoRipperRipHeart(weapActor);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case RIPPER2_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_RIPPER_SLASH);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
if (PlayerCheckDeath(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlaySound(DIGI_RIPPERHEARTOUT,actor->user.PlayerP,v3df_dontpan|v3df_doppler);
|
|
|
|
DoRipper2RipHeart(weapActor);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case BUNNY_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_RIPPER_SLASH);
|
|
damage /= 3;
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
if (PlayerCheckDeath(actor->user.PlayerP, weapActor))
|
|
{
|
|
DoBunnyRipHeart(weapActor);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case SERP_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_SERP_SLASH);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage/4, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
SpawnBlood(actor, weapActor);
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case BLADE1:
|
|
case BLADE2:
|
|
case BLADE3:
|
|
case 5011:
|
|
|
|
if (weapActor->user.ID == 5011)
|
|
damage = -(3 + (RandomRange(4<<8)>>8));
|
|
else
|
|
damage = GetDamage(actor, weapActor, DMG_BLADE);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
if ((actor->user.BladeDamageTics -= synctics) < 0)
|
|
{
|
|
actor->user.BladeDamageTics = DAMAGE_BLADE_TIME;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((actor->user.BladeDamageTics -= ACTORMOVETICS) < 0)
|
|
{
|
|
actor->user.BladeDamageTics = DAMAGE_BLADE_TIME;
|
|
ActorHealth(actor, damage);
|
|
}
|
|
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
break;
|
|
|
|
case STAR1:
|
|
case CROSSBOLT:
|
|
damage = GetDamage(actor, weapActor, WPN_STAR);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
// Is the player blocking?
|
|
if (actor->user.PlayerP->WpnKungFuMove == 3)
|
|
damage /= 3;
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
StarBlood(actor, weapActor);
|
|
|
|
weapActor->user.ID = 0;
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case SPEAR_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_SPEAR_TRAP);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
weapActor->user.ID = 0;
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case LAVA_BOULDER:
|
|
damage = GetDamage(actor, weapActor, DMG_LAVA_BOULDER);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
weapActor->user.ID = 0;
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case LAVA_SHARD:
|
|
damage = GetDamage(actor, weapActor, DMG_LAVA_SHARD);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBlood(actor, weapActor);
|
|
|
|
weapActor->user.ID = 0;
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case UZI_SMOKE:
|
|
case UZI_SMOKE+2:
|
|
if (weapActor->user.ID == UZI_SMOKE)
|
|
damage = GetDamage(actor, weapActor, WPN_UZI);
|
|
else
|
|
damage = GetDamage(actor, weapActor, WPN_UZI)/3; // Enemy Uzi, 1/3 damage
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
//PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
PlayerDamageSlide(actor->user.PlayerP, damage/2, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
switch (actor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
case ZILLA_RUN_R0:
|
|
break;
|
|
default:
|
|
if (RandomRange(1000) > 900)
|
|
InitBloodSpray(actor,false,105);
|
|
if (RandomRange(1000) > 900)
|
|
SpawnMidSplash(actor);
|
|
break;
|
|
}
|
|
}
|
|
|
|
//SpawnBlood(actor, weapActor);
|
|
// reset id so no more damage is taken
|
|
weapActor->user.ID = 0;
|
|
break;
|
|
|
|
case SHOTGUN_SMOKE:
|
|
damage = GetDamage(actor, weapActor, WPN_SHOTGUN);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
//SpawnBlood(actor, weapActor);
|
|
switch (actor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
case ZILLA_RUN_R0:
|
|
break;
|
|
default:
|
|
if (RandomRange(1000) > 950)
|
|
SpawnMidSplash(actor);
|
|
break;
|
|
}
|
|
|
|
// reset id so no more damage is taken
|
|
weapActor->user.ID = 0;
|
|
break;
|
|
|
|
case MIRV_METEOR:
|
|
|
|
//damage = -DAMAGE_MIRV_METEOR;
|
|
damage = GetDamage(actor, weapActor, DMG_MIRV_METEOR);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
}
|
|
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case SERP_METEOR:
|
|
|
|
//damage = -DAMAGE_SERP_METEOR;
|
|
damage = GetDamage(actor, weapActor, DMG_SERP_METEOR);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
}
|
|
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case BOLT_THINMAN_R0:
|
|
damage = GetDamage(actor, weapActor, WPN_ROCKET);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
}
|
|
|
|
if (weapActor->user.Radius == NUKE_RADIUS)
|
|
SpawnNuclearExp(weapActor);
|
|
else
|
|
SpawnBoltExp(weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case BOLT_THINMAN_R1:
|
|
//damage = -(2000 + (65 + RandomRange(40))); // -2000 makes armor not count
|
|
damage = -(65 + RandomRange(40));
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
// this is special code to prevent the Zombie from taking out the Bosses to quick
|
|
// if rail gun weapon Owner is not player
|
|
auto own = GetOwner(weapActor);
|
|
if (own && own->hasU() && !own->user.PlayerP)
|
|
{
|
|
// if actor is a boss
|
|
if (actor->user.ID == ZILLA_RUN_R0 || actor->user.ID == SERP_RUN_R0 || actor->user.ID == SUMO_RUN_R0)
|
|
damage /= 2;
|
|
}
|
|
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage>>1, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
weapActor->user.ID = 0; // No more damage
|
|
SpawnTracerExp(weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case BOLT_THINMAN_R2:
|
|
damage = (GetDamage(actor, weapActor, WPN_ROCKET)/2);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
}
|
|
|
|
if (weapActor->user.Radius == NUKE_RADIUS)
|
|
SpawnNuclearExp(weapActor);
|
|
else
|
|
SpawnBoltExp(weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case BOLT_THINMAN_R4:
|
|
damage = GetDamage(actor, weapActor, DMG_GRENADE_EXP);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnBunnyExp(weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case SUMO_RUN_R0:
|
|
damage = GetDamage(actor, weapActor, DMG_FLASHBOMB);
|
|
|
|
damage /= 3;
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case BOLT_EXP:
|
|
damage = GetDamage(actor, weapActor, DMG_BOLT_EXP);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case BLOOD_WORM:
|
|
|
|
// Don't hurt blood worm zombies!
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
break;
|
|
|
|
damage = GetDamage(actor, weapActor, WPN_HEART);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
//PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
if (PlayerCheckDeath(actor->user.PlayerP, weapActor))
|
|
{
|
|
// degrade blood worm life
|
|
weapActor->user.Counter3 += (4*120)/MISSILEMOVETICS;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
// degrade blood worm life
|
|
weapActor->user.Counter3 += (2*120)/MISSILEMOVETICS;
|
|
|
|
break;
|
|
|
|
|
|
case TANK_SHELL_EXP:
|
|
damage = GetDamage(actor, weapActor, DMG_TANK_SHELL_EXP);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case MUSHROOM_CLOUD:
|
|
case GRENADE_EXP:
|
|
if (weapActor->user.Radius == NUKE_RADIUS) // Special Nuke stuff
|
|
damage = (GetDamage(actor, weapActor, DMG_NUCLEAR_EXP));
|
|
else
|
|
damage = GetDamage(actor, weapActor, DMG_GRENADE_EXP);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Don't let it hurt the SUMO
|
|
if (OwnerIs(weapActor, SUMO_RUN_R0)) break;
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case MICRO_EXP:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_MINE_EXP);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
break;
|
|
|
|
case MINE_EXP:
|
|
{
|
|
damage = GetDamage(actor, weapActor, DMG_MINE_EXP);
|
|
if (OwnerIs(weapActor, SERP_RUN_R0))
|
|
{
|
|
damage /= 6;
|
|
}
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto own = GetOwner(weapActor);
|
|
if (own && own->hasU())
|
|
{
|
|
// Don't let serp skulls hurt the Serpent God
|
|
if (OwnerIs(weapActor, SERP_RUN_R0)) break;
|
|
// Don't let it hurt the SUMO
|
|
if (OwnerIs(weapActor, SUMO_RUN_R0)) break;
|
|
}
|
|
if (actor->user.ID == TRASHCAN)
|
|
ActorHealth(actor, -500);
|
|
else
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
// reset id so no more damage is taken
|
|
weapActor->user.ID = 0;
|
|
break;
|
|
}
|
|
#if 0
|
|
case MINE_SHRAP:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_MINE_SHRAP);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->spr.angle);
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->spr.angle);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
// reset id so no more damage is taken
|
|
weapActor->user.ID = 0;
|
|
break;
|
|
#endif
|
|
|
|
case NAP_EXP:
|
|
{
|
|
damage = GetDamage(actor, weapActor, DMG_NAPALM_EXP);
|
|
|
|
// Sumo Nap does less
|
|
if (OwnerIs(weapActor, SUMO_RUN_R0))
|
|
damage /= 4;
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Don't let it hurt the SUMO
|
|
if (OwnerIs(weapActor, SUMO_RUN_R0)) break;
|
|
ActorHealth(actor, damage);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SetSuicide(weapActor);
|
|
break;
|
|
}
|
|
case Vomit1:
|
|
case Vomit2:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_VOMIT);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case COOLG_FIRE:
|
|
damage = GetDamage(actor, weapActor, DMG_COOLG_FIRE);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
// actor->user.ID = 0;
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
// Skull Exp
|
|
case SKULL_R0:
|
|
case BETTY_R0:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_SKULL_EXP);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
if (actor->user.sop_parent->flags & (SOBJ_DIE_HARD))
|
|
break;
|
|
SopDamage(actor->user.sop_parent, damage);
|
|
SopCheckKill(actor->user.sop_parent);
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, AngToPlayer(actor->user.PlayerP, weapActor));
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorDamageSlide(actor, damage, AngToSprite(actor, weapActor));
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
// Serp ring of skull
|
|
case SKULL_SERP:
|
|
//DoSkullBeginDeath(Weapon);
|
|
break;
|
|
|
|
case FIREBALL1:
|
|
{
|
|
damage = GetDamage(actor, weapActor, WPN_HOTHEAD);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
auto own = GetOwner(weapActor);
|
|
if (own && own->hasU()) // For SerpGod Ring
|
|
own->user.Counter--;
|
|
SpawnFireballFlames(weapActor, actor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
}
|
|
case FIREBALL:
|
|
case GORO_FIREBALL:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_GORO_FIREBALL);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
PlayerDamageSlide(actor->user.PlayerP, damage, weapActor->int_ang());
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
if (actor->user.PlayerP->Armor)
|
|
PlaySound(DIGI_ARMORHIT,actor->user.PlayerP,v3df_dontpan|v3df_follow|v3df_doppler);
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorDamageSlide(actor, damage, weapActor->int_ang());
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SpawnGoroFireballExp(weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
|
|
case FIREBALL_FLAMES:
|
|
|
|
damage = -DamageData[DMG_FIREBALL_FLAMES].damage_lo;
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
break;
|
|
|
|
case RADIATION_CLOUD:
|
|
|
|
damage = GetDamage(actor, weapActor, DMG_RADIATION_CLOUD);
|
|
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
PLAYER* pp = actor->user.PlayerP;
|
|
|
|
PlayerSound(DIGI_GASHURT, v3df_dontpan|v3df_follow|v3df_doppler,pp);
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage-1000);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Don't let it hurt the SUMO
|
|
if (OwnerIs(weapActor, SUMO_RUN_R0)) break;
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
// actor->user.ID = 0;
|
|
weapActor->user.ID = 0;
|
|
break;
|
|
|
|
case PLASMA:
|
|
|
|
//damage = GetDamage(actor, weapActor, WPN_HEART);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
//PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
//PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.ID == SKULL_R0 || actor->user.ID == BETTY_R0)
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
SetSuicide(weapActor);
|
|
break;
|
|
}
|
|
else if (actor->user.ID == RIPPER_RUN_R0)
|
|
{
|
|
DoRipperGrow(actor);
|
|
break;
|
|
}
|
|
|
|
ActorPainPlasma(actor);
|
|
}
|
|
|
|
InitPlasmaFountain(weapActor, actor);
|
|
|
|
SetSuicide(weapActor);
|
|
|
|
break;
|
|
|
|
case CALTROPS:
|
|
{
|
|
damage = GetDamage(actor, weapActor, DMG_MINE_SHRAP);
|
|
if (actor->user.sop_parent)
|
|
{
|
|
break;
|
|
}
|
|
else if (actor->user.PlayerP)
|
|
{
|
|
if (PlayerTakeDamage(actor->user.PlayerP, weapActor))
|
|
{
|
|
if (RANDOM_P2(1024<<4)>>4 < 800)
|
|
PlayerSound(DIGI_STEPONCALTROPS, v3df_follow|v3df_dontpan, actor->user.PlayerP);
|
|
PlayerUpdateHealth(actor->user.PlayerP, damage);
|
|
PlayerCheckDeath(actor->user.PlayerP, weapActor);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ActorHealth(actor, damage);
|
|
ActorPain(actor);
|
|
ActorStdMissile(actor, weapActor);
|
|
ActorChooseDeath(actor, weapActor);
|
|
}
|
|
|
|
SetSuicide(weapActor);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If player take alot of damage, make him yell
|
|
if (actor->hasU() && actor->user.PlayerP)
|
|
{
|
|
if (damage <= -40 && RandomRange(1000) > 700)
|
|
PlayerSound(DIGI_SONOFABITCH, v3df_dontpan|v3df_follow, actor->user.PlayerP);
|
|
else if (damage <= -40 && RandomRange(1000) > 700)
|
|
PlayerSound(DIGI_PAINFORWEAK, v3df_dontpan|v3df_follow, actor->user.PlayerP);
|
|
else if (damage <= -10)
|
|
PlayerSound(PlayerPainVocs[RandomRange(MAX_PAIN)], v3df_dontpan|v3df_follow, actor->user.PlayerP);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// Select death text based on ID
|
|
const char *DeathString(DSWActor* actor)
|
|
{
|
|
if (!actor->hasU()) return " ";
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case NINJA_RUN_R0:
|
|
return " ";
|
|
case ZOMBIE_RUN_R0:
|
|
return GStrings("Zombie");
|
|
case BLOOD_WORM:
|
|
return GStrings("Blood Worm");
|
|
case SKEL_RUN_R0:
|
|
return GStrings("Skeletor Priest");
|
|
case COOLG_RUN_R0:
|
|
return GStrings("Coolie Ghost");
|
|
case GORO_RUN_R0:
|
|
return GStrings("Guardian");
|
|
case HORNET_RUN_R0:
|
|
return GStrings("Hornet");
|
|
case RIPPER_RUN_R0:
|
|
return GStrings("Ripper Hatchling");
|
|
case RIPPER2_RUN_R0:
|
|
return GStrings("Ripper");
|
|
case BUNNY_RUN_R0:
|
|
return GStrings("Killer Rabbit");
|
|
case SERP_RUN_R0:
|
|
return GStrings("Serpent god");
|
|
case GIRLNINJA_RUN_R0:
|
|
return GStrings("Girl Ninja");
|
|
case BLADE1:
|
|
case BLADE2:
|
|
case BLADE3:
|
|
case 5011:
|
|
return GStrings("blade");
|
|
case STAR1:
|
|
if (sw_darts) return GStrings("dart");
|
|
else return GStrings("shuriken");
|
|
case CROSSBOLT:
|
|
return GStrings("crossbow bolt");
|
|
case SPEAR_R0:
|
|
return GStrings("spear");
|
|
case LAVA_BOULDER:
|
|
case LAVA_SHARD:
|
|
return GStrings("lava boulder");
|
|
case UZI_SMOKE:
|
|
return GStrings("Uzi");
|
|
case UZI_SMOKE+2:
|
|
return GStrings("Evil Ninja Uzi");
|
|
case SHOTGUN_SMOKE:
|
|
return GStrings("shotgun");
|
|
case MIRV_METEOR:
|
|
case SERP_METEOR:
|
|
return GStrings("meteor");
|
|
case BOLT_THINMAN_R0:
|
|
return GStrings("rocket");
|
|
case BOLT_THINMAN_R1:
|
|
return GStrings("rail gun");
|
|
case BOLT_THINMAN_R2:
|
|
return GStrings("enemy rocket");
|
|
case BOLT_THINMAN_R4: // BunnyRocket
|
|
return GStrings("bunny rocket");
|
|
case BOLT_EXP:
|
|
return GStrings("explosion");
|
|
case TANK_SHELL_EXP:
|
|
return GStrings("tank shell");
|
|
case MUSHROOM_CLOUD:
|
|
return GStrings("nuclear bomb");
|
|
case GRENADE_EXP:
|
|
return GStrings("40mm grenade");
|
|
case MICRO_EXP:
|
|
return GStrings("micro missile");
|
|
case MINE_EXP:
|
|
//case MINE_SHRAP:
|
|
return GStrings("sticky bomb");
|
|
case NAP_EXP:
|
|
return GStrings("napalm");
|
|
case Vomit1:
|
|
case Vomit2:
|
|
return GStrings("vomit");
|
|
case COOLG_FIRE:
|
|
return GStrings("Coolie Ghost phlem");
|
|
case SKULL_R0:
|
|
return GStrings("Accursed Head");
|
|
case BETTY_R0:
|
|
return GStrings("Bouncing Betty");
|
|
case SKULL_SERP:
|
|
return GStrings("Serpent god Protector");
|
|
case FIREBALL1:
|
|
case FIREBALL:
|
|
case GORO_FIREBALL:
|
|
case FIREBALL_FLAMES:
|
|
return GStrings("flames");
|
|
case RADIATION_CLOUD:
|
|
return GStrings("radiation");
|
|
case CALTROPS:
|
|
return GStrings("useitem 7");
|
|
}
|
|
return "";
|
|
}
|
|
|
|
int DoDamageTest(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, tx, ty;
|
|
int tmin;
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
if ((unsigned)dist > actor->user.Radius + itActor->user.Radius)
|
|
continue;
|
|
|
|
if (actor == itActor)
|
|
continue;
|
|
|
|
if (!(itActor->spr.cstat & CSTAT_SPRITE_BLOCK))
|
|
continue;
|
|
|
|
// !JIM! Put in a cansee so that you don't take damage through walls and such
|
|
// For speed's sake, try limiting check only to radius weapons!
|
|
if (actor->user.Radius > 200)
|
|
{
|
|
if (!FAFcansee(ActorUpperVect(itActor), itActor->sector(),actor->spr.pos,actor->sector()))
|
|
continue;
|
|
}
|
|
|
|
if (GetOwner(actor) != itActor && SpriteOverlap(actor, itActor))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void DoHitscanDamage(DSWActor* weaponActor, DSWActor* hitActor)
|
|
{
|
|
if (hitActor == nullptr)
|
|
return;
|
|
|
|
unsigned stat;
|
|
|
|
// this routine needs some sort of sprite generated from the hitscan
|
|
// such as a smoke or spark sprite - reason is because of DoDamage()
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
if (hitActor->spr.statnum == StatDamageList[stat])
|
|
{
|
|
DoDamage(hitActor, weaponActor);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
int DoFlamesDamageTest(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, tx, ty;
|
|
int tmin;
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
switch (itActor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
continue;
|
|
}
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
|
|
if ((unsigned)dist > actor->user.Radius + itActor->user.Radius)
|
|
continue;
|
|
|
|
if (actor == itActor)
|
|
continue;
|
|
|
|
if (!(itActor->spr.cstat & (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN)))
|
|
continue;
|
|
|
|
if (actor->spr.cstat & (CSTAT_SPRITE_INVISIBLE))
|
|
continue;
|
|
|
|
if (actor->user.Radius > 200) // Note: No weaps have bigger radius than 200 cept explosion stuff
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(itActor),itActor->sector(),ActorVectOfMiddle(actor),actor->sector()))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
else if (SpriteOverlap(actor, itActor))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
walltype* PrevWall(walltype* wall_num)
|
|
{
|
|
for(auto&wal : wallsofsector(wall_num->sectorp()))
|
|
{
|
|
if (wal.point2Wall() == wall_num) return &wal;
|
|
}
|
|
return wall_num; // should never happen unless the sector is malformed.
|
|
}
|
|
|
|
|
|
short StatBreakList[] =
|
|
{
|
|
STAT_DEFAULT,
|
|
STAT_BREAKABLE,
|
|
STAT_NO_STATE,
|
|
STAT_DEAD_ACTOR,
|
|
};
|
|
|
|
void TraverseBreakableWalls(sectortype* start_sect, int x, int y, int z, short ang, int radius)
|
|
{
|
|
int k;
|
|
int xmid,ymid;
|
|
int dist;
|
|
int break_count;
|
|
|
|
// limit radius
|
|
if (radius > 2000)
|
|
radius = 2000;
|
|
|
|
break_count = 0;
|
|
|
|
BFSSectorSearch search(start_sect);
|
|
while (auto sect = search.GetNext())
|
|
{
|
|
for(auto& wal : wallsofsector(sect))
|
|
{
|
|
// see if this wall should be broken
|
|
if (wal.lotag == TAG_WALL_BREAK)
|
|
{
|
|
// find midpoint
|
|
xmid = (wal.wall_int_pos().X + wal.point2Wall()->wall_int_pos().X) >> 1;
|
|
ymid = (wal.wall_int_pos().Y + wal.point2Wall()->wall_int_pos().Y) >> 1;
|
|
|
|
// don't need to go further if wall is too far out
|
|
|
|
dist = Distance(xmid, ymid, x, y);
|
|
if (dist > radius)
|
|
continue;
|
|
|
|
sectortype* sectp = nullptr;
|
|
DVector3 hitpos;
|
|
DAngle wall_ang;
|
|
if (WallBreakPosition(&wal, §p, hitpos, wall_ang))
|
|
{
|
|
if (sectp != nullptr && FAFcansee(DVector3(x* inttoworld, y* inttoworld, z* zinttoworld), start_sect, hitpos, sectp))
|
|
{
|
|
HitBreakWall(&wal, DVector3(INT32_MAX, INT32_MAX, INT32_MAX), DAngle::fromBuild(ang), 0);
|
|
|
|
break_count++;
|
|
if (break_count > 4)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (wal.twoSided())
|
|
search.Add(wal.nextSector());
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
int DoExpDamageTest(DSWActor* actor)
|
|
{
|
|
short i, stat;
|
|
int dist, tx, ty;
|
|
int tmin;
|
|
int max_stat;
|
|
short break_count;
|
|
|
|
DSWActor* found_act = nullptr;
|
|
int found_dist = 999999;
|
|
int DoWallMoveMatch(short match);
|
|
|
|
// crack sprites
|
|
if (actor->user.ID != MUSHROOM_CLOUD)
|
|
WeaponExplodeSectorInRange(actor);
|
|
|
|
// Just like DoDamageTest() except that it doesn't care about the Owner
|
|
|
|
max_stat = SIZ(StatDamageList);
|
|
// don't check for mines if the weapon is a mine
|
|
if (actor->spr.statnum == STAT_MINE_STUCK)
|
|
max_stat--;
|
|
|
|
for (stat = 0; stat < max_stat; stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
|
|
if ((unsigned)dist > actor->user.Radius + itActor->user.Radius)
|
|
continue;
|
|
|
|
if (itActor == actor)
|
|
continue;
|
|
|
|
if (StatDamageList[stat] == STAT_SO_SP_CHILD)
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
else
|
|
{
|
|
if ((unsigned)FindDistance3D(itActor->int_pos().X - actor->int_pos().X, itActor->int_pos().Y - actor->int_pos().Y, itActor->int_pos().Z - actor->int_pos().Z) > actor->user.Radius + itActor->user.Radius)
|
|
continue;
|
|
|
|
// added hitscan block because mines no long clip against actors/players
|
|
if (!(itActor->spr.cstat & (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN)))
|
|
continue;
|
|
|
|
// Second parameter MUST have blocking bits set or cansee won't work
|
|
// added second check for FAF water - hitscans were hitting ceiling
|
|
if (!FAFcansee(actor->spr.pos, actor->sector(), ActorUpperVect(itActor), itActor->sector()) &&
|
|
!FAFcansee(actor->spr.pos, actor->sector(), ActorLowerVect(itActor), itActor->sector()))
|
|
continue;
|
|
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (actor->user.ID == MUSHROOM_CLOUD) return 0; // Central Nuke doesn't break stuff
|
|
// Only secondaries do that
|
|
|
|
TraverseBreakableWalls(actor->sector(), actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), actor->user.Radius);
|
|
|
|
break_count = 0;
|
|
max_stat = SIZ(StatBreakList);
|
|
// Breakable stuff
|
|
for (stat = 0; stat < max_stat; stat++)
|
|
{
|
|
SWStatIterator it(StatBreakList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
if ((unsigned)dist > actor->user.Radius)
|
|
continue;
|
|
|
|
dist = (ActorVectOfMiddle(itActor) - actor->spr.pos).Length() * worldtoint;
|
|
if ((unsigned)dist > actor->user.Radius)
|
|
continue;
|
|
|
|
if (!FAFcansee(ActorVectOfMiddle(itActor), itActor->sector(), actor->spr.pos, actor->sector()))
|
|
continue;
|
|
|
|
if ((itActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(itActor, actor->user.ID);
|
|
break_count++;
|
|
if (break_count > 6)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (actor->user.ID == BLOOD_WORM)
|
|
return 0;
|
|
|
|
// wall damaging
|
|
SWStatIterator it(STAT_WALL_MOVE);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
if ((unsigned)dist > actor->user.Radius/4)
|
|
continue;
|
|
|
|
if (TEST_BOOL1(actor))
|
|
continue;
|
|
|
|
if (!CanSeeWallMove(actor, SP_TAG2(actor)))
|
|
continue;
|
|
|
|
if (dist < found_dist)
|
|
{
|
|
found_dist = dist;
|
|
found_act = itActor;
|
|
}
|
|
}
|
|
|
|
if (found_act)
|
|
{
|
|
if (SP_TAG2(found_act) == 0)
|
|
{
|
|
// just do one
|
|
DoWallMove(found_act);
|
|
}
|
|
else
|
|
{
|
|
if (DoWallMoveMatch(SP_TAG2(found_act)))
|
|
{
|
|
DoSpawnSpotsForDamage(SP_TAG2(found_act));
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int DoMineExpMine(DSWActor* actor)
|
|
{
|
|
int i;
|
|
int dist, tx, ty;
|
|
int tmin;
|
|
int zdist;
|
|
|
|
SWStatIterator it(STAT_MINE_STUCK);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
if ((unsigned)dist > actor->user.Radius + itActor->user.Radius)
|
|
continue;
|
|
|
|
if (itActor == actor)
|
|
continue;
|
|
|
|
if (!(itActor->spr.cstat & CSTAT_SPRITE_BLOCK_HITSCAN))
|
|
continue;
|
|
|
|
// Explosions are spherical, not planes, so let's check that way, well cylindrical at least.
|
|
zdist = abs(itActor->int_pos().Z - actor->int_pos().Z)>>4;
|
|
if (SpriteOverlap(actor, itActor) || (unsigned)zdist < actor->user.Radius + itActor->user.Radius)
|
|
{
|
|
DoDamage(itActor, actor);
|
|
// only explode one mine at a time
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoStar(DSWActor* actor)
|
|
{
|
|
const int STAR_STICK_RNUM = 400;
|
|
const int STAR_BOUNCE_RNUM = 600;
|
|
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
actor->user.motion_blur_num = 0;
|
|
ScaleSpriteVector(actor, 54000);
|
|
|
|
auto velsq = actor->user.change.XY().LengthSquared();
|
|
|
|
if (velsq > 6.25 * 6.25)
|
|
{
|
|
if ((RANDOM_P2(1024 << 4) >> 4) < 128)
|
|
SpawnBubble(actor);
|
|
}
|
|
|
|
actor->spr.pos.Z += 0.5 * MISSILEMOVETICS;
|
|
|
|
DoActorZrange(actor);
|
|
MissileWaterAdjust(actor);
|
|
|
|
if (actor->spr.pos.Z > actor->user.loz)
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto velsq = actor->user.change.XY().LengthSquared();
|
|
|
|
|
|
if (velsq < 50 * 50)
|
|
{
|
|
actor->user.Counter += 50;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone && !(actor->user.Flags & SPR_UNDERWATER))
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
default:
|
|
break;
|
|
|
|
case kHitWall:
|
|
{
|
|
short nw,wall_ang;
|
|
walltype* wph;
|
|
|
|
wph = actor->user.coll.hitWall;
|
|
|
|
if (wph->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(wph, actor->spr.pos, actor->spr.angle, actor->user.ID);
|
|
actor->user.coll.setNone();
|
|
break;
|
|
}
|
|
|
|
// special case with MissileSetPos - don't queue star
|
|
// from this routine
|
|
if (actor->user.Flags & (SPR_SET_POS_DONT_KILL))
|
|
break;
|
|
|
|
// chance of sticking
|
|
if (!(actor->user.Flags & SPR_BOUNCE) && RANDOM_P2(1024) < STAR_STICK_RNUM)
|
|
{
|
|
actor->user.motion_blur_num = 0;
|
|
ChangeState(actor, s_StarStuck);
|
|
actor->spr.xrepeat -= 16;
|
|
actor->spr.yrepeat -= 16;
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->spr.clipdist = 16L >> 2;
|
|
actor->user.ceiling_dist = (2);
|
|
actor->user.floor_dist = (2);
|
|
// treat this just like a KillSprite but don't kill
|
|
QueueStar(actor);
|
|
return 0;
|
|
}
|
|
|
|
// chance of bouncing
|
|
if (RANDOM_P2(1024) < STAR_BOUNCE_RNUM)
|
|
break;
|
|
|
|
WallBounce(actor, wph->delta().Angle() + DAngle90);
|
|
ScaleSpriteVector(actor, 36000);
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.motion_blur_num = 0;
|
|
actor->user.coll.setNone();
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
bool did_hit_wall;
|
|
auto hit_sect = actor->user.coll.hitSector;
|
|
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
if (hit_sect->hasU() && FixedToInt(hit_sect->depth_fixed) > 0)
|
|
{
|
|
SpawnSplash(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
// hit water - will be taken care of in WeaponMoveHit
|
|
//break;
|
|
}
|
|
}
|
|
|
|
if (actor->user.lowActor)
|
|
if (actor->user.lowActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
break;
|
|
if (actor->user.highActor)
|
|
if (actor->user.highActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
break;
|
|
|
|
ScaleSpriteVector(actor, 58000);
|
|
|
|
auto velsq = actor->user.change.XY().LengthSquared();
|
|
if (velsq < 31.25 * 31.25)
|
|
break; // will be killed below - actor != 0
|
|
|
|
// 32000 to 96000
|
|
int xscale = 64000 + (RandomRange(64000) - 32000);
|
|
int yscale = 64000 + (RandomRange(64000) - 32000);
|
|
int zscale;
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5)) zscale = 50000; // floor
|
|
else zscale = 40000; // ceiling
|
|
ScaleSpriteVector(actor, xscale, yscale, zscale);
|
|
|
|
if (SlopeBounce(actor, &did_hit_wall))
|
|
{
|
|
if (did_hit_wall)
|
|
{
|
|
// chance of sticking
|
|
if (RANDOM_P2(1024) < STAR_STICK_RNUM)
|
|
break;
|
|
|
|
// chance of bouncing
|
|
if (RANDOM_P2(1024) < STAR_BOUNCE_RNUM)
|
|
break;
|
|
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.motion_blur_num = 0;
|
|
actor->user.coll.setNone();
|
|
|
|
}
|
|
else
|
|
{
|
|
// hit a sloped sector < 45 degrees
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.motion_blur_num = 0;
|
|
actor->user.coll.setNone();
|
|
}
|
|
|
|
// BREAK HERE - LOOOK !!!!!!!!!!!!!!!!!!!!!!!!
|
|
break; // hit a slope
|
|
}
|
|
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.motion_blur_num = 0;
|
|
actor->user.coll.setNone();
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
|
|
// 32000 to 96000
|
|
xscale = 64000 + (RandomRange(64000) - 32000);
|
|
yscale = 64000 + (RandomRange(64000) - 32000);
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5)) zscale = 50000; // floor
|
|
else zscale = 40000; // ceiling
|
|
ScaleSpriteVector(actor, xscale, yscale, zscale);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (actor->user.coll.type == kHitSprite && actor->user.coll.actor()->hasU())
|
|
{
|
|
auto su = actor->user.coll.actor();
|
|
if (su->user.ID == TRASHCAN || su->user.ID == ZILLA_RUN_R0)
|
|
PlaySound(DIGI_STARCLINK, actor, v3df_none);
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitSprite) // Don't clank on sprites
|
|
PlaySound(DIGI_STARCLINK, actor, v3df_none);
|
|
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
int DoCrossBolt(DSWActor* actor)
|
|
{
|
|
DoBlurExtend(actor, 0, 2);
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, 16, 16, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
int DoPlasmaDone(DSWActor* actor)
|
|
{
|
|
actor->spr.xrepeat += actor->user.Counter;
|
|
actor->spr.yrepeat -= 4;
|
|
actor->user.Counter += 2;
|
|
|
|
if (actor->spr.yrepeat < 6)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
DSWActor* PickEnemyTarget(DSWActor* actor, short aware_range)
|
|
{
|
|
TARGET_SORT* ts;
|
|
|
|
DoPickTarget(actor, aware_range, false);
|
|
|
|
for (ts = TargetSort; ts < &TargetSort[TargetSortCount]; ts++)
|
|
{
|
|
if (ts->actor != nullptr)
|
|
{
|
|
if (ts->actor == GetOwner(actor) || GetOwner(ts->actor) == GetOwner(actor))
|
|
continue;
|
|
|
|
return ts->actor;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
int MissileSeek(DSWActor* actor, int16_t delay_tics, int16_t aware_range/*, int16_t dang_shift, int16_t turn_limit, int16_t z_limit*/)
|
|
{
|
|
if (actor->user.WaitTics <= delay_tics)
|
|
actor->user.WaitTics += MISSILEMOVETICS;
|
|
|
|
if (actor->user.WpnGoalActor == nullptr)
|
|
{
|
|
if (actor->user.WaitTics > delay_tics)
|
|
{
|
|
DSWActor* hitActor;
|
|
|
|
if (actor->user.Flags2 & (SPR2_DONT_TARGET_OWNER))
|
|
{
|
|
if ((hitActor = PickEnemyTarget(actor, aware_range)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
else if ((hitActor = DoPickTarget(actor, aware_range, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
}
|
|
|
|
DSWActor* goal = actor->user.WpnGoalActor;
|
|
if (goal != nullptr)
|
|
{
|
|
// move to correct angle
|
|
auto ang2tgt = VecToAngle(goal->spr.pos - actor->spr.pos);
|
|
auto delta_ang = clamp(deltaangle(ang2tgt, actor->spr.angle), -DAngle45 / 8, DAngle45 / 8);
|
|
actor->spr.angle -= delta_ang;
|
|
|
|
double zh = ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25);
|
|
auto vel = clamp((zh - actor->spr.pos.Z)* 0.5, -16., 16.);
|
|
|
|
actor->set_int_zvel(vel * zworldtoint);
|
|
|
|
UpdateChange(actor);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// combination of vector manipulation
|
|
int ComboMissileSeek(DSWActor* actor, int16_t delay_tics, int16_t aware_range/*, int16_t dang_shift, int16_t turn_limit, int16_t z_limit*/)
|
|
{
|
|
if (actor->user.WaitTics <= delay_tics)
|
|
actor->user.WaitTics += MISSILEMOVETICS;
|
|
|
|
if (actor->user.WpnGoalActor == nullptr)
|
|
{
|
|
if (actor->user.WaitTics > delay_tics)
|
|
{
|
|
DSWActor* hitActor;
|
|
|
|
if ((hitActor = DoPickTarget(actor, aware_range, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
}
|
|
|
|
DSWActor* goal = actor->user.WpnGoalActor;
|
|
if (goal != nullptr)
|
|
{
|
|
// move to correct angle
|
|
auto ang2tgt = VecToAngle(goal->spr.pos - actor->spr.pos);
|
|
auto delta_ang = clamp(deltaangle(ang2tgt, actor->spr.angle), -DAngle45 / 8, DAngle45 / 8);
|
|
actor->spr.angle -= delta_ang;
|
|
UpdateChangeXY(actor);
|
|
|
|
double zdiff = actor->spr.pos.Z - ActorZOfTop(goal) - (ActorSizeZ(goal) * 0.25);
|
|
double dist = g_sqrt((actor->spr.pos.XY() - goal->spr.pos.XY()).LengthSquared() + zdiff * zdiff);
|
|
|
|
actor->user.change.Z = (actor->vel.X) * zdiff / dist + actor->user.change.Z * (15 / 16.);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void SetAngleFromChange(DSWActor* actor)
|
|
{
|
|
actor->spr.angle = VecToAngle(actor->user.change);
|
|
}
|
|
|
|
// completely vector manipulation
|
|
int VectorMissileSeek(DSWActor* actor, int16_t delay_tics, int16_t turn_speed, int16_t aware_range1, int16_t aware_range2)
|
|
{
|
|
if (actor->user.WaitTics <= delay_tics)
|
|
actor->user.WaitTics += MISSILEMOVETICS;
|
|
|
|
if (actor->user.WpnGoalActor == nullptr)
|
|
{
|
|
if (actor->user.WaitTics > delay_tics)
|
|
{
|
|
DSWActor* hitActor;
|
|
|
|
if (actor->user.Flags2 & (SPR2_DONT_TARGET_OWNER))
|
|
{
|
|
if ((hitActor = PickEnemyTarget(actor, aware_range1)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
else if ((hitActor = PickEnemyTarget(actor, aware_range2)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((hitActor = DoPickTarget(actor, aware_range1, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
else if ((hitActor = DoPickTarget(actor, aware_range2, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DSWActor* goal = actor->user.WpnGoalActor;
|
|
if (goal != nullptr)
|
|
{
|
|
auto delta = (goal->spr.pos.XY() - actor->spr.pos.XY());
|
|
double zdiff = ActorZOfTop(goal) + (ActorSizeZ(goal) * 0.25) - actor->spr.pos.Z;
|
|
double dist = g_sqrt(delta.LengthSquared() + zdiff * zdiff);
|
|
|
|
auto oc = actor->user.change;
|
|
|
|
auto vel = (actor->int_xvel() * zinttoworld) / dist;
|
|
actor->user.change = DVector3(delta, -zdiff) * vel;
|
|
|
|
// the large turn_speed is the slower the turn
|
|
|
|
actor->user.change += (oc * (turn_speed-1)) / turn_speed;
|
|
|
|
SetAngleFromChange(actor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// completely vector manipulation
|
|
int VectorWormSeek(DSWActor* actor, int16_t delay_tics, int16_t aware_range1, int16_t aware_range2)
|
|
{
|
|
if (actor->user.WaitTics <= delay_tics)
|
|
actor->user.WaitTics += MISSILEMOVETICS;
|
|
|
|
if (actor->user.WpnGoalActor == nullptr)
|
|
{
|
|
if (actor->user.WaitTics > delay_tics)
|
|
{
|
|
DSWActor* hitActor;
|
|
if ((hitActor = DoPickTarget(actor, aware_range1, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
else if ((hitActor = DoPickTarget(actor, aware_range2, false)) != nullptr)
|
|
{
|
|
actor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
}
|
|
}
|
|
|
|
DSWActor* goal = actor->user.WpnGoalActor;
|
|
if (goal != nullptr)
|
|
{
|
|
auto delta = (goal->spr.pos.XY() - actor->spr.pos.XY());
|
|
double zdiff = ActorZOfTop(goal) + (ActorSizeZ(goal) * 0.25) - actor->spr.pos.Z;
|
|
double dist = g_sqrt(delta.LengthSquared() + zdiff * zdiff);
|
|
|
|
auto oc = actor->user.change;
|
|
|
|
auto vel = (actor->int_xvel() * zinttoworld) / dist;
|
|
actor->user.change = DVector3(delta, zdiff) * vel + oc * (7. / 8);
|
|
|
|
SetAngleFromChange(actor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoBlurExtend(DSWActor* actor, int16_t interval, int16_t blur_num)
|
|
{
|
|
if (actor->user.motion_blur_num >= blur_num)
|
|
return 0;
|
|
|
|
actor->user.Counter2++;
|
|
if (actor->user.Counter2 > interval)
|
|
actor->user.Counter2 = 0;
|
|
|
|
if (!actor->user.Counter2)
|
|
{
|
|
actor->user.motion_blur_num++;
|
|
if (actor->user.motion_blur_num > blur_num)
|
|
actor->user.motion_blur_num = blur_num;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitPlasmaFountain(DSWActor* wActor, DSWActor* sActor)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PLASMA_FOUNTAIN, s_PlasmaFountain, sActor->sector(),
|
|
DVector3(sActor->spr.pos.XY(), ActorZOfBottom(sActor)), sActor->spr.angle, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
if (wActor)
|
|
SetOwner(GetOwner(wActor), actorNew);
|
|
SetAttach(sActor, actorNew);
|
|
actorNew->spr.yrepeat = 0;
|
|
actorNew->spr.clipdist = 8>>2;
|
|
|
|
actorNew->user.WaitTics = 120+60;
|
|
actorNew->user.Radius = 50;
|
|
return 0;
|
|
}
|
|
|
|
int DoPlasmaFountain(DSWActor* actor)
|
|
{
|
|
// if no Owner then die
|
|
if (actor->user.attachActor == nullptr)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
DSWActor* attachActor = actor->user.attachActor;
|
|
if (!attachActor) return 0;
|
|
|
|
// move with sprite
|
|
SetActorZ(actor, attachActor->spr.pos);
|
|
actor->spr.angle = attachActor->spr.angle;
|
|
|
|
actor->user.Counter++;
|
|
if (actor->user.Counter > 3)
|
|
actor->user.Counter = 0;
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
SpawnBlood(attachActor, actor);
|
|
if (RandomRange(1000) > 600)
|
|
InitBloodSpray(attachActor, false, 105);
|
|
}
|
|
}
|
|
|
|
// kill the fountain
|
|
if ((actor->user.WaitTics-=MISSILEMOVETICS) <= 0)
|
|
{
|
|
actor->user.WaitTics = 0;
|
|
|
|
auto bak_cstat = actor->spr.cstat;
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK);
|
|
actor->spr.cstat = bak_cstat;
|
|
|
|
KillActor(actor);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DoPlasma(DSWActor* actor)
|
|
{
|
|
auto oldv = actor->spr.pos;
|
|
DoBlurExtend(actor, 0, 4);
|
|
|
|
auto vec = MOVExy(actor->int_xvel(), actor->spr.angle);
|
|
double daz = actor->vel.Z;
|
|
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), 16, 16, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
{
|
|
// this sprite is supposed to go through players/enemys
|
|
// if hit a player/enemy back up and do it again with blocking reset
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_BLOCK) && !(hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
auto hcstat = hitActor->spr.cstat;
|
|
|
|
if (hitActor->hasU() && hitActor != actor->user.WpnGoalActor)
|
|
{
|
|
actor->spr.pos = oldv;
|
|
|
|
hitActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), 16, 16, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
hitActor->spr.cstat = hcstat;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
MissileHitDiveArea(actor);
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
actor->user.Counter = 4;
|
|
ChangeState(actor, s_PlasmaDone);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoCoolgFire(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
PlaySound(DIGI_CGMAGICHIT, actor, v3df_follow);
|
|
ChangeState(actor, s_CoolgFireDone);
|
|
auto own = GetOwner(actor);
|
|
if (own && own->hasU() && own->user.ID != RIPPER_RUN_R0) // JBF: added range check
|
|
SpawnDemonFist(actor); // Just a red magic circle flash
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoEelFire(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
void ScaleSpriteVector(DSWActor* actor, int scalex, int scaley, int scalez)
|
|
{
|
|
actor->user.change.X *= FixedToFloat(scalex);
|
|
actor->user.change.Y *= FixedToFloat(scaley);
|
|
actor->user.change.Z *= FixedToFloat(scalez);
|
|
}
|
|
|
|
void ScaleSpriteVector(DSWActor* actor, int scale)
|
|
{
|
|
actor->user.change *= FixedToFloat(scale);
|
|
}
|
|
|
|
void WallBounce(DSWActor* actor, DAngle ang)
|
|
{
|
|
actor->user.bounce++;
|
|
|
|
double k = ang.Cos() * ang.Sin() * 2;
|
|
double l = (ang * 2).Cos();
|
|
|
|
auto davec = -actor->user.change.XY();
|
|
|
|
actor->user.change.X = davec.Y * k + davec.X * l;
|
|
actor->user.change.Y = davec.X * k - davec.Y * l;
|
|
|
|
auto old_ang = actor->spr.angle;
|
|
SetAngleFromChange(actor);
|
|
|
|
// hack to prevent missile from sticking to a wall
|
|
//
|
|
if (old_ang == actor->spr.angle)
|
|
{
|
|
actor->user.change.XY() = -actor->user.change.XY();
|
|
SetAngleFromChange(actor);
|
|
}
|
|
}
|
|
|
|
|
|
bool SlopeBounce(DSWActor* actor, bool* hit_wall)
|
|
{
|
|
double hiz, loz;
|
|
int slope;
|
|
|
|
auto hit_sector = actor->user.coll.hitSector;
|
|
|
|
getzsofslopeptr(hit_sector, actor->spr.pos, &hiz, &loz);
|
|
|
|
// detect the ceiling and the hit_wall
|
|
if (actor->spr.pos.Z < ((hiz + loz) * 0.5))
|
|
{
|
|
if (!(hit_sector->ceilingstat & CSTAT_SECTOR_SLOPE))
|
|
slope = 0;
|
|
else
|
|
slope = hit_sector->ceilingheinum;
|
|
}
|
|
else
|
|
{
|
|
if (!(hit_sector->floorstat & CSTAT_SECTOR_SLOPE))
|
|
slope = 0;
|
|
else
|
|
slope = hit_sector->floorheinum;
|
|
}
|
|
|
|
if (!slope)
|
|
return false;
|
|
|
|
// if greater than a 45 degree angle
|
|
if (abs(slope) > 4096)
|
|
*hit_wall = true;
|
|
else
|
|
*hit_wall = false;
|
|
|
|
// get angle of the first wall of the sector
|
|
auto wallp = hit_sector->firstWall();
|
|
DAngle daang = VecToAngle(wallp->delta());
|
|
|
|
// k is now the slope of the ceiling or floor
|
|
|
|
// normal vector of the slope
|
|
double dax = slope * daang.Sin();
|
|
double day = slope * -daang.Cos();
|
|
double daz = 256; // 4096/256 = 45 degrees
|
|
|
|
// reflection code
|
|
double k = actor->user.change.X * dax + actor->user.change.Y * day + actor->user.change.Z * daz;
|
|
double l = (dax * dax) + (day * day) + (daz * daz);
|
|
|
|
k /= l;
|
|
|
|
actor->user.change.X = dax * k;
|
|
actor->user.change.Y = day * k;
|
|
actor->user.change.Z = daz * k * 16;
|
|
|
|
SetAngleFromChange(actor);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
extern STATE s_Phosphorus[];
|
|
|
|
int DoGrenade(DSWActor* actor)
|
|
{
|
|
short i;
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
ScaleSpriteVector(actor, 50000);
|
|
|
|
actor->user.Counter += 20;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
else
|
|
{
|
|
actor->user.Counter += 20;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitVoid:
|
|
KillActor(actor);
|
|
return true;
|
|
case kHitSprite:
|
|
{
|
|
short wall_ang;
|
|
|
|
PlaySound(DIGI_40MMBNCE, actor, v3df_dontpan);
|
|
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
// special case so grenade can ring gong
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (SP_TAG8(hitActor) & BIT(3))
|
|
DoMatchEverything(nullptr, hitActor->spr.hitag, -1);
|
|
}
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
WallBounce(actor, hitActor->spr.angle);
|
|
ScaleSpriteVector(actor, 32000);
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.Counter2 == 1) // It's a phosphorus grenade!
|
|
{
|
|
for (i=0; i<5; i++)
|
|
{
|
|
actor->set_int_ang(NORM_ANGLE(RandomRange(2048)));
|
|
InitPhosphorus(actor);
|
|
}
|
|
}
|
|
SpawnGrenadeExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
short nw,wall_ang;
|
|
walltype* wph;
|
|
|
|
wph = actor->user.coll.hitWall;
|
|
|
|
if (wph->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(wph, actor->spr.pos, actor->spr.angle, actor->user.ID);
|
|
actor->user.coll.setNone();
|
|
break;
|
|
}
|
|
|
|
PlaySound(DIGI_40MMBNCE, actor, v3df_dontpan);
|
|
|
|
WallBounce(actor, wph->delta().Angle() + DAngle90);
|
|
ScaleSpriteVector(actor, 22000);
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
bool did_hit_wall;
|
|
if (SlopeBounce(actor, &did_hit_wall))
|
|
{
|
|
if (did_hit_wall)
|
|
{
|
|
// hit a wall
|
|
ScaleSpriteVector(actor, 22000); // 28000
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
}
|
|
else
|
|
{
|
|
// hit a sector
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
// hit a floor
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
ScaleSpriteVector(actor, 40000); // 18000
|
|
actor->user.coll.setNone();
|
|
actor->user.change.Z /= 4;
|
|
actor->user.Counter = 0;
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.Counter2 == 1) // It's a phosphorus grenade!
|
|
{
|
|
for (i=0; i<5; i++)
|
|
{
|
|
actor->set_int_ang(NORM_ANGLE(RandomRange(2048)));
|
|
InitPhosphorus(actor);
|
|
}
|
|
}
|
|
SpawnGrenadeExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// hit a ceiling
|
|
ScaleSpriteVector(actor, 22000);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// hit floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actor->user.Flags |= (SPR_BOUNCE); // no bouncing underwater
|
|
|
|
if (actor->user.lo_sectp && actor->sector()->hasU() && FixedToInt(actor->sector()->depth_fixed))
|
|
actor->user.Flags |= (SPR_BOUNCE); // no bouncing on shallow water
|
|
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 40000); // 18000
|
|
actor->user.change.Z /= 4;
|
|
PlaySound(DIGI_40MMBNCE, actor, v3df_dontpan);
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.Counter2 == 1) // It's a phosphorus grenade!
|
|
{
|
|
for (i=0; i<5; i++)
|
|
{
|
|
actor->set_int_ang(NORM_ANGLE(RandomRange(2048)));
|
|
InitPhosphorus(actor);
|
|
}
|
|
}
|
|
SpawnGrenadeExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
// hit something above
|
|
{
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 22000);
|
|
PlaySound(DIGI_40MMBNCE, actor, v3df_dontpan);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (actor->user.bounce > 10)
|
|
{
|
|
SpawnGrenadeExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
// if you haven't bounced or your going slow do some puffs
|
|
if (!(actor->user.Flags & (SPR_BOUNCE|SPR_UNDERWATER)))
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_Puff, actor->sector(),
|
|
actor->spr.pos, actor->spr.angle, 100);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->opos = actor->opos;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.change = actor->user.change;
|
|
|
|
ScaleSpriteVector(actorNew, 22000);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoVulcanBoulder(DSWActor* actor)
|
|
{
|
|
actor->user.Counter += 40;
|
|
actor->user.addCounterToChange();
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
auto velsq = actor->user.change.XY().LengthSquared();
|
|
|
|
if (velsq < 1.875 * 1.875)
|
|
{
|
|
SpawnLittleExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitVoid:
|
|
KillActor(actor);
|
|
return true;
|
|
case kHitSprite:
|
|
{
|
|
short wall_ang;
|
|
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
WallBounce(actor, hitActor->spr.angle);
|
|
ScaleSpriteVector(actor, 40000);
|
|
}
|
|
else
|
|
{
|
|
// hit an actor
|
|
SpawnLittleExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
short nw,wall_ang;
|
|
walltype* wph;
|
|
|
|
wph = actor->user.coll.hitWall;
|
|
|
|
if (wph->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(wph, actor->spr.pos, actor->spr.angle, actor->user.ID);
|
|
actor->user.coll.setNone();
|
|
break;
|
|
}
|
|
|
|
WallBounce(actor, wph->delta().Angle() + DAngle90);
|
|
ScaleSpriteVector(actor, 40000);
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
bool did_hit_wall;
|
|
|
|
if (SlopeBounce(actor, &did_hit_wall))
|
|
{
|
|
if (did_hit_wall)
|
|
{
|
|
// hit a sloped sector - treated as a wall because of large slope
|
|
ScaleSpriteVector(actor, 30000);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
}
|
|
else
|
|
{
|
|
// hit a sloped sector
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
// hit a floor
|
|
ScaleSpriteVector(actor, 30000, 30000, 12000);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
|
|
// limit to a reasonable bounce value
|
|
if (actor->user.change.Z > 32)
|
|
actor->user.change.Z = 32;
|
|
}
|
|
else
|
|
{
|
|
// hit a sloped ceiling
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
ScaleSpriteVector(actor, 30000);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// hit unsloped floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
|
|
ScaleSpriteVector(actor, 20000, 20000, 32000);
|
|
|
|
// limit to a reasonable bounce value
|
|
if (actor->user.change.Z > 24)
|
|
actor->user.change.Z = 24;
|
|
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
|
|
}
|
|
else
|
|
// hit unsloped ceiling
|
|
{
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 30000);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool OwnerIsPlayer(DSWActor* actor)
|
|
{
|
|
auto own = GetOwner(actor);
|
|
return (own && own->hasU() && own->user.PlayerP != nullptr);
|
|
}
|
|
|
|
int DoMineRangeTest(DSWActor* actor, int range)
|
|
{
|
|
unsigned stat;
|
|
int dist, tx, ty;
|
|
int tmin;
|
|
bool ownerisplayer = false;
|
|
|
|
ownerisplayer = OwnerIsPlayer(actor);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
|
|
if (dist > range)
|
|
continue;
|
|
|
|
if (actor == itActor)
|
|
continue;
|
|
|
|
if (!(itActor->spr.cstat & CSTAT_SPRITE_BLOCK))
|
|
continue;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
if (itActor->user.ID == GIRLNINJA_RUN_R0 && !ownerisplayer)
|
|
continue;
|
|
|
|
dist = FindDistance3D(actor->int_pos() - itActor->int_pos());
|
|
if (dist > range)
|
|
continue;
|
|
|
|
if (!FAFcansee(ActorUpperVect(actor),itActor->sector(),actor->spr.pos,actor->sector()))
|
|
continue;
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoMineStuck(DSWActor* actor)
|
|
{
|
|
constexpr int MINE_DETONATE_STATE = 99;
|
|
|
|
// if no Owner then die
|
|
DSWActor* attachActor = actor->user.attachActor;
|
|
if (attachActor != nullptr)
|
|
{
|
|
ASSERT(attachActor->hasU());
|
|
|
|
// Is it attached to a dead actor? Blow it up if so.
|
|
if ((attachActor->user.Flags & SPR_DEAD) && actor->user.Counter2 < MINE_DETONATE_STATE)
|
|
{
|
|
actor->user.Counter2 = MINE_DETONATE_STATE;
|
|
actor->user.WaitTics = SEC(1)/2;
|
|
}
|
|
|
|
SetActorZ(actor, attachActor->spr.pos.plusZ(-actor->user.pos.Z));
|
|
actor->spr.pos.Z = attachActor->spr.pos.Z - (ActorSizeZ(attachActor) * 0.5);
|
|
}
|
|
|
|
// not activated yet
|
|
if (!(actor->user.Flags & SPR_ACTIVE))
|
|
{
|
|
if ((actor->user.WaitTics -= (MISSILEMOVETICS*2)) > 0)
|
|
return false;
|
|
|
|
// activate it
|
|
//actor->user.WaitTics = 65536;
|
|
actor->user.WaitTics = 32767;
|
|
actor->user.Counter2 = 0;
|
|
actor->user.Flags |= (SPR_ACTIVE);
|
|
}
|
|
|
|
// limit the number of times DoMineRangeTest is called
|
|
actor->user.Counter++;
|
|
if (actor->user.Counter > 1)
|
|
actor->user.Counter = 0;
|
|
|
|
if (actor->user.Counter2 != MINE_DETONATE_STATE)
|
|
{
|
|
if ((actor->user.Counter2++) > 30)
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.WaitTics = 32767; // Keep reseting tics to make it stay forever
|
|
actor->user.Counter2 = 0;
|
|
}
|
|
}
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
// not already in detonate state
|
|
if (actor->user.Counter2 < MINE_DETONATE_STATE)
|
|
{
|
|
// if something came into range - detonate
|
|
if (DoMineRangeTest(actor, 3000))
|
|
{
|
|
// move directly to detonate state
|
|
actor->user.Counter2 = MINE_DETONATE_STATE;
|
|
actor->user.WaitTics = SEC(1)/2;
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
|
|
|
|
|
|
// start beeping with pauses
|
|
// quick and dirty beep countdown code
|
|
switch (actor->user.Counter2)
|
|
{
|
|
#if 0
|
|
case 0:
|
|
if (actor->user.WaitTics < SEC(45))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 1:
|
|
if (actor->user.WaitTics < SEC(38))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 2:
|
|
if (actor->user.WaitTics < SEC(30))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 3:
|
|
if (actor->user.WaitTics < SEC(20))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 4:
|
|
if (actor->user.WaitTics < SEC(15))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 5:
|
|
if (actor->user.WaitTics < SEC(12))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 6:
|
|
if (actor->user.WaitTics < SEC(10))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
case 7:
|
|
if (actor->user.WaitTics < SEC(8))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2++;
|
|
}
|
|
break;
|
|
#endif
|
|
case 30:
|
|
if (actor->user.WaitTics < SEC(6))
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
actor->user.Counter2 = MINE_DETONATE_STATE;
|
|
}
|
|
break;
|
|
case MINE_DETONATE_STATE:
|
|
if (actor->user.WaitTics < 0)
|
|
{
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
|
|
SpawnMineExp(actor);
|
|
KillActor(actor);
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void SetMineStuck(DSWActor* actor)
|
|
{
|
|
// stuck
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
// not yet active for 1 sec
|
|
actor->user.Flags &= ~(SPR_ACTIVE);
|
|
actor->user.WaitTics = SEC(3);
|
|
//actor->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->spr.cstat |= (CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.Counter = 0;
|
|
change_actor_stat(actor, STAT_MINE_STUCK);
|
|
ChangeState(actor, s_MineStuck);
|
|
}
|
|
|
|
int DoMine(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
// decrease velocity
|
|
ScaleSpriteVector(actor, 50000);
|
|
|
|
actor->user.Counter += 20;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
else
|
|
{
|
|
//actor->user.Counter += 75;
|
|
actor->user.Counter += 40;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
// check to see if you hit a sprite
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitVoid:
|
|
KillActor(actor);
|
|
return 0;
|
|
case kHitSprite:
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
SetMineStuck(actor);
|
|
// Set the Z position
|
|
actor->spr.pos.Z = hitActor->spr.pos.Z - (ActorSizeZ(hitActor) * 0.5);
|
|
|
|
// If it's not alive, don't stick it
|
|
if (hitActor->hasU() && hitActor->user.Health <= 0) return false; // JBF: added null check
|
|
|
|
// check to see if sprite is player or enemy
|
|
if ((hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
PLAYER* pp;
|
|
|
|
// attach weapon to sprite
|
|
SetAttach(hitActor, actor);
|
|
actor->user.pos.Z = hitActor->spr.pos.Z - actor->spr.pos.Z;
|
|
|
|
auto own = GetOwner(actor);
|
|
if (own && own->hasU())
|
|
{
|
|
if (own->user.PlayerP)
|
|
{
|
|
pp = own->user.PlayerP;
|
|
|
|
if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_STICKYGOTU1, v3df_follow|v3df_dontpan,pp);
|
|
else if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_STICKYGOTU2, v3df_follow|v3df_dontpan,pp);
|
|
else if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_STICKYGOTU3, v3df_follow|v3df_dontpan,pp);
|
|
else if (RandomRange(1000) > 800)
|
|
PlayerSound(DIGI_STICKYGOTU4, v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
actor->user.Flags2 |= (SPR2_ATTACH_WALL);
|
|
}
|
|
else if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_FLOOR))
|
|
{
|
|
// hit floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
actor->user.Flags2 |= (SPR2_ATTACH_FLOOR);
|
|
else
|
|
actor->user.Flags2 |= (SPR2_ATTACH_CEILING);
|
|
}
|
|
else
|
|
{
|
|
SpawnMineExp(actor);
|
|
KillActor(actor);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
auto hit_wall = actor->user.coll.hitWall;
|
|
|
|
if (hit_wall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit_wall, actor->spr.pos, actor->spr.angle, actor->user.ID);
|
|
actor->user.coll.setNone();
|
|
break;
|
|
}
|
|
|
|
SetMineStuck(actor);
|
|
|
|
actor->user.Flags2 |= (SPR2_ATTACH_WALL);
|
|
|
|
if ((hit_wall->extra & WALLFX_SECTOR_OBJECT))
|
|
{
|
|
}
|
|
|
|
if ((hit_wall->extra & WALLFX_DONT_STICK))
|
|
{
|
|
SpawnMineExp(actor);
|
|
KillActor(actor);
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
auto hit_sect = actor->user.coll.hitSector;
|
|
|
|
SetMineStuck(actor);
|
|
|
|
// hit floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
actor->user.Flags2 |= (SPR2_ATTACH_FLOOR);
|
|
else
|
|
actor->user.Flags2 |= (SPR2_ATTACH_CEILING);
|
|
|
|
|
|
if ((hit_sect->extra & SECTFX_SECTOR_OBJECT))
|
|
{
|
|
SpawnMineExp(actor);
|
|
KillActor(actor);
|
|
return false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
actor->user.coll.setNone();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoPuff(DSWActor* actor)
|
|
{
|
|
actor->spr.pos += actor->user.change;
|
|
return 0;
|
|
}
|
|
|
|
int DoRailPuff(DSWActor* actor)
|
|
{
|
|
actor->spr.xrepeat += 4;
|
|
actor->spr.yrepeat += 4;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoBoltThinMan(DSWActor* actor)
|
|
{
|
|
DoBlurExtend(actor, 0, 4);
|
|
|
|
auto vec = MOVExy(actor->int_xvel(), actor->spr.angle);
|
|
double daz = actor->vel.Z;
|
|
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), CEILING_DIST, FLOOR_DIST, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return true;
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnBoltExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoTracer(DSWActor* actor)
|
|
{
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_INVISIBLE);
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoEMP(DSWActor* actor)
|
|
{
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (RandomRange(1000) > 500)
|
|
{
|
|
actor->spr.xrepeat = 52;
|
|
actor->spr.yrepeat = 10;
|
|
}
|
|
else
|
|
{
|
|
actor->spr.xrepeat = 8;
|
|
actor->spr.yrepeat = 38;
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_INVISIBLE);
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoEMPBurst(DSWActor* actor)
|
|
{
|
|
DSWActor* attachActor = actor->user.attachActor;
|
|
if (attachActor != nullptr)
|
|
{
|
|
SetActorZ(actor, attachActor->spr.pos.plusZ(-actor->user.pos.Z));
|
|
actor->set_int_ang(NORM_ANGLE(attachActor->int_ang() + 1024));
|
|
}
|
|
|
|
// not activated yet
|
|
if (!(actor->user.Flags & SPR_ACTIVE))
|
|
{
|
|
// activate it
|
|
actor->user.WaitTics = SEC(7);
|
|
actor->user.Flags |= (SPR_ACTIVE);
|
|
}
|
|
|
|
if (RandomRange(1000) > 500)
|
|
{
|
|
actor->spr.xrepeat = 52;
|
|
actor->spr.yrepeat = 10;
|
|
}
|
|
else
|
|
{
|
|
actor->spr.xrepeat = 8;
|
|
actor->spr.yrepeat = 38;
|
|
}
|
|
|
|
if ((RANDOM_P2(1024<<6)>>6) < 700)
|
|
{
|
|
SpawnShrapX(actor);
|
|
}
|
|
|
|
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
|
|
|
|
if (actor->user.WaitTics < 0)
|
|
{
|
|
// Spawn a big radius burst of sparks here and check for final damage amount
|
|
KillActor(actor);
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoTankShell(DSWActor* actor)
|
|
{
|
|
short i;
|
|
|
|
for (i = 0; i < 4; i++)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnTankShellExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoTracerStart(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoLaser(DSWActor* actor)
|
|
{
|
|
short spawn_count = 0;
|
|
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
while (true)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnBoltExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
spawn_count++;
|
|
if (spawn_count < 256)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_LaserPuff, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 16;
|
|
actorNew->spr.pal = actorNew->user.spal = PALETTE_RED_LIGHTING;
|
|
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actor->user.change.Zero();
|
|
}
|
|
}
|
|
}
|
|
|
|
int DoLaserStart(DSWActor* actor)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnBoltExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoRail(DSWActor* actor)
|
|
{
|
|
short spawn_count = 0;
|
|
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
while (true)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor) && actor->user.coll.type != kHitNone) // beware of side effects of WeaponMoveHit!
|
|
{
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if (hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY)
|
|
{
|
|
auto cstat_save = hitActor->spr.cstat;
|
|
|
|
hitActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN|CSTAT_SPRITE_BLOCK_MISSILE);
|
|
DoRail(actor);
|
|
hitActor->spr.cstat = cstat_save;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
SpawnTracerExp(actor);
|
|
SpawnShrapX(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SpawnTracerExp(actor);
|
|
SpawnShrapX(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
spawn_count++;
|
|
if (spawn_count < 128)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, &s_RailPuff[0][0], actor->sector(), actor->spr.pos, actor->spr.angle, 20);
|
|
|
|
actorNew->add_int_xvel( (RandomRange(140)-RandomRange(140)));
|
|
actorNew->add_int_zvel( (RandomRange(140)-RandomRange(140)));
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, sg_RailPuff);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 10;
|
|
actorNew->spr.yrepeat = 10;
|
|
actorNew->opos = actor->opos;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.change = actor->user.change;
|
|
|
|
ScaleSpriteVector(actorNew, 1500);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
}
|
|
}
|
|
|
|
int DoRailStart(DSWActor* actor)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnTracerExp(actor);
|
|
SpawnShrapX(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int DoRocket(DSWActor* actor)
|
|
{
|
|
int dist,a,b,c;
|
|
auto pos = actor->int_pos();
|
|
|
|
|
|
if ((actor->user.FlagOwner -= ACTORMOVETICS)<=0 && actor->user.spal == 20)
|
|
{
|
|
DISTANCE(actor->spr.pos, actor->user.targetActor->spr.pos, dist, a, b, c);
|
|
actor->user.FlagOwner = dist>>6;
|
|
// Special warn sound attached to each seeker spawned
|
|
PlaySound(DIGI_MINEBEEP, actor, v3df_follow);
|
|
}
|
|
|
|
if (actor->user.Flags & (SPR_FIND_PLAYER))
|
|
{
|
|
VectorMissileSeek(actor, 30, 16, 128, 768);
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return true;
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor) && actor->user.coll.type != kHitNone)
|
|
{
|
|
if (actor->user.ID == BOLT_THINMAN_R4)
|
|
{
|
|
SpawnBunnyExp(actor);
|
|
}
|
|
else if (actor->user.Radius == NUKE_RADIUS)
|
|
SpawnNuclearExp(actor);
|
|
else
|
|
SpawnBoltExp(actor);
|
|
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_Puff, actor->sector(),
|
|
pos.X, pos.Y, pos.Z, actor->int_ang(), 100);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->opos = actor->opos;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.change = actor->user.change;
|
|
|
|
ScaleSpriteVector(actorNew, 20000);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int DoMicroMini(DSWActor* actor)
|
|
{
|
|
short i;
|
|
|
|
for (i = 0; i < 3; i++)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnMicroExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int SpawnExtraMicroMini(DSWActor* actor)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Micro[0][0], actor->sector(),
|
|
actor->spr.pos, actor->spr.angle, actor->int_xvel());
|
|
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.yrepeat = actorNew->spr.xrepeat = actor->spr.xrepeat;
|
|
actorNew->spr.shade = actor->spr.shade;
|
|
actorNew->spr.clipdist = actor->spr.clipdist;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_MicroMini[0]);
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = actor->user.Radius;
|
|
actorNew->user.ceiling_dist = actor->user.ceiling_dist;
|
|
actorNew->user.floor_dist = actor->user.floor_dist;
|
|
actorNew->spr.cstat = actor->spr.cstat;
|
|
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + RandomRange(64) - 32));
|
|
actorNew->vel.Z = -actor->vel.Z;
|
|
actorNew->add_int_zvel( RandomRange(Z(16)) - Z(8));
|
|
|
|
UpdateChange(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
int DoMicro(DSWActor* actor)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_Puff, actor->sector(),
|
|
actor->spr.pos, actor->spr.angle, 100);
|
|
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 20;
|
|
actorNew->spr.yrepeat = 20;
|
|
actorNew->opos = actor->opos;
|
|
actorNew->vel.Z = -actor->vel.Z;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.change = actor->user.change;
|
|
|
|
ScaleSpriteVector(actorNew, 20000);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// last smoke
|
|
if ((actor->user.WaitTics -= MISSILEMOVETICS) <= 0)
|
|
{
|
|
SetActorZ(actorNew, actorNew->spr.pos);
|
|
NewStateGroup(actor, &sg_MicroMini[0]);
|
|
actor->spr.xrepeat = actor->spr.yrepeat = 10;
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_INVISIBLE);
|
|
SpawnExtraMicroMini(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
// hit something
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnMicroExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoUziBullet(DSWActor* actor)
|
|
{
|
|
// call move_sprite twice for each movement
|
|
// otherwize the moves are in too big an increment
|
|
for (int i = 0; i < 2; i++)
|
|
{
|
|
auto vec = MOVExy((actor->int_xvel() >> 1), actor->spr.angle);
|
|
double daz = (actor->int_zvel() >> 1) * zinttoworld;
|
|
|
|
auto spos = actor->spr.pos.XY();
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
actor->user.Dist += (spos - actor->spr.pos.XY()). Length();
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 128)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
WeaponMoveHit(actor);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT;
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 128 >> 2;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
|
|
if (!(actor->user.Flags & SPR_UNDERWATER))
|
|
{
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, actorNew->sector(), actorNew->spr.pos, nullAngle, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT;
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
}
|
|
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
else if (actor->user.Dist > 500)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoBoltSeeker(DSWActor* actor)
|
|
{
|
|
MissileSeek(actor, 30, 768/*, 4, 48, 6*/);
|
|
DoBlurExtend(actor, 0, 4);
|
|
|
|
auto vec = MOVExy(actor->int_xvel(), actor->spr.angle);
|
|
double daz = actor->vel.Z;
|
|
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), CEILING_DIST, FLOOR_DIST, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnBoltExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int DoBoltShrapnel(DSWActor* actor)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int DoBoltFatMan(DSWActor* actor)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int DoElectro(DSWActor* actor)
|
|
{
|
|
DoBlurExtend(actor, 0, 4);
|
|
|
|
// only seek on Electro's after a hit on an actor
|
|
if (actor->user.Counter > 0)
|
|
MissileSeek(actor, 30, 512/*, 3, 52, 2*/);
|
|
|
|
auto vec = MOVExy(actor->int_xvel(), actor->spr.angle);
|
|
double daz = actor->vel.Z;
|
|
|
|
actor->user.coll = move_missile(actor, DVector3(vec, daz), CEILING_DIST, FLOOR_DIST, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return true;
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitSprite:
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if (!(hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY) || hitActor->user.ID == SKULL_R0 || hitActor->user.ID == BETTY_R0)
|
|
SpawnShrap(actor, nullptr);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
SpawnShrap(actor, nullptr);
|
|
break;
|
|
}
|
|
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int DoLavaBoulder(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return true;
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnShrap(actor, nullptr);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoSpear(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return true;
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int SpawnCoolieExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
actor->user.Counter = RandomRange(120); // This is the wait til birth time!
|
|
|
|
auto vect = actor->spr.pos + MOVExy(64, actor->spr.angle + DAngle180);
|
|
vect.Z -= ActorSizeZ(actor) * 0.75;
|
|
|
|
PlaySound(DIGI_COOLIEEXPLODE, actor, v3df_none);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_EXP, s_BoltExp, actor->sector(), vect, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.Radius = DamageData[DMG_BOLT_EXP].radius;
|
|
DoExpDamageTest(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
void SpawnFireballFlames(DSWActor* actor, DSWActor* enemyActor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
return;
|
|
|
|
if (enemyActor != nullptr)
|
|
{
|
|
// test for already burned
|
|
if ((enemyActor->spr.extra & SPRX_BURNABLE) && enemyActor->spr.shade > 40)
|
|
return;
|
|
|
|
if (!enemyActor->hasU())
|
|
{
|
|
ASSERT(true == false);
|
|
return;
|
|
}
|
|
|
|
auto flameActor = enemyActor->user.flameActor;
|
|
if (flameActor != nullptr)
|
|
{
|
|
double sizez = ActorSizeZ(enemyActor) + (ActorSizeZ(enemyActor) * 0.25);
|
|
|
|
if ((enemyActor->spr.extra & SPRX_BURNABLE))
|
|
return;
|
|
|
|
if (flameActor->user.Counter >= GetRepeatFromHeight(flameActor, sizez))
|
|
{
|
|
// keep flame only slightly bigger than the enemy itself
|
|
flameActor->user.Counter = GetRepeatFromHeight(flameActor, sizez);
|
|
}
|
|
else
|
|
{
|
|
//increase max size
|
|
flameActor->user.Counter += GetRepeatFromHeight(flameActor, 8);
|
|
}
|
|
|
|
// Counter is max size
|
|
if (flameActor->user.Counter >= 230)
|
|
{
|
|
// this is far too big
|
|
flameActor->user.Counter = 230;
|
|
}
|
|
|
|
if (flameActor->user.WaitTics < 2*120)
|
|
flameActor->user.WaitTics = 2*120; // allow it to grow again
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL_FLAMES, s_FireballFlames, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
if (enemyActor != nullptr)
|
|
enemyActor->user.flameActor = actorNew;
|
|
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 16;
|
|
if (enemyActor != nullptr)
|
|
{
|
|
// large flame for trees and such
|
|
if ((enemyActor->spr.extra & SPRX_BURNABLE))
|
|
{
|
|
double sizez = ActorSizeZ(enemyActor) + (ActorSizeZ(enemyActor) * 0.25);
|
|
actorNew->user.Counter = GetRepeatFromHeight(actorNew, sizez);
|
|
}
|
|
else
|
|
{
|
|
actorNew->user.Counter = GetRepeatFromHeight(actorNew, ActorSizeZ(enemyActor) * 0.5);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
actorNew->user.Counter = 48; // max flame size
|
|
}
|
|
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
//actorNew->user.Radius = DamageData[DMG_FIREBALL_FLAMES].radius;
|
|
actorNew->user.Radius = 200;
|
|
|
|
if (enemyActor != nullptr)
|
|
{
|
|
SetAttach(enemyActor, actorNew);
|
|
}
|
|
else
|
|
{
|
|
if (TestDontStickSector(actorNew->sector()))
|
|
{
|
|
KillActor(actorNew);
|
|
return;
|
|
}
|
|
|
|
actorNew->user.floor_dist = actorNew->user.ceiling_dist = 0;
|
|
|
|
DoFindGround(actorNew);
|
|
actorNew->user.jump_speed = 0;
|
|
DoBeginJump(actorNew);
|
|
}
|
|
|
|
PlaySound(DIGI_FIRE1,actorNew,v3df_dontpan|v3df_doppler);
|
|
}
|
|
|
|
|
|
int SpawnBreakFlames(DSWActor* actor)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL_FLAMES+1, s_BreakFlames, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 16;
|
|
actorNew->user.Counter = 48; // max flame size
|
|
|
|
actorNew->spr.shade = -40;
|
|
if (actor->hasU())
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.Radius = 200;
|
|
|
|
actorNew->user.floor_dist = actorNew->user.ceiling_dist = 0;
|
|
|
|
DoFindGround(actorNew);
|
|
actorNew->user.jump_speed = 0;
|
|
DoBeginJump(actorNew);
|
|
|
|
PlaySound(DIGI_FIRE1,actorNew,v3df_dontpan|v3df_doppler);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
void SpawnBreakStaticFlames(DSWActor* actor)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_STATIC_FIRE, FIREBALL_FLAMES, nullptr, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
if (RandomRange(1000) > 500)
|
|
actorNew->spr.picnum = 3143;
|
|
else
|
|
actorNew->spr.picnum = 3157;
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.yrepeat = 32;
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.floor_dist = actorNew->user.ceiling_dist = 0;
|
|
actorNew->spr.pos.Z = getflorzofslopeptrf(actorNew->sector(), actorNew->spr.pos);
|
|
|
|
PlaySound(DIGI_FIRE1,actorNew,v3df_dontpan|v3df_doppler);
|
|
}
|
|
|
|
|
|
void SpawnFireballExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_SMALLEXP, actor, v3df_none);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL_EXP, s_FireballExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.yrepeat = 52;
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->user.Flags |= (actor->user.Flags & (SPR_UNDERWATER));
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
//
|
|
// All this stuff assures that explosions do not go into floors &
|
|
// ceilings
|
|
//
|
|
|
|
SpawnExpZadjust(actor, actorNew, 15, 15);
|
|
|
|
if (RANDOM_P2(1024) < 150)
|
|
SpawnFireballFlames(actorNew, nullptr);
|
|
}
|
|
|
|
void SpawnGoroFireballExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_MEDIUMEXP, actor, v3df_none);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, 0, s_FireballExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 16;
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.pal = actorNew->user.spal = actor->user.spal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
//
|
|
// All this stuff assures that explosions do not go into floors &
|
|
// ceilings
|
|
//
|
|
|
|
SpawnExpZadjust(actor, actorNew, 15, 15);
|
|
}
|
|
|
|
void SpawnBoltExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_BOLTEXPLODE, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, BOLT_EXP, s_BoltExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 76;
|
|
expActor->spr.yrepeat = 76;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
expActor->user.Radius = DamageData[DMG_BOLT_EXP].radius;
|
|
|
|
SpawnExpZadjust(actor, expActor, 40, 40);
|
|
|
|
DoExpDamageTest(expActor);
|
|
|
|
SetExpQuake(actor); // !JIM! made rocket launcher shake things
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
}
|
|
|
|
int SpawnBunnyExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return -1;
|
|
|
|
PlaySound(DIGI_BUNNYDIE3, actor, v3df_none);
|
|
|
|
actor->user.ID = BOLT_EXP; // Change id
|
|
InitBloodSpray(actor, true, -1);
|
|
InitBloodSpray(actor, true, -1);
|
|
InitBloodSpray(actor, true, -1);
|
|
DoExpDamageTest(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void SpawnTankShellExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_BOLTEXPLODE, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, TANK_SHELL_EXP, s_TankShellExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 64+32;
|
|
expActor->spr.yrepeat = 64+32;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
expActor->user.Radius = DamageData[DMG_TANK_SHELL_EXP].radius;
|
|
|
|
SpawnExpZadjust(actor, expActor, 40, 40);
|
|
DoExpDamageTest(expActor);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
}
|
|
|
|
|
|
void SpawnNuclearSecondaryExp(DSWActor* actor, short ang)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, GRENADE_EXP, s_GrenadeExp, actor->sector(), actor->spr.pos, actor->spr.angle, 512);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -128;
|
|
expActor->spr.xrepeat = 218;
|
|
expActor->spr.yrepeat = 152;
|
|
expActor->spr.clipdist = actor->spr.clipdist;
|
|
expActor->user.ceiling_dist = (16);
|
|
expActor->user.floor_dist = (16);
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
//ang = RANDOM_P2(2048);
|
|
int32_t const vel = (2048+128) + RandomRange(2048);
|
|
expActor->user.change.XY() = MOVExy(vel, DAngle::fromBuild(ang));
|
|
expActor->user.Radius = 200; // was NUKE_RADIUS
|
|
expActor->user.coll = move_missile(expActor, DVector3(expActor->user.change.XY(), 0),
|
|
expActor->user.ceiling_dist,expActor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (FindDistance3D(expActor->int_pos() - actor->int_pos()) < 1024)
|
|
{
|
|
KillActor(expActor);
|
|
return;
|
|
}
|
|
|
|
SpawnExpZadjust(actor, expActor, 50, 10);
|
|
InitChemBomb(expActor);
|
|
}
|
|
|
|
void SpawnNuclearExp(DSWActor* actor)
|
|
{
|
|
short ang=0;
|
|
PLAYER* pp = nullptr;
|
|
short rnd_rng;
|
|
|
|
ASSERT(actor->hasU());
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_NUCLEAREXP, actor, v3df_dontpan | v3df_doppler);
|
|
|
|
auto own = GetOwner(actor);
|
|
if (own && own->hasU())
|
|
{
|
|
pp = own->user.PlayerP;
|
|
rnd_rng = RandomRange(1000);
|
|
|
|
if (rnd_rng > 990)
|
|
PlayerSound(DIGI_LIKEHIROSHIMA, v3df_follow|v3df_dontpan,pp);
|
|
else if (rnd_rng > 980)
|
|
PlayerSound(DIGI_LIKENAGASAKI, v3df_follow|v3df_dontpan,pp);
|
|
else if (rnd_rng > 970)
|
|
PlayerSound(DIGI_LIKEPEARL, v3df_follow|v3df_dontpan,pp);
|
|
}
|
|
|
|
// Spawn big mushroom cloud
|
|
auto expActor = SpawnActor(STAT_MISSILE, MUSHROOM_CLOUD, s_NukeMushroom, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(own, expActor);
|
|
expActor->spr.shade = -128;
|
|
expActor->spr.xrepeat = 255;
|
|
expActor->spr.yrepeat = 255;
|
|
expActor->spr.clipdist = actor->spr.clipdist;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.spal = expActor->spr.pal = PALETTE_PLAYER1; // Set nuke puff to gray
|
|
|
|
InitChemBomb(expActor);
|
|
|
|
|
|
// Do central explosion
|
|
expActor = SpawnActor(STAT_MISSILE, MUSHROOM_CLOUD, s_GrenadeExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
SetOwner(own, expActor);
|
|
expActor->spr.shade = -128;
|
|
expActor->spr.xrepeat = 218;
|
|
expActor->spr.yrepeat = 152;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
|
|
expActor->user.Radius = NUKE_RADIUS;
|
|
|
|
SpawnExpZadjust(actor, expActor, 30, 30);
|
|
|
|
DoExpDamageTest(expActor);
|
|
|
|
// Nuclear effects
|
|
SetNuclearQuake(actor);
|
|
|
|
SetFadeAmt(pp, -80, 1); // Nuclear flash
|
|
|
|
// Secondary blasts
|
|
ang = RANDOM_P2(2048);
|
|
SpawnNuclearSecondaryExp(expActor, ang);
|
|
ang = ang + 512 + RANDOM_P2(256);
|
|
SpawnNuclearSecondaryExp(expActor, ang);
|
|
ang = ang + 512 + RANDOM_P2(256);
|
|
SpawnNuclearSecondaryExp(expActor, ang);
|
|
ang = ang + 512 + RANDOM_P2(256);
|
|
SpawnNuclearSecondaryExp(expActor, ang);
|
|
}
|
|
|
|
void SpawnTracerExp(DSWActor* actor)
|
|
{
|
|
DSWActor* expActor;
|
|
|
|
ASSERT(actor->hasU());
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return ;
|
|
|
|
if (actor->user.ID == BOLT_THINMAN_R1)
|
|
expActor = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R1, s_TracerExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
else
|
|
expActor = SpawnActor(STAT_MISSILE, TRACER_EXP, s_TracerExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 4;
|
|
expActor->spr.yrepeat = 4;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
|
|
if (actor->user.ID == BOLT_THINMAN_R1)
|
|
{
|
|
expActor->user.Radius = DamageData[DMG_BASIC_EXP].radius;
|
|
DoExpDamageTest(expActor);
|
|
}
|
|
else
|
|
expActor->user.Radius = DamageData[DMG_BOLT_EXP].radius;
|
|
}
|
|
|
|
void SpawnMicroExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return ;
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, MICRO_EXP, s_MicroExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 32;
|
|
expActor->spr.yrepeat = 32;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
expActor->user.Radius = DamageData[DMG_BOLT_EXP].radius;
|
|
|
|
//
|
|
// All this stuff assures that explosions do not go into floors &
|
|
// ceilings
|
|
//
|
|
|
|
SpawnExpZadjust(actor, expActor, 20, 20);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
}
|
|
|
|
void AddSpriteToSectorObject(DSWActor* actor, SECTOR_OBJECT* sop)
|
|
{
|
|
unsigned sn;
|
|
|
|
// make sure it has a user
|
|
if (!actor->hasU())
|
|
{
|
|
SpawnUser(actor, 0, nullptr);
|
|
}
|
|
|
|
// find a free place on this list
|
|
for (sn = 0; sn < SIZ(sop->so_actors); sn++)
|
|
{
|
|
if (sop->so_actors[sn] == nullptr)
|
|
break;
|
|
}
|
|
|
|
if (sn >= SIZ(sop->so_actors) - 1) return;
|
|
sop->so_actors[sn] = actor;
|
|
so_setspriteinterpolation(sop, actor);
|
|
|
|
actor->user.Flags |= (SPR_ON_SO_SECTOR|SPR_SO_ATTACHED);
|
|
|
|
actor->user.pos.XY() = sop->pmid.XY() - actor->spr.pos.XY();
|
|
actor->user.pos.Z = sop->mid_sector->floorz - actor->spr.pos.Z;
|
|
|
|
actor->user.sang = actor->spr.angle;
|
|
}
|
|
|
|
void SpawnBigGunFlames(DSWActor* actor, DSWActor* Operator, SECTOR_OBJECT* sop, bool smallflames)
|
|
{
|
|
unsigned sn;
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, MICRO_EXP, s_BigGunFlame, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(Operator, expActor);
|
|
expActor->spr.shade = -40;
|
|
if (smallflames)
|
|
{
|
|
expActor->spr.xrepeat = 12;
|
|
expActor->spr.yrepeat = 12;
|
|
}
|
|
else
|
|
{
|
|
expActor->spr.xrepeat = 34;
|
|
expActor->spr.yrepeat = 34;
|
|
}
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) > 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
// place all sprites on list
|
|
for (sn = 0; sn < SIZ(sop->so_actors); sn++)
|
|
{
|
|
if (sop->so_actors[sn] == nullptr)
|
|
break;
|
|
}
|
|
|
|
if (sn >= SIZ(sop->so_actors) - 1) return;
|
|
|
|
sop->so_actors[sn] = expActor;
|
|
so_setspriteinterpolation(sop, expActor);
|
|
|
|
expActor->user.Flags |= (actor->user.Flags & (SPR_ON_SO_SECTOR|SPR_SO_ATTACHED));
|
|
|
|
if (actor->user.Flags & (SPR_ON_SO_SECTOR))
|
|
{
|
|
// move with sector its on
|
|
expActor->spr.pos.Z = actor->sector()->floorz - actor->user.pos.Z;
|
|
expActor->backupz();
|
|
}
|
|
else
|
|
{
|
|
// move with the mid sector
|
|
expActor->spr.pos.Z = sop->mid_sector->floorz - actor->user.pos.Z;
|
|
expActor->backupz();
|
|
}
|
|
|
|
expActor->user.pos = actor->user.pos;
|
|
}
|
|
|
|
void SpawnGrenadeSecondaryExp(DSWActor* actor, int ang)
|
|
{
|
|
int vel;
|
|
|
|
ASSERT(actor->hasU());
|
|
auto expActor = SpawnActor(STAT_MISSILE, GRENADE_EXP, s_GrenadeSmallExp, actor->sector(), actor->spr.pos, actor->spr.angle, 1024);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 32;
|
|
expActor->spr.yrepeat = 32;
|
|
expActor->spr.clipdist = actor->spr.clipdist;
|
|
expActor->user.ceiling_dist = (16);
|
|
expActor->user.floor_dist = (16);
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
//ang = RANDOM_P2(2048);
|
|
vel = (1024+512) + RandomRange(1024);
|
|
expActor->user.change.XY() = MOVExy(vel, DAngle::fromBuild(ang));
|
|
|
|
expActor->user.coll = move_missile(expActor, DVector3(expActor->user.change.XY(), 0),
|
|
expActor->user.ceiling_dist,expActor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (FindDistance3D(expActor->int_pos() - actor->int_pos()) < 1024)
|
|
{
|
|
KillActor(expActor);
|
|
return;
|
|
}
|
|
|
|
SpawnExpZadjust(actor, expActor, 50, 10);
|
|
expActor->backuppos();
|
|
}
|
|
|
|
int SpawnGrenadeSmallExp(DSWActor* actor)
|
|
{
|
|
int ang = RANDOM_P2(2048);
|
|
SpawnGrenadeSecondaryExp(actor, ang);
|
|
return 0;
|
|
}
|
|
|
|
void SpawnGrenadeExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return;
|
|
|
|
PlaySound(DIGI_30MMEXPLODE, actor, v3df_none);
|
|
|
|
if (RandomRange(1000) > 990)
|
|
{
|
|
auto own = GetOwner(actor);
|
|
if (own != nullptr && own->hasU() && own->user.PlayerP)
|
|
{
|
|
PlayerSound(DIGI_LIKEFIREWORKS, v3df_follow|v3df_dontpan, own->user.PlayerP);
|
|
}
|
|
}
|
|
|
|
auto pos = actor->spr.pos;
|
|
|
|
if (actor->user.ID == ZILLA_RUN_R0)
|
|
{
|
|
pos.X += (RandomRange(1000)-RandomRange(1000)) * maptoworld;
|
|
pos.Y += (RandomRange(1000) - RandomRange(1000)) * maptoworld;
|
|
pos.Z = ActorZOfMiddle(actor) + (RandomRange(1000)-RandomRange(1000)) * zmaptoworld;
|
|
}
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, GRENADE_EXP, s_GrenadeExp, actor->sector(), pos, actor->spr.angle, 0);
|
|
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 64 + 32;
|
|
expActor->spr.yrepeat = 64 + 32;
|
|
expActor->spr.clipdist = actor->spr.clipdist;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_GRENADE_EXP].radius;
|
|
|
|
//
|
|
// All this stuff assures that explosions do not go into floors &
|
|
// ceilings
|
|
//
|
|
|
|
SpawnExpZadjust(actor, expActor, 100, 30);
|
|
|
|
DoExpDamageTest(expActor);
|
|
|
|
SetExpQuake(expActor);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 0);
|
|
}
|
|
|
|
void SpawnExpZadjust(DSWActor* actor, DSWActor* expActor, double upper_zsize, double lower_zsize)
|
|
{
|
|
double tos_z, bos_z;
|
|
|
|
ASSERT(expActor->hasU());
|
|
|
|
if (actor->hasU())
|
|
{
|
|
tos_z = expActor->spr.pos.Z - upper_zsize;
|
|
bos_z = expActor->spr.pos.Z + lower_zsize;
|
|
|
|
if (tos_z <= actor->user.hiz + 4)
|
|
{
|
|
expActor->spr.pos.Z = actor->user.hiz + upper_zsize;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
}
|
|
else if (bos_z > actor->user.loz)
|
|
{
|
|
expActor->spr.pos.Z = actor->user.loz - lower_zsize;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
double cz,fz;
|
|
|
|
getzsofslopeptr(expActor->sector(), expActor->spr.pos, &cz, &fz);
|
|
|
|
tos_z = expActor->spr.pos.Z - upper_zsize;
|
|
bos_z = expActor->spr.pos.Z + lower_zsize;
|
|
|
|
if (tos_z <= cz + 4)
|
|
{
|
|
expActor->spr.pos.Z = cz + upper_zsize;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
}
|
|
else if (bos_z > fz)
|
|
{
|
|
expActor->spr.pos.Z = fz - lower_zsize;
|
|
}
|
|
}
|
|
|
|
expActor->backupz();
|
|
}
|
|
|
|
void SpawnMineExp(DSWActor* actor)
|
|
{
|
|
ASSERT(actor->hasU());
|
|
if (actor->hasU() && (actor->user.Flags & SPR_SUICIDE))
|
|
return;
|
|
|
|
change_actor_stat(actor, STAT_MISSILE);
|
|
|
|
PlaySound(DIGI_MINEBLOW, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, MINE_EXP, s_MineExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(GetOwner(actor), expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 64 + 44;
|
|
expActor->spr.yrepeat = 64 + 44;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_MINE_EXP].radius;
|
|
|
|
//
|
|
// All this stuff assures that explosions do not go into floors &
|
|
// ceilings
|
|
//
|
|
|
|
SpawnExpZadjust(actor, expActor, 100, 20);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
|
|
SetExpQuake(expActor);
|
|
}
|
|
|
|
|
|
int DoMineExp(DSWActor* actor)
|
|
{
|
|
DoExpDamageTest(actor);
|
|
return 0;
|
|
}
|
|
|
|
int DoSectorExp(DSWActor* actor)
|
|
{
|
|
actor->spr.pos += actor->user.change.XY();
|
|
return 0;
|
|
}
|
|
|
|
DSWActor* SpawnSectorExp(DSWActor* actor)
|
|
{
|
|
short explosion;
|
|
|
|
ASSERT(actor->hasU());
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return nullptr;
|
|
|
|
PlaySound(DIGI_30MMEXPLODE, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, GRENADE_EXP, s_SectorExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 90; // was 40,40
|
|
expActor->spr.yrepeat = 90;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_SECTOR_EXP].radius;
|
|
|
|
DoExpDamageTest(expActor);
|
|
SetExpQuake(expActor);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
|
|
return expActor;
|
|
}
|
|
|
|
// called from SpawnShrap
|
|
DSWActor* SpawnLargeExp(DSWActor* actor)
|
|
{
|
|
PlaySound(DIGI_30MMEXPLODE, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, GRENADE_EXP, s_SectorExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 90; // was 40,40
|
|
expActor->spr.yrepeat = 90;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_SECTOR_EXP].radius;
|
|
|
|
SpawnExpZadjust(actor, expActor, 50, 50);
|
|
|
|
// Should not cause other sectors to explode
|
|
DoExpDamageTest(expActor);
|
|
SetExpQuake(expActor);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
|
|
return expActor;
|
|
}
|
|
|
|
void SpawnMeteorExp(DSWActor* actor)
|
|
{
|
|
DSWActor* expActor;
|
|
|
|
ASSERT(actor->hasU());
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return;
|
|
|
|
if (actor->user.spal == 25) // Serp ball
|
|
{
|
|
expActor = SpawnActor(STAT_MISSILE, METEOR_EXP, s_TeleportEffect2, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
}
|
|
else
|
|
{
|
|
PlaySound(DIGI_MEDIUMEXP, actor, v3df_none);
|
|
expActor = SpawnActor(STAT_MISSILE, METEOR_EXP, s_MeteorExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
}
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
expActor->spr.shade = -40;
|
|
if (actor->spr.yrepeat < 64)
|
|
{
|
|
// small
|
|
expActor->spr.xrepeat = 64;
|
|
expActor->spr.yrepeat = 64;
|
|
}
|
|
else
|
|
{
|
|
// large - boss
|
|
expActor->spr.xrepeat = 80;
|
|
expActor->spr.yrepeat = 80;
|
|
}
|
|
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_BASIC_EXP].radius;
|
|
}
|
|
|
|
void SpawnLittleExp(DSWActor* actor)
|
|
{
|
|
short explosion;
|
|
|
|
PlaySound(DIGI_HEADSHOTHIT, actor, v3df_none);
|
|
auto expActor = SpawnActor(STAT_MISSILE, BOLT_EXP, s_SectorExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
expActor->spr.shade = -127;
|
|
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_BASIC_EXP].radius;
|
|
DoExpDamageTest(expActor);
|
|
SpawnVis(nullptr, expActor->sector(), expActor->spr.pos, 16);
|
|
}
|
|
|
|
int DoFireball(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
actor->spr.xrepeat = actor->spr.yrepeat -= 1;
|
|
if (actor->spr.xrepeat <= 37)
|
|
{
|
|
SpawnSmokePuff(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
bool hit_burn = false;
|
|
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitSprite:
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if ((hitActor->spr.extra & SPRX_BURNABLE))
|
|
{
|
|
if (!hitActor->hasU())
|
|
SpawnUser(hitActor, hitActor->spr.picnum, nullptr);
|
|
SpawnFireballFlames(actor, hitActor);
|
|
hit_burn = true;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!hit_burn)
|
|
{
|
|
if (actor->user.ID == GORO_FIREBALL)
|
|
SpawnGoroFireballExp(actor);
|
|
else
|
|
SpawnFireballExp(actor);
|
|
}
|
|
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int DoFindGround(DSWActor* actor)
|
|
{
|
|
Collision ceilhit, florhit;
|
|
|
|
// recursive routine to find the ground - either sector or floor sprite
|
|
// skips over enemy and other types of sprites
|
|
|
|
auto save_cstat = actor->spr.cstat;
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
FAFgetzrange(actor->int_pos(), actor->sector(), &actor->user.hiz, &ceilhit, &actor->user.loz, &florhit, (((int) actor->spr.clipdist) << 2) - GETZRANGE_CLIP_ADJ, CLIPMASK_PLAYER);
|
|
actor->spr.cstat = save_cstat;
|
|
|
|
switch (florhit.type)
|
|
{
|
|
case kHitSprite:
|
|
{
|
|
auto florActor = florhit.actor();
|
|
|
|
if ((florActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_FLOOR))
|
|
{
|
|
// found a sprite floor
|
|
actor->user.lowActor = florActor;
|
|
actor->user.lo_sectp = nullptr;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
// reset the blocking bit of what you hit and try again -
|
|
// recursive
|
|
auto bak_cstat = florActor->spr.cstat;
|
|
florActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
DoFindGround(actor);
|
|
florActor->spr.cstat = bak_cstat;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
case kHitSector:
|
|
{
|
|
actor->user.lo_sectp = florhit.hitSector;
|
|
actor->user.lowActor = nullptr;
|
|
return true;
|
|
}
|
|
|
|
default:
|
|
ASSERT(true == false);
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoFindGroundPoint(DSWActor* actor)
|
|
{
|
|
Collision ceilhit, florhit;
|
|
|
|
// recursive routine to find the ground - either sector or floor sprite
|
|
// skips over enemy and other types of sprites
|
|
|
|
auto save_cstat = actor->spr.cstat;
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
FAFgetzrangepoint(actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->sector(), &actor->user.hiz, &ceilhit, &actor->user.loz, &florhit);
|
|
actor->spr.cstat = save_cstat;
|
|
|
|
switch (florhit.type)
|
|
{
|
|
case kHitSprite:
|
|
{
|
|
auto florActor = florhit.actor();
|
|
|
|
if ((florActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_FLOOR))
|
|
{
|
|
// found a sprite floor
|
|
actor->user.lowActor = florActor;
|
|
actor->user.lo_sectp = nullptr;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
// reset the blocking bit of what you hit and try again -
|
|
// recursive
|
|
auto bak_cstat = florActor->spr.cstat;
|
|
florActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
DoFindGroundPoint(actor);
|
|
florActor->spr.cstat = bak_cstat;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
case kHitSector:
|
|
{
|
|
actor->user.lo_sectp = florhit.hitSector;
|
|
actor->user.lowActor = nullptr;
|
|
return true;
|
|
}
|
|
|
|
default:
|
|
ASSERT(true == false);
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int DoNapalm(DSWActor* actor)
|
|
{
|
|
DoBlurExtend(actor, 1, 7);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
actor->spr.xrepeat = actor->spr.yrepeat -= 1;
|
|
if (actor->spr.xrepeat <= 30)
|
|
{
|
|
SpawnSmokePuff(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
auto oldv = actor->spr.pos;
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
{
|
|
// this sprite is suPlayerosed to go through players/enemys
|
|
// if hit a player/enemy back up and do it again with blocking reset
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_BLOCK) && !(hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
auto hcstat = hitActor->spr.cstat;
|
|
|
|
actor->spr.pos = oldv;
|
|
|
|
hitActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
hitActor->spr.cstat = hcstat;
|
|
}
|
|
}
|
|
}
|
|
|
|
actor->user.Counter++;
|
|
if (actor->user.Counter > 2)
|
|
actor->user.Counter = 0;
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
PlaySound(DIGI_NAPPUFF, actor, v3df_none);
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, NAP_EXP, s_NapExp, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(actor, expActor);
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.cstat = actor->spr.cstat;
|
|
expActor->spr.xrepeat = 48;
|
|
expActor->spr.yrepeat = 64;
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
if (RANDOM_P2(1024) < 512)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_TRANSLUCENT);
|
|
expActor->user.Radius = 1500;
|
|
|
|
DoFindGroundPoint(expActor);
|
|
MissileWaterAdjust(expActor);
|
|
expActor->spr.pos.Z = expActor->user.loz;
|
|
expActor->backupz();
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
expActor->user.Flags |= SPR_UNDERWATER;
|
|
|
|
ASSERT(expActor->spr.picnum == 3072);
|
|
ASSERT(expActor->user.Tics == 0);
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoBloodWorm(DSWActor* actor)
|
|
{
|
|
short ang;
|
|
int xvect,yvect;
|
|
int amt;
|
|
|
|
actor->user.coll = move_ground_missile(actor, actor->user.change.XY(), actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
actor->user.change.XY() = -actor->user.change.XY();
|
|
actor->user.coll.setNone();
|
|
actor->spr.angle += DAngle180;
|
|
return true;
|
|
}
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (!actor->user.z_tgt)
|
|
{
|
|
// stay alive for 10 seconds
|
|
if (++actor->user.Counter3 > 3)
|
|
{
|
|
InitBloodSpray(actor, false, 1);
|
|
InitBloodSpray(actor, false, 1);
|
|
InitBloodSpray(actor, false, 1);
|
|
|
|
// Kill any old zombies you own
|
|
SWStatIterator it(STAT_ENEMY);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (!itActor->hasU()) continue;
|
|
|
|
if (itActor->user.ID == ZOMBIE_RUN_R0 && GetOwner(itActor) == GetOwner(actor))
|
|
{
|
|
InitBloodSpray(itActor, true, 105);
|
|
InitBloodSpray(itActor, true, 105);
|
|
SetSuicide(itActor);
|
|
break;
|
|
}
|
|
}
|
|
|
|
SpawnZombie2(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
ang = NORM_ANGLE(actor->int_ang() + 512);
|
|
|
|
xvect = bcos(ang);
|
|
yvect = bsin(ang);
|
|
|
|
auto bpos = actor->spr.pos.XY();
|
|
|
|
amt = RANDOM_P2(2048) - 1024;
|
|
actor->add_int_pos({ MulScale(amt,xvect, 15), MulScale(amt,yvect, 15), 0 });
|
|
|
|
auto sect = actor->sector();
|
|
updatesectorz(actor->spr.pos, §);
|
|
if (sect)
|
|
{
|
|
GlobalSkipZrange = true;
|
|
InitBloodSpray(actor, false, 1);
|
|
GlobalSkipZrange = false;
|
|
}
|
|
|
|
actor->spr.pos.XY() = bpos;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoMeteor(DSWActor* actor)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
int DoSerpMeteor(DSWActor* actor)
|
|
{
|
|
auto oldv = actor->spr.pos;
|
|
|
|
actor->spr.xrepeat += MISSILEMOVETICS * 2;
|
|
if (actor->spr.xrepeat > 80)
|
|
actor->spr.xrepeat = 80;
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
// this sprite is supposed to go through players/enemys
|
|
// if hit a player/enemy back up and do it again with blocking reset
|
|
if (actor->user.coll.type == kHitSprite)
|
|
{
|
|
auto hitActor = actor->user.coll.actor();
|
|
|
|
if (hitActor->hasU() && hitActor->user.ID >= SKULL_R0 && hitActor->user.ID <= SKULL_SERP)
|
|
{
|
|
auto hcstat = hitActor->spr.cstat;
|
|
|
|
actor->spr.pos = oldv;
|
|
|
|
hitActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
hitActor->spr.cstat = hcstat;
|
|
}
|
|
}
|
|
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnMeteorExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoMirvMissile(DSWActor* actor)
|
|
{
|
|
actor->spr.xrepeat += MISSILEMOVETICS * 2;
|
|
if (actor->spr.xrepeat > 80)
|
|
actor->spr.xrepeat = 80;
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
if (WeaponMoveHit(actor))
|
|
{
|
|
SpawnMeteorExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int DoMirv(DSWActor* actor)
|
|
{
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
|
|
SpawnBubble(actor);
|
|
|
|
|
|
actor->user.Counter++;
|
|
actor->user.Counter &= 1;
|
|
|
|
if (!actor->user.Counter)
|
|
{
|
|
int i;
|
|
static short angs[] =
|
|
{
|
|
512,
|
|
-512
|
|
};
|
|
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, MIRV_METEOR, &sg_MirvMeteor[0][0], actor->sector(),
|
|
actor->spr.pos, actor->spr.angle + DAngle::fromBuild(angs[i]), 800);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_MirvMeteor[0]);
|
|
actorNew->user.StateEnd = s_MirvMeteorExp;
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->spr.clipdist = 32L >> 2;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.ceiling_dist = (16);
|
|
actorNew->user.floor_dist = (16);
|
|
actorNew->user.Dist = 12.5;
|
|
//actorNew->user.Dist = 0;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
|
|
}
|
|
|
|
if (actor->user.coll.type != kHitNone)
|
|
{
|
|
SpawnMeteorExp(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool MissileSetPos(DSWActor* actor, ANIMATOR* DoWeapon, int dist)
|
|
{
|
|
int oldvel, oldzvel;
|
|
bool retval = false;
|
|
|
|
// backup values
|
|
auto oldc = actor->user.change;
|
|
oldvel = actor->int_xvel();
|
|
oldzvel = actor->int_zvel();
|
|
|
|
// make missile move in smaller increments
|
|
actor->set_int_xvel(short((dist * 6) / MISSILEMOVETICS));
|
|
actor->set_int_zvel(short((actor->int_zvel() * 6) / MISSILEMOVETICS));
|
|
|
|
// some Weapon Animators use this
|
|
UpdateChange(actor);
|
|
|
|
actor->user.Flags |= (SPR_SET_POS_DONT_KILL);
|
|
if ((*DoWeapon)(actor))
|
|
retval = true;
|
|
actor->user.Flags &= ~(SPR_SET_POS_DONT_KILL);
|
|
|
|
// reset values
|
|
actor->user.change = oldc;
|
|
actor->set_int_xvel(oldvel);
|
|
actor->set_int_zvel(oldzvel);
|
|
|
|
// update for interpolation
|
|
actor->backuppos();
|
|
|
|
return retval;
|
|
}
|
|
|
|
bool TestMissileSetPos(DSWActor* actor, ANIMATOR* DoWeapon, int dist, int zvel)
|
|
{
|
|
int oldvel, oldzvel;
|
|
bool retval = false;
|
|
|
|
// backup values
|
|
auto oldc = actor->user.change;
|
|
oldvel = actor->int_xvel();
|
|
oldzvel = actor->int_zvel();
|
|
|
|
// make missile move in smaller increments
|
|
actor->set_int_xvel(short((dist * 6) / MISSILEMOVETICS));
|
|
zvel = short((zvel*6) / MISSILEMOVETICS);
|
|
|
|
// some Weapon Animators use this
|
|
UpdateChangeXY(actor);
|
|
actor->user.set_int_change_z(zvel);
|
|
|
|
actor->user.Flags |= (SPR_SET_POS_DONT_KILL);
|
|
if ((*DoWeapon)(actor))
|
|
retval = true;
|
|
actor->user.Flags &= ~(SPR_SET_POS_DONT_KILL);
|
|
|
|
// reset values
|
|
actor->user.change = oldc;
|
|
actor->set_int_xvel(oldvel);
|
|
actor->set_int_zvel(oldzvel);
|
|
|
|
// update for interpolation
|
|
actor->backuppos();
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
constexpr int RINGMOVETICS = (MISSILEMOVETICS * 2);
|
|
constexpr double RING_OUTER_DIST = 200;
|
|
constexpr double RING_INNER_DIST = 50;
|
|
|
|
int DoRing(DSWActor* actor)
|
|
{
|
|
auto own = GetOwner(actor);
|
|
if (!own) return 0; // this would crash.
|
|
PLAYER* pp = own->user.PlayerP;;
|
|
double cz,fz;
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
actor->spr.xrepeat = actor->spr.yrepeat -= 1;
|
|
if (actor->spr.xrepeat <= 30)
|
|
{
|
|
SpawnSmokePuff(actor);
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
double z;
|
|
// move the center with the player
|
|
if (pp)
|
|
z = pp->pos.Z + 20;
|
|
else
|
|
z = ActorZOfMiddle(own) + 30;
|
|
|
|
actor->spr.pos = DVector3(own->spr.pos.XY(), z);
|
|
|
|
// go out until its time to come back in
|
|
if (actor->user.Counter2 == false)
|
|
{
|
|
actor->user.Dist += 0.5 * RINGMOVETICS;
|
|
|
|
if (actor->user.Dist > RING_OUTER_DIST)
|
|
{
|
|
actor->user.Counter2 = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
actor->user.Dist -= 0.5 * RINGMOVETICS;
|
|
|
|
if (actor->user.Dist <= RING_INNER_DIST)
|
|
{
|
|
if (!pp)
|
|
own->user.Counter--;
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// rotate the ring
|
|
actor->set_int_ang(NORM_ANGLE(actor->int_ang() + (4 * RINGMOVETICS) + RINGMOVETICS));
|
|
|
|
// put it out there
|
|
actor->spr.pos += actor->spr.angle.ToVector() * actor->user.Dist;
|
|
if (pp) actor->spr.pos.Z += (actor->user.Dist * (-pp->horizon.horiz.asq16() >> 9)) * (1/ (512.*16.)); // horizon math sucks...
|
|
|
|
SetActor(actor, actor->spr.pos);
|
|
|
|
ASSERT(actor->insector());
|
|
|
|
getzsofslopeptr(actor->sector(), actor->spr.pos, &cz, &fz);
|
|
|
|
// bound the sprite by the sectors ceiling and floor
|
|
if (actor->spr.pos.Z > fz)
|
|
{
|
|
actor->spr.pos.Z = fz;
|
|
}
|
|
|
|
if (actor->spr.pos.Z < cz + ActorSizeZ(actor))
|
|
{
|
|
actor->spr.pos.Z = (cz + ActorSizeZ(actor));
|
|
}
|
|
|
|
// Done last - check for damage
|
|
DoDamageTest(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
void InitSpellRing(PLAYER* pp)
|
|
{
|
|
short ang, ang_diff, ang_start, missiles;
|
|
short max_missiles = 16;
|
|
short ammo;
|
|
|
|
ammo = NAPALM_MIN_AMMO;
|
|
|
|
if (pp->WpnAmmo[WPN_HOTHEAD] < ammo)
|
|
return;
|
|
else
|
|
PlayerUpdateAmmo(pp, WPN_HOTHEAD, -ammo);
|
|
|
|
ang_diff = 2048 / max_missiles;
|
|
|
|
ang_start = NORM_ANGLE(pp->angle.ang.Buildang() - (2048 / 2));
|
|
|
|
if (!SW_SHAREWARE)
|
|
PlaySound(DIGI_RFWIZ, pp, v3df_none);
|
|
|
|
if (!pp->insector())
|
|
return;
|
|
|
|
for (missiles = 0, ang = ang_start; missiles < max_missiles; ang += ang_diff, missiles++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE_SKIP4, FIREBALL1, s_Ring, pp->cursector, pp->pos, DAngle::fromBuild(ang), 0);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->set_int_xvel(500);
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->vel.Z = 0;
|
|
|
|
actorNew->user.pos.Z = 20;
|
|
actorNew->user.Dist = RING_INNER_DIST;
|
|
actorNew->user.Counter = max_missiles;
|
|
actorNew->user.Counter2 = 0;
|
|
actorNew->user.ceiling_dist = 10;
|
|
actorNew->user.floor_dist = 10;
|
|
|
|
// put it out there
|
|
actorNew->spr.pos += actorNew->spr.angle.ToVector() * actorNew->user.Dist;
|
|
actorNew->spr.pos.Z += pp->pos.Z + 20 + (actorNew->user.Dist * (-pp->horizon.horiz.asq16() >> 9)) * (1 / (512. * 16.)); // horizon math sucks...
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
|
|
actorNew->backuppos();
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
}
|
|
|
|
int DoSerpRing(DSWActor* actor)
|
|
{
|
|
int dist,a,b,c;
|
|
double cz,fz;
|
|
|
|
auto own = GetOwner(actor);
|
|
// if Owner does not exist or he's dead on the floor
|
|
// kill off all of his skull children
|
|
if (own == nullptr || own->user.RotNum < 5)
|
|
{
|
|
UpdateSinglePlayKills(actor);
|
|
DoSkullBeginDeath(actor);
|
|
// +2 does not spawn shrapnel
|
|
actor->user.ID = SKULL_SERP;
|
|
return 0;
|
|
}
|
|
|
|
double zz = actor->int_pos().Z + actor->vel.Z;
|
|
if (zz > own->spr.pos.Z - actor->user.pos.Z)
|
|
zz = own->spr.pos.Z - actor->user.pos.Z;
|
|
|
|
// move the center with the player
|
|
actor->spr.pos = DVector3(own->spr.pos.XY(), zz);
|
|
|
|
|
|
// go out until its time to come back in
|
|
if (actor->user.Counter2 == false)
|
|
{
|
|
actor->user.Dist += 0.5 * RINGMOVETICS;
|
|
|
|
if (actor->user.Dist > actor->user.TargetDist)
|
|
actor->user.Counter2 = true;
|
|
}
|
|
|
|
// rotate the ring
|
|
actor->user.slide_ang += DAngle::fromBuild(actor->spr.yint);
|
|
|
|
// rotate the heads
|
|
if (actor->user.Flags & (SPR_BOUNCE))
|
|
actor->set_int_ang(NORM_ANGLE(actor->int_ang() + (28 * RINGMOVETICS)));
|
|
else
|
|
actor->set_int_ang(NORM_ANGLE(actor->int_ang() - (28 * RINGMOVETICS)));
|
|
|
|
// put it out there
|
|
actor->spr.pos += actor->user.slide_ang.ToVector() * actor->user.Dist;
|
|
|
|
SetActor(actor, actor->spr.pos);
|
|
|
|
ASSERT(actor->insector());
|
|
|
|
getzsofslopeptr(actor->sector(), actor->spr.pos, &cz, &fz);
|
|
|
|
// bound the sprite by the sectors ceiling and floor
|
|
if (actor->spr.pos.Z > fz)
|
|
{
|
|
actor->spr.pos.Z = fz;
|
|
}
|
|
|
|
if (actor->spr.pos.Z < cz + ActorSizeZ(actor))
|
|
{
|
|
actor->spr.pos.Z = cz + ActorSizeZ(actor);
|
|
}
|
|
|
|
if (actor->user.Counter2 > 0)
|
|
{
|
|
DSWActor* tActor = own->user.targetActor;
|
|
if (!tActor->hasU() ||
|
|
!tActor->user.PlayerP ||
|
|
!(tActor->user.PlayerP->Flags & PF_DEAD))
|
|
{
|
|
actor->user.targetActor = own->user.targetActor;
|
|
DISTANCE(actor->spr.pos, actor->user.targetActor->spr.pos, dist, a,b,c);
|
|
|
|
// if ((dist ok and random ok) OR very few skulls left)
|
|
if ((dist < 18000 && (RANDOM_P2(2048<<5)>>5) < 16) || own->user.Counter < 4)
|
|
{
|
|
auto sect = actor->sector();
|
|
updatesector(actor->spr.pos, §);
|
|
|
|
// if (valid sector and can see target)
|
|
if (sect != nullptr && CanSeePlayer(actor))
|
|
{
|
|
extern STATE* sg_SkullJump[];
|
|
actor->user.ID = SKULL_R0;
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
actor->set_int_xvel(dist>>5);
|
|
actor->add_int_xvel( (actor->int_xvel() >> 1));
|
|
actor->add_int_xvel( (RANDOM_P2(128<<8)>>8));
|
|
actor->user.jump_speed = -800;
|
|
change_actor_stat(actor, STAT_ENEMY);
|
|
NewStateGroup(actor, sg_SkullJump);
|
|
DoBeginJump(actor);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitLavaFlame(DSWActor* actor)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void SetZVelFromTarget(DSWActor* actorNew, DSWActor* actor, bool setchange = false, double offset = 0)
|
|
{
|
|
// find the distance to the target (player)
|
|
double dist = (actorNew->spr.pos.XY() - actor->user.targetActor->spr.pos.XY()).Length();
|
|
|
|
if (dist != 0)
|
|
{
|
|
double zdist = (ActorUpperZ(actor->user.targetActor) - actorNew->spr.pos.Z - offset) / dist;
|
|
double change = zdist * actorNew->int_xvel() * inttoworld;
|
|
actorNew->set_int_zvel((change * zworldtoint));
|
|
if (setchange) actorNew->user.change.Z = change;
|
|
}
|
|
}
|
|
|
|
int InitLavaThrow(DSWActor* actor)
|
|
{
|
|
short w;
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
double nz = ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25);
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, LAVA_BOULDER, s_LavaBoulder, actor->sector(),
|
|
DVector3(actor->spr.pos.XY(), nz), actor->spr.angle, NINJA_BOLT_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.yrepeat = 72;
|
|
actorNew->spr.xrepeat = 72;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
|
|
if (RANDOM_P2(1024) > 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) > 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 256>>2;
|
|
actorNew->user.ceiling_dist = (14);
|
|
actorNew->user.floor_dist = (14);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoLavaBoulder, 1200);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
return 0;
|
|
}
|
|
|
|
void InitVulcanBoulder(DSWActor* actor)
|
|
{
|
|
DAngle nang;
|
|
double zsize;
|
|
int zvel, zvel_rand;
|
|
int delta;
|
|
int vel;
|
|
|
|
if (SP_TAG7(actor))
|
|
{
|
|
delta = SP_TAG5(actor);
|
|
nang = actor->spr.angle + DAngle::fromBuild((RandomRange(delta) - (delta >> 1)));
|
|
}
|
|
else
|
|
{
|
|
nang = RandomAngle();
|
|
}
|
|
|
|
if (SP_TAG6(actor))
|
|
vel = SP_TAG6(actor);
|
|
else
|
|
vel = 800;
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, LAVA_BOULDER, s_VulcanBoulder, actor->sector(), actor->spr.pos.plusZ(-40), nang, (vel/2 + vel/4) + RandomRange(vel/4));
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.xrepeat = actorNew->spr.yrepeat = 8 + RandomRange(72);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.angle = nang;
|
|
actorNew->user.Counter = 0;
|
|
|
|
zsize = ActorSizeZ(actorNew);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 256>>2;
|
|
actorNew->user.ceiling_dist = zsize/2;
|
|
actorNew->user.floor_dist = zsize/2;
|
|
if (RANDOM_P2(1024) > 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) > 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
if (SP_TAG7(actor))
|
|
{
|
|
zvel = SP_TAG7(actor);
|
|
zvel_rand = SP_TAG8(actor);
|
|
}
|
|
else
|
|
{
|
|
zvel = 50;
|
|
zvel_rand = 40;
|
|
}
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.change.Z = -zvel -RandomRange(zvel_rand);
|
|
}
|
|
|
|
int InitSerpRing(DSWActor* actor)
|
|
{
|
|
short ang, ang_diff, ang_start, missiles;
|
|
short max_missiles;
|
|
|
|
const int SERP_RING_DIST = 175;
|
|
|
|
extern STATE s_SkullExplode[];
|
|
extern STATE s_SkullRing[5][1];
|
|
extern STATE* sg_SkullRing[];
|
|
|
|
max_missiles = 12;
|
|
|
|
actor->user.Counter = max_missiles;
|
|
|
|
ang_diff = 2048 / max_missiles;
|
|
|
|
ang_start = NORM_ANGLE(actor->int_ang() - (2048 / 2));
|
|
|
|
PlaySound(DIGI_SERPSUMMONHEADS, actor, v3df_none);
|
|
|
|
for (missiles = 0, ang = ang_start; missiles < max_missiles; ang += ang_diff, missiles++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_SKIP4, SKULL_SERP, &s_SkullRing[0][0], actor->sector(), actor->spr.pos, DAngle::fromBuild(ang), 0);
|
|
|
|
actorNew->set_int_xvel(500);
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -20;
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.yint = 2*RINGMOVETICS;
|
|
actorNew->set_int_zvel(Z(3));
|
|
actorNew->spr.pal = 0;
|
|
|
|
actorNew->spr.pos.Z = ActorZOfTop(actor) - 20;
|
|
actorNew->user.pos.Z = 50;
|
|
|
|
// ang around the serp is now slide_ang
|
|
actorNew->user.slide_ang = actorNew->spr.angle;
|
|
// randomize the head turning angle
|
|
actorNew->set_int_ang(RANDOM_P2(2048<<5)>>5);
|
|
|
|
// control direction of spinning
|
|
actor->user.Flags ^= SPR_BOUNCE;
|
|
actorNew->user.Flags |= (actor->user.Flags & (SPR_BOUNCE));
|
|
|
|
actorNew->user.Dist = 37.5;
|
|
actorNew->user.TargetDist = SERP_RING_DIST;
|
|
actorNew->user.Counter2 = 0;
|
|
|
|
actorNew->user.StateEnd = s_SkullExplode;
|
|
actorNew->user.Rot = sg_SkullRing;
|
|
|
|
// defaults do change the statnum
|
|
EnemyDefaults(actorNew, nullptr, nullptr);
|
|
change_actor_stat(actorNew, STAT_SKIP4);
|
|
actorNew->spr.extra &= ~(SPRX_PLAYER_OR_ENEMY);
|
|
|
|
actorNew->spr.clipdist = (128+64) >> 2;
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.Radius = 400;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void InitSpellNapalm(PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
unsigned i;
|
|
short ammo;
|
|
|
|
static const MISSILE_PLACEMENT mp[] =
|
|
{
|
|
{600 * 6, 400, 512},
|
|
{0, 1100, 0},
|
|
{600 * 6, 400, -512},
|
|
};
|
|
|
|
ammo = NAPALM_MIN_AMMO;
|
|
|
|
if (pp->WpnAmmo[WPN_HOTHEAD] < ammo)
|
|
return;
|
|
else
|
|
PlayerUpdateAmmo(pp, WPN_HOTHEAD, -ammo);
|
|
|
|
PlaySound(DIGI_NAPFIRE, pp, v3df_none);
|
|
|
|
if (!pp->insector())
|
|
return;
|
|
|
|
for (i = 0; i < SIZ(mp); i++)
|
|
{
|
|
auto actor = SpawnActor(STAT_MISSILE, FIREBALL1, s_Napalm, pp->cursector,
|
|
pp->pos.plusZ(12), pp->angle.ang, NAPALM_VELOCITY*2);
|
|
|
|
actor->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
if (i==0) // Only attach sound to first projectile
|
|
{
|
|
PlaySound(DIGI_NAPWIZ, actor, v3df_follow);
|
|
}
|
|
|
|
SetOwner(pp->actor, actor);
|
|
actor->spr.shade = -40;
|
|
actor->spr.xrepeat = 32;
|
|
actor->spr.yrepeat = 32;
|
|
actor->spr.clipdist = 0;
|
|
actor->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actor->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.Flags2 |= (SPR2_BLUR_TAPER_FAST);
|
|
|
|
actor->user.floor_dist = (1);
|
|
actor->user.ceiling_dist = (1);
|
|
actor->user.Dist = 12.5;
|
|
|
|
auto oclipdist = plActor->spr.clipdist;
|
|
plActor->spr.clipdist = 1;
|
|
|
|
if (mp[i].dist_over != 0)
|
|
{
|
|
actor->set_int_ang(NORM_ANGLE(actor->int_ang() + mp[i].ang));
|
|
HelpMissileLateral(actor, mp[i].dist_over);
|
|
actor->set_int_ang(NORM_ANGLE(actor->int_ang() - mp[i].ang));
|
|
}
|
|
|
|
UpdateChange(actor);
|
|
|
|
if (MissileSetPos(actor, DoNapalm, mp[i].dist_out))
|
|
{
|
|
plActor->spr.clipdist = oclipdist;
|
|
KillActor(actor);
|
|
continue;
|
|
}
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actor))
|
|
actor->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
plActor->spr.clipdist = oclipdist;
|
|
|
|
actor->user.Counter = 0;
|
|
|
|
}
|
|
}
|
|
|
|
int InitEnemyNapalm(DSWActor* actor)
|
|
{
|
|
unsigned i;
|
|
|
|
static const MISSILE_PLACEMENT mp[] =
|
|
{
|
|
{600 * 6, 400, 512},
|
|
{0, 1100, 0},
|
|
{600 * 6, 400, -512},
|
|
};
|
|
|
|
PlaySound(DIGI_NAPFIRE, actor, v3df_none);
|
|
|
|
for (i = 0; i < SIZ(mp); i++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL1, s_Napalm, actor->sector(),
|
|
DVector3(actor->spr.pos, ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25)), actor->spr.angle, NAPALM_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
if (i==0) // Only attach sound to first projectile
|
|
{
|
|
PlaySound(DIGI_NAPWIZ, actorNew, v3df_follow);
|
|
}
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.clipdist = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.Flags2 |= (SPR2_BLUR_TAPER_FAST);
|
|
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 1;
|
|
|
|
if (mp[i].dist_over != 0)
|
|
{
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + mp[i].ang));
|
|
HelpMissileLateral(actorNew, mp[i].dist_over);
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() - mp[i].ang));
|
|
}
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoNapalm, mp[i].dist_out);
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actor->user.Counter = 0;
|
|
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int InitSpellMirv(PLAYER* pp)
|
|
{
|
|
PlaySound(DIGI_MIRVFIRE, pp, v3df_none);
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL1, s_Mirv, pp->cursector, pp->pos.plusZ(12), pp->angle.ang, MIRV_VELOCITY);
|
|
|
|
PlaySound(DIGI_MIRVWIZ, actorNew, v3df_follow);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 72;
|
|
actorNew->spr.yrepeat = 72;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.floor_dist = (16);
|
|
actorNew->user.ceiling_dist = (16);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
DSWActor* plActor = pp->actor;
|
|
auto oclipdist = plActor->spr.clipdist;
|
|
plActor->spr.clipdist = 0;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoMirv, 600);
|
|
plActor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->user.Counter = 0;
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyMirv(DSWActor* actor)
|
|
{
|
|
PlaySound(DIGI_MIRVFIRE, actor, v3df_none);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, MIRV_METEOR, s_Mirv, actor->sector(),
|
|
DVector3(actor->spr.pos, ActorZOfTop(actor) + (ActorSizeZ(actor) * 0.25)), actor->spr.angle, MIRV_VELOCITY);
|
|
|
|
PlaySound(DIGI_MIRVWIZ, actorNew, v3df_follow);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 72;
|
|
actorNew->spr.yrepeat = 72;
|
|
actorNew->spr.clipdist = 32L >> 2;
|
|
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->user.floor_dist = (16);
|
|
actorNew->user.ceiling_dist = (16);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoMirv, 600);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitSwordAttack(PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
unsigned stat;
|
|
int dist;
|
|
short reach, face;
|
|
|
|
PlaySound(DIGI_SWORDSWOOSH, pp, v3df_dontpan | v3df_doppler);
|
|
|
|
if (pp->Flags & (PF_DIVING))
|
|
{
|
|
DSWActor* bubble;
|
|
|
|
static const DAngle dangs[] =
|
|
{
|
|
-DAngle45, -DAngle22_5, nullAngle, DAngle22_5, DAngle45
|
|
};
|
|
|
|
for (size_t i = 0; i < countof(dangs); i++)
|
|
{
|
|
if (RandomRange(1000) < 500) continue; // Don't spawn bubbles every time
|
|
bubble = SpawnBubble(pp->actor);
|
|
if (bubble != nullptr)
|
|
{
|
|
bubble->set_int_ang(pp->angle.ang.Buildang());
|
|
|
|
auto random_amt = RandomAngle(DAngle22_5 / 4) - DAngle22_5 / 8;
|
|
|
|
// back it up a bit to get it out of your face
|
|
auto vec = MOVExy((1024 + 256) * 3, bubble->spr.angle + dangs[i] + random_amt);
|
|
|
|
move_missile(bubble, DVector3(vec, 0), plActor->user.ceiling_dist, plActor->user.floor_dist, CLIPMASK_PLAYER, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (!itActor->hasU())
|
|
break;
|
|
|
|
if (itActor->user.PlayerP == pp)
|
|
break;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
dist = DistanceI(pp->pos, itActor->spr.pos);
|
|
|
|
reach = 1000; // !JIM! was 800
|
|
face = 200;
|
|
|
|
if (dist < CloseRangeDist(itActor, plActor, reach) && PlayerFacingRange(pp, itActor, face))
|
|
{
|
|
if (SpriteOverlapZ(pp->actor, itActor, 20))
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(itActor), itActor->sector(), ActorVectOfMiddle(plActor), plActor->sector()))
|
|
DoDamage(itActor, pp->actor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// all this is to break glass
|
|
{
|
|
HitInfo hit{};
|
|
short daang;
|
|
int daz;
|
|
|
|
daang = pp->angle.ang.Buildang();
|
|
daz = -MulScale(pp->horizon.horiz.asq16(), 2000, 16) + (RandomRange(24000) - 12000);
|
|
|
|
FAFhitscan(pp->int_ppos().X, pp->int_ppos().Y, pp->int_ppos().Z, pp->cursector, // Start position
|
|
bcos(daang), // X vector of 3D ang
|
|
bsin(daang), // Y vector of 3D ang
|
|
daz, // Z vector of 3D ang
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
return 0;
|
|
|
|
if (FindDistance3D(pp->int_ppos() - hit.int_hitpos()) < 700)
|
|
{
|
|
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
extern STATE s_TrashCanPain[];
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->hasU()) // JBF: added null check
|
|
{
|
|
switch (hitActor->user.ID)
|
|
{
|
|
case ZILLA_RUN_R0:
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_SWORDCLANK, hit.int_hitpos(), v3df_none);
|
|
break;
|
|
case TRASHCAN:
|
|
if (hitActor->user.WaitTics <= 0)
|
|
{
|
|
hitActor->user.WaitTics = SEC(2);
|
|
ChangeState(hitActor, s_TrashCanPain);
|
|
}
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_SWORDCLANK, hit.int_hitpos(), v3df_none);
|
|
PlaySound(DIGI_TRASHLID, hitActor, v3df_none);
|
|
break;
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_SWORDCLANK, hit.int_hitpos(), v3df_none);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_STAR, hitActor))
|
|
return 0;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hitActor, 0);
|
|
}
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
}
|
|
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, pp->angle.ang, plActor->user.ID);
|
|
}
|
|
// hit non breakable wall - do sound and puff
|
|
else
|
|
{
|
|
SpawnSwordSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_SWORDCLANK, hit.int_hitpos(), v3df_none);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int InitFistAttack(PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
unsigned stat;
|
|
int dist;
|
|
short reach,face;
|
|
|
|
PlaySound(DIGI_STAR, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
if (pp->Flags & (PF_DIVING))
|
|
{
|
|
DSWActor* bubble;
|
|
|
|
static const DAngle dangs[] =
|
|
{
|
|
-DAngle22_5, DAngle22_5
|
|
};
|
|
|
|
for (size_t i = 0; i < countof(dangs); i++)
|
|
{
|
|
bubble = SpawnBubble(pp->actor);
|
|
if (bubble != nullptr)
|
|
{
|
|
bubble->set_int_ang(pp->angle.ang.Buildang());
|
|
|
|
auto random_amt = RandomAngle(DAngle22_5 / 4) - DAngle22_5 / 8;
|
|
|
|
// back it up a bit to get it out of your face
|
|
auto vec = MOVExy((1024+256)*3, bubble->spr.angle + dangs[i] + random_amt);
|
|
|
|
move_missile(bubble, DVector3(vec, 0), plActor->user.ceiling_dist, plActor->user.floor_dist, CLIPMASK_PLAYER, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor->user.PlayerP == pp)
|
|
break;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
dist = DistanceI(pp->pos, itActor->spr.pos);
|
|
|
|
if (pp->InventoryActive[2]) // Shadow Bombs give you demon fist
|
|
{
|
|
face = 190;
|
|
reach = 2300;
|
|
}
|
|
else
|
|
{
|
|
reach = 1000;
|
|
face = 200;
|
|
}
|
|
|
|
if (dist < CloseRangeDist(itActor, plActor, reach) && PlayerFacingRange(pp, itActor, face))
|
|
{
|
|
if (SpriteOverlapZ(pp->actor, itActor, 20) || face == 190)
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(itActor), itActor->sector(), ActorVectOfMiddle(plActor), plActor->sector()))
|
|
DoDamage(itActor, plActor);
|
|
if (face == 190)
|
|
{
|
|
SpawnDemonFist(itActor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// all this is to break glass
|
|
{
|
|
HitInfo hit{};
|
|
short daang;
|
|
int daz;
|
|
|
|
daang = pp->angle.ang.Buildang();
|
|
daz = -MulScale(pp->horizon.horiz.asq16(), 2000, 16) + (RandomRange(24000) - 12000);
|
|
|
|
FAFhitscan(pp->int_ppos().X, pp->int_ppos().Y, pp->int_ppos().Z, pp->cursector, // Start position
|
|
bcos(daang), // X vector of 3D ang
|
|
bsin(daang), // Y vector of 3D ang
|
|
daz, // Z vector of 3D ang
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
return 0;
|
|
|
|
if (FindDistance3D(pp->int_ppos() - hit.int_hitpos()) < 700)
|
|
{
|
|
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
extern STATE s_TrashCanPain[];
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->hasU()) // JBF: added null check
|
|
{
|
|
switch (hitActor->user.ID)
|
|
{
|
|
case ZILLA_RUN_R0:
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_ARMORHIT, hit.int_hitpos(), v3df_none);
|
|
break;
|
|
case TRASHCAN:
|
|
if (hitActor->user.WaitTics <= 0)
|
|
{
|
|
hitActor->user.WaitTics = SEC(2);
|
|
ChangeState(hitActor, s_TrashCanPain);
|
|
}
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_ARMORHIT, hit.int_hitpos(), v3df_none);
|
|
PlaySound(DIGI_TRASHLID, hitActor, v3df_none);
|
|
break;
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_ARMORHIT, hit.int_hitpos(), v3df_none);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_STAR, hitActor))
|
|
return 0;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hitActor,0);
|
|
}
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
}
|
|
|
|
switch (hitActor->spr.picnum)
|
|
{
|
|
case 5062:
|
|
case 5063:
|
|
case 4947:
|
|
SpawnSwordSparks(pp, hit.hitSector, nullptr, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_ARMORHIT, hit.int_hitpos(), v3df_none);
|
|
if (RandomRange(1000) > 700)
|
|
PlayerUpdateHealth(pp,1); // Give some health
|
|
hitActor->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, pp->angle.ang, plActor->user.ID);
|
|
}
|
|
// hit non breakable wall - do sound and puff
|
|
else
|
|
{
|
|
SpawnSwordSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
PlaySound(DIGI_ARMORHIT, hit.int_hitpos(), v3df_none);
|
|
if (PlayerTakeDamage(pp, nullptr))
|
|
{
|
|
PlayerUpdateHealth(pp, -(RandomRange(2<<8)>>8));
|
|
PlayerCheckDeath(pp, nullptr);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int InitSumoNapalm(DSWActor* actor)
|
|
{
|
|
short dist;
|
|
short ang;
|
|
|
|
static const MISSILE_PLACEMENT mp[] =
|
|
{
|
|
{0, 1100, 0},
|
|
};
|
|
|
|
PlaySound(DIGI_NAPFIRE, actor, v3df_none);
|
|
|
|
ang = actor->int_ang();
|
|
for (int j = 0; j < 4; j++)
|
|
{
|
|
for (size_t i = 0; i < countof(mp); i++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL1, s_Napalm, actor->sector(),
|
|
DVector3(actor->spr.pos.XY(), ActorZOfTop(actor)), actor->spr.angle, NAPALM_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
if (i == 0) // Only attach sound to first projectile
|
|
{
|
|
PlaySound(DIGI_NAPWIZ, actorNew, v3df_follow);
|
|
}
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.clipdist = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.Flags2 |= (SPR2_BLUR_TAPER_FAST);
|
|
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 1;
|
|
|
|
if (mp[i].dist_over != 0)
|
|
{
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + mp[i].ang));
|
|
HelpMissileLateral(actorNew, mp[i].dist_over);
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() - mp[i].ang));
|
|
}
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoNapalm, mp[i].dist_out);
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actor->user.Counter = 0;
|
|
|
|
}
|
|
ang += 512;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int InitSumoSkull(DSWActor* actor)
|
|
{
|
|
extern STATE s_SkullExplode[];
|
|
extern STATE s_SkullWait[5][1];
|
|
extern STATE* sg_SkullWait[];
|
|
extern ATTRIBUTE SkullAttrib;
|
|
|
|
|
|
PlaySound(DIGI_SERPSUMMONHEADS, actor, v3df_none);
|
|
|
|
auto actorNew = SpawnActor(STAT_ENEMY, SKULL_R0, &s_SkullWait[0][0], actor->sector(), DVector3(actor->spr.pos, ActorZOfMiddle(actor)), actor->spr.angle, 0);
|
|
|
|
actorNew->set_int_xvel(500);
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -20;
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.pal = 0;
|
|
|
|
// randomize the head turning angle
|
|
actorNew->set_int_ang(RANDOM_P2(2048<<5)>>5);
|
|
|
|
// control direction of spinning
|
|
actor->user.Flags ^= SPR_BOUNCE;
|
|
actorNew->user.Flags |= (actor->user.Flags & (SPR_BOUNCE));
|
|
|
|
actorNew->user.StateEnd = s_SkullExplode;
|
|
actorNew->user.Rot = sg_SkullWait;
|
|
|
|
actorNew->user.Attrib = &SkullAttrib;
|
|
DoActorSetSpeed(actor, NORM_SPEED);
|
|
actorNew->user.Counter = RANDOM_P2(2048);
|
|
actorNew->user.pos.Z = actorNew->spr.pos.Z;
|
|
actorNew->user.Health = 100;
|
|
|
|
// defaults do change the statnum
|
|
EnemyDefaults(actorNew, nullptr, nullptr);
|
|
actorNew->spr.extra |= SPRX_PLAYER_OR_ENEMY;
|
|
|
|
actorNew->spr.clipdist = (128+64) >> 2;
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.Radius = 400;
|
|
return 0;
|
|
}
|
|
|
|
int InitSumoStompAttack(DSWActor* actor)
|
|
{
|
|
unsigned stat;
|
|
int dist;
|
|
short reach;
|
|
|
|
|
|
PlaySound(DIGI_30MMEXPLODE, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor != actor->user.targetActor)
|
|
break;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
dist = DistanceI(actor->spr.pos, itActor->spr.pos);
|
|
|
|
reach = 16384;
|
|
|
|
if (dist < CloseRangeDist(itActor, actor, reach))
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(itActor), itActor->sector(), ActorVectOfMiddle(actor), actor->sector()))
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitMiniSumoClap(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
short reach;
|
|
|
|
auto targetActor = actor->user.targetActor;
|
|
if (!targetActor) return 0;
|
|
|
|
dist = DistanceI(actor->spr.pos, targetActor->spr.pos);
|
|
|
|
reach = 10000;
|
|
|
|
if (dist < CloseRangeDist(targetActor, actor, 1000))
|
|
{
|
|
if (SpriteOverlapZ(actor, targetActor, 20))
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(targetActor), targetActor->sector(), ActorVectOfMiddle(actor), actor->sector()))
|
|
{
|
|
PlaySound(DIGI_CGTHIGHBONE, actor, v3df_follow | v3df_dontpan);
|
|
DoDamage(targetActor, actor);
|
|
}
|
|
}
|
|
}
|
|
else if (dist < CloseRangeDist(targetActor, actor, reach))
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(targetActor), targetActor->sector(), ActorVectOfMiddle(actor), actor->sector()))
|
|
{
|
|
PlaySound(DIGI_30MMEXPLODE, actor, v3df_none);
|
|
SpawnFireballFlames(actor, targetActor);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int WeaponAutoAim(DSWActor* actor, DSWActor* mislActor, short ang, bool test)
|
|
{
|
|
if (actor->hasU() && actor->user.PlayerP)
|
|
{
|
|
if (Autoaim(actor->user.PlayerP->pnum) != 1)
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
DSWActor* hitActor;
|
|
if ((hitActor = DoPickTarget(actor, ang, test)) != nullptr)
|
|
{
|
|
mislActor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
|
|
auto delta = hitActor->spr.pos.XY() - mislActor->spr.pos.XY();
|
|
mislActor->spr.angle = VecToAngle(delta);
|
|
double dist = delta.Length();
|
|
|
|
if (dist != 0)
|
|
{
|
|
double zh;
|
|
double tos = ActorZOfTop(hitActor);
|
|
double diff = mislActor->spr.pos.Z - tos;
|
|
double siz = ActorSizeZ(hitActor);
|
|
|
|
// hit_sprite is below
|
|
if (diff < -50)
|
|
zh = tos + (siz * 0.5);
|
|
else
|
|
// hit_sprite is above
|
|
if (diff > 50)
|
|
zh = tos + (siz * 0.125);
|
|
else
|
|
zh = tos + (siz * 0.25);
|
|
|
|
mislActor->set_int_zvel(int((mislActor->int_xvel() * (zh - mislActor->spr.pos.Z)) / dist) * (zworldtoint / worldtoint));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int WeaponAutoAimZvel(DSWActor* actor, DSWActor* missileActor, int *zvel, short ang, bool test)
|
|
{
|
|
int dist;
|
|
|
|
if (actor->hasU() && actor->user.PlayerP)
|
|
{
|
|
if (!Autoaim(actor->user.PlayerP->pnum))
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
DSWActor* hitActor;
|
|
if ((hitActor = DoPickTarget(actor, ang, test)) != nullptr)
|
|
{
|
|
missileActor->user.WpnGoalActor = hitActor;
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
|
|
missileActor->set_int_ang(NORM_ANGLE(getangle(hitActor->int_pos().X - missileActor->int_pos().X, hitActor->int_pos().Y - missileActor->int_pos().Y)));
|
|
dist = FindDistance2D(missileActor->int_pos().vec2 - hitActor->int_pos().vec2);
|
|
|
|
if (dist != 0)
|
|
{
|
|
double zh;
|
|
double tos = ActorZOfTop(hitActor);
|
|
double diff = missileActor->spr.pos.Z - tos;
|
|
double siz = ActorSizeZ(hitActor);
|
|
|
|
// hit_sprite is below
|
|
if (diff < -50)
|
|
zh = tos + (siz * 0.5);
|
|
else
|
|
// hit_sprite is above
|
|
if (diff > 50)
|
|
zh = tos + (siz * 0.125);
|
|
else
|
|
zh = tos + (siz * 0.25);
|
|
|
|
*zvel = int((missileActor->int_xvel() * (zh - missileActor->spr.pos.Z)) / dist * (zworldtoint));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
DSWActor* AimHitscanToTarget(DSWActor* actor, double *z, DAngle *ang, double z_ratio)
|
|
{
|
|
DSWActor* hitActor = actor->user.targetActor;
|
|
if (hitActor == nullptr)
|
|
return nullptr;
|
|
|
|
hitActor->user.Flags |= (SPR_TARGETED);
|
|
hitActor->user.Flags |= (SPR_ATTACKED);
|
|
|
|
auto delta = hitActor->spr.pos.XY() - actor->spr.pos.XY();
|
|
*ang = VecToAngle(delta);
|
|
|
|
// find the distance to the target
|
|
double dist = delta.Length();
|
|
|
|
if (dist != 0)
|
|
{
|
|
double zh = ActorUpperZ(hitActor);
|
|
|
|
// This doesn't look like it makes much sense...
|
|
auto vect = ang->ToVector() * 1024;
|
|
|
|
if (delta.X != 0)
|
|
*z = vect.X * (zh - *z) / delta.X;
|
|
else if (delta.Y != 0)
|
|
*z = vect.Y * (zh - *z) / delta.Y;
|
|
else
|
|
*z = 0;
|
|
|
|
// so actors won't shoot straight up at you
|
|
// need to be a bit of a distance away
|
|
// before they have a valid shot
|
|
if (abs(*z / dist) > z_ratio)
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
return hitActor;
|
|
}
|
|
|
|
DSWActor* WeaponAutoAimHitscan(DSWActor* actor, int *z, short *ang, bool test)
|
|
{
|
|
int dist;
|
|
int zh;
|
|
int xvect;
|
|
int yvect;
|
|
|
|
if (actor->hasU() && actor->user.PlayerP)
|
|
{
|
|
if (!Autoaim(actor->user.PlayerP->pnum))
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
DSWActor* picked;
|
|
if ((picked = DoPickTarget(actor, *ang, test)) != nullptr)
|
|
{
|
|
|
|
picked->user.Flags |= (SPR_TARGETED);
|
|
picked->user.Flags |= (SPR_ATTACKED);
|
|
|
|
*ang = NORM_ANGLE(getangle(picked->int_pos().X - actor->int_pos().X, picked->int_pos().Y - actor->int_pos().Y));
|
|
|
|
// find the distance to the target
|
|
dist = ksqrt(SQ(actor->int_pos().X - picked->int_pos().X) + SQ(actor->int_pos().Y - picked->int_pos().Y));
|
|
|
|
if (dist != 0)
|
|
{
|
|
zh = (ActorZOfTop(picked) + (ActorSizeZ(picked) * 0.5)) * zworldtoint;
|
|
|
|
xvect = bcos(*ang);
|
|
yvect = bsin(*ang);
|
|
|
|
if (picked->int_pos().X - actor->int_pos().X != 0)
|
|
*z = Scale(xvect,zh - *z,picked->int_pos().X - actor->int_pos().X);
|
|
else if (picked->int_pos().Y - actor->int_pos().Y != 0)
|
|
*z = Scale(yvect,zh - *z,picked->int_pos().Y - actor->int_pos().Y);
|
|
else
|
|
*z = 0;
|
|
}
|
|
}
|
|
|
|
return picked;
|
|
}
|
|
|
|
void WeaponHitscanShootFeet(DSWActor* actor, DSWActor* hitActor, double *zvect)
|
|
{
|
|
auto delta = hitActor->spr.pos.XY() - actor->spr.pos.XY();
|
|
DAngle ang = VecToAngle(delta);
|
|
|
|
// find the distance to the target
|
|
double dist = delta.Length();
|
|
|
|
if (dist != 0)
|
|
{
|
|
double zh = ActorZOfBottom(hitActor) + 20;
|
|
double z = actor->spr.pos.Z;
|
|
auto vect = ang.ToVector() * 1024;
|
|
|
|
if (delta.X != 0)
|
|
*zvect = vect.X * (zh - z) / delta.X;
|
|
else if (delta.Y != 0)
|
|
*zvect = vect.Y * (zh - z) / delta.Y;
|
|
else
|
|
*zvect = 0;
|
|
}
|
|
}
|
|
|
|
int InitStar(PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
int zvel;
|
|
|
|
static DAngle dang[] = { DAngle::fromBuild(-12), DAngle::fromBuild(12) };
|
|
const int STAR_REPEAT = 26;
|
|
const int STAR_HORIZ_ADJ = 100;
|
|
|
|
PlayerUpdateAmmo(pp, plActor->user.WeaponNum, -3);
|
|
|
|
PlaySound(DIGI_STAR, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, STAR1, s_Star, pp->cursector, pos, pp->angle.ang, STAR_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = actorNew->spr.xrepeat = STAR_REPEAT;
|
|
actorNew->spr.shade = -25;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
// zvel was overflowing with this calculation - had to move to a local long var
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT+STAR_HORIZ_ADJ, 16);
|
|
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.WeaponNum = plActor->user.WeaponNum;
|
|
actorNew->user.Radius = 100;
|
|
actorNew->user.Counter = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
// zvel had to be tweaked alot for this weapon
|
|
// MissileSetPos seemed to be pushing the sprite too far up or down when
|
|
// the horizon was tilted. Never figured out why.
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (MissileSetPos(actorNew, DoStar, 1000))
|
|
{
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) != -1)
|
|
{
|
|
zvel = actorNew->int_zvel();
|
|
}
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
actorNew->backuppos();
|
|
|
|
for (size_t i = 0; i < countof(dang); i++)
|
|
{
|
|
auto actorNew2 = SpawnActor(STAT_MISSILE, STAR1, s_Star, pp->cursector,pos, actorNew->spr.angle + dang[i], actorNew->int_xvel());
|
|
|
|
SetOwner(GetOwner(actorNew), actorNew2);
|
|
actorNew2->spr.yrepeat = actorNew2->spr.xrepeat = STAR_REPEAT;
|
|
actorNew2->spr.shade = actorNew->spr.shade;
|
|
|
|
actorNew2->spr.extra = actorNew->spr.extra;
|
|
actorNew2->spr.clipdist = actorNew->spr.clipdist;
|
|
actorNew2->user.WeaponNum = actorNew->user.WeaponNum;
|
|
actorNew2->user.Radius = actorNew->user.Radius;
|
|
actorNew2->user.ceiling_dist = actorNew->user.ceiling_dist;
|
|
actorNew2->user.floor_dist = actorNew->user.floor_dist;
|
|
actorNew2->user.Flags2 = actorNew->user.Flags2 & ~(SPR2_FLAMEDIE); // mask out any new flags here for safety.
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew2))
|
|
actorNew2->user.Flags |= SPR_UNDERWATER;
|
|
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT+STAR_HORIZ_ADJ, 16);
|
|
actorNew2->set_int_zvel(zvel >> 1);
|
|
|
|
if (MissileSetPos(actorNew2, DoStar, 1000))
|
|
{
|
|
KillActor(actorNew2);
|
|
return 0;
|
|
}
|
|
|
|
// move the same as middle star
|
|
actorNew2->user.change.Z = actorNew->user.change.Z;
|
|
UpdateChangeXY(actorNew2);
|
|
|
|
actorNew2->backuppos();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void InitHeartAttack(PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
short i = 0;
|
|
|
|
static const MISSILE_PLACEMENT mp[] =
|
|
{
|
|
{0, 1100, 0},
|
|
};
|
|
|
|
PlayerUpdateAmmo(pp, WPN_HEART, -1);
|
|
|
|
if (!pp->insector())
|
|
return;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE_SKIP4, BLOOD_WORM, s_BloodWorm, pp->cursector,
|
|
pp->pos.plusZ(12), pp->angle.ang, BLOOD_WORM_VELOCITY*2);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.shade = -10;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.clipdist = 0;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.Flags2 |= (SPR2_DONT_TARGET_OWNER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
auto oclipdist = plActor->spr.clipdist;
|
|
plActor->spr.clipdist = 1;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoBloodWorm, mp[i].dist_out);
|
|
|
|
plActor->spr.clipdist = oclipdist;
|
|
actorNew->user.Counter = 0;
|
|
actorNew->user.Counter2 = 0;
|
|
actorNew->user.Counter3 = 0;
|
|
actorNew->user.WaitTics = 0;
|
|
}
|
|
|
|
int ContinueHitscan(PLAYER* pp, sectortype* sect, int x, int y, int z, short ang, int xvect, int yvect, int zvect)
|
|
{
|
|
HitInfo hit{};
|
|
DSWActor* actor = pp->actor;
|
|
|
|
FAFhitscan(x, y, z, sect,
|
|
xvect, yvect, zvect,
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
return 0;
|
|
|
|
if (hit.actor() == nullptr && hit.hitWall == nullptr)
|
|
{
|
|
if (abs(hit.hitpos.Z - hit.hitSector->ceilingz) <= 1)
|
|
{
|
|
hit.hitpos.Z += 16;
|
|
if ((hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
return 0;
|
|
}
|
|
else if (abs(hit.hitpos.Z - hit.hitSector->floorz) <= 1)
|
|
{
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, DAngle::fromBuild(ang), actor->user.ID);
|
|
return 0;
|
|
}
|
|
|
|
QueueHole(hit.hitSector, hit.hitWall, hit.hitpos);
|
|
}
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_SHOTGUN, hit.actor()))
|
|
return 0;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hit.actor(),0);
|
|
return 0;
|
|
}
|
|
|
|
if (BulletHitSprite(pp->actor, hit.actor(), hit.hitpos, 0))
|
|
return 0;
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hit.actor());
|
|
}
|
|
}
|
|
|
|
auto j = SpawnShotgunSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, ang);
|
|
DoHitscanDamage(j, hit.actor());
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitShotgun(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
short daang,ndaang, i;
|
|
HitInfo hit{};
|
|
int daz, ndaz;
|
|
short cstat = 0;
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
|
|
PlaySound(DIGI_RIOTFIRE2, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (pp->WpnShotgunAuto)
|
|
{
|
|
switch (pp->WpnShotgunType)
|
|
{
|
|
case 1:
|
|
pp->WpnShotgunAuto--;
|
|
}
|
|
}
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z);
|
|
daz = pos.Z * zworldtoint;
|
|
|
|
daang = 64;
|
|
if (WeaponAutoAimHitscan(pp->actor, &daz, &daang, false) != nullptr)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
daz = -MulScale(pp->horizon.horiz.asq16(), 2000, 16);
|
|
daang = pp->angle.ang.Buildang();
|
|
}
|
|
|
|
for (i = 0; i < 12; i++)
|
|
{
|
|
if (pp->WpnShotgunType == 0)
|
|
{
|
|
ndaz = daz + (RandomRange(Z(120)) - Z(45));
|
|
ndaang = NORM_ANGLE(daang + (RandomRange(30) - 15));
|
|
}
|
|
else
|
|
{
|
|
ndaz = daz + (RandomRange(Z(200)) - Z(65));
|
|
ndaang = NORM_ANGLE(daang + (RandomRange(70) - 30));
|
|
}
|
|
|
|
DVector3 vect;
|
|
vect.XY() = DAngle::fromBuild(ndaang).ToVector() * 1024;
|
|
vect.Z = ndaz * zinttoworld;
|
|
FAFhitscan(pos, pp->cursector, vect, hit, CLIPMASK_MISSILE);
|
|
|
|
// still needed. Must go away later
|
|
int xvect = bcos(ndaang);
|
|
int yvect = bsin(ndaang);
|
|
int zvect = ndaz;
|
|
|
|
|
|
if (hit.hitSector == nullptr)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (hit.actor() == nullptr && hit.hitWall == nullptr)
|
|
{
|
|
if (abs(hit.hitpos.Z - hit.hitSector->ceilingz) <= 1)
|
|
{
|
|
hit.hitpos.Z += 16;
|
|
cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
if ((hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
continue;
|
|
|
|
if (SectorIsUnderwaterArea(hit.hitSector))
|
|
{
|
|
WarpToSurface(hit.hitpos, &hit.hitSector);
|
|
ContinueHitscan(pp, hit.hitSector, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, ndaang, xvect, yvect, zvect);
|
|
continue;
|
|
}
|
|
}
|
|
else if (abs(hit.hitpos.Z - hit.hitSector->floorz) <= 1)
|
|
{
|
|
if ((hit.hitSector->extra & SECTFX_LIQUID_MASK) != SECTFX_LIQUID_NONE)
|
|
{
|
|
SpawnSplashXY(hit.int_hitpos().X,hit.int_hitpos().Y,hit.int_hitpos().Z,hit.hitSector);
|
|
|
|
if (SectorIsDiveArea(hit.hitSector))
|
|
{
|
|
WarpToUnderwater(hit.hitpos, &hit.hitSector);
|
|
ContinueHitscan(pp, hit.hitSector, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, ndaang, xvect, yvect, zvect);
|
|
}
|
|
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, DAngle::fromBuild(ndaang), actor->user.ID);
|
|
continue;
|
|
}
|
|
|
|
QueueHole(hit.hitSector, hit.hitWall, hit.hitpos);
|
|
}
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->hasU() && hitActor->user.ID == TRASHCAN) // JBF: added null check
|
|
{
|
|
extern STATE s_TrashCanPain[];
|
|
|
|
PlaySound(DIGI_TRASHLID, hitActor, v3df_none);
|
|
if (hitActor->user.WaitTics <= 0)
|
|
{
|
|
hitActor->user.WaitTics = SEC(2);
|
|
ChangeState(hitActor,s_TrashCanPain);
|
|
}
|
|
}
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_SHOTGUN, hitActor))
|
|
continue;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hitActor,0);
|
|
continue;
|
|
}
|
|
|
|
if (BulletHitSprite(pp->actor, hitActor, hit.hitpos, SHOTGUN_SMOKE))
|
|
continue;
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
}
|
|
}
|
|
|
|
auto j = SpawnShotgunSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, ndaang);
|
|
DoHitscanDamage(j, hit.actor());
|
|
}
|
|
|
|
DoPlayerBeginRecoil(pp, SHOTGUN_RECOIL_AMT);
|
|
return 0;
|
|
}
|
|
|
|
int InitLaser(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
|
|
DoPlayerBeginRecoil(pp, RAIL_RECOIL_AMT);
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
PlaySound(DIGI_RIOTFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, s_Laser, pp->cursector, pos, pp->angle.ang, 300);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
// the slower the missile travels the less of a zvel it needs
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 11);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 900);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// the slower the missile travels the less of a zvel it needs
|
|
// move it 1200 dist in increments - works better
|
|
if (MissileSetPos(actorNew, DoLaserStart, 300))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
if (MissileSetPos(actorNew, DoLaserStart, 300))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
if (MissileSetPos(actorNew, DoLaserStart, 300))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
if (MissileSetPos(actorNew, DoLaserStart, 300))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(5);
|
|
}
|
|
|
|
UpdateChange(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
int InitRail(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int nx, ny, nz;
|
|
int zvel;
|
|
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
DoPlayerBeginRecoil(pp, RAIL_RECOIL_AMT);
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
|
|
PlaySound(DIGI_RAILFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 11);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R1, &s_Rail[0][0], pp->cursector, pos, pp->angle.ang, 1200);
|
|
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT + 17, 16);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rail[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = RAIL_RADIUS;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER|CSTAT_SPRITE_INVISIBLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
actorNew->spr.clipdist = 32L>>2;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 700);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
if (TestMissileSetPos(actorNew, DoRailStart, 1200, zvel))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(4);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitZillaRail(DSWActor* actor)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
PlaySound(DIGI_RAILFIRE, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
auto pos = ActorVectOfTop(actor);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R1, &s_Rail[0][0], actor->sector(), pos, actor->spr.angle, 1200);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
int zvel = (100 * (HORIZ_MULT+17));
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rail[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = RAIL_RADIUS;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER|CSTAT_SPRITE_INVISIBLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 700);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
if (TestMissileSetPos(actorNew, DoRailStart, 1200, zvel))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(4);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitRocket(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int zvel;
|
|
|
|
DoPlayerBeginRecoil(pp, ROCKET_RECOIL_AMT);
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
auto const WpnRocketHeat = pp->WpnRocketHeat;
|
|
if (WpnRocketHeat)
|
|
{
|
|
switch (pp->WpnRocketType)
|
|
{
|
|
case 1:
|
|
pp->WpnRocketHeat--;
|
|
break;
|
|
}
|
|
}
|
|
|
|
PlaySound(DIGI_RIOTFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Rocket[0][0], pp->cursector, pos, pp->angle.ang, ROCKET_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 90;
|
|
actorNew->spr.xrepeat = 90;
|
|
actorNew->spr.shade = -15;
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT + 35, 16);
|
|
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 2000;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
// Set default palette
|
|
actorNew->spr.pal = actorNew->user.spal = 17; // White
|
|
|
|
if (WpnRocketHeat)
|
|
{
|
|
switch (pp->WpnRocketType)
|
|
{
|
|
case 1:
|
|
actorNew->user.Flags |= (SPR_FIND_PLAYER);
|
|
actorNew->spr.pal = actorNew->user.spal = 20; // Yellow
|
|
break;
|
|
}
|
|
}
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 900);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// cancel smoke trail
|
|
actorNew->user.Counter = 1;
|
|
if (TestMissileSetPos(actorNew, DoRocket, 1200, zvel))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
// inable smoke trail
|
|
actorNew->user.Counter = 0;
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(5);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitBunnyRocket(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int nx, ny, nz;
|
|
int zvel;
|
|
|
|
DoPlayerBeginRecoil(pp, ROCKET_RECOIL_AMT);
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
auto const WpnRocketHeat = pp->WpnRocketHeat;
|
|
if (WpnRocketHeat)
|
|
{
|
|
switch (pp->WpnRocketType)
|
|
{
|
|
case 1:
|
|
pp->WpnRocketHeat--;
|
|
break;
|
|
}
|
|
}
|
|
|
|
PlaySound(DIGI_BUNNYATTACK, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R4, &s_BunnyRocket[0][0], pp->cursector, pos, pp->angle.ang, ROCKET_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.shade = -15;
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT + 35, 16);
|
|
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_BunnyRocket[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 2000;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
|
|
if (WpnRocketHeat)
|
|
{
|
|
switch (pp->WpnRocketType)
|
|
{
|
|
case 1:
|
|
actorNew->user.Flags |= (SPR_FIND_PLAYER);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 900);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// cancel smoke trail
|
|
actorNew->user.Counter = 1;
|
|
if (TestMissileSetPos(actorNew, DoRocket, 1200, zvel))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
// inable smoke trail
|
|
actorNew->user.Counter = 0;
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(5);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_PLAYER1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitNuke(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int zvel;
|
|
|
|
if (pp->WpnRocketNuke > 0)
|
|
pp->WpnRocketNuke = 0; // Bye Bye little nukie.
|
|
else
|
|
return 0;
|
|
|
|
DoPlayerBeginRecoil(pp, ROCKET_RECOIL_AMT*12);
|
|
|
|
|
|
PlaySound(DIGI_RIOTFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Rocket[0][0], pp->cursector, pos, pp->angle.ang, 700);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 128;
|
|
actorNew->spr.xrepeat = 128;
|
|
actorNew->spr.shade = -15;
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), HORIZ_MULT + 36, 16);
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
// Set to red palette
|
|
actorNew->spr.pal = actorNew->user.spal = 19;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = NUKE_RADIUS;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 900);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// cancel smoke trail
|
|
actorNew->user.Counter = 1;
|
|
if (TestMissileSetPos(actorNew, DoRocket, 1200, zvel))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
// inable smoke trail
|
|
actorNew->user.Counter = 0;
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(5);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
PlayerDamageSlide(pp, -40, NORM_ANGLE(pp->angle.ang.Buildang()+1024)); // Recoil slide
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyNuke(DSWActor* actor)
|
|
{
|
|
int nx, ny, nz;
|
|
int zvel;
|
|
|
|
PlaySound(DIGI_RIOTFIRE, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
nx = actor->int_pos().X;
|
|
ny = actor->int_pos().Y;
|
|
|
|
// Spawn a shot
|
|
nz = actor->int_pos().Z + Z(40);
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Rocket[0][0], actor->sector(),
|
|
nx, ny, nz, actor->int_ang(), 700);
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
|
|
actorNew->spr.yrepeat = 128;
|
|
actorNew->spr.xrepeat = 128;
|
|
actorNew->spr.shade = -15;
|
|
zvel = (100 * (HORIZ_MULT-36));
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
// Set to red palette
|
|
actorNew->spr.pal = actorNew->user.spal = 19;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = NUKE_RADIUS;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 500);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// cancel smoke trail
|
|
actorNew->user.Counter = 1;
|
|
if (TestMissileSetPos(actorNew, DoRocket, 1200, zvel))
|
|
{
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
// enable smoke trail
|
|
actorNew->user.Counter = 0;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAim(actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->spr.angle -= DAngle::fromBuild(5);
|
|
}
|
|
else
|
|
zvel = actorNew->int_zvel(); // Let autoaiming set zvel now
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitMicro(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int dist;
|
|
short i;
|
|
DAngle angle;
|
|
TARGET_SORT* ts = TargetSort;
|
|
DSWActor* picked = nullptr;
|
|
|
|
|
|
const int MAX_MICRO = 1;
|
|
|
|
DoPickTarget(pp->actor, 256, false);
|
|
|
|
if (TargetSortCount > MAX_MICRO)
|
|
TargetSortCount = MAX_MICRO;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
for (i = 0; i < MAX_MICRO; i++)
|
|
{
|
|
if (ts < &TargetSort[TargetSortCount] && ts->actor != nullptr)
|
|
{
|
|
picked = ts->actor;
|
|
|
|
angle = VecToAngle(picked->spr.pos.XY() - pp->pos.XY());
|
|
|
|
ts++;
|
|
}
|
|
else
|
|
{
|
|
picked = nullptr;
|
|
angle = pp->angle.ang;
|
|
}
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 4 + RandomRange(20));
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Micro[0][0], pp->cursector, pos, angle, 1200);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 24;
|
|
actorNew->spr.xrepeat = 24;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
// randomize zvelocity
|
|
actorNew->add_int_zvel( RandomRange(Z(8)) - Z(5));
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Micro[0]);
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (2);
|
|
actorNew->user.floor_dist = (2);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->user.WaitTics = 10 + RandomRange(40);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
const int MICRO_LATERAL = 5000;
|
|
HelpMissileLateral(actorNew, 1000 + (RandomRange(MICRO_LATERAL) - (MICRO_LATERAL / 2)));
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
// cancel smoke trail
|
|
actorNew->user.Counter = 1;
|
|
if (MissileSetPos(actorNew, DoMicro, 700))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
continue;
|
|
}
|
|
// inable smoke trail
|
|
actorNew->user.Counter = 0;
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
const int MICRO_ANG = 400;
|
|
|
|
if (picked)
|
|
{
|
|
dist = DistanceI(actorNew->spr.pos, picked->spr.pos);
|
|
if (dist != 0)
|
|
{
|
|
double zh = ActorZOfTop(picked) + (ActorSizeZ(picked) * 0.25);
|
|
actorNew->set_int_zvel((actorNew->int_xvel() * (zh - actorNew->spr.pos.Z) * zworldtoint) / dist);
|
|
}
|
|
|
|
actorNew->user.WpnGoalActor = ts->actor;
|
|
picked->user.Flags |= (SPR_TARGETED);
|
|
picked->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
else
|
|
{
|
|
actorNew->spr.angle += DAngle::fromBuild((RandomRange(MICRO_ANG) - (MICRO_ANG / 2)) - 16);
|
|
}
|
|
|
|
UpdateChange(actorNew);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitRipperSlash(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_RIPPER2ATTACK, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
if ((unsigned)FindDistance3D(actor->int_pos() - itActor->int_pos()) > itActor->user.Radius + actor->user.Radius)
|
|
continue;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 600) && FacingRange(itActor, actor,150))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitBunnySlash(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_BUNNYATTACK, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 600) && FacingRange(itActor, actor,150))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitSerpSlash(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_SERPSWORDATTACK, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 800) && FacingRange(itActor, actor,150))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool WallSpriteInsideSprite(DSWActor* wactor, DSWActor* actor)
|
|
{
|
|
DVector2 out[2];
|
|
GetWallSpritePosition(&wactor->spr, wactor->spr.pos, out);
|
|
return IsCloseToLine(actor->spr.pos.XY(), out[0], out[1], (((int) actor->spr.clipdist) << 2) * inttoworld) != EClose::Outside;
|
|
}
|
|
|
|
|
|
int DoBladeDamage(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist > 2000)
|
|
continue;
|
|
|
|
dist = FindDistance3D(actor->int_pos() - itActor->int_pos());
|
|
|
|
if (dist > 2000)
|
|
continue;
|
|
|
|
if (WallSpriteInsideSprite(actor, itActor))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoStaticFlamesDamage(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist > 2000)
|
|
continue;
|
|
|
|
dist = FindDistance3D(actor->int_pos() - itActor->int_pos());
|
|
|
|
if (dist > 2000)
|
|
continue;
|
|
|
|
if (SpriteOverlap(actor, itActor)) // If sprites are overlapping, cansee will fail!
|
|
DoDamage(itActor, actor);
|
|
else if (actor->user.Radius > 200)
|
|
{
|
|
if (FAFcansee(ActorVectOfMiddle(actor), actor->sector(), ActorVectOfMiddle(itActor), itActor->sector()))
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitCoolgBash(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_CGTHIGHBONE, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
// don't set off mine
|
|
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
continue;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 600) && FacingRange(itActor, actor,150))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitSkelSlash(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_SPBLADE, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 600) && FacingRange(itActor, actor,150))
|
|
{
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitGoroChop(DSWActor* actor)
|
|
{
|
|
int i;
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
PlaySound(DIGI_GRDSWINGAXE, actor, v3df_none);
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
break;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 700) && FacingRange(itActor, actor,150))
|
|
{
|
|
PlaySound(DIGI_GRDAXEHIT, actor, v3df_none);
|
|
DoDamage(itActor, actor);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitHornetSting(DSWActor* actor)
|
|
{
|
|
DoDamage(actor->user.coll.actor(), actor);
|
|
InitActorReposition(actor);
|
|
return 0;
|
|
}
|
|
|
|
int InitSerpSpell(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
short i;
|
|
|
|
static const DAngle lat_ang[] =
|
|
{
|
|
DAngle90, -DAngle90
|
|
};
|
|
|
|
static const DAngle delta_ang[] =
|
|
{
|
|
DAngle::fromBuild(-10), DAngle::fromBuild(10)
|
|
};
|
|
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, SERP_METEOR, &sg_SerpMeteor[0][0], actor->sector(), actor->spr.pos, actor->spr.angle, 1500);
|
|
|
|
actorNew->spr.pos.Z = ActorZOfTop(actor);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_SerpMeteor[0]);
|
|
actorNew->user.StateEnd = s_MirvMeteorExp;
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
PlaySound(DIGI_SERPMAGICLAUNCH, actor, v3df_none);
|
|
actorNew->user.spal = actorNew->spr.pal = 27; // Bright Green
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.clipdist = 32L >> 2;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.ceiling_dist = (16);
|
|
actorNew->user.floor_dist = (16);
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 1;
|
|
|
|
actorNew->spr.angle += lat_ang[i];
|
|
HelpMissileLateral(actorNew, 4200);
|
|
actorNew->spr.angle -= lat_ang[i];
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor);
|
|
|
|
actorNew->spr.angle += delta_ang[i];
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoMirvMissile, 400);
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int SpawnDemonFist(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_SUICIDE))
|
|
return -1;
|
|
|
|
auto expActor = SpawnActor(STAT_MISSILE, 0, s_TeleportEffect, actor->sector(), DVector3(actor->spr.pos.XY(), ActorZOfMiddle(actor)), actor->spr.angle, 0);
|
|
|
|
expActor->spr.hitag = LUMINOUS; //Always full brightness
|
|
expActor->spr.shade = -40;
|
|
expActor->spr.xrepeat = 32;
|
|
expActor->spr.yrepeat = 32;
|
|
expActor->user.spal = expActor->spr.pal = 25;
|
|
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
expActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
expActor->user.Radius = DamageData[DMG_BASIC_EXP].radius;
|
|
|
|
if (RANDOM_P2(1024<<8)>>8 > 600)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024<<8)>>8 > 600)
|
|
expActor->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitSerpMonstSpell(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
short i;
|
|
|
|
static const DAngle lat_ang[] =
|
|
{
|
|
DAngle90, -DAngle90
|
|
};
|
|
|
|
static const DAngle delta_ang[] =
|
|
{
|
|
DAngle::fromBuild(-10), DAngle::fromBuild(10)
|
|
};
|
|
|
|
PlaySound(DIGI_MISSLFIRE, actor, v3df_none);
|
|
|
|
for (i = 0; i < 1; i++)
|
|
{
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, SERP_METEOR, &sg_SerpMeteor[0][0], actor->sector(), actor->spr.pos, actor->spr.angle, 500);
|
|
|
|
actorNew->user.spal = actorNew->spr.pal = 25; // Bright Red
|
|
actorNew->spr.pos.Z = ActorZOfTop(actor);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_SerpMeteor[0]);
|
|
actorNew->user.StateEnd = s_TeleportEffect2;
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 122;
|
|
actorNew->spr.yrepeat = 116;
|
|
actorNew->spr.clipdist = 32L >> 2;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.ceiling_dist = (16);
|
|
actorNew->user.floor_dist = (16);
|
|
|
|
actorNew->user.Dist = 12.5;
|
|
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 1;
|
|
|
|
actorNew->spr.angle += lat_ang[i];
|
|
HelpMissileLateral(actorNew, 4200);
|
|
actorNew->spr.angle -= lat_ang[i];
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor);
|
|
|
|
actorNew->spr.angle += delta_ang[i];
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoMirvMissile, 400);
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoTeleRipper(DSWActor* actor)
|
|
{
|
|
PlaySound(DIGI_ITEM_SPAWN, actor, v3df_none);
|
|
Ripper2Hatch(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitEnemyRocket(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
|
|
PlaySound(DIGI_NINJARIOTATTACK, actor, v3df_none);
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R2, &s_Rocket[0][0], actor->sector(),
|
|
actor->spr.pos.plusZ(-(ActorSizeZ(actor) * 0.5)-16), actor->user.targetActor->spr.angle, NINJA_BOLT_VELOCITY);
|
|
|
|
// Set default palette
|
|
actorNew->spr.pal = actorNew->user.spal = 17; // White
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
|
|
actorNew->spr.yrepeat = 28;
|
|
actorNew->spr.xrepeat = 28;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
actorNew->user.Radius = 200;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (actor->user.spal == PAL_XLAT_LT_TAN)
|
|
{
|
|
actorNew->user.Flags |= (SPR_FIND_PLAYER);
|
|
actorNew->spr.pal = actorNew->user.spal = 20; // Yellow
|
|
}
|
|
|
|
MissileSetPos(actorNew, DoBoltThinMan, 400);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyRail(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
short pnum=0;
|
|
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
// if co-op don't hurt teammate
|
|
if (gNet.MultiGameType == MULTI_GAME_COOPERATIVE && actor->user.ID == ZOMBIE_RUN_R0)
|
|
{
|
|
PLAYER* pp;
|
|
|
|
// Check all players
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
pp = &Player[pnum];
|
|
if (actor->user.targetActor == pp->actor)
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
PlaySound(DIGI_RAILFIRE, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
// add a bit of randomness
|
|
if (RANDOM_P2(1024) < 512)
|
|
actor->spr.angle += DAngle::fromBuild(RANDOM_P2(128) - 64);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R1, &s_Rail[0][0], actor->sector(),
|
|
actor->spr.pos.plusZ(-(ActorSizeZ(actor) * 0.5) - 8), actor->spr.angle, 1200);
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->vel.Z = 0;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rail[0]);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER|CSTAT_SPRITE_INVISIBLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->spr.clipdist = 64 >> 2;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (TestMissileSetPos(actorNew, DoRailStart, 600, actorNew->int_zvel()))
|
|
{
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitZillaRocket(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
short w;
|
|
|
|
static const MISSILE_PLACEMENT mp[] =
|
|
{
|
|
{600 * 6, 400, 512},
|
|
{900 * 6, 400, 512},
|
|
{1100 * 6, 400, 512},
|
|
{600 * 6, 400, -512},
|
|
{900 * 6, 400, -512},
|
|
{1100 * 6, 400, -512},
|
|
};
|
|
|
|
PlaySound(DIGI_NINJARIOTATTACK, actor, v3df_none);
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
for (int i = 0; i < (int)SIZ(mp); i++)
|
|
{
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R2, &s_Rocket[0][0], actor->sector(),
|
|
actor->spr.pos.plusZ(-(ActorSizeZ(actor) * 0.5) - 16), actor->user.targetActor->spr.angle, NINJA_BOLT_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 28;
|
|
actorNew->spr.xrepeat = 28;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 64 >>2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
actorNew->user.Radius = 200;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
// Zilla has seekers!
|
|
if (i != 1 && i != 4)
|
|
actorNew->spr.pal = actorNew->user.spal = 17; // White
|
|
else
|
|
{
|
|
actorNew->user.Flags |= (SPR_FIND_PLAYER);
|
|
actorNew->spr.pal = actorNew->user.spal = 20; // Yellow
|
|
}
|
|
|
|
if (mp[i].dist_over != 0)
|
|
{
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + mp[i].ang));
|
|
HelpMissileLateral(actorNew, mp[i].dist_over);
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() - mp[i].ang));
|
|
}
|
|
|
|
MissileSetPos(actorNew, DoBoltThinMan, mp[i].dist_out);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyStar(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, STAR1, s_Star, actor->sector(),
|
|
ActorVectOfMiddle(actor), actor->user.targetActor->spr.angle, NINJA_STAR_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 16;
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.shade = -25;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
MissileSetPos(actorNew, DoStar, 400);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
|
|
PlaySound(DIGI_STAR, actor, v3df_none);
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyCrossbow(DSWActor* actor)
|
|
{
|
|
int dist;
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, CROSSBOLT, &s_CrossBolt[0][0], actor->sector(),
|
|
ActorVectOfMiddle(actor).plusZ(-14), actor->user.targetActor->spr.angle, 800);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 26;
|
|
actorNew->spr.shade = -25;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 64>>2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_CrossBolt[0]);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
|
|
MissileSetPos(actorNew, DoStar, 400);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, true);
|
|
|
|
PlaySound(DIGI_STAR, actor, v3df_none);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitSkelSpell(DSWActor* actor)
|
|
{
|
|
PlaySound(DIGI_SPELEC, actor, v3df_none);
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, ELECTRO_ENEMY, s_Electro, actor->sector(),
|
|
actor->spr.pos.plusZ(-(ActorSizeZ(actor) * 0.5)), actor->user.targetActor->spr.angle, SKEL_ELECTRO_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.xrepeat -= 20;
|
|
actorNew->spr.yrepeat -= 20;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, false, ActorSizeZ(actor) * 0.5);
|
|
UpdateChange(actorNew);
|
|
MissileSetPos(actorNew, DoElectro, 400);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitCoolgFire(DSWActor* actor)
|
|
{
|
|
int nx, ny, nz, dist;
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->spr.angle = VecToAngle(actor->user.targetActor->spr.pos.XY() - actor->spr.pos.XY());
|
|
|
|
nz = actor->int_pos().Z - Z(16);
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
PlaySound(DIGI_CGMAGIC, actor, v3df_follow);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, COOLG_FIRE, s_CoolgFire, actor->sector(),
|
|
actor->spr.pos.plusZ(-16), actor->user.targetActor->spr.angle, COOLG_FIRE_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.hitag = LUMINOUS;
|
|
actorNew->spr.yrepeat = 18;
|
|
actorNew->spr.xrepeat = 18;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
actorNew->spr.clipdist = 32>>2;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
if (actor->user.ID == RIPPER_RUN_R0)
|
|
actorNew->user.spal = actorNew->spr.pal = 27; // Bright Green
|
|
else
|
|
actorNew->user.spal = actorNew->spr.pal = 25; // Bright Red
|
|
|
|
PlaySound(DIGI_MAGIC1, actorNew, v3df_follow|v3df_doppler);
|
|
|
|
// find the distance to the target (player)
|
|
SetZVelFromTarget(actorNew, actor, false, -16);
|
|
UpdateChange(actorNew);
|
|
|
|
auto vec = actor->spr.angle.ToVector() * 45.5;
|
|
|
|
move_missile(actorNew, DVector3(vec, 0), actorNew->user.ceiling_dist, actorNew->user.floor_dist, 0, 3);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoCoolgDrip(DSWActor* actor)
|
|
{
|
|
actor->user.Counter += 220;
|
|
actor->spr.pos.Z += actor->user.Counter * maptoworld;
|
|
|
|
if (actor->spr.pos.Z > actor->user.loz - actor->user.floor_dist)
|
|
{
|
|
actor->spr.pos.Z = actor->user.loz - actor->user.floor_dist;
|
|
actor->spr.yrepeat = actor->spr.xrepeat = 32;
|
|
ChangeState(actor, s_GoreFloorSplash);
|
|
if (actor->user.spal == PALETTE_BLUE_LIGHTING)
|
|
PlaySound(DIGI_DRIP, actor, v3df_none);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int InitCoolgDrip(DSWActor* actor)
|
|
{
|
|
short w;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, COOLG_DRIP, s_CoolgDrip, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = actorNew->spr.xrepeat = 20;
|
|
actorNew->spr.shade = -5;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 16L>>2;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
DoFindGroundPoint(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int GenerateDrips(DSWActor* actor)
|
|
{
|
|
short w = 0;
|
|
|
|
if ((actor->user.WaitTics-=ACTORMOVETICS) <= 0)
|
|
{
|
|
if (actor->spr.lotag == 0)
|
|
actor->user.WaitTics = RANDOM_P2(256<<8)>>8;
|
|
else
|
|
actor->user.WaitTics = (actor->spr.lotag * 120) + SEC(RandomRange(3<<8)>>8);
|
|
|
|
if (TEST_BOOL2(actor))
|
|
{
|
|
auto ww = SpawnBubble(actor);
|
|
return 1;
|
|
}
|
|
|
|
auto actorNew = SpawnActor(STAT_SHRAP, COOLG_DRIP, s_CoolgDrip, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = actorNew->spr.xrepeat = 20;
|
|
actorNew->spr.shade = -10;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 16L>>2;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
if (TEST_BOOL1(actor))
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_BLUE_LIGHTING;
|
|
|
|
DoFindGroundPoint(actor);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int InitEelFire(DSWActor* actor)
|
|
{
|
|
unsigned stat;
|
|
int dist, a, b, c;
|
|
|
|
for (stat = 0; stat < SIZ(StatDamageList); stat++)
|
|
{
|
|
SWStatIterator it(StatDamageList[stat]);
|
|
while (auto itActor = it.Next())
|
|
{
|
|
if (itActor == actor)
|
|
continue;
|
|
|
|
if (itActor != actor->user.targetActor)
|
|
continue;
|
|
|
|
if ((unsigned)FindDistance3D(actor->int_pos() - itActor->int_pos()) > itActor->user.Radius + actor->user.Radius)
|
|
continue;
|
|
|
|
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, a, b, c);
|
|
|
|
if (dist < CloseRangeDist(actor, itActor, 600) && FacingRange(itActor, actor,150))
|
|
{
|
|
PlaySound(DIGI_GIBS1, actor, v3df_none);
|
|
DoDamage(itActor, actor);
|
|
}
|
|
else
|
|
InitActorReposition(actor);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void InitFireballTrap(DSWActor* actor)
|
|
{
|
|
short w;
|
|
|
|
PlaySound(DIGI_FIREBALL1, actor, v3df_none);
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL, s_Fireball, actor->sector(), actor->spr.pos.plusZ(-ActorSizeZ(actor)), actor->spr.angle, FIREBALL_TRAP_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.xrepeat -= 20;
|
|
actorNew->spr.yrepeat -= 20;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.clipdist = 32>>2;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->user.WeaponNum = WPN_HOTHEAD;
|
|
|
|
UpdateChange(actorNew);
|
|
}
|
|
|
|
void InitBoltTrap(DSWActor* actor)
|
|
{
|
|
short w;
|
|
|
|
PlaySound(DIGI_RIOTFIRE, actor, v3df_none);
|
|
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Rocket[0][0], actor->sector(), actor->spr.pos.plusZ(-ActorSizeZ(actor)), actor->spr.angle, BOLT_TRAP_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rocket[0]);
|
|
actorNew->user.Radius = 200;
|
|
|
|
UpdateChange(actorNew);
|
|
}
|
|
|
|
|
|
void InitSpearTrap(DSWActor* actor)
|
|
{
|
|
// Spawn a shot
|
|
auto actorNew = SpawnActor(STAT_MISSILE, CROSSBOLT, &s_CrossBolt[0][0], actor->sector(), ActorVectOfMiddle(actor), actor->spr.angle, 750);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.xrepeat = 16;
|
|
actorNew->spr.yrepeat = 26;
|
|
actorNew->spr.shade = -25;
|
|
actorNew->spr.clipdist = 64 >> 2;
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_CrossBolt[0]);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
|
|
PlaySound(DIGI_STAR, actor, v3df_none);
|
|
}
|
|
|
|
int DoSuicide(DSWActor* actor)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
int DoDefaultStat(DSWActor* actor)
|
|
{
|
|
change_actor_stat(actor, STAT_DEFAULT);
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitTracerUzi(PLAYER* pp)
|
|
{
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
DSWActor* actor = pp->actor;
|
|
|
|
int oclipdist;
|
|
|
|
static const short lat_dist[] = {800,-800};
|
|
|
|
double nz = 8 - MulScaleF(pp->horizon.horiz.asq16(), 72, 24);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, 0, s_Tracer, pp->cursector, pp->pos.plusZ(nz), pp->angle.ang, TRACER_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 10;
|
|
actorNew->spr.xrepeat = 10;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
DSWActor* plActor = pp->actor;
|
|
oclipdist = plActor->spr.clipdist;
|
|
plActor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
if (pp->Flags & (PF_TWO_UZI) && pp->WpnUziType == 0)
|
|
HelpMissileLateral(actorNew, lat_dist[RANDOM_P2(2<<8)>>8]);
|
|
else
|
|
HelpMissileLateral(actorNew, lat_dist[0]);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (MissileSetPos(actorNew, DoTracerStart, 800))
|
|
{
|
|
plActor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
actorNew->set_int_zvel(int(-pp->horizon.horiz.asbuildf() * actorNew->int_xvel() * (1. / 8.)));
|
|
|
|
plActor->spr.clipdist = oclipdist;
|
|
|
|
WeaponAutoAim(pp->actor, actorNew, 32, false);
|
|
|
|
// a bit of randomness
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + RandomRange(30) - 15));
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTracerTurret(DSWActor* actor, DSWActor* Operator, fixed_t q16horiz)
|
|
{
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, 0, s_Tracer, actor->sector(),
|
|
actor->spr.pos.plusZ(-MulScaleF(q16horiz, 72, 24)), actor->spr.angle, TRACER_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
if (Operator!= nullptr)
|
|
SetOwner(Operator, actorNew);
|
|
actorNew->spr.yrepeat = 10;
|
|
actorNew->spr.xrepeat = 10;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->set_int_zvel(int(-MulScaleF(q16horiz, actorNew->int_xvel() * (1. / 8.), 16)));
|
|
|
|
WeaponAutoAim(actor, actorNew, 32, false);
|
|
|
|
// a bit of randomness
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + RandomRange(30) - 15));
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTracerAutoTurret(DSWActor* actor, int xchange, int ychange, int zchange)
|
|
{
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, 0, s_Tracer, actor->sector(), actor->spr.pos, actor->spr.angle, TRACER_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.yrepeat = 10;
|
|
actorNew->spr.xrepeat = 10;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->user.set_int_change_x(xchange);
|
|
actorNew->user.set_int_change_y(ychange);
|
|
actorNew->user.set_int_change_z(zchange);
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int BulletHitSprite(DSWActor* actor, DSWActor* hitActor, const DVector3& hit_pos, short ID)
|
|
{
|
|
short id;
|
|
|
|
// hit a NPC or PC?
|
|
if ((hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
|
|
{
|
|
// spawn a red splotch
|
|
// !FRANK! this if was incorrect - its not who is HIT, its who is SHOOTING
|
|
//if(!hitActor->user.PlayerP)
|
|
if (actor->user.PlayerP)
|
|
id = UZI_SMOKE;
|
|
else if (actor->user.Flags & (SPR_SO_ATTACHED))
|
|
id = UZI_SMOKE;
|
|
else // Spawn NPC uzi with less damage
|
|
id = UZI_SMOKE+2;
|
|
|
|
if (ID>0) id = ID;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, id, s_UziSmoke, §or[0], hit_pos, actor->spr.angle, 0);
|
|
actorNew->spr.shade = -40;
|
|
|
|
if (hitActor->user.PlayerP)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
switch (hitActor->user.ID)
|
|
{
|
|
case TRASHCAN:
|
|
case PACHINKO1:
|
|
case PACHINKO2:
|
|
case PACHINKO3:
|
|
case PACHINKO4:
|
|
case 623:
|
|
case ZILLA_RUN_R0:
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT;
|
|
if (RANDOM_P2(1024) > 800)
|
|
SpawnShrapX(hitActor);
|
|
break;
|
|
default:
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT/3;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT/3;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
//actorNew->user.spal = actorNew->spr.pal = PALETTE_RED_LIGHTING;
|
|
break;
|
|
}
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.angle = actor->spr.angle;
|
|
|
|
SetActorZ(actorNew, hit_pos);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
int hit_x = int(hit_pos.X * worldtoint);
|
|
int hit_y = int(hit_pos.Y * worldtoint);
|
|
int hit_z = int(hit_pos.Z * zworldtoint);
|
|
if ((RANDOM_P2(1024<<5)>>5) < 512+128)
|
|
{
|
|
if (!hitActor->user.PlayerP)
|
|
SpawnBlood(hitActor, nullptr, actor->spr.angle, &hit_pos);
|
|
else
|
|
{
|
|
auto hpp = hit_pos.plusZ(20);
|
|
SpawnBlood(hitActor, nullptr, actor->spr.angle, &hpp);
|
|
}
|
|
|
|
// blood comes out the other side?
|
|
if ((RANDOM_P2(1024<<5)>>5) < 256)
|
|
{
|
|
if (!hitActor->user.PlayerP)
|
|
SpawnBlood(hitActor, nullptr, actor->spr.angle + DAngle180, &hit_pos);
|
|
if (hitActor->user.ID != TRASHCAN && hitActor->user.ID != ZILLA_RUN_R0)
|
|
QueueWallBlood(hitActor, actor->spr.angle); //QueueWallBlood needs bullet angle.
|
|
}
|
|
}
|
|
|
|
DoHitscanDamage(actorNew, hitActor);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool HitscanSpriteAdjust(DSWActor* actor, walltype* hit_wall)
|
|
{
|
|
int16_t ang;
|
|
int xvect,yvect;
|
|
|
|
if (hit_wall)
|
|
{
|
|
int16_t const wall_ang = NORM_ANGLE(getangle(hit_wall->delta()));
|
|
actor->set_int_ang(NORM_ANGLE(wall_ang + 512));
|
|
}
|
|
ang = actor->int_ang();
|
|
|
|
xvect = bcos(ang, 4);
|
|
yvect = bsin(ang, 4);
|
|
|
|
// must have this
|
|
auto sect = actor->sector();
|
|
|
|
Collision coll;
|
|
clipmove(actor->spr.pos, §, xvect, yvect, 4, 4 << 8, 4 << 8, CLIPMASK_MISSILE, coll);
|
|
|
|
if (actor->sector() != sect)
|
|
ChangeActorSect(actor, sect);
|
|
|
|
return true;
|
|
}
|
|
|
|
int InitUzi(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
short daang;
|
|
HitInfo hit{};
|
|
int daz, nz;
|
|
int xvect,yvect,zvect;
|
|
ESpriteFlags cstat = 0;
|
|
uint8_t pal = 0;
|
|
static int uziclock=0;
|
|
int clockdiff=0;
|
|
bool FireSnd = false;
|
|
const int UZIFIRE_WAIT = 20;
|
|
|
|
void InitUziShell(PLAYER*);
|
|
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
|
|
if (uziclock > PlayClock)
|
|
{
|
|
uziclock = PlayClock;
|
|
FireSnd = true;
|
|
}
|
|
|
|
clockdiff = PlayClock - uziclock;
|
|
if (clockdiff > UZIFIRE_WAIT)
|
|
{
|
|
uziclock = PlayClock;
|
|
FireSnd = true;
|
|
}
|
|
|
|
if (FireSnd)
|
|
PlaySound(DIGI_UZIFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (RANDOM_P2(1024) < 400)
|
|
InitTracerUzi(pp);
|
|
|
|
nz = pp->int_ppos().Z + pp->bob_z * zworldtoint;
|
|
daz = nz;
|
|
daang = 32;
|
|
if (WeaponAutoAimHitscan(pp->actor, &daz, &daang, false) != nullptr)
|
|
{
|
|
daang += RandomRange(24) - 12;
|
|
daang = NORM_ANGLE(daang);
|
|
daz += RandomRange(10000) - 5000;
|
|
}
|
|
else
|
|
{
|
|
//daang = NORM_ANGLE(pp->angle.ang.Buildang() + (RandomRange(50) - 25));
|
|
daang = NORM_ANGLE(pp->angle.ang.Buildang() + (RandomRange(24) - 12));
|
|
daz = -MulScale(pp->horizon.horiz.asq16(), 2000, 16) + (RandomRange(24000) - 12000);
|
|
}
|
|
|
|
|
|
xvect = bcos(daang);
|
|
yvect = bsin(daang);
|
|
zvect = daz;
|
|
FAFhitscan(pp->int_ppos().X, pp->int_ppos().Y, nz, pp->cursector, // Start position
|
|
xvect,yvect,zvect,
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
SetVisHigh();
|
|
|
|
// check to see what you hit
|
|
if (hit.actor() == nullptr && hit.hitWall == nullptr)
|
|
{
|
|
if (abs(hit.hitpos.Z - hit.hitSector->ceilingz) <= 1)
|
|
{
|
|
hit.hitpos.Z += 16;
|
|
cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
if ((hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
return 0;
|
|
|
|
if (SectorIsUnderwaterArea(hit.hitSector))
|
|
{
|
|
WarpToSurface(hit.hitpos, &hit.hitSector);
|
|
ContinueHitscan(pp, hit.hitSector, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang, xvect, yvect, zvect);
|
|
return 0;
|
|
}
|
|
}
|
|
else if (abs(hit.hitpos.Z - hit.hitSector->floorz) <= 1)
|
|
{
|
|
if ((hit.hitSector->extra & SECTFX_LIQUID_MASK) != SECTFX_LIQUID_NONE)
|
|
{
|
|
SpawnSplashXY(hit.int_hitpos().X,hit.int_hitpos().Y,hit.int_hitpos().Z,hit.hitSector);
|
|
|
|
if (SectorIsDiveArea(hit.hitSector))
|
|
{
|
|
WarpToUnderwater(hit.hitpos, &hit.hitSector);
|
|
ContinueHitscan(pp, hit.hitSector, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang, xvect, yvect, zvect);
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, DAngle::fromBuild(daang), actor->user.ID);
|
|
return 0;
|
|
}
|
|
|
|
QueueHole(hit.hitSector, hit.hitWall, hit.hitpos);
|
|
}
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->hasU()) // JBF: added null check
|
|
if (hitActor->user.ID == TRASHCAN)
|
|
{
|
|
extern STATE s_TrashCanPain[];
|
|
|
|
PlaySound(DIGI_TRASHLID, hitActor, v3df_none);
|
|
if (hitActor->user.WaitTics <= 0)
|
|
{
|
|
hitActor->user.WaitTics = SEC(2);
|
|
ChangeState(hitActor,s_TrashCanPain);
|
|
}
|
|
}
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_UZI, hitActor))
|
|
return 0;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE) && HitBreakSprite(hitActor,0))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (BulletHitSprite(pp->actor, hitActor, hit.hitpos, 0))
|
|
return 0;
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hitActor);
|
|
}
|
|
}
|
|
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, hit.hitSector, hit.hitpos, DAngle::fromBuild(daang), 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.cstat |= (cstat | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit.hitWall);
|
|
DoHitscanDamage(actorNew, hit.actor());
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit.hitSector, hit.hitpos, DAngle::fromBuild(daang), 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->user.spal = actorNew->spr.pal = pal;
|
|
actorNew->spr.cstat |= (cstat | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit.hitWall);
|
|
|
|
if (RANDOM_P2(1024) < 100)
|
|
{
|
|
PlaySound(DIGI_RICHOCHET1,actorNew, v3df_none);
|
|
}
|
|
else if (RANDOM_P2(1024) < 100)
|
|
PlaySound(DIGI_RICHOCHET2,actorNew, v3df_none);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitTankShell(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
if (!SW_SHAREWARE)
|
|
PlaySound(DIGI_CANNON, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, 0, s_TankShell, actor->sector(), actor->spr.pos, actor->spr.angle, TANK_SHELL_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 8;
|
|
actorNew->spr.xrepeat = 8;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->set_int_zvel(int(-pp->horizon.horiz.asbuildf() * actorNew->int_xvel() * (1. / 8.)));
|
|
|
|
WeaponAutoAim(actor, actorNew, 64, false);
|
|
// a bit of randomness
|
|
actorNew->add_int_ang(RandomRange(30) - 15);
|
|
actorNew->norm_ang();
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTurretMicro(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
DSWActor* plActor = pp->actor;
|
|
int nx, ny, nz, dist;
|
|
short i,ang;
|
|
TARGET_SORT* ts = TargetSort;
|
|
DSWActor* picked = nullptr;
|
|
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
|
|
nx = actor->int_pos().X;
|
|
ny = actor->int_pos().Y;
|
|
|
|
const int MAX_TURRET_MICRO = 10;
|
|
|
|
DoPickTarget(plActor, 256, false);
|
|
|
|
if (TargetSortCount > MAX_TURRET_MICRO)
|
|
TargetSortCount = MAX_TURRET_MICRO;
|
|
|
|
for (i = 0; i < MAX_TURRET_MICRO; i++)
|
|
{
|
|
if (ts < &TargetSort[TargetSortCount] && ts->actor != nullptr)
|
|
{
|
|
picked = ts->actor;
|
|
|
|
ang = getangle(picked->int_pos().X - nx, picked->int_pos().Y - ny);
|
|
|
|
ts++;
|
|
}
|
|
else
|
|
{
|
|
picked = nullptr;
|
|
ang = actor->int_ang();
|
|
}
|
|
|
|
nz = actor->int_pos().Z;
|
|
nz += Z(RandomRange(20)) - Z(10);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Micro[0][0], actor->sector(),
|
|
nx, ny, nz, ang, 1200);
|
|
|
|
SetOwner(plActor, actorNew);
|
|
actorNew->spr.yrepeat = 24;
|
|
actorNew->spr.xrepeat = 24;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
// randomize zvelocity
|
|
actorNew->add_int_zvel( RandomRange(Z(8)) - Z(5));
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Micro[0]);
|
|
|
|
actorNew->user.WeaponNum = plActor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (2);
|
|
actorNew->user.floor_dist = (2);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->user.WaitTics = 10 + RandomRange(40);
|
|
|
|
const int MICRO_ANG = 400;
|
|
|
|
if (picked)
|
|
{
|
|
dist = DistanceI(actorNew->spr.pos, picked->spr.pos);
|
|
if (dist != 0)
|
|
{
|
|
double zh = ActorZOfTop(picked) + (ActorSizeZ(picked) * 0.25);
|
|
actorNew->set_int_zvel((actorNew->int_xvel() * (zh - actorNew->spr.pos.Z) * zworldtoint) / dist);
|
|
}
|
|
|
|
actorNew->user.WpnGoalActor = ts->actor;
|
|
picked->user.Flags |= (SPR_TARGETED);
|
|
picked->user.Flags |= (SPR_ATTACKED);
|
|
}
|
|
else
|
|
{
|
|
actorNew->spr.angle += DAngle::fromBuild((RandomRange(MICRO_ANG) - (MICRO_ANG / 2)) - 16);
|
|
}
|
|
|
|
UpdateChange(actorNew);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitTurretRocket(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, &s_Rocket[0][0], actor->sector(), actor->spr.pos, actor->spr.angle, ROCKET_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->set_int_zvel(int(-pp->horizon.horiz.asbuildf() * actorNew->int_xvel() * (1. / 8.)));
|
|
|
|
WeaponAutoAim(actor, actorNew, 64, false);
|
|
// a bit of randomness
|
|
//actorNew->spr.angle += RandomRange(30) - 15;
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTurretFireball(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL, s_Fireball, actor->sector(), actor->spr.pos, actor->spr.angle, FIREBALL_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 50;
|
|
actorNew->user.ceiling_dist = (4);
|
|
actorNew->user.floor_dist = (4);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->set_int_zvel(int(-pp->horizon.horiz.asbuildf() * actorNew->int_xvel() * (1. / 8.)));
|
|
|
|
WeaponAutoAim(actor, actorNew, 64, false);
|
|
// a bit of randomness
|
|
actorNew->add_int_ang(RandomRange(30) - 15);
|
|
actorNew->norm_ang();
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTurretRail(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R1, &s_Rail[0][0], pp->cursector, actor->spr.pos, actor->spr.angle, 1200);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Rail[0]);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER|CSTAT_SPRITE_INVISIBLE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->norm_ang();
|
|
}
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitTurretLaser(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
if (SW_SHAREWARE) return false; // JBF: verify
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BOLT_THINMAN_R0, s_Laser, pp->cursector, actor->spr.pos, actor->spr.angle, 300);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 52;
|
|
actorNew->spr.xrepeat = 52;
|
|
actorNew->spr.shade = -15;
|
|
|
|
// the slower the missile travels the less of a zvel it needs
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 11);
|
|
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (1);
|
|
actorNew->user.floor_dist = (1);
|
|
actorNew->user.Flags2 |= (SPR2_SO_MISSILE);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->spr.clipdist = 64L>>2;
|
|
|
|
if (WeaponAutoAim(actor, actorNew, 32, false) == -1)
|
|
{
|
|
actorNew->norm_ang();
|
|
}
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitSobjMachineGun(DSWActor* actor, PLAYER* pp)
|
|
{
|
|
short daang;
|
|
HitInfo hit{};
|
|
int daz;
|
|
int nx,ny,nz;
|
|
short cstat = 0;
|
|
DSWActor* spark;
|
|
|
|
PlaySound(DIGI_BOATFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
nx = actor->int_pos().X;
|
|
ny = actor->int_pos().Y;
|
|
daz = nz = actor->int_pos().Z;
|
|
|
|
if (RANDOM_P2(1024) < 200)
|
|
InitTracerTurret(actor, pp->actor, pp->horizon.horiz.asq16());
|
|
|
|
daang = 64;
|
|
if (WeaponAutoAimHitscan(actor, &daz, &daang, false) != nullptr)
|
|
{
|
|
daz += RandomRange(Z(30)) - Z(15);
|
|
//daz += 0;
|
|
}
|
|
else
|
|
{
|
|
fixed_t q16horiz = pp->horizon.horiz.asq16();
|
|
fixed_t horizmin = IntToFixed(-25);
|
|
if (q16horiz < horizmin)
|
|
q16horiz = horizmin;
|
|
|
|
daz = -MulScale(q16horiz, 2000, 16) + (RandomRange(Z(80)) - Z(40));
|
|
daang = actor->int_ang();
|
|
}
|
|
|
|
FAFhitscan(nx, ny, nz, actor->sector(), // Start position
|
|
bcos(daang), // X vector of 3D ang
|
|
bsin(daang), // Y vector of 3D ang
|
|
daz, // Z vector of 3D ang
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (hit.actor() == nullptr && hit.hitWall == nullptr)
|
|
{
|
|
if (abs(hit.hitpos.Z - hit.hitSector->ceilingz) <= 1)
|
|
{
|
|
hit.hitpos.Z += 16;
|
|
cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
if ((hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
return 0;
|
|
}
|
|
else if (abs(hit.hitpos.Z - hit.hitSector->floorz) <= 1)
|
|
{
|
|
if ((hit.hitSector->extra & SECTFX_LIQUID_MASK) != SECTFX_LIQUID_NONE)
|
|
{
|
|
SpawnSplashXY(hit.int_hitpos().X,hit.int_hitpos().Y,hit.int_hitpos().Z,hit.hitSector);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
auto hitActor = hit.actor();
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
// spawn sparks here and pass the sprite as SO_MISSILE
|
|
spark = SpawnBoatSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
spark->user.Flags2 |= SPR2_SO_MISSILE;
|
|
if (MissileHitMatch(spark, -1, hit.actor()))
|
|
return 0;
|
|
return 0;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hit.actor(), 0);
|
|
return 0;
|
|
}
|
|
|
|
if (BulletHitSprite(pp->actor, hit.actor(), hit.hitpos, 0))
|
|
return 0;
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hit.actor());
|
|
}
|
|
}
|
|
|
|
spark = SpawnBoatSparks(pp, hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
DoHitscanDamage(spark, hit.actor());
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitSobjGun(PLAYER* pp)
|
|
{
|
|
short i;
|
|
bool first = false;
|
|
|
|
for (i = 0; pp->sop->so_actors[i] != nullptr; i++)
|
|
{
|
|
DSWActor* actor = pp->sop->so_actors[i];
|
|
if (!actor) continue;
|
|
if (actor->spr.statnum == STAT_SO_SHOOT_POINT)
|
|
{
|
|
// match when firing
|
|
if (SP_TAG2(actor))
|
|
{
|
|
DoMatchEverything(pp, SP_TAG2(actor), -1);
|
|
if (TEST_BOOL1(actor))
|
|
{
|
|
SP_TAG2(actor) = 0;
|
|
}
|
|
}
|
|
|
|
// inert shoot point
|
|
if ((uint8_t)SP_TAG3(actor) == 255)
|
|
return 0;
|
|
|
|
if (!first)
|
|
{
|
|
first = true;
|
|
if (SP_TAG6(actor))
|
|
DoSoundSpotMatch(SP_TAG6(actor), 1, SOUND_OBJECT_TYPE);
|
|
}
|
|
|
|
switch (SP_TAG3(actor))
|
|
{
|
|
case 32:
|
|
case 0:
|
|
SpawnVis(actor, nullptr, {}, 8);
|
|
SpawnBigGunFlames(actor, pp->actor, pp->sop, false);
|
|
SetGunQuake(actor);
|
|
InitTankShell(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 80;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 1:
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
SpawnBigGunFlames(actor, pp->actor, pp->sop, true);
|
|
InitSobjMachineGun(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 10;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 2:
|
|
if (SW_SHAREWARE) break;
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
InitTurretLaser(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 120;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 3:
|
|
if (SW_SHAREWARE) break;
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
InitTurretRail(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 120;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 4:
|
|
if (SW_SHAREWARE) break;
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
InitTurretFireball(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 20;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 5:
|
|
if (SW_SHAREWARE) break;
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
InitTurretRocket(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 100;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
case 6:
|
|
if (SW_SHAREWARE) break;
|
|
SpawnVis(actor, nullptr, {}, 32);
|
|
InitTurretMicro(actor, pp);
|
|
if (!SP_TAG5(actor))
|
|
pp->FirePause = 100;
|
|
else
|
|
pp->FirePause = SP_TAG5(actor);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
DSWActor* SpawnBoatSparks(PLAYER* pp, sectortype* hit_sect, walltype* hit_wall, int hit_x, int hit_y, int hit_z, short hit_ang)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT + 12;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT + 12;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
// Sprite starts out with center exactly on wall.
|
|
// This moves it back enough to see it at all angles.
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT + 10;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT + 10;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_DEFAULT;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
if (RANDOM_P2(1024) < 100)
|
|
PlaySound(DIGI_RICHOCHET1, actorNew, v3df_none);
|
|
|
|
return actorNew;
|
|
}
|
|
|
|
int SpawnSwordSparks(PLAYER* pp, sectortype* hit_sect, walltype* hit_wall, int hit_x, int hit_y, int hit_z, short hit_ang)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = actorNew->spr.yrepeat = 20;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
// Sprite starts out with center exactly on wall.
|
|
// This moves it back enough to see it at all angles.
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
if (hit_wall != nullptr)
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = actorNew->spr.yrepeat = 20;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_DEFAULT;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
if (actor->user.WeaponNum == WPN_FIST)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_INVISIBLE);
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
if (hit_wall != nullptr)
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
return 0;
|
|
}
|
|
|
|
DSWActor* SpawnTurretSparks(sectortype* hit_sect, walltype* hit_wall, int hit_x, int hit_y, int hit_z, short hit_ang)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT + 12;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT + 12;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
|
|
// Sprite starts out with center exactly on wall.
|
|
// This moves it back enough to see it at all angles.
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT + 10;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT + 10;
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_DEFAULT;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
if (RANDOM_P2(1024) < 100)
|
|
PlaySound(DIGI_RICHOCHET1, actorNew, v3df_none);
|
|
|
|
return actorNew;
|
|
}
|
|
|
|
DSWActor* SpawnShotgunSparks(PLAYER* pp, sectortype* hit_sect, walltype* hit_wall, int hit_x, int hit_y, int hit_z, short hit_ang)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_DEFAULT;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, SHOTGUN_SMOKE, s_ShotgunSmoke, hit_sect, hit_x, hit_y, hit_z, hit_ang, 0);
|
|
actorNew->spr.xrepeat = SHOTGUN_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = SHOTGUN_SMOKE_REPEAT;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
// Sprite starts out with center exactly on wall.
|
|
// This moves it back enough to see it at all angles.
|
|
|
|
actorNew->spr.clipdist = 32 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit_wall);
|
|
|
|
return actorNew;
|
|
}
|
|
|
|
int InitTurretMgun(SECTOR_OBJECT* sop)
|
|
{
|
|
short daang, i;
|
|
HitInfo hit{};
|
|
int daz;
|
|
int nx,ny,nz;
|
|
short cstat = 0;
|
|
short delta;
|
|
int xvect,yvect,zvect;
|
|
|
|
PlaySound(DIGI_BOATFIRE, sop->pmid, v3df_dontpan|v3df_doppler);
|
|
|
|
for (i = 0; sop->so_actors[i] != nullptr; i++)
|
|
{
|
|
DSWActor* actor = sop->so_actors[i];
|
|
if (!actor) continue;
|
|
if (actor->spr.statnum == STAT_SO_SHOOT_POINT)
|
|
{
|
|
nx = actor->int_pos().X;
|
|
ny = actor->int_pos().Y;
|
|
daz = nz = actor->int_pos().Z;
|
|
|
|
// if its not operated by a player
|
|
if (sop->Animator)
|
|
{
|
|
// only auto aim for Z
|
|
daang = 512;
|
|
auto hitt = WeaponAutoAimHitscan(actor, &daz, &daang, false);
|
|
hit.hitActor = hitt;
|
|
if (hitt != nullptr)
|
|
{
|
|
delta = short(abs(getincangle(actor->int_ang(), daang)));
|
|
if (delta > 128)
|
|
{
|
|
// don't shoot if greater than 128
|
|
return 0;
|
|
}
|
|
else if (delta > 24)
|
|
{
|
|
// always shoot the ground when tracking
|
|
// and not close
|
|
double fdaz = daz * zinttoworld;
|
|
WeaponHitscanShootFeet(actor, hitt, &fdaz);
|
|
daz = fdaz * zworldtoint;
|
|
|
|
daang = actor->int_ang();
|
|
daang = NORM_ANGLE(daang + RANDOM_P2(32) - 16);
|
|
}
|
|
else
|
|
{
|
|
// randomize the z for shots
|
|
daz += RandomRange(Z(120)) - Z(60);
|
|
// never auto aim the angle
|
|
daang = actor->int_ang();
|
|
daang = NORM_ANGLE(daang + RANDOM_P2(64) - 32);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
daang = 64;
|
|
if (WeaponAutoAimHitscan(actor, &daz, &daang, false) != nullptr)
|
|
{
|
|
daz += RandomRange(Z(30)) - Z(15);
|
|
}
|
|
}
|
|
|
|
xvect = bcos(daang);
|
|
yvect = bsin(daang);
|
|
zvect = daz;
|
|
|
|
FAFhitscan(nx, ny, nz, actor->sector(), // Start position
|
|
xvect, yvect, zvect,
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (RANDOM_P2(1024) < 400)
|
|
{
|
|
InitTracerAutoTurret(sop->so_actors[i],
|
|
xvect>>4, yvect>>4, zvect>>4);
|
|
}
|
|
|
|
if (hit.hitSector == nullptr)
|
|
continue;
|
|
|
|
if (hit.actor() == nullptr && hit.hitWall == nullptr)
|
|
{
|
|
if (abs(hit.hitpos.Z - hit.hitSector->ceilingz) <= 1)
|
|
{
|
|
hit.hitpos.Z += 16;
|
|
cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
if ((hit.hitSector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
continue;
|
|
}
|
|
else if (abs(hit.hitpos.Z - hit.hitSector->floorz) <= 1)
|
|
{
|
|
if ((hit.hitSector->extra & SECTFX_LIQUID_MASK) != SECTFX_LIQUID_NONE)
|
|
{
|
|
SpawnSplashXY(hit.int_hitpos().X,hit.int_hitpos().Y,hit.int_hitpos().Z,hit.hitSector);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, DAngle::fromBuild(daang), 0);
|
|
continue;
|
|
}
|
|
|
|
QueueHole(hit.hitSector, hit.hitWall, hit.hitpos);
|
|
}
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
auto hitActor = hit.actor();
|
|
|
|
if (hitActor->spr.lotag == TAG_SPRITE_HIT_MATCH)
|
|
{
|
|
if (MissileHitMatch(nullptr, WPN_UZI, hit.actor()))
|
|
continue;
|
|
}
|
|
|
|
if ((hitActor->spr.extra & SPRX_BREAKABLE))
|
|
{
|
|
HitBreakSprite(hit.actor(), 0);
|
|
continue;
|
|
}
|
|
|
|
if (BulletHitSprite(actor, hit.actor(), hit.hitpos, 0))
|
|
continue;
|
|
|
|
// hit a switch?
|
|
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL) && (hitActor->spr.lotag || hitActor->spr.hitag))
|
|
{
|
|
ShootableSwitch(hit.actor());
|
|
}
|
|
}
|
|
|
|
|
|
auto j = SpawnTurretSparks(hit.hitSector, hit.hitWall, hit.int_hitpos().X, hit.int_hitpos().Y, hit.int_hitpos().Z, daang);
|
|
DoHitscanDamage(j, hit.actor());
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitEnemyUzi(DSWActor* actor)
|
|
{
|
|
DAngle daang;
|
|
HitInfo hit{};
|
|
double zh;
|
|
void InitUziShell(PLAYER*);
|
|
static short alternate;
|
|
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
SetActorZ(actor, actor->spr.pos);
|
|
|
|
if (actor->user.ID == ZILLA_RUN_R0)
|
|
{
|
|
zh = ActorZOfTop(actor) + 20;
|
|
}
|
|
else
|
|
{
|
|
zh = ActorSizeZ(actor) * 0.5;
|
|
}
|
|
double daz = actor->spr.pos.Z - zh;
|
|
|
|
if (AimHitscanToTarget(actor, &daz, &daang, 200) != nullptr)
|
|
{
|
|
// set angle to player and also face player when attacking
|
|
actor->spr.angle = daang;
|
|
daang += DAngle::fromBuild(RandomRange(24) - 12);
|
|
daz += RandomRange(40 * 256) / 256. - 20;
|
|
}
|
|
else
|
|
{
|
|
// couldn't shoot target for some reason
|
|
|
|
// don't bother wasting processing 50% of the time
|
|
if (RANDOM_P2(1024) < 512)
|
|
return 0;
|
|
|
|
daz = 0;
|
|
daang = actor->spr.angle + RandomAngle(DAngle22_5) - DAngle22_5/2;
|
|
}
|
|
|
|
// todo: confirm the ToVector factor.
|
|
FAFhitscan(actor->spr.pos.plusZ(-zh), actor->sector(), DVector3(daang.ToVector() * 1024, daz), hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
return 0;
|
|
|
|
if (RANDOM_P2(1024<<4)>>4 > 700)
|
|
{
|
|
if (actor->user.ID == TOILETGIRL_R0 || actor->user.ID == WASHGIRL_R0 || actor->user.ID == CARGIRL_R0)
|
|
SpawnShell(actor,-3);
|
|
else
|
|
SpawnShell(actor,-2); // Enemy Uzi shell
|
|
}
|
|
|
|
if ((alternate++)>2) alternate = 0;
|
|
if (!alternate)
|
|
{
|
|
if (actor->spr.pal == PALETTE_PLAYER3 || actor->spr.pal == PALETTE_PLAYER5 ||
|
|
actor->spr.pal == PAL_XLAT_LT_GREY || actor->spr.pal == PAL_XLAT_LT_TAN)
|
|
PlaySound(DIGI_M60, actor, v3df_none);
|
|
else
|
|
PlaySound(DIGI_NINJAUZIATTACK, actor, v3df_none);
|
|
}
|
|
|
|
if (hit.hitWall != nullptr)
|
|
{
|
|
if (hit.hitWall->twoSided())
|
|
{
|
|
if ((hit.hitWall->nextSector()->ceilingstat & CSTAT_SECTOR_SKY))
|
|
{
|
|
if (hit.hitpos.Z < hit.hitWall->nextSector()->ceilingz)
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hit.hitWall->lotag == TAG_WALL_BREAK)
|
|
{
|
|
HitBreakWall(hit.hitWall, hit.hitpos, daang, actor->user.ID);
|
|
return 0;
|
|
}
|
|
|
|
QueueHole(hit.hitSector, hit.hitWall, hit.hitpos);
|
|
}
|
|
|
|
if (hit.actor() != nullptr)
|
|
{
|
|
if (BulletHitSprite(actor, hit.actor(), hit.hitpos, 0))
|
|
return 0;
|
|
}
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE+2, s_UziSmoke, hit.hitSector, hit.hitpos, daang, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT;
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
|
|
actorNew->user.WaitTics = 63;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
|
|
actorNew->spr.clipdist = 32L >> 2;
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SMOKE, s_UziSmoke, hit.hitSector, hit.hitpos, daang, 0);
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SMOKE_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SMOKE_REPEAT;
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit.hitWall);
|
|
DoHitscanDamage(actorNew, hit.actor());
|
|
|
|
actorNew = SpawnActor(STAT_MISSILE, UZI_SPARK, s_UziSpark, hit.hitSector, hit.hitpos, daang, 0);
|
|
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = UZI_SPARK_REPEAT;
|
|
actorNew->spr.yrepeat = UZI_SPARK_REPEAT;
|
|
SetOwner(actor, actorNew);
|
|
actorNew->user.spal = actorNew->spr.pal;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.clipdist = 8 >> 2;
|
|
|
|
HitscanSpriteAdjust(actorNew, hit.hitWall);
|
|
|
|
if (RANDOM_P2(1024) < 100)
|
|
{
|
|
PlaySound(DIGI_RICHOCHET1,actorNew, v3df_none);
|
|
}
|
|
else if (RANDOM_P2(1024) < 100)
|
|
PlaySound(DIGI_RICHOCHET2,actorNew, v3df_none);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitGrenade(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int zvel;
|
|
bool auto_aim = false;
|
|
|
|
DoPlayerBeginRecoil(pp, GRENADE_RECOIL_AMT);
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
|
|
PlaySound(DIGI_30MMFIRE, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, GRENADE, &s_Grenade[0][0], pp->cursector, pos, pp->angle.ang, GRENADE_VELOCITY);
|
|
|
|
// don't throw it as far if crawling
|
|
if (pp->Flags & (PF_CRAWLING))
|
|
{
|
|
actorNew->add_int_xvel(-(actorNew->int_xvel() >> 2));
|
|
}
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Grenade[0]);
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.shade = -15;
|
|
//actorNew->spr.clipdist = 80L>>2;
|
|
actorNew->spr.clipdist = 32L>>2;
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->user.Counter = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK);
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 800);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
// don't do smoke for this movement
|
|
actorNew->user.Flags |= (SPR_BOUNCE);
|
|
MissileSetPos(actorNew, DoGrenade, 1000);
|
|
actorNew->user.Flags &= ~(SPR_BOUNCE);
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
zvel = actorNew->int_zvel();
|
|
if (WeaponAutoAim(pp->actor, actorNew, 32, false) >= 0)
|
|
{
|
|
auto_aim = true;
|
|
}
|
|
actorNew->set_int_zvel(zvel);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
if (!auto_aim)
|
|
{
|
|
// adjust xvel according to player velocity
|
|
actorNew->user.change += pp->vect;
|
|
}
|
|
|
|
actorNew->user.Counter2 = true; // Phosphorus Grenade
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitSpriteGrenade(DSWActor* actor)
|
|
{
|
|
PlaySound(DIGI_30MMFIRE, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
auto actorNew = SpawnActor(STAT_MISSILE, GRENADE, &s_Grenade[0][0], actor->sector(),
|
|
actor->spr.pos.plusZ(-40), actor->spr.angle, GRENADE_VELOCITY);
|
|
|
|
actorNew->user.RotNum = 5;
|
|
NewStateGroup(actorNew, &sg_Grenade[0]);
|
|
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
|
|
|
|
if (actor->user.ID == ZOMBIE_RUN_R0)
|
|
SetOwner(GetOwner(actor), actorNew);
|
|
else
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.shade = -15;
|
|
//actorNew->spr.clipdist = 80L>>2;
|
|
actorNew->spr.clipdist = 32L>>2;
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (3);
|
|
actorNew->user.floor_dist = (3);
|
|
actorNew->user.Counter = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK);
|
|
|
|
|
|
actorNew->set_int_zvel(-2000);
|
|
|
|
UpdateChange(actorNew);
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 800);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
// don't do smoke for this movement
|
|
actorNew->user.Flags |= (SPR_BOUNCE);
|
|
MissileSetPos(actorNew, DoGrenade, 400);
|
|
actorNew->user.Flags &= ~(SPR_BOUNCE);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitMine(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
|
|
PlayerUpdateAmmo(pp, actor->user.WeaponNum, -1);
|
|
|
|
PlaySound(DIGI_MINETHROW, pp, v3df_dontpan|v3df_doppler);
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 8);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, MINE, s_Mine, pp->cursector, pos, pp->angle.ang, MINE_VELOCITY);
|
|
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->spr.clipdist = 128L>>2;
|
|
actorNew->set_int_zvel(-pp->horizon.horiz.asq16() >> 9);
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (5);
|
|
actorNew->user.floor_dist = (5);
|
|
actorNew->user.Counter = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.spal = actorNew->spr.pal = actor->user.spal; // Set sticky color
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
MissileSetPos(actorNew, DoMine, 800);
|
|
|
|
UpdateChange(actorNew, 0.5);
|
|
|
|
double dot = pp->vect.dot(pp->angle.ang.ToVector());
|
|
|
|
// don't adjust for strafing
|
|
// not really sure what to do here as the original formula was very likely to overflow, creating a Q0.32 value.
|
|
if (abs(dot) > 10000./0xffffffff)
|
|
{
|
|
// adjust xvel according to player velocity
|
|
actorNew->user.change += 2 * pp->vect;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyMine(DSWActor* actor)
|
|
{
|
|
int nx, ny, nz;
|
|
|
|
PlaySound(DIGI_MINETHROW, actor, v3df_dontpan|v3df_doppler);
|
|
|
|
nx = actor->int_pos().X;
|
|
ny = actor->int_pos().Y;
|
|
nz = actor->int_pos().Z - Z(40);
|
|
|
|
// Spawn a shot
|
|
// Inserting and setting up variables
|
|
auto actorNew = SpawnActor(STAT_MISSILE, MINE, s_Mine, actor->sector(), actor->spr.pos.plusZ(-40), actor->spr.angle, MINE_VELOCITY);
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->spr.yrepeat = 32;
|
|
actorNew->spr.xrepeat = 32;
|
|
actorNew->spr.shade = -15;
|
|
actorNew->spr.clipdist = 128L>>2;
|
|
|
|
actorNew->user.WeaponNum = actor->user.WeaponNum;
|
|
actorNew->user.Radius = 200;
|
|
actorNew->user.ceiling_dist = (5);
|
|
actorNew->user.floor_dist = (5);
|
|
actorNew->user.Counter = 0;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actorNew->user.spal = actorNew->spr.pal = actor->user.spal; // Set sticky color
|
|
|
|
if (SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
MissileSetPos(actorNew, DoMine, 300);
|
|
actorNew->spr.angle -= DAngle90;
|
|
MissileSetPos(actorNew, DoMine, 300);
|
|
actorNew->spr.angle += DAngle90;
|
|
|
|
actorNew->user.change. Z = -5000 / 256.;
|
|
UpdateChangeXY(actorNew);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int HelpMissileLateral(DSWActor* actor, int dist)
|
|
{
|
|
auto old_xvel = actor->int_xvel();
|
|
auto old_clipdist = actor->spr.clipdist;
|
|
|
|
actor->set_int_xvel(dist);
|
|
|
|
auto vec = MOVExy(actor->int_xvel(), actor->spr.angle);
|
|
|
|
actor->spr.clipdist = 32L >> 2;
|
|
|
|
actor->user.coll = move_missile(actor, DVector3(vec, 0), 16, 16, 0, 1);
|
|
|
|
actor->set_int_xvel(old_xvel);
|
|
actor->spr.clipdist = old_clipdist;
|
|
|
|
actor->backuppos();
|
|
return 0;
|
|
}
|
|
|
|
|
|
int InitFireball(PLAYER* pp)
|
|
{
|
|
DSWActor* actor = pp->actor;
|
|
int zvel;
|
|
|
|
PlayerUpdateAmmo(pp, WPN_HOTHEAD, -1);
|
|
|
|
PlaySound(DIGI_HEADFIRE, pp, v3df_none);
|
|
|
|
// Make sprite shade brighter
|
|
actor->user.Vis = 128;
|
|
|
|
if (!pp->insector())
|
|
return 0;
|
|
|
|
auto pos = pp->pos.plusZ(pp->bob_z + 15);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL1, s_Fireball, pp->cursector, pos, pp->angle.ang, FIREBALL_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.xrepeat = 40;
|
|
actorNew->spr.yrepeat = 40;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.clipdist = 32>>2;
|
|
SetOwner(pp->actor, actorNew);
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->user.Radius = 100;
|
|
|
|
actorNew->user.ceiling_dist = (6);
|
|
actorNew->user.floor_dist = (6);
|
|
zvel = -MulScale(pp->horizon.horiz.asq16(), 240, 16);
|
|
|
|
// at certain angles the clipping box was big enough to block the
|
|
// initial positioning of the fireball.
|
|
auto oclipdist = actor->spr.clipdist;
|
|
actor->spr.clipdist = 0;
|
|
|
|
actorNew->spr.angle += DAngle90;
|
|
HelpMissileLateral(actorNew, 2100);
|
|
actorNew->spr.angle -= DAngle90;
|
|
|
|
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
|
|
if (TestMissileSetPos(actorNew, DoFireball, 1200, MulScale(zvel,44000, 16)))
|
|
{
|
|
actor->spr.clipdist = oclipdist;
|
|
KillActor(actorNew);
|
|
return 0;
|
|
}
|
|
|
|
actor->spr.clipdist = oclipdist;
|
|
|
|
actorNew->set_int_zvel(zvel >> 1);
|
|
if (WeaponAutoAimZvel(pp->actor, actorNew, &zvel, 32, false) == -1)
|
|
{
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() - 9));
|
|
}
|
|
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->user.set_int_change_z(zvel);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int InitEnemyFireball(DSWActor* actor)
|
|
{
|
|
int nz, dist;
|
|
int size_z;
|
|
int i, targ_z, xchange, ychange;
|
|
|
|
static short lat_ang[] =
|
|
{
|
|
512, -512
|
|
};
|
|
|
|
auto targetActor = actor->user.targetActor;
|
|
if (!targetActor) return 0;
|
|
|
|
PlaySound(DIGI_FIREBALL1, actor, v3df_none);
|
|
|
|
// get angle to player and also face player when attacking
|
|
actor->set_int_ang(NORM_ANGLE(getangle(targetActor->int_pos().X - actor->int_pos().X, targetActor->int_pos().Y - actor->int_pos().Y)));
|
|
|
|
size_z = Z(ActorSizeY(actor));
|
|
nz = actor->int_pos().Z - size_z + (size_z >> 2) + (size_z >> 3) + Z(4);
|
|
|
|
xchange = MOVEx(GORO_FIREBALL_VELOCITY, actor->int_ang());
|
|
ychange = MOVEy(GORO_FIREBALL_VELOCITY, actor->int_ang());
|
|
|
|
int lastvel = 0;
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, GORO_FIREBALL, s_Fireball, actor->sector(),
|
|
actor->int_pos().X, actor->int_pos().Y, nz, actor->int_ang(), GORO_FIREBALL_VELOCITY);
|
|
|
|
actorNew->spr.hitag = LUMINOUS; //Always full brightness
|
|
actorNew->spr.xrepeat = 20;
|
|
actorNew->spr.yrepeat = 20;
|
|
actorNew->spr.shade = -40;
|
|
|
|
SetOwner(actor, actorNew);
|
|
actorNew->vel.Z = 0;
|
|
actorNew->spr.clipdist = 16>>2;
|
|
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() + lat_ang[i]));
|
|
HelpMissileLateral(actorNew, 500);
|
|
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang() - lat_ang[i]));
|
|
|
|
actorNew->user.set_int_change_x(xchange);
|
|
actorNew->user.set_int_change_y(ychange);
|
|
|
|
MissileSetPos(actorNew, DoFireball, 700);
|
|
|
|
if (i == 0)
|
|
{
|
|
// find the distance to the target (player)
|
|
dist = ksqrt(SQ(actorNew->int_pos().X - targetActor->int_pos().X) + SQ(actorNew->int_pos().Y - targetActor->int_pos().Y));
|
|
|
|
// Determine target Z value
|
|
targ_z = targetActor->int_pos().Z - (Z(ActorSizeY(actor)) >> 1);
|
|
|
|
// (velocity * difference between the target and the throwing star) /
|
|
// distance
|
|
if (dist != 0)
|
|
{
|
|
actorNew->set_int_zvel((GORO_FIREBALL_VELOCITY * (targ_z - actorNew->int_pos().Z)) / dist);
|
|
actorNew->user.change.Z = actorNew->vel.Z;
|
|
}
|
|
// back up first one
|
|
lastvel = actorNew->int_zvel();
|
|
}
|
|
else
|
|
{
|
|
// use the first calculations so the balls stay together
|
|
actorNew->user.set_int_change_z(lastvel);
|
|
actorNew->set_int_zvel(lastvel);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// for hitscans or other uses
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool WarpToUnderwater(DVector3& pos, sectortype** psectu)
|
|
{
|
|
int i;
|
|
auto sectu = *psectu;
|
|
bool Found = false;
|
|
DVector2 spos;
|
|
DSWActor* overActor = nullptr;
|
|
DSWActor* underActor = nullptr;
|
|
|
|
// 0 not valid for water match tags
|
|
if (!sectu->hasU() || sectu->number == 0)
|
|
return false;
|
|
|
|
// search for DIVE_AREA "over" sprite for reference point
|
|
SWStatIterator it(STAT_DIVE_AREA);
|
|
while ((overActor = it.Next()))
|
|
{
|
|
if ((overActor->sector()->extra & SECTFX_DIVE_AREA) &&
|
|
overActor->sector()->hasU() &&
|
|
overActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
Found = false;
|
|
|
|
// search for UNDERWATER "under" sprite for reference point
|
|
it.Reset(STAT_UNDERWATER);
|
|
while ((underActor = it.Next()))
|
|
{
|
|
if ((underActor->sector()->extra & SECTFX_UNDERWATER) &&
|
|
underActor->sector()->hasU() &&
|
|
underActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
|
|
// get the offset from the sprite
|
|
spos = overActor->spr.pos.XY() - pos.XY();
|
|
|
|
// update to the new x y position
|
|
pos.XY() = underActor->spr.pos - spos;
|
|
|
|
auto over = overActor->sector();
|
|
auto under = underActor->sector();
|
|
|
|
if (GetOverlapSector(pos.XY(), &over, &under) == 2)
|
|
{
|
|
*psectu = under;
|
|
}
|
|
else
|
|
{
|
|
*psectu = under;
|
|
}
|
|
|
|
pos.Z = underActor->sector()->ceilingz+ 1;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool WarpToSurface(DVector3& pos, sectortype** psectu)
|
|
{
|
|
int i;
|
|
auto sectu = *psectu;
|
|
DSWActor* overActor = nullptr;
|
|
DSWActor* underActor = nullptr;
|
|
bool Found = false;
|
|
|
|
// 0 not valid for water match tags
|
|
if (!sectu->hasU() || sectu->number == 0)
|
|
return false;
|
|
|
|
// search for UNDERWATER "under" sprite for reference point
|
|
SWStatIterator it(STAT_UNDERWATER);
|
|
while ((underActor = it.Next()))
|
|
{
|
|
if ((underActor->sector()->extra & SECTFX_UNDERWATER) &&
|
|
underActor->sector()->hasU() &&
|
|
underActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
Found = false;
|
|
|
|
// search for DIVE_AREA "over" sprite for reference point
|
|
it.Reset(STAT_DIVE_AREA);
|
|
while ((overActor = it.Next()))
|
|
{
|
|
if ((overActor->sector()->extra & SECTFX_DIVE_AREA) &&
|
|
overActor->sector()->hasU() &&
|
|
overActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
|
|
// get the offset from the under sprite
|
|
DVector2 spos = underActor->spr.pos.XY() - pos.XY();
|
|
|
|
// update to the new x y position
|
|
pos.XY() = overActor->spr.pos.XY() - spos;
|
|
|
|
auto over = overActor->sector();
|
|
auto under = underActor->sector();
|
|
|
|
if (GetOverlapSector(pos.XY(), &over, &under))
|
|
{
|
|
*psectu = over;
|
|
}
|
|
|
|
pos.Z = overActor->sector()->floorz- 2;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool SpriteWarpToUnderwater(DSWActor* actor)
|
|
{
|
|
int i;
|
|
auto sectu = actor->sector();
|
|
bool Found = false;
|
|
DSWActor* overActor = nullptr;
|
|
DSWActor* underActor = nullptr;
|
|
|
|
// 0 not valid for water match tags
|
|
if (!sectu->hasU() || sectu->number == 0)
|
|
return false;
|
|
|
|
// search for DIVE_AREA "over" sprite for reference point
|
|
SWStatIterator it(STAT_DIVE_AREA);
|
|
while ((overActor = it.Next()))
|
|
{
|
|
if ((overActor->sector()->extra & SECTFX_DIVE_AREA) &&
|
|
overActor->sector()->hasU() &&
|
|
overActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
Found = false;
|
|
|
|
// search for UNDERWATER "under" sprite for reference point
|
|
it.Reset(STAT_UNDERWATER);
|
|
while ((underActor = it.Next()))
|
|
{
|
|
if ((underActor->sector()->extra & SECTFX_UNDERWATER) &&
|
|
underActor->sector()->hasU() &&
|
|
underActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(Found);
|
|
|
|
// update to the new x y position
|
|
actor->spr.pos.XY() += (underActor->spr.pos.XY() - overActor->spr.pos.XY());
|
|
|
|
auto over = overActor->sector();
|
|
auto under = underActor->sector();
|
|
|
|
if (GetOverlapSector(actor->spr.pos, &over, &under) == 2)
|
|
{
|
|
ChangeActorSect(actor, under);
|
|
}
|
|
else
|
|
{
|
|
ChangeActorSect(actor, over);
|
|
}
|
|
|
|
actor->spr.pos.Z = underActor->sector()->ceilingz + actor->user.ceiling_dist+ 1;
|
|
|
|
actor->backuppos();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SpriteWarpToSurface(DSWActor* actor)
|
|
{
|
|
auto sectu = actor->sector();
|
|
int sx, sy;
|
|
DSWActor* overActor = nullptr;
|
|
DSWActor* underActor = nullptr;
|
|
bool Found = false;
|
|
|
|
// 0 not valid for water match tags
|
|
if (!sectu->hasU() || sectu->number == 0)
|
|
return false;
|
|
|
|
// search for UNDERWATER "under" sprite for reference point
|
|
SWStatIterator it(STAT_UNDERWATER);
|
|
while ((underActor = it.Next()))
|
|
{
|
|
if ((underActor->sector()->extra & SECTFX_UNDERWATER) &&
|
|
underActor->sector()->hasU() &&
|
|
underActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!Found) return false;
|
|
|
|
if (underActor->spr.lotag == 0)
|
|
return false;
|
|
|
|
Found = false;
|
|
|
|
// search for DIVE_AREA "over" sprite for reference point
|
|
it.Reset(STAT_DIVE_AREA);
|
|
while ((overActor = it.Next()))
|
|
{
|
|
if ((overActor->sector()->extra & SECTFX_DIVE_AREA) &&
|
|
overActor->sector()->hasU() &&
|
|
overActor->sector()->number == sectu->number)
|
|
{
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!Found) return false;
|
|
|
|
// get the offset from the under sprite
|
|
// update to the new x y position
|
|
actor->spr.pos += overActor->spr.pos.XY() - underActor->spr.pos.XY();
|
|
|
|
auto over = overActor->sector();
|
|
auto under = underActor->sector();
|
|
|
|
if (GetOverlapSector(actor->spr.pos, &over, &under))
|
|
{
|
|
ChangeActorSect(actor, over);
|
|
}
|
|
|
|
actor->spr.pos.Z = overActor->sector()->floorz - 2;
|
|
|
|
// set z range and wade depth so we know how high to set view
|
|
DoActorZrange(actor);
|
|
MissileWaterAdjust(actor);
|
|
|
|
|
|
actor->backuppos();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
int SpawnSplash(DSWActor* actor)
|
|
{
|
|
auto sectu = actor->sector();
|
|
sectortype* sectp = actor->sector();
|
|
|
|
if (Prediction)
|
|
return 0;
|
|
|
|
if (sectu && ((sectp->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_NONE))
|
|
return 0;
|
|
|
|
if (sectu && (sectp->floorstat & CSTAT_SECTOR_SKY))
|
|
return 0;
|
|
|
|
PlaySound(DIGI_SPLASH1, actor, v3df_none);
|
|
|
|
DoActorZrange(actor);
|
|
MissileWaterAdjust(actor);
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, SPLASH, s_Splash, actor->sector(), DVector3(actor->spr.pos.XY(), actor->user.loz), actor->spr.angle, 0);
|
|
|
|
if (sectu && (sectp->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_RED_LIGHTING;
|
|
|
|
actorNew->spr.xrepeat = 45;
|
|
actorNew->spr.yrepeat = 42;
|
|
actorNew->spr.shade = actor->sector()->floorshade - 10;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int SpawnSplashXY(int hit_x, int hit_y, int hit_z, sectortype* sectp)
|
|
{
|
|
if (Prediction)
|
|
return 0;
|
|
|
|
if (sectp->hasU() && ((sectp->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_NONE))
|
|
return 0;
|
|
|
|
if (sectp->hasU() && (sectp->floorstat & CSTAT_SECTOR_SKY))
|
|
return 0;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, SPLASH, s_Splash, sectp, hit_x, hit_y, hit_z, 0, 0);
|
|
|
|
if (sectp->hasU() && (sectp->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
actorNew->user.spal = actorNew->spr.pal = PALETTE_RED_LIGHTING;
|
|
|
|
actorNew->spr.xrepeat = 45;
|
|
actorNew->spr.yrepeat = 42;
|
|
actorNew->spr.shade = actorNew->sector()->floorshade - 10;
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool MissileHitDiveArea(DSWActor* actor)
|
|
{
|
|
// correctly set underwater bit for missiles
|
|
// in Stacked water areas.
|
|
if (FAF_ConnectArea(actor->sector()))
|
|
{
|
|
if (SectorIsUnderwaterArea(actor->sector()))
|
|
actor->user.Flags |= (SPR_UNDERWATER);
|
|
else
|
|
actor->user.Flags &= ~(SPR_UNDERWATER);
|
|
}
|
|
|
|
if (actor->user.coll.type == kHitSector)
|
|
{
|
|
auto hit_sect = actor->user.coll.hitSector;
|
|
|
|
if (SpriteInDiveArea(actor))
|
|
{
|
|
// make sure you are close to the floor
|
|
if (actor->spr.pos.Z < (actor->user.hiz + actor->user.loz) * 0.5)
|
|
return false;
|
|
|
|
// Check added by Jim because of sprite bridge over water
|
|
if (actor->spr.pos.Z < hit_sect->floorz - 20)
|
|
return false;
|
|
|
|
actor->user.Flags |= (SPR_UNDERWATER);
|
|
SpawnSplash(actor);
|
|
SpriteWarpToUnderwater(actor);
|
|
actor->user.coll.setNone();
|
|
PlaySound(DIGI_PROJECTILEWATERHIT, actor, v3df_none);
|
|
return true;
|
|
}
|
|
else if (SpriteInUnderwaterArea(actor))
|
|
{
|
|
// make sure you are close to the ceiling
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
return false;
|
|
|
|
actor->user.Flags &= ~(SPR_UNDERWATER);
|
|
if (!SpriteWarpToSurface(actor))
|
|
{
|
|
return false;
|
|
}
|
|
SpawnSplash(actor);
|
|
actor->user.coll.setNone();
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
DSWActor* SpawnBubble(DSWActor* actor)
|
|
{
|
|
if (Prediction)
|
|
return nullptr;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, BUBBLE, s_Bubble, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
actorNew->spr.xrepeat = 8 + (RANDOM_P2(8 << 8) >> 8);
|
|
actorNew->spr.yrepeat = actorNew->spr.xrepeat;
|
|
// notreallypos
|
|
actorNew->user.pos.X = actorNew->spr.xrepeat;
|
|
actorNew->user.pos.Y = actorNew->spr.yrepeat;
|
|
actorNew->user.ceiling_dist = 1;
|
|
actorNew->user.floor_dist = 1;
|
|
actorNew->spr.shade = actor->sector()->floorshade - 10;
|
|
actorNew->user.WaitTics = 120 * 120;
|
|
actorNew->vel.Z = 2;
|
|
actorNew->spr.clipdist = 12 >> 2;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->user.Flags |= (SPR_UNDERWATER);
|
|
actorNew->spr.shade = -60; // Make em brighter
|
|
|
|
return actorNew;
|
|
}
|
|
|
|
int DoVehicleSmoke(DSWActor* actor)
|
|
{
|
|
actor->spr.pos.XY() += actor->user.change.XY();
|
|
actor->spr.pos.Z -= actor->vel.Z;
|
|
return false;
|
|
}
|
|
|
|
int DoWaterSmoke(DSWActor* actor)
|
|
{
|
|
actor->spr.pos.Z -= actor->vel.Z;
|
|
return false;
|
|
}
|
|
|
|
int SpawnVehicleSmoke(DSWActor* actor)
|
|
{
|
|
if (MoveSkip2 != 0)
|
|
return false;
|
|
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_VehicleSmoke, actor->sector(), actor->spr.pos.plusZ(-RANDOM_P2F(8, 8)), actor->spr.angle, 0);
|
|
|
|
actorNew->user.WaitTics = 1*120;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
if (RANDOM_P2(1024) < 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) < 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
actorNew->spr.angle = RandomAngle();
|
|
actorNew->set_int_xvel(RANDOM_P2(32));
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->set_int_zvel(Z(4) + RANDOM_P2(Z(4)));
|
|
|
|
return false;
|
|
}
|
|
|
|
int SpawnSmokePuff(DSWActor* actor)
|
|
{
|
|
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_WaterSmoke, actor->sector(), actor->spr.pos.plusZ(-RANDOM_P2F(8, 8)), actor->spr.angle, 0);
|
|
|
|
actorNew->user.WaitTics = 1*120;
|
|
actorNew->spr.shade = -40;
|
|
actorNew->spr.xrepeat = 64;
|
|
actorNew->spr.yrepeat = 64;
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
if (RANDOM_P2(1024) < 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
if (RANDOM_P2(1024) < 512)
|
|
actorNew->spr.cstat |= (CSTAT_SPRITE_YFLIP);
|
|
|
|
actorNew->spr.angle = RandomAngle();
|
|
actorNew->set_int_xvel(RANDOM_P2(32));
|
|
UpdateChangeXY(actorNew);
|
|
actorNew->set_int_zvel(Z(1) + RANDOM_P2(Z(2)));
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int DoBubble(DSWActor* actor)
|
|
{
|
|
actor->spr.pos.Z -= actor->vel.Z;
|
|
actor->add_int_zvel( 32);
|
|
|
|
if (actor->vel.Z > 3)
|
|
actor->set_int_zvel(768);
|
|
|
|
// notreallypos
|
|
actor->user.pos.X += 1;
|
|
actor->user.pos.Y += 1;
|
|
|
|
if (actor->user.pos.X > 32)
|
|
{
|
|
actor->user.pos.X = 32;
|
|
actor->user.pos.Y = 32;
|
|
}
|
|
|
|
actor->spr.xrepeat = int(actor->user.pos.X) + (RANDOM_P2(8 << 8) >> 8) - 4;
|
|
actor->spr.yrepeat = int(actor->user.pos.Y) + (RANDOM_P2(8 << 8) >> 8) - 4;
|
|
|
|
if (actor->spr.pos.Z < actor->sector()->ceilingz)
|
|
{
|
|
if (SectorIsUnderwaterArea(actor->user.hi_sectp))
|
|
{
|
|
if (!SpriteWarpToSurface(actor))
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
actor->user.Flags &= ~(SPR_UNDERWATER);
|
|
// stick around above water for this long
|
|
actor->user.WaitTics = (RANDOM_P2(64 << 8) >> 8);
|
|
}
|
|
else
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (!(actor->user.Flags & SPR_UNDERWATER))
|
|
{
|
|
if ((actor->user.WaitTics -= MISSILEMOVETICS) <= 0)
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
// just in case its stuck somewhere kill it after a while
|
|
{
|
|
if ((actor->user.WaitTics -= MISSILEMOVETICS) <= 0)
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// this needs to be called before killsprite
|
|
// whenever killing a sprite that you aren't completely sure what it is, like
|
|
// with the drivables, copy sectors, break sprites, etc
|
|
void SpriteQueueDelete(DSWActor* actor)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < MAX_STAR_QUEUE; i++)
|
|
if (StarQueue[i] == actor)
|
|
StarQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_HOLE_QUEUE; i++)
|
|
if (HoleQueue[i] == actor)
|
|
HoleQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_WALLBLOOD_QUEUE; i++)
|
|
if (WallBloodQueue[i] == actor)
|
|
WallBloodQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_FLOORBLOOD_QUEUE; i++)
|
|
if (FloorBloodQueue[i] == actor)
|
|
FloorBloodQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_GENERIC_QUEUE; i++)
|
|
if (GenericQueue[i] == actor)
|
|
GenericQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_LOWANGS_QUEUE; i++)
|
|
if (LoWangsQueue[i] == actor)
|
|
LoWangsQueue[i] = nullptr;
|
|
}
|
|
|
|
|
|
void QueueReset(void)
|
|
{
|
|
size_t i;
|
|
StarQueueHead=0;
|
|
HoleQueueHead=0;
|
|
WallBloodQueueHead=0;
|
|
FloorBloodQueueHead=0;
|
|
GenericQueueHead=0;
|
|
LoWangsQueueHead=0;
|
|
|
|
|
|
for (i = 0; i < MAX_STAR_QUEUE; i++)
|
|
StarQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_HOLE_QUEUE; i++)
|
|
HoleQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_WALLBLOOD_QUEUE; i++)
|
|
WallBloodQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_FLOORBLOOD_QUEUE; i++)
|
|
FloorBloodQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_GENERIC_QUEUE; i++)
|
|
GenericQueue[i] = nullptr;
|
|
|
|
for (i = 0; i < MAX_LOWANGS_QUEUE; i++)
|
|
LoWangsQueue[i] = nullptr;
|
|
}
|
|
|
|
bool TestDontStick(DSWActor* actor, walltype* hit_wall)
|
|
{
|
|
if (hit_wall == nullptr)
|
|
{
|
|
ASSERT(actor != nullptr);
|
|
if (actor->user.coll.type != kHitWall) return true; // ain't got a wall here.
|
|
hit_wall = actor->user.coll.hitWall;
|
|
}
|
|
|
|
|
|
if ((hit_wall->extra & WALLFX_DONT_STICK))
|
|
return true;
|
|
|
|
// if blocking red wallo
|
|
if ((hit_wall->cstat & CSTAT_WALL_BLOCK) && hit_wall->twoSided())
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
bool TestDontStickSector(sectortype* hit_sect)
|
|
{
|
|
if ((hit_sect->extra & (SECTFX_DYNAMIC_AREA|SECTFX_SECTOR_OBJECT)))
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
int QueueStar(DSWActor* actor)
|
|
{
|
|
if (TestDontStick(actor, nullptr))
|
|
{
|
|
KillActor(actor);
|
|
return -1;
|
|
}
|
|
|
|
// can and should kill the user portion of the star
|
|
if (StarQueue[StarQueueHead] == nullptr)
|
|
{
|
|
// new star
|
|
actor->clearUser();
|
|
change_actor_stat(actor, STAT_STAR_QUEUE);
|
|
StarQueue[StarQueueHead] = actor;
|
|
}
|
|
else
|
|
{
|
|
// move old star to new stars place
|
|
auto osp = StarQueue[StarQueueHead];
|
|
osp->spr.pos = actor->spr.pos;
|
|
ChangeActorSect(osp, actor->sector());
|
|
KillActor(actor);
|
|
actor = osp;
|
|
}
|
|
|
|
StarQueueHead = (StarQueueHead+1) & (MAX_STAR_QUEUE-1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void QueueHole(sectortype* hit_sect, walltype* hit_wall, const DVector3& pos)
|
|
{
|
|
short w,nw,wall_ang;
|
|
DSWActor* spawnedActor;
|
|
int nx,ny;
|
|
|
|
if (TestDontStick(nullptr, hit_wall))
|
|
return;
|
|
|
|
if (HoleQueue[HoleQueueHead] == nullptr)
|
|
HoleQueue[HoleQueueHead] = spawnedActor = insertActor(hit_sect, STAT_HOLE_QUEUE);
|
|
else
|
|
spawnedActor = HoleQueue[HoleQueueHead];
|
|
|
|
HoleQueueHead = (HoleQueueHead+1) & (MAX_HOLE_QUEUE-1);
|
|
|
|
spawnedActor->spr.xrepeat = spawnedActor->spr.yrepeat = 16;
|
|
spawnedActor->spr.cstat = 0;
|
|
spawnedActor->spr.pal = 0;
|
|
spawnedActor->spr.shade = 0;
|
|
spawnedActor->spr.extra = 0;
|
|
spawnedActor->spr.clipdist = 0;
|
|
spawnedActor->spr.xoffset = spawnedActor->spr.yoffset = 0;
|
|
spawnedActor->spr.pos = pos;
|
|
spawnedActor->spr.picnum = 2151;
|
|
ChangeActorSect(spawnedActor, hit_sect);
|
|
|
|
ASSERT(spawnedActor->spr.statnum != MAXSTATUS);
|
|
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ALIGNMENT_WALL);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ONE_SIDE);
|
|
spawnedActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
wall_ang = NORM_ANGLE(getangle(hit_wall->delta())+512);
|
|
spawnedActor->set_int_ang(wall_ang);
|
|
|
|
// move it back some
|
|
nx = bcos(spawnedActor->int_ang(), 4);
|
|
ny = bsin(spawnedActor->int_ang(), 4);
|
|
|
|
auto sect = spawnedActor->sector();
|
|
|
|
Collision coll;
|
|
clipmove(spawnedActor->spr.pos, §, nx, ny, 0, 0, 0, CLIPMASK_MISSILE, coll, 1);
|
|
|
|
if (spawnedActor->sector() != sect)
|
|
ChangeActorSect(spawnedActor, sect);
|
|
|
|
}
|
|
|
|
enum { FLOORBLOOD_RATE = 30 };
|
|
ANIMATOR DoFloorBlood;
|
|
STATE s_FloorBlood1[] =
|
|
{
|
|
{FLOORBLOOD1, SF_QUICK_CALL, DoFloorBlood, &s_FloorBlood1[1]},
|
|
{FLOORBLOOD1, FLOORBLOOD_RATE, NullAnimator, &s_FloorBlood1[0]},
|
|
};
|
|
|
|
int QueueFloorBlood(DSWActor* actor)
|
|
{
|
|
sectortype* sectp = actor->sector();
|
|
DSWActor* spawnedActor = nullptr;
|
|
|
|
|
|
if ((sectp->extra & SECTFX_SINK)||(sectp->extra & SECTFX_CURRENT))
|
|
return -1; // No blood in water or current areas
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) || SpriteInUnderwaterArea(actor) || SpriteInDiveArea(actor))
|
|
return -1; // No blood underwater!
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_WATER)
|
|
return -1; // No prints liquid areas!
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
return -1; // Not in lave either
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
return -1; // Not on special sectors you don't
|
|
|
|
if (FloorBloodQueue[FloorBloodQueueHead] != nullptr)
|
|
KillActor(FloorBloodQueue[FloorBloodQueueHead]);
|
|
|
|
FloorBloodQueue[FloorBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_SKIP4, FLOORBLOOD1, s_FloorBlood1, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
FloorBloodQueueHead = (FloorBloodQueueHead+1) & (MAX_FLOORBLOOD_QUEUE-1);
|
|
|
|
// Stupid hack to fix the blood under the skull to not show through
|
|
// x,y repeat of floor blood MUST be smaller than the sprite above it or clipping probs.
|
|
if (actor->user.ID == GORE_Head)
|
|
spawnedActor->spr.hitag = 9995;
|
|
else
|
|
spawnedActor->spr.hitag = 0;
|
|
spawnedActor->spr.xrepeat = spawnedActor->spr.yrepeat = 8;
|
|
spawnedActor->spr.cstat = 0;
|
|
spawnedActor->spr.pal = 0;
|
|
spawnedActor->spr.shade = 0;
|
|
spawnedActor->spr.extra = 0;
|
|
spawnedActor->spr.clipdist = 0;
|
|
spawnedActor->spr.xoffset = spawnedActor->spr.yoffset = 0;
|
|
spawnedActor->spr.pos = actor->spr.pos.plusZ(1);
|
|
spawnedActor->spr.angle = RandomAngle(); // Just make it any old angle
|
|
spawnedActor->spr.shade -= 5; // Brighten it up just a bit
|
|
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ALIGNMENT_FLOOR);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ONE_SIDE);
|
|
spawnedActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
actor->user.Flags &= ~(SPR_SHADOW);
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum
|
|
{
|
|
FOOTPRINT1 = 2490,
|
|
FOOTPRINT2 = 2491,
|
|
FOOTPRINT3 = 2492,
|
|
FOOTPRINT_RATE = 30,
|
|
};
|
|
ANIMATOR DoFootPrint;
|
|
STATE s_FootPrint1[] =
|
|
{
|
|
{FOOTPRINT1, FOOTPRINT_RATE, NullAnimator, &s_FootPrint1[0]},
|
|
};
|
|
STATE s_FootPrint2[] =
|
|
{
|
|
{FOOTPRINT2, FOOTPRINT_RATE, NullAnimator, &s_FootPrint2[0]},
|
|
};
|
|
STATE s_FootPrint3[] =
|
|
{
|
|
{FOOTPRINT3, FOOTPRINT_RATE, NullAnimator, &s_FootPrint3[0]},
|
|
};
|
|
|
|
int QueueFootPrint(DSWActor* actor)
|
|
{
|
|
DSWActor* spawnedActor;
|
|
short rnd_num=0;
|
|
bool Found=false;
|
|
sectortype* sectp = actor->sector();
|
|
|
|
|
|
if ((sectp->extra & SECTFX_SINK)||(sectp->extra & SECTFX_CURRENT))
|
|
return -1; // No blood in water or current areas
|
|
|
|
if (actor->user.PlayerP)
|
|
{
|
|
if ((actor->user.PlayerP->Flags & PF_DIVING))
|
|
Found = true;
|
|
|
|
// Stupid masked floor stuff! Damn your weirdness!
|
|
if ((actor->user.PlayerP->cursector->ceilingstat & CSTAT_SECTOR_SKY))
|
|
Found = true;
|
|
if ((actor->user.PlayerP->cursector->floorstat & CSTAT_SECTOR_SKY))
|
|
Found = true;
|
|
}
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) || SpriteInUnderwaterArea(actor) || Found || SpriteInDiveArea(actor))
|
|
return -1; // No prints underwater!
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_WATER)
|
|
return -1; // No prints liquid areas!
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
return -1; // Not in lave either
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
return -1; // Not on special sectors you don't
|
|
|
|
// So, are we like, done checking now!?
|
|
if (FloorBloodQueue[FloorBloodQueueHead] != nullptr)
|
|
KillActor(FloorBloodQueue[FloorBloodQueueHead]);
|
|
|
|
rnd_num = RandomRange(1024);
|
|
|
|
if (rnd_num > 683)
|
|
FloorBloodQueue[FloorBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, FOOTPRINT1, s_FootPrint1, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
else if (rnd_num > 342)
|
|
FloorBloodQueue[FloorBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, FOOTPRINT2, s_FootPrint2, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
else
|
|
FloorBloodQueue[FloorBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, FOOTPRINT3, s_FootPrint3, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
|
|
FloorBloodQueueHead = (FloorBloodQueueHead+1) & (MAX_FLOORBLOOD_QUEUE-1);
|
|
|
|
// Decrease footprint count
|
|
if (actor->user.PlayerP)
|
|
actor->user.PlayerP->NumFootPrints--;
|
|
|
|
|
|
spawnedActor->spr.hitag = 0;
|
|
spawnedActor->spr.xrepeat = 48;
|
|
spawnedActor->spr.yrepeat = 54;
|
|
spawnedActor->spr.cstat = 0;
|
|
spawnedActor->spr.pal = 0;
|
|
spawnedActor->spr.shade = 0;
|
|
spawnedActor->spr.extra = 0;
|
|
spawnedActor->spr.clipdist = 0;
|
|
spawnedActor->spr.xoffset = spawnedActor->spr.yoffset = 0;
|
|
spawnedActor->spr.pos = actor->spr.pos;
|
|
spawnedActor->spr.angle = actor->spr.angle;
|
|
spawnedActor->user.Flags &= ~(SPR_SHADOW);
|
|
switch (FootMode)
|
|
{
|
|
case BLOOD_FOOT:
|
|
spawnedActor->user.spal = spawnedActor->spr.pal = PALETTE_PLAYER3; // Turn blue to blood red
|
|
break;
|
|
default:
|
|
spawnedActor->user.spal = spawnedActor->spr.pal = PALETTE_PLAYER1; // Gray water
|
|
break;
|
|
}
|
|
|
|
|
|
// Alternate the feet
|
|
left_foot = !left_foot;
|
|
if (left_foot)
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_XFLIP);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ALIGNMENT_FLOOR);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ONE_SIDE);
|
|
spawnedActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum
|
|
{
|
|
WALLBLOOD1 = 2500,
|
|
WALLBLOOD2 = 2501,
|
|
WALLBLOOD3 = 2502,
|
|
WALLBLOOD4 = 2503,
|
|
WALLBLOOD_RATE = 30,
|
|
};
|
|
ANIMATOR DoWallBlood;
|
|
STATE s_WallBlood1[] =
|
|
{
|
|
{WALLBLOOD1, SF_QUICK_CALL, DoWallBlood, &s_WallBlood1[1]},
|
|
{WALLBLOOD1, WALLBLOOD_RATE, NullAnimator, &s_WallBlood1[0]},
|
|
};
|
|
STATE s_WallBlood2[] =
|
|
{
|
|
{WALLBLOOD2, SF_QUICK_CALL, DoWallBlood, &s_WallBlood2[1]},
|
|
{WALLBLOOD2, WALLBLOOD_RATE, NullAnimator, &s_WallBlood2[0]},
|
|
};
|
|
STATE s_WallBlood3[] =
|
|
{
|
|
{WALLBLOOD3, SF_QUICK_CALL, DoWallBlood, &s_WallBlood3[1]},
|
|
{WALLBLOOD3, WALLBLOOD_RATE, NullAnimator, &s_WallBlood3[0]},
|
|
};
|
|
STATE s_WallBlood4[] =
|
|
{
|
|
{WALLBLOOD4, SF_QUICK_CALL, DoWallBlood, &s_WallBlood4[1]},
|
|
{WALLBLOOD4, WALLBLOOD_RATE, NullAnimator, &s_WallBlood4[0]},
|
|
};
|
|
|
|
|
|
DSWActor* QueueWallBlood(DSWActor* actor, DAngle bang)
|
|
{
|
|
short w,nw,wall_ang,dang;
|
|
DSWActor* spawnedActor;
|
|
int nx,ny;
|
|
short rndnum;
|
|
int daz;
|
|
HitInfo hit{};
|
|
|
|
if (actor->user.Flags & (SPR_UNDERWATER) || SpriteInUnderwaterArea(actor) || SpriteInDiveArea(actor))
|
|
return nullptr; // No blood underwater!
|
|
|
|
daz = Z(RANDOM_P2(128))<<3;
|
|
daz -= (Z(128)<<2);
|
|
dang = (bang.Buildang() + (RANDOM_P2(128 << 5) >> 5)) - (64);
|
|
|
|
FAFhitscan(actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z - Z(30), actor->sector(), // Start position
|
|
bcos(dang), // X vector of 3D ang
|
|
bsin(dang), // Y vector of 3D ang
|
|
daz, // Z vector of 3D ang
|
|
hit, CLIPMASK_MISSILE);
|
|
|
|
if (hit.hitSector == nullptr)
|
|
return nullptr;
|
|
|
|
const double WALLBLOOD_DIST_MAX = 156.25;
|
|
if ((hit.hitpos.XY(), actor->spr.pos.XY()).Length() > WALLBLOOD_DIST_MAX)
|
|
return nullptr;
|
|
|
|
// hit a sprite?
|
|
if (hit.actor() != nullptr)
|
|
return nullptr; // Don't try to put blood on a sprite
|
|
|
|
if (hit.hitWall != nullptr) // Don't check if blood didn't hit a wall, otherwise the ASSERT fails!
|
|
{
|
|
if (TestDontStick(nullptr, hit.hitWall))
|
|
return nullptr;
|
|
}
|
|
else
|
|
return nullptr;
|
|
|
|
|
|
if (WallBloodQueue[WallBloodQueueHead] != nullptr)
|
|
KillActor(WallBloodQueue[WallBloodQueueHead]);
|
|
|
|
// Randomly choose a wall blood sprite
|
|
rndnum = RandomRange(1024);
|
|
if (rndnum > 768)
|
|
{
|
|
WallBloodQueue[WallBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, WALLBLOOD1, s_WallBlood1, hit.hitSector, hit.hitpos, bang, 0);
|
|
}
|
|
else if (rndnum > 512)
|
|
{
|
|
WallBloodQueue[WallBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, WALLBLOOD2, s_WallBlood2, hit.hitSector, hit.hitpos, bang, 0);
|
|
}
|
|
else if (rndnum > 128)
|
|
{
|
|
WallBloodQueue[WallBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, WALLBLOOD3, s_WallBlood3, hit.hitSector, hit.hitpos, bang, 0);
|
|
}
|
|
else
|
|
{
|
|
WallBloodQueue[WallBloodQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_WALLBLOOD_QUEUE, WALLBLOOD4, s_WallBlood4, hit.hitSector, hit.hitpos, bang, 0);
|
|
}
|
|
|
|
WallBloodQueueHead = (WallBloodQueueHead+1) & (MAX_WALLBLOOD_QUEUE-1);
|
|
|
|
spawnedActor->spr.xrepeat = 30;
|
|
spawnedActor->spr.yrepeat = 40; // yrepeat will grow towards 64, it's default size
|
|
spawnedActor->spr.cstat = 0;
|
|
spawnedActor->spr.pal = 0;
|
|
spawnedActor->spr.shade = 0;
|
|
spawnedActor->spr.extra = 0;
|
|
spawnedActor->spr.clipdist = 0;
|
|
spawnedActor->spr.xoffset = spawnedActor->spr.yoffset = 0;
|
|
spawnedActor->spr.pos = hit.hitpos;
|
|
spawnedActor->spr.shade -= 5; // Brighten it up just a bit
|
|
spawnedActor->tempwall = hit.hitWall; // pass hitinfo.wall
|
|
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ALIGNMENT_WALL);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_ONE_SIDE);
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_YCENTER);
|
|
spawnedActor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK|CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
wall_ang = NORM_ANGLE(getangle(hit.hitWall->delta()) + 512);
|
|
spawnedActor->set_int_ang(wall_ang);
|
|
|
|
// move it back some
|
|
nx = bcos(spawnedActor->int_ang(), 4);
|
|
ny = bsin(spawnedActor->int_ang(), 4);
|
|
|
|
auto sect = spawnedActor->sector();
|
|
|
|
Collision coll;
|
|
clipmove(spawnedActor->spr.pos, §, nx, ny, 0, 0, 0, CLIPMASK_MISSILE, coll, 1);
|
|
|
|
if (spawnedActor->sector() != sect)
|
|
ChangeActorSect(spawnedActor, sect);
|
|
|
|
return spawnedActor;
|
|
}
|
|
|
|
int DoFloorBlood(DSWActor* actor)
|
|
{
|
|
const int FEET_IN_BLOOD_DIST = 300;
|
|
|
|
int dist, near_dist = FEET_IN_BLOOD_DIST, a,b,c;
|
|
short pnum;
|
|
PLAYER* pp;
|
|
short xsiz,ysiz;
|
|
|
|
|
|
if (actor->spr.hitag == 9995)
|
|
{
|
|
xsiz = 12;
|
|
ysiz = 12;
|
|
}
|
|
else
|
|
{
|
|
xsiz = 40;
|
|
ysiz = 40;
|
|
}
|
|
|
|
// Make pool of blood seem to grow
|
|
if (actor->spr.xrepeat < xsiz && actor->spr.xrepeat != 4)
|
|
{
|
|
actor->spr.xrepeat++;
|
|
}
|
|
|
|
if (actor->spr.yrepeat < ysiz && actor->spr.xrepeat != xsiz && actor->spr.xrepeat != 4)
|
|
{
|
|
actor->spr.yrepeat++;
|
|
}
|
|
|
|
// See if any players stepped in blood
|
|
if (actor->spr.xrepeat != 4 && actor->spr.yrepeat > 4)
|
|
{
|
|
TRAVERSE_CONNECT(pnum)
|
|
{
|
|
pp = &Player[pnum];
|
|
|
|
DISTANCE(actor->spr.pos, pp->pos, dist, a, b, c);
|
|
|
|
if (dist < near_dist)
|
|
{
|
|
if (pp->NumFootPrints <= 0 || FootMode != BLOOD_FOOT)
|
|
{
|
|
pp->NumFootPrints = RandomRange(10)+3;
|
|
FootMode = BLOOD_FOOT;
|
|
}
|
|
|
|
// If blood has already grown to max size, we can shrink it
|
|
if (actor->spr.xrepeat == 40 && actor->spr.yrepeat > 10)
|
|
{
|
|
actor->spr.yrepeat -= 10;
|
|
if (actor->spr.yrepeat <= 10) // Shrink it down and don't use it anymore
|
|
actor->spr.xrepeat = actor->spr.yrepeat = 4;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int DoWallBlood(DSWActor* actor)
|
|
{
|
|
// Make blood drip down the wall
|
|
if (actor->spr.yrepeat < 80)
|
|
{
|
|
actor->spr.yrepeat++;
|
|
actor->spr.pos.Z += 0.5;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// This is the FAST queue, it doesn't call any animator functions or states
|
|
void QueueGeneric(DSWActor* actor, short pic)
|
|
{
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_WATER)
|
|
{
|
|
KillActor(actor);
|
|
return;
|
|
}
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
{
|
|
KillActor(actor);
|
|
return;
|
|
}
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
{
|
|
KillActor(actor);
|
|
return;
|
|
}
|
|
|
|
auto xrepeat = actor->spr.xrepeat;
|
|
auto yrepeat = actor->spr.yrepeat;
|
|
|
|
// can and should kill the user portion
|
|
if (GenericQueue[GenericQueueHead] == nullptr)
|
|
{
|
|
actor->clearUser();
|
|
change_actor_stat(actor, STAT_GENERIC_QUEUE);
|
|
GenericQueue[GenericQueueHead] = actor;
|
|
}
|
|
else
|
|
{
|
|
// move old sprite to new sprite's place
|
|
auto osp = GenericQueue[GenericQueueHead];
|
|
osp->spr.pos = actor->spr.pos;
|
|
ChangeActorSect(osp, actor->sector());
|
|
KillActor(actor);
|
|
actor = GenericQueue[GenericQueueHead];
|
|
ASSERT(actor->spr.statnum != MAXSTATUS);
|
|
}
|
|
|
|
actor->spr.picnum = pic;
|
|
actor->spr.xrepeat = xrepeat;
|
|
actor->spr.yrepeat = yrepeat;
|
|
actor->spr.cstat = 0;
|
|
switch (actor->spr.picnum)
|
|
{
|
|
case 900:
|
|
case 901:
|
|
case 902:
|
|
case 915:
|
|
case 916:
|
|
case 917:
|
|
case 930:
|
|
case 931:
|
|
case 932:
|
|
case GORE_Head:
|
|
change_actor_stat(actor,STAT_DEFAULT); // Breakable
|
|
actor->spr.cstat |= (CSTAT_SPRITE_BREAKABLE);
|
|
actor->spr.extra |= (SPRX_BREAKABLE);
|
|
break;
|
|
default:
|
|
actor->spr.cstat &= ~(CSTAT_SPRITE_BREAKABLE);
|
|
actor->spr.extra &= ~(SPRX_BREAKABLE);
|
|
break;
|
|
}
|
|
|
|
GenericQueueHead = (GenericQueueHead+1) & (MAX_GENERIC_QUEUE-1);
|
|
}
|
|
|
|
#if 0
|
|
int DoShellShrap(DSWActor* actor)
|
|
{
|
|
// If the shell doesn't fall in the allowable range, kill it.
|
|
if (actor->user.ShellNum < (ShellCount-MAXSHELLS))
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
// Get rid of shell if they fall in non-divable liquid areas
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_WATER)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
{
|
|
KillActor(actor);
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
int DoShrapVelocity(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER) || SpriteInUnderwaterArea(actor))
|
|
{
|
|
ScaleSpriteVector(actor, 20000);
|
|
|
|
actor->user.Counter += 8*4; // These are MoveSkip4 now
|
|
actor->user.addCounterToChange();
|
|
}
|
|
else
|
|
{
|
|
actor->user.Counter += 60*4;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS*2);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitVoid:
|
|
KillActor(actor);
|
|
return true;
|
|
case kHitSprite:
|
|
{
|
|
short wall_ang;
|
|
|
|
auto hit_sprite = actor->user.coll.actor();
|
|
|
|
WallBounce(actor, hit_sprite->spr.angle);
|
|
ScaleSpriteVector(actor, 32000);
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
WallBounce(actor, actor->user.coll.hitWall->delta().Angle() + DAngle90);
|
|
ScaleSpriteVector(actor, 32000);
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
bool did_hit_wall;
|
|
|
|
if (SlopeBounce(actor, &did_hit_wall))
|
|
{
|
|
if (did_hit_wall)
|
|
{
|
|
// hit a wall
|
|
ScaleSpriteVector(actor, 28000);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
}
|
|
else
|
|
{
|
|
// hit a sector
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
// hit a floor
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
ScaleSpriteVector(actor, 18000);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.ID == GORE_Drip)
|
|
ChangeState(actor, s_GoreFloorSplash);
|
|
else
|
|
ShrapKillSprite(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// hit a ceiling
|
|
ScaleSpriteVector(actor, 22000);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// hit floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
actor->spr.pos.Z = actor->user.loz;
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
actor->user.Flags |= (SPR_BOUNCE); // no bouncing underwater
|
|
|
|
if (actor->user.lo_sectp && actor->sector()->hasU() && FixedToInt(actor->sector()->depth_fixed))
|
|
actor->user.Flags |= (SPR_BOUNCE); // no bouncing on shallow water
|
|
|
|
if (!(actor->user.Flags & SPR_BOUNCE))
|
|
{
|
|
actor->user.Flags |= (SPR_BOUNCE);
|
|
actor->user.coll.setNone();
|
|
actor->user.Counter = 0;
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 18000);
|
|
switch (actor->user.ID)
|
|
{
|
|
case UZI_SHELL:
|
|
PlaySound(DIGI_SHELL, actor, v3df_none);
|
|
break;
|
|
case SHOT_SHELL:
|
|
PlaySound(DIGI_SHOTSHELLSPENT, actor, v3df_none);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (actor->user.ID == GORE_Drip)
|
|
ChangeState(actor, s_GoreFloorSplash);
|
|
else
|
|
ShrapKillSprite(actor);
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
// hit something above
|
|
{
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 22000);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// just outright kill it if its boucing around alot
|
|
if (actor->user.bounce > 10)
|
|
{
|
|
KillActor(actor);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int ShrapKillSprite(DSWActor* actor)
|
|
{
|
|
short rnd_num;
|
|
|
|
rnd_num = RandomRange(1024);
|
|
|
|
ASSERT(actor->hasU());
|
|
|
|
switch (actor->user.ID)
|
|
{
|
|
case UZI_SHELL:
|
|
if (rnd_num > 854)
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 0);
|
|
}
|
|
else if (rnd_num > 684)
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 1);
|
|
}
|
|
else if (rnd_num > 514)
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 2);
|
|
}
|
|
else if (rnd_num > 344)
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 3);
|
|
}
|
|
else if (rnd_num > 174)
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 4);
|
|
}
|
|
else
|
|
{
|
|
QueueGeneric(actor,UZI_SHELL + 5);
|
|
}
|
|
|
|
return 0;
|
|
break;
|
|
case SHOT_SHELL:
|
|
if (rnd_num > 683)
|
|
{
|
|
QueueGeneric(actor,SHOT_SHELL + 1);
|
|
}
|
|
else if (rnd_num > 342)
|
|
{
|
|
QueueGeneric(actor,SHOT_SHELL + 3);
|
|
}
|
|
else
|
|
{
|
|
QueueGeneric(actor,SHOT_SHELL + 7);
|
|
}
|
|
return 0;
|
|
break;
|
|
case GORE_Lung:
|
|
if (RandomRange(1000) > 500) break;
|
|
actor->spr.clipdist = ActorSizeX(actor);
|
|
SpawnFloorSplash(actor);
|
|
if (RandomRange(1000) < 500)
|
|
PlaySound(DIGI_GIBS1, actor, v3df_none);
|
|
else
|
|
PlaySound(DIGI_GIBS2, actor, v3df_none);
|
|
if (rnd_num > 683)
|
|
{
|
|
QueueGeneric(actor,900);
|
|
}
|
|
else if (rnd_num > 342)
|
|
{
|
|
QueueGeneric(actor,901);
|
|
}
|
|
else
|
|
{
|
|
QueueGeneric(actor,902);
|
|
}
|
|
return 0;
|
|
break;
|
|
case GORE_Liver:
|
|
if (RandomRange(1000) > 500) break;
|
|
actor->spr.clipdist = ActorSizeX(actor);
|
|
SpawnFloorSplash(actor);
|
|
if (RandomRange(1000) < 500)
|
|
PlaySound(DIGI_GIBS1, actor, v3df_none);
|
|
else
|
|
PlaySound(DIGI_GIBS2, actor, v3df_none);
|
|
if (rnd_num > 683)
|
|
{
|
|
QueueGeneric(actor,915);
|
|
}
|
|
else if (rnd_num > 342)
|
|
{
|
|
QueueGeneric(actor,916);
|
|
}
|
|
else
|
|
{
|
|
QueueGeneric(actor,917);
|
|
}
|
|
return 0;
|
|
break;
|
|
case GORE_SkullCap:
|
|
if (RandomRange(1000) > 500) break;
|
|
actor->spr.clipdist = ActorSizeX(actor);
|
|
SpawnFloorSplash(actor);
|
|
if (rnd_num > 683)
|
|
{
|
|
QueueGeneric(actor,930);
|
|
}
|
|
else if (rnd_num > 342)
|
|
{
|
|
QueueGeneric(actor,931);
|
|
}
|
|
else
|
|
{
|
|
QueueGeneric(actor,932);
|
|
}
|
|
return 0;
|
|
break;
|
|
case GORE_Head:
|
|
if (RandomRange(1000) > 500) break;
|
|
actor->spr.clipdist = ActorSizeX(actor);
|
|
QueueFloorBlood(actor);
|
|
QueueGeneric(actor,GORE_Head);
|
|
return 0;
|
|
break;
|
|
}
|
|
|
|
// If it wasn't in the switch statement, kill it.
|
|
KillActor(actor);
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool CheckBreakToughness(BREAK_INFO* break_info, int ID)
|
|
{
|
|
if ((break_info->flags & BF_TOUGH))
|
|
{
|
|
switch (ID)
|
|
{
|
|
case LAVA_BOULDER:
|
|
case MIRV_METEOR:
|
|
case SERP_METEOR:
|
|
case BOLT_THINMAN_R0:
|
|
case BOLT_THINMAN_R1:
|
|
case BOLT_THINMAN_R2:
|
|
case BOLT_EXP:
|
|
case TANK_SHELL_EXP:
|
|
case GRENADE_EXP:
|
|
case MICRO_EXP:
|
|
case MINE_EXP:
|
|
case NAP_EXP:
|
|
case SKULL_R0:
|
|
case BETTY_R0:
|
|
case SKULL_SERP:
|
|
case FIREBALL1:
|
|
case GORO_FIREBALL:
|
|
return true; // All the above stuff will break tough things
|
|
}
|
|
return false; // False means it won't break with current weapon
|
|
}
|
|
|
|
return true; // It wasn't tough, go ahead and break it
|
|
}
|
|
|
|
int DoItemFly(DSWActor* actor)
|
|
{
|
|
if (actor->user.Flags & (SPR_UNDERWATER))
|
|
{
|
|
ScaleSpriteVector(actor, 50000);
|
|
|
|
actor->user.Counter += 20*2;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
else
|
|
{
|
|
actor->user.Counter += 60*2;
|
|
actor->user.addCounterToChange();
|
|
}
|
|
|
|
actor->user.coll = move_missile(actor, actor->user.change, actor->user.ceiling_dist, actor->user.floor_dist, CLIPMASK_MISSILE, MISSILEMOVETICS*2);
|
|
|
|
MissileHitDiveArea(actor);
|
|
|
|
{
|
|
switch (actor->user.coll.type)
|
|
{
|
|
case kHitSprite:
|
|
{
|
|
short wall_ang;
|
|
auto hit_sprite = actor->user.coll.actor();
|
|
|
|
if ((hit_sprite->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
|
|
{
|
|
WallBounce(actor, hit_sprite->spr.angle);
|
|
ScaleSpriteVector(actor, 32000);
|
|
}
|
|
else
|
|
{
|
|
actor->user.change.XY() = -actor->user.change.XY();
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case kHitWall:
|
|
{
|
|
WallBounce(actor, actor->user.coll.hitWall->delta().Angle() + DAngle90);
|
|
ScaleSpriteVector(actor, 32000);
|
|
break;
|
|
}
|
|
|
|
case kHitSector:
|
|
{
|
|
// hit floor
|
|
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
|
|
{
|
|
actor->spr.pos.Z = actor->user.loz;
|
|
actor->user.Counter = 0;
|
|
actor->vel.X = 0;
|
|
actor->user.change.Zero();
|
|
return false;
|
|
}
|
|
else
|
|
// hit something above
|
|
{
|
|
actor->user.change.Z = -actor->user.change.Z;
|
|
ScaleSpriteVector(actor, 22000);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// This is the FAST queue, it doesn't call any animator functions or states
|
|
void QueueLoWangs(DSWActor* actor)
|
|
{
|
|
DSWActor* spawnedActor;
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_WATER)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ((actor->sector()->extra & SECTFX_LIQUID_MASK) == SECTFX_LIQUID_LAVA)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (TestDontStickSector(actor->sector()))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (LoWangsQueue[LoWangsQueueHead] == nullptr)
|
|
{
|
|
LoWangsQueue[LoWangsQueueHead] = spawnedActor =
|
|
SpawnActor(STAT_GENERIC_QUEUE, actor->spr.picnum, s_DeadLoWang, actor->sector(), actor->spr.pos, actor->spr.angle, 0);
|
|
}
|
|
else
|
|
{
|
|
// move old sprite to new sprite's place
|
|
SetActorZ(LoWangsQueue[LoWangsQueueHead], actor->spr.pos);
|
|
spawnedActor = LoWangsQueue[LoWangsQueueHead];
|
|
ASSERT(spawnedActor->spr.statnum != MAXSTATUS);
|
|
}
|
|
|
|
// Point passed in sprite to ps
|
|
spawnedActor->spr.cstat = 0;
|
|
spawnedActor->spr.xrepeat = actor->spr.xrepeat;
|
|
spawnedActor->spr.yrepeat = actor->spr.yrepeat;
|
|
spawnedActor->spr.shade = actor->spr.shade;
|
|
spawnedActor->user.spal = spawnedActor->spr.pal = actor->spr.pal;
|
|
change_actor_stat(spawnedActor, STAT_DEFAULT); // Breakable
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_BREAKABLE);
|
|
spawnedActor->spr.extra |= SPRX_BREAKABLE;
|
|
spawnedActor->spr.cstat |= (CSTAT_SPRITE_BLOCK_HITSCAN);
|
|
|
|
LoWangsQueueHead = (LoWangsQueueHead+1) & (MAX_LOWANGS_QUEUE-1);
|
|
}
|
|
|
|
|
|
#include "saveable.h"
|
|
|
|
static saveable_code saveable_weapon_code[] =
|
|
{
|
|
SAVE_CODE(SpawnShrapX),
|
|
SAVE_CODE(DoLavaErupt),
|
|
SAVE_CODE(DoVomit),
|
|
SAVE_CODE(DoVomitSplash),
|
|
SAVE_CODE(DoFastShrapJumpFall),
|
|
SAVE_CODE(DoTracerShrap),
|
|
SAVE_CODE(DoShrapJumpFall),
|
|
SAVE_CODE(DoShrapDamage),
|
|
SAVE_CODE(DoUziSmoke),
|
|
SAVE_CODE(DoShotgunSmoke),
|
|
SAVE_CODE(DoMineSpark),
|
|
SAVE_CODE(DoFireballFlames),
|
|
SAVE_CODE(DoBreakFlames),
|
|
SAVE_CODE(DoActorScale),
|
|
SAVE_CODE(DoRipperGrow),
|
|
SAVE_CODE(DoDamageTest),
|
|
SAVE_CODE(DoStar),
|
|
SAVE_CODE(DoCrossBolt),
|
|
SAVE_CODE(DoPlasmaDone),
|
|
SAVE_CODE(DoPlasmaFountain),
|
|
SAVE_CODE(DoPlasma),
|
|
SAVE_CODE(DoCoolgFire),
|
|
SAVE_CODE(DoEelFire),
|
|
SAVE_CODE(DoGrenade),
|
|
SAVE_CODE(DoVulcanBoulder),
|
|
SAVE_CODE(DoMineStuck),
|
|
SAVE_CODE(DoMine),
|
|
SAVE_CODE(DoPuff),
|
|
SAVE_CODE(DoRailPuff),
|
|
SAVE_CODE(DoBoltThinMan),
|
|
SAVE_CODE(DoTracer),
|
|
SAVE_CODE(DoEMP),
|
|
SAVE_CODE(DoEMPBurst),
|
|
SAVE_CODE(DoTankShell),
|
|
SAVE_CODE(DoTracerStart),
|
|
SAVE_CODE(DoLaser),
|
|
SAVE_CODE(DoLaserStart),
|
|
SAVE_CODE(DoRail),
|
|
SAVE_CODE(DoRailStart),
|
|
SAVE_CODE(DoRocket),
|
|
SAVE_CODE(DoMicroMini),
|
|
SAVE_CODE(SpawnExtraMicroMini),
|
|
SAVE_CODE(DoMicro),
|
|
SAVE_CODE(DoUziBullet),
|
|
SAVE_CODE(DoBoltSeeker),
|
|
SAVE_CODE(DoBoltShrapnel),
|
|
SAVE_CODE(DoBoltFatMan),
|
|
SAVE_CODE(DoElectro),
|
|
SAVE_CODE(DoLavaBoulder),
|
|
SAVE_CODE(DoSpear),
|
|
SAVE_CODE(SpawnGrenadeSmallExp),
|
|
SAVE_CODE(SpawnGrenadeExp),
|
|
SAVE_CODE(SpawnMineExp),
|
|
SAVE_CODE(DoSectorExp),
|
|
SAVE_CODE(SpawnSectorExp),
|
|
SAVE_CODE(SpawnLargeExp),
|
|
SAVE_CODE(SpawnMeteorExp),
|
|
SAVE_CODE(SpawnLittleExp),
|
|
SAVE_CODE(DoFireball),
|
|
SAVE_CODE(DoFindGround),
|
|
SAVE_CODE(DoFindGroundPoint),
|
|
SAVE_CODE(DoNapalm),
|
|
SAVE_CODE(DoBloodWorm),
|
|
SAVE_CODE(DoMeteor),
|
|
SAVE_CODE(DoSerpMeteor),
|
|
SAVE_CODE(DoMirvMissile),
|
|
SAVE_CODE(DoMirv),
|
|
SAVE_CODE(DoRing),
|
|
SAVE_CODE(DoSerpRing),
|
|
SAVE_CODE(InitLavaFlame),
|
|
SAVE_CODE(InitLavaThrow),
|
|
SAVE_CODE(InitVulcanBoulder),
|
|
SAVE_CODE(InitSerpRing),
|
|
SAVE_CODE(InitSerpRing),
|
|
SAVE_CODE(InitSpellNapalm),
|
|
SAVE_CODE(InitEnemyNapalm),
|
|
SAVE_CODE(InitSpellMirv),
|
|
SAVE_CODE(InitEnemyMirv),
|
|
SAVE_CODE(InitSwordAttack),
|
|
SAVE_CODE(InitFistAttack),
|
|
SAVE_CODE(InitSumoNapalm),
|
|
SAVE_CODE(InitSumoSkull),
|
|
SAVE_CODE(InitSumoStompAttack),
|
|
SAVE_CODE(InitMiniSumoClap),
|
|
SAVE_CODE(InitZillaRail),
|
|
SAVE_CODE(InitEnemyNuke),
|
|
SAVE_CODE(InitRipperSlash),
|
|
SAVE_CODE(InitBunnySlash),
|
|
SAVE_CODE(InitSerpSlash),
|
|
SAVE_CODE(InitCoolgBash),
|
|
SAVE_CODE(InitSkelSlash),
|
|
SAVE_CODE(InitGoroChop),
|
|
SAVE_CODE(InitHornetSting),
|
|
SAVE_CODE(InitSerpSpell),
|
|
SAVE_CODE(InitSerpMonstSpell),
|
|
SAVE_CODE(DoTeleRipper),
|
|
SAVE_CODE(InitEnemyRocket),
|
|
SAVE_CODE(InitEnemyRail),
|
|
SAVE_CODE(InitZillaRocket),
|
|
SAVE_CODE(InitEnemyStar),
|
|
SAVE_CODE(InitEnemyCrossbow),
|
|
SAVE_CODE(InitSkelSpell),
|
|
SAVE_CODE(InitCoolgFire),
|
|
SAVE_CODE(InitCoolgDrip),
|
|
SAVE_CODE(GenerateDrips),
|
|
SAVE_CODE(InitEelFire),
|
|
SAVE_CODE(InitFireballTrap),
|
|
SAVE_CODE(InitBoltTrap),
|
|
SAVE_CODE(InitEnemyCrossbow),
|
|
SAVE_CODE(InitSpearTrap),
|
|
SAVE_CODE(DoSuicide),
|
|
SAVE_CODE(DoDefaultStat),
|
|
SAVE_CODE(InitEnemyUzi),
|
|
SAVE_CODE(InitSpriteGrenade),
|
|
SAVE_CODE(InitEnemyMine),
|
|
SAVE_CODE(InitEnemyFireball),
|
|
SAVE_CODE(DoVehicleSmoke),
|
|
SAVE_CODE(DoWaterSmoke),
|
|
SAVE_CODE(SpawnVehicleSmoke),
|
|
SAVE_CODE(SpawnSmokePuff),
|
|
SAVE_CODE(DoBubble),
|
|
SAVE_CODE(DoFloorBlood),
|
|
SAVE_CODE(DoWallBlood),
|
|
SAVE_CODE(DoItemFly),
|
|
};
|
|
|
|
static saveable_data saveable_weapon_data[] =
|
|
{
|
|
SAVE_DATA(s_NotRestored),
|
|
SAVE_DATA(s_Suicide),
|
|
SAVE_DATA(s_DeadLoWang),
|
|
SAVE_DATA(s_BreakLight),
|
|
SAVE_DATA(s_BreakBarrel),
|
|
SAVE_DATA(s_BreakPedistal),
|
|
SAVE_DATA(s_BreakBottle1),
|
|
SAVE_DATA(s_BreakBottle2),
|
|
SAVE_DATA(s_Puff),
|
|
SAVE_DATA(s_RailPuff),
|
|
SAVE_DATA(sg_RailPuff),
|
|
SAVE_DATA(s_LaserPuff),
|
|
SAVE_DATA(s_Tracer),
|
|
SAVE_DATA(s_EMP),
|
|
SAVE_DATA(s_EMPBurst),
|
|
SAVE_DATA(s_EMPShrap),
|
|
SAVE_DATA(s_TankShell),
|
|
SAVE_DATA(s_VehicleSmoke),
|
|
SAVE_DATA(s_WaterSmoke),
|
|
SAVE_DATA(s_UziSmoke),
|
|
SAVE_DATA(s_ShotgunSmoke),
|
|
SAVE_DATA(s_UziBullet),
|
|
SAVE_DATA(s_UziSpark),
|
|
SAVE_DATA(s_UziPowerSpark),
|
|
SAVE_DATA(s_Bubble),
|
|
SAVE_DATA(s_Splash),
|
|
SAVE_DATA(s_CrossBolt),
|
|
SAVE_DATA(sg_CrossBolt),
|
|
SAVE_DATA(s_Star),
|
|
SAVE_DATA(s_StarStuck),
|
|
SAVE_DATA(s_StarDown),
|
|
SAVE_DATA(s_StarDownStuck),
|
|
SAVE_DATA(s_LavaBoulder),
|
|
SAVE_DATA(s_LavaShard),
|
|
SAVE_DATA(s_VulcanBoulder),
|
|
SAVE_DATA(s_Grenade),
|
|
SAVE_DATA(s_Grenade),
|
|
SAVE_DATA(sg_Grenade),
|
|
SAVE_DATA(s_MineStuck),
|
|
SAVE_DATA(s_Mine),
|
|
SAVE_DATA(s_MineSpark),
|
|
SAVE_DATA(s_Meteor),
|
|
SAVE_DATA(sg_Meteor),
|
|
SAVE_DATA(s_MeteorExp),
|
|
SAVE_DATA(s_MirvMeteor),
|
|
SAVE_DATA(sg_MirvMeteor),
|
|
SAVE_DATA(s_MirvMeteorExp),
|
|
SAVE_DATA(s_SerpMeteor),
|
|
SAVE_DATA(sg_SerpMeteor),
|
|
SAVE_DATA(s_SerpMeteorExp),
|
|
SAVE_DATA(s_Spear),
|
|
SAVE_DATA(sg_Spear),
|
|
SAVE_DATA(s_Rocket),
|
|
SAVE_DATA(sg_Rocket),
|
|
SAVE_DATA(s_BunnyRocket),
|
|
SAVE_DATA(sg_BunnyRocket),
|
|
SAVE_DATA(s_Rail),
|
|
SAVE_DATA(sg_Rail),
|
|
SAVE_DATA(s_Laser),
|
|
//SAVE_DATA(s_MicroPuff),
|
|
SAVE_DATA(s_Micro),
|
|
SAVE_DATA(sg_Micro),
|
|
SAVE_DATA(s_MicroMini),
|
|
SAVE_DATA(sg_MicroMini),
|
|
SAVE_DATA(s_BoltThinMan),
|
|
SAVE_DATA(sg_BoltThinMan),
|
|
SAVE_DATA(s_BoltSeeker),
|
|
SAVE_DATA(sg_BoltSeeker),
|
|
SAVE_DATA(s_BoltFatMan),
|
|
SAVE_DATA(s_BoltShrapnel),
|
|
SAVE_DATA(s_CoolgFire),
|
|
SAVE_DATA(s_CoolgFireDone),
|
|
SAVE_DATA(s_CoolgDrip),
|
|
SAVE_DATA(s_GoreFloorSplash),
|
|
SAVE_DATA(s_GoreSplash),
|
|
SAVE_DATA(s_Plasma),
|
|
SAVE_DATA(s_PlasmaFountain),
|
|
SAVE_DATA(s_PlasmaDrip),
|
|
SAVE_DATA(s_PlasmaDone),
|
|
SAVE_DATA(s_TeleportEffect),
|
|
SAVE_DATA(s_TeleportEffect2),
|
|
SAVE_DATA(s_Electro),
|
|
SAVE_DATA(s_ElectroShrap),
|
|
SAVE_DATA(s_GrenadeExp),
|
|
SAVE_DATA(s_GrenadeSmallExp),
|
|
SAVE_DATA(s_GrenadeExp),
|
|
SAVE_DATA(s_MineExp),
|
|
SAVE_DATA(s_BasicExp),
|
|
SAVE_DATA(s_MicroExp),
|
|
SAVE_DATA(s_BigGunFlame),
|
|
SAVE_DATA(s_BoltExp),
|
|
SAVE_DATA(s_TankShellExp),
|
|
SAVE_DATA(s_TracerExp),
|
|
SAVE_DATA(s_SectorExp),
|
|
SAVE_DATA(s_FireballExp),
|
|
SAVE_DATA(s_NapExp),
|
|
SAVE_DATA(s_FireballFlames),
|
|
SAVE_DATA(s_BreakFlames),
|
|
SAVE_DATA(s_Fireball),
|
|
SAVE_DATA(s_Fireball),
|
|
//SAVE_DATA(sg_Fireball),
|
|
SAVE_DATA(s_Ring),
|
|
//SAVE_DATA(sg_Ring),
|
|
SAVE_DATA(s_Ring),
|
|
SAVE_DATA(s_Ring2),
|
|
SAVE_DATA(s_Napalm),
|
|
SAVE_DATA(s_BloodWorm),
|
|
SAVE_DATA(s_BloodWorm),
|
|
SAVE_DATA(s_PlasmaExp),
|
|
SAVE_DATA(s_PlasmaExp),
|
|
SAVE_DATA(s_Mirv),
|
|
SAVE_DATA(s_MirvMissile),
|
|
SAVE_DATA(s_Vomit1),
|
|
SAVE_DATA(s_Vomit2),
|
|
SAVE_DATA(s_VomitSplash),
|
|
SAVE_DATA(s_GoreHead),
|
|
SAVE_DATA(s_GoreLeg),
|
|
SAVE_DATA(s_GoreEye),
|
|
SAVE_DATA(s_GoreTorso),
|
|
SAVE_DATA(s_GoreArm),
|
|
SAVE_DATA(s_GoreLung),
|
|
SAVE_DATA(s_GoreLiver),
|
|
SAVE_DATA(s_GoreSkullCap),
|
|
SAVE_DATA(s_GoreChunkS),
|
|
SAVE_DATA(s_GoreDrip),
|
|
SAVE_DATA(s_FastGoreDrip),
|
|
SAVE_DATA(s_GoreFlame),
|
|
SAVE_DATA(s_TracerShrap),
|
|
SAVE_DATA(s_UziShellShrap),
|
|
SAVE_DATA(s_UziShellShrapStill1),
|
|
SAVE_DATA(s_UziShellShrapStill2),
|
|
SAVE_DATA(s_UziShellShrapStill3),
|
|
SAVE_DATA(s_UziShellShrapStill4),
|
|
SAVE_DATA(s_UziShellShrapStill5),
|
|
SAVE_DATA(s_UziShellShrapStill6),
|
|
SAVE_DATA(s_ShotgunShellShrap),
|
|
SAVE_DATA(s_ShotgunShellShrapStill1),
|
|
SAVE_DATA(s_ShotgunShellShrapStill2),
|
|
SAVE_DATA(s_ShotgunShellShrapStill3),
|
|
SAVE_DATA(s_GoreFlameChunkA),
|
|
SAVE_DATA(s_GoreFlameChunkB),
|
|
SAVE_DATA(s_CoinShrap),
|
|
SAVE_DATA(s_Marbel),
|
|
SAVE_DATA(s_GlassShrapA),
|
|
SAVE_DATA(s_GlassShrapB),
|
|
SAVE_DATA(s_GlassShrapC),
|
|
SAVE_DATA(s_WoodShrapA),
|
|
SAVE_DATA(s_WoodShrapB),
|
|
SAVE_DATA(s_WoodShrapC),
|
|
SAVE_DATA(s_StoneShrapA),
|
|
SAVE_DATA(s_StoneShrapB),
|
|
SAVE_DATA(s_StoneShrapC),
|
|
SAVE_DATA(s_MetalShrapA),
|
|
SAVE_DATA(s_MetalShrapB),
|
|
SAVE_DATA(s_MetalShrapC),
|
|
SAVE_DATA(s_PaperShrapA),
|
|
SAVE_DATA(s_PaperShrapB),
|
|
SAVE_DATA(s_PaperShrapC),
|
|
SAVE_DATA(s_FloorBlood1),
|
|
SAVE_DATA(s_FootPrint1),
|
|
SAVE_DATA(s_FootPrint2),
|
|
SAVE_DATA(s_FootPrint3),
|
|
SAVE_DATA(s_WallBlood1),
|
|
SAVE_DATA(s_WallBlood2),
|
|
SAVE_DATA(s_WallBlood3),
|
|
SAVE_DATA(s_WallBlood4),
|
|
};
|
|
|
|
saveable_module saveable_weapon =
|
|
{
|
|
// code
|
|
saveable_weapon_code,
|
|
SIZ(saveable_weapon_code),
|
|
|
|
// data
|
|
saveable_weapon_data,
|
|
SIZ(saveable_weapon_data)
|
|
};
|
|
END_SW_NS
|