raze/source/games/sw/src/jweapon.cpp

2287 lines
73 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
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"
2020-08-05 22:18:45 +00:00
#include "misc.h"
#include "tags.h"
#include "break.h"
#include "network.h"
#include "pal.h"
#include "ai.h"
#include "weapon.h"
#include "sprite.h"
#include "sector.h"
BEGIN_SW_NS
ANIMATOR DoSuicide;
ANIMATOR DoBloodSpray;
2021-10-31 08:04:24 +00:00
void SpawnFlashBombOnActor(DSWActor* actor);
ANIMATOR DoPuff, BloodSprayFall;
extern STATE s_Puff[];
extern STATE s_FireballFlames[];
extern STATE s_GoreFloorSplash[];
extern STATE s_GoreSplash[];
2020-09-09 18:32:24 +00:00
extern bool GlobalSkipZrange;
#define CHEMTICS SEC(40)
#define GOREDrip 1562 //2430
#define BLOODSPRAY_RATE 20
STATE s_BloodSpray[] =
{
{GOREDrip + 0, BLOODSPRAY_RATE, BloodSprayFall, &s_BloodSpray[1]},
{GOREDrip + 1, BLOODSPRAY_RATE, BloodSprayFall, &s_BloodSpray[2]},
{GOREDrip + 2, BLOODSPRAY_RATE, BloodSprayFall, &s_BloodSpray[3]},
{GOREDrip + 3, BLOODSPRAY_RATE, BloodSprayFall, &s_BloodSpray[4]},
{GOREDrip + 3, 100, DoSuicide, &s_BloodSpray[0]}
};
#define EXP_RATE 2
STATE s_PhosphorExp[] =
{
{EXP + 0, EXP_RATE, NullAnimator, &s_PhosphorExp[1]},
{EXP + 1, EXP_RATE, NullAnimator, &s_PhosphorExp[2]},
{EXP + 2, EXP_RATE, NullAnimator, &s_PhosphorExp[3]},
{EXP + 3, EXP_RATE, NullAnimator, &s_PhosphorExp[4]},
{EXP + 4, EXP_RATE, NullAnimator, &s_PhosphorExp[5]},
{EXP + 5, EXP_RATE, NullAnimator, &s_PhosphorExp[6]},
{EXP + 6, EXP_RATE, NullAnimator, &s_PhosphorExp[7]},
{EXP + 7, EXP_RATE, NullAnimator, &s_PhosphorExp[8]},
{EXP + 8, EXP_RATE, NullAnimator, &s_PhosphorExp[9]},
{EXP + 9, EXP_RATE, NullAnimator, &s_PhosphorExp[10]},
{EXP + 10, EXP_RATE, NullAnimator, &s_PhosphorExp[11]},
{EXP + 11, EXP_RATE, NullAnimator, &s_PhosphorExp[12]},
{EXP + 12, EXP_RATE, NullAnimator, &s_PhosphorExp[13]},
{EXP + 13, EXP_RATE, NullAnimator, &s_PhosphorExp[14]},
{EXP + 14, EXP_RATE, NullAnimator, &s_PhosphorExp[15]},
{EXP + 15, EXP_RATE, NullAnimator, &s_PhosphorExp[16]},
{EXP + 16, EXP_RATE, NullAnimator, &s_PhosphorExp[17]},
{EXP + 17, EXP_RATE, NullAnimator, &s_PhosphorExp[18]},
{EXP + 18, EXP_RATE, NullAnimator, &s_PhosphorExp[19]},
{EXP + 19, EXP_RATE, NullAnimator, &s_PhosphorExp[20]},
{EXP + 20, 100, DoSuicide, &s_PhosphorExp[0]}
};
#define MUSHROOM_RATE 25
STATE s_NukeMushroom[] =
{
{MUSHROOM_CLOUD + 0, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[1]},
{MUSHROOM_CLOUD + 1, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[2]},
{MUSHROOM_CLOUD + 2, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[3]},
{MUSHROOM_CLOUD + 3, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[4]},
{MUSHROOM_CLOUD + 4, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[5]},
{MUSHROOM_CLOUD + 5, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[6]},
{MUSHROOM_CLOUD + 6, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[7]},
{MUSHROOM_CLOUD + 7, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[8]},
{MUSHROOM_CLOUD + 8, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[9]},
{MUSHROOM_CLOUD + 9, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[10]},
{MUSHROOM_CLOUD + 10, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[11]},
{MUSHROOM_CLOUD + 11, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[12]},
{MUSHROOM_CLOUD + 12, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[13]},
{MUSHROOM_CLOUD + 13, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[14]},
{MUSHROOM_CLOUD + 14, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[15]},
{MUSHROOM_CLOUD + 15, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[16]},
{MUSHROOM_CLOUD + 16, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[17]},
{MUSHROOM_CLOUD + 17, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[18]},
{MUSHROOM_CLOUD + 18, MUSHROOM_RATE, NullAnimator, &s_NukeMushroom[19]},
{MUSHROOM_CLOUD + 19, 100, DoSuicide, &s_NukeMushroom[0]},
};
ANIMATOR DoRadiationCloud;
#define RADIATION_RATE 16
STATE s_RadiationCloud[] =
{
{RADIATION_CLOUD + 0, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[1]},
{RADIATION_CLOUD + 1, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[2]},
{RADIATION_CLOUD + 2, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[3]},
{RADIATION_CLOUD + 3, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[4]},
{RADIATION_CLOUD + 4, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[5]},
{RADIATION_CLOUD + 5, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[6]},
{RADIATION_CLOUD + 6, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[7]},
{RADIATION_CLOUD + 7, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[8]},
{RADIATION_CLOUD + 8, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[9]},
{RADIATION_CLOUD + 9, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[10]},
{RADIATION_CLOUD + 10, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[11]},
{RADIATION_CLOUD + 11, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[12]},
{RADIATION_CLOUD + 12, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[13]},
{RADIATION_CLOUD + 13, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[14]},
{RADIATION_CLOUD + 14, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[15]},
{RADIATION_CLOUD + 15, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[16]},
{RADIATION_CLOUD + 16, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[17]},
{RADIATION_CLOUD + 17, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[18]},
{RADIATION_CLOUD + 18, RADIATION_RATE, DoRadiationCloud, &s_RadiationCloud[19]},
{RADIATION_CLOUD + 19, 100, DoSuicide, &s_RadiationCloud[0]},
};
#define CHEMBOMB_FRAMES 1
#define CHEMBOMB_R0 3038
#define CHEMBOMB_R1 CHEMBOMB_R0 + (CHEMBOMB_FRAMES * 1)
#define CHEMBOMB_R2 CHEMBOMB_R0 + (CHEMBOMB_FRAMES * 2)
#define CHEMBOMB_R3 CHEMBOMB_R0 + (CHEMBOMB_FRAMES * 3)
#define CHEMBOMB_R4 CHEMBOMB_R0 + (CHEMBOMB_FRAMES * 4)
#define CHEMBOMB CHEMBOMB_R0
#define CHEMBOMB_RATE 8
ANIMATOR DoChemBomb;
STATE s_ChemBomb[5] =
{
{CHEMBOMB_R0 + 0, CHEMBOMB_RATE, DoChemBomb, &s_ChemBomb[1]},
{CHEMBOMB_R1 + 0, CHEMBOMB_RATE, DoChemBomb, &s_ChemBomb[2]},
{CHEMBOMB_R2 + 0, CHEMBOMB_RATE, DoChemBomb, &s_ChemBomb[3]},
{CHEMBOMB_R3 + 0, CHEMBOMB_RATE, DoChemBomb, &s_ChemBomb[4]},
{CHEMBOMB_R4 + 0, CHEMBOMB_RATE, DoChemBomb, &s_ChemBomb[0]},
};
#define CALTROPS_FRAMES 1
#define CALTROPS_R0 CALTROPS-1
#define CALTROPS_RATE 8
ANIMATOR DoCaltrops, DoCaltropsStick;
STATE s_Caltrops[] =
{
{CALTROPS_R0 + 0, CALTROPS_RATE, DoCaltrops, &s_Caltrops[1]},
{CALTROPS_R0 + 1, CALTROPS_RATE, DoCaltrops, &s_Caltrops[2]},
{CALTROPS_R0 + 2, CALTROPS_RATE, DoCaltrops, &s_Caltrops[0]},
};
STATE s_CaltropsStick[] =
{
{CALTROPS_R0 + 2, CALTROPS_RATE, DoCaltropsStick, &s_CaltropsStick[0]},
};
//////////////////////
//
// CAPTURE FLAG
//
//////////////////////
ANIMATOR DoFlag, DoCarryFlag, DoCarryFlagNoDet;
#undef FLAG
#define FLAG 2520
#define FLAG_RATE 16
STATE s_CarryFlag[] =
{
{FLAG + 0, FLAG_RATE, DoCarryFlag, &s_CarryFlag[1]},
{FLAG + 1, FLAG_RATE, DoCarryFlag, &s_CarryFlag[2]},
{FLAG + 2, FLAG_RATE, DoCarryFlag, &s_CarryFlag[0]}
};
STATE s_CarryFlagNoDet[] =
{
{FLAG + 0, FLAG_RATE, DoCarryFlagNoDet, &s_CarryFlagNoDet[1]},
{FLAG + 1, FLAG_RATE, DoCarryFlagNoDet, &s_CarryFlagNoDet[2]},
{FLAG + 2, FLAG_RATE, DoCarryFlagNoDet, &s_CarryFlagNoDet[0]}
};
STATE s_Flag[] =
{
{FLAG + 0, FLAG_RATE, DoFlag, &s_Flag[1]},
{FLAG + 1, FLAG_RATE, DoFlag, &s_Flag[2]},
{FLAG + 2, FLAG_RATE, DoFlag, &s_Flag[0]}
};
#define PHOSPHORUS_RATE 8
ANIMATOR DoPhosphorus;
STATE s_Phosphorus[] =
{
{PHOSPHORUS + 0, PHOSPHORUS_RATE, DoPhosphorus, &s_Phosphorus[1]},
{PHOSPHORUS + 1, PHOSPHORUS_RATE, DoPhosphorus, &s_Phosphorus[0]},
};
ANIMATOR DoBloodSpray;
#define CHUNK1 1685
STATE s_BloodSprayChunk[] =
{
{CHUNK1 + 0, 8, DoBloodSpray, &s_BloodSprayChunk[1]},
{CHUNK1 + 1, 8, DoBloodSpray, &s_BloodSprayChunk[2]},
{CHUNK1 + 2, 8, DoBloodSpray, &s_BloodSprayChunk[3]},
{CHUNK1 + 3, 8, DoBloodSpray, &s_BloodSprayChunk[4]},
{CHUNK1 + 4, 8, DoBloodSpray, &s_BloodSprayChunk[5]},
{CHUNK1 + 5, 8, DoBloodSpray, &s_BloodSprayChunk[0]},
};
ANIMATOR DoWallBloodDrip;
#define DRIP 1566
STATE s_BloodSprayDrip[] =
{
{DRIP + 0, PHOSPHORUS_RATE, DoWallBloodDrip, &s_BloodSprayDrip[1]},
{DRIP + 1, PHOSPHORUS_RATE, DoWallBloodDrip, &s_BloodSprayDrip[2]},
{DRIP + 2, PHOSPHORUS_RATE, DoWallBloodDrip, &s_BloodSprayDrip[0]},
};
/////////////////////////////////////////////////////////////////////////////////////////////
2021-11-01 08:23:52 +00:00
int DoWallBloodDrip(DSWActor* actor)
{
// sy & sz are the ceiling and floor of the sector you are sliding down
if (actor->user.pos.Z != actor->user.pos.Y)
{
// if you are between the ceiling and floor fall fast
if (actor->spr.pos.Z > actor->user.pos.Y && actor->spr.pos.Z < actor->user.pos.Z)
{
2021-12-24 15:05:27 +00:00
actor->spr.zvel += 300;
2022-02-01 19:19:46 +00:00
actor->add_int_z(actor->spr.zvel);
}
else
{
2021-12-24 15:05:27 +00:00
actor->spr.zvel = (300+RandomRange(2300)) >> 1;
2022-02-01 19:19:46 +00:00
actor->add_int_z(actor->spr.zvel);
}
}
else
{
2021-12-24 15:05:27 +00:00
actor->spr.zvel = (300+RandomRange(2300)) >> 1;
2022-02-01 19:19:46 +00:00
actor->add_int_z(actor->spr.zvel);
}
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z >= actor->user.loz)
{
2022-08-20 20:49:30 +00:00
actor->spr.pos.Z = actor->user.loz;
2021-11-01 08:23:52 +00:00
SpawnFloorSplash(actor);
2021-10-30 20:53:24 +00:00
KillActor(actor);
return 0;
}
return 0;
}
2021-11-01 08:31:36 +00:00
void SpawnMidSplash(DSWActor* actor)
{
auto actorNew = SpawnActor(STAT_MISSILE, GOREDrip, s_GoreSplash, actor->sector(),
2022-10-05 15:57:38 +00:00
actor->int_pos().X, actor->int_pos().Y, int_ActorZOfMiddle(actor), actor->int_ang(), 0);
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -12;
actorNew->spr.xrepeat = 70-RandomRange(20);
actorNew->spr.yrepeat = 70-RandomRange(20);
2021-12-30 17:21:02 +00:00
actorNew->opos = actor->opos;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
if (RANDOM_P2(1024) < 512)
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
actorNew->user.change.X = 0;
actorNew->user.change.Y = 0;
actorNew->user.change.Z = 0;
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
actorNew->user.Flags |= (SPR_UNDERWATER);
}
2021-11-01 08:23:52 +00:00
void SpawnFloorSplash(DSWActor* actor)
{
auto actorNew = SpawnActor(STAT_MISSILE, GOREDrip, s_GoreFloorSplash, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), 0);
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -12;
actorNew->spr.xrepeat = 70-RandomRange(20);
actorNew->spr.yrepeat = 70-RandomRange(20);
2021-12-30 17:21:02 +00:00
actorNew->opos = actor->opos;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
if (RANDOM_P2(1024) < 512)
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
actorNew->user.change.X = 0;
actorNew->user.change.Y = 0;
actorNew->user.change.Z = 0;
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
actorNew->user.Flags |= (SPR_UNDERWATER);
}
2021-11-01 08:31:36 +00:00
int DoBloodSpray(DSWActor* actor)
{
int cz,fz;
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
{
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 50000);
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20; // These are STAT_SKIIP4 now, so * 2
actor->user.change.Z += actor->user.Counter;
}
else
{
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20;
actor->user.change.Z += actor->user.Counter;
}
2021-12-24 15:05:27 +00:00
if (actor->spr.xvel <= 2)
{
// special stuff for blood worm
2022-02-01 19:19:46 +00:00
actor->add_int_z((actor->user.change.Z >> 1));
getzsofslopeptr(actor->sector(), actor->int_pos().X, actor->int_pos().Y, &cz, &fz);
// pretend like we hit a sector
if (actor->int_pos().Z >= fz)
{
2022-02-01 19:04:41 +00:00
actor->set_int_z(fz);
2021-11-01 08:23:52 +00:00
SpawnFloorSplash(actor);
KillActor(actor);
return true;
}
}
else
{
actor->user.coll = move_missile(actor, actor->user.change.X, actor->user.change.Y, actor->user.change.Z,
2022-08-20 21:30:18 +00:00
actor->user.int_ceiling_dist(), actor->user.int_floor_dist(), CLIPMASK_MISSILE, MISSILEMOVETICS);
}
2021-11-03 16:52:16 +00:00
MissileHitDiveArea(actor);
{
2021-12-25 21:06:35 +00:00
switch (actor->user.coll.type)
{
2021-11-25 20:42:49 +00:00
case kHitVoid:
KillActor(actor);
return true;
case kHitSprite:
{
short wall_ang;
2021-12-25 21:06:35 +00:00
auto hitActor = actor->user.coll.actor();
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
{
2022-08-16 21:15:49 +00:00
wall_ang = NORM_ANGLE(hitActor->int_ang());
2021-11-01 08:23:52 +00:00
SpawnMidSplash(actor);
2022-08-16 21:15:49 +00:00
QueueWallBlood(actor, hitActor->int_ang());
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000);
}
else
{
actor->user.change.X = actor->user.change.Y = 0;
2021-11-01 08:23:52 +00:00
SpawnMidSplash(actor);
2022-08-16 21:15:49 +00:00
QueueWallBlood(actor, hitActor->int_ang());
KillActor(actor);
return true;
}
break;
}
case kHitWall:
{
short hit_wall, nw, wall_ang;
2021-12-31 14:41:39 +00:00
walltype* wph;
short wb;
2021-12-25 21:06:35 +00:00
wph = actor->user.coll.hitWall;
if (wph->lotag == TAG_WALL_BREAK)
{
2022-08-16 21:17:01 +00:00
HitBreakWall(wph, actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), actor->user.ID);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
break;
}
2021-11-24 15:55:24 +00:00
wall_ang = NORM_ANGLE(getangle(wph->delta()) + 512);
2021-11-01 08:23:52 +00:00
SpawnMidSplash(actor);
2021-11-02 17:51:14 +00:00
auto bldActor = QueueWallBlood(actor, NORM_ANGLE(wall_ang+1024));
2021-11-02 17:51:14 +00:00
if (bldActor== nullptr)
{
KillActor(actor);
return 0;
}
else
{
if (FAF_Sector(bldActor->sector()) || FAF_ConnectArea(bldActor->sector()))
{
KillActor(actor);
return 0;
}
actor->spr.xvel = actor->spr.yvel = actor->user.change.X = actor->user.change.Y = 0;
2021-12-24 15:05:27 +00:00
actor->spr.xrepeat = actor->spr.yrepeat = 70 - RandomRange(25);
2022-02-07 07:47:46 +00:00
actor->copyXY(bldActor);
// !FRANK! bit of a hack
// yvel is the hit_wall
2021-11-24 19:11:50 +00:00
if (bldActor->tempwall)
{
// sy & sz are the ceiling and floor of the sector you are sliding down
2021-11-24 19:11:50 +00:00
if (bldActor->tempwall->twoSided())
getzsofslopeptr(bldActor->tempwall->nextSector(), actor->spr.pos.X, actor->spr.pos.Y, &actor->user.pos.Y, &actor->user.pos.Z);
else
actor->user.pos.Y = actor->user.pos.Z; // ceiling and floor are equal - white wall
}
2021-12-27 18:07:39 +00:00
actor->spr.cstat &= ~(CSTAT_SPRITE_INVISIBLE);
2021-11-01 08:31:36 +00:00
ChangeState(actor, s_BloodSprayDrip);
}
break;
}
case kHitSector:
{
// hit floor
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
2021-12-27 18:34:06 +00:00
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 0
if (!(actor->user.Flags & SPR_BOUNCE))
{
2021-11-01 08:23:52 +00:00
SpawnFloorSplash(actor);
actor->user.Flags |= (SPR_BOUNCE);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
actor->user.zchange = -actor->user.zchange;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // Was 18000
2021-12-25 21:06:35 +00:00
actor->user.zchange /= 6;
}
else
#endif
{
actor->user.change.X = actor->user.change.Y = 0;
2021-11-01 08:23:52 +00:00
SpawnFloorSplash(actor);
KillActor(actor);
return true;
}
}
else
// hit something above
{
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 22000
}
break;
}
}
}
// 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, GOREDrip, s_BloodSpray, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), 100);
2021-11-01 08:31:36 +00:00
SetOwner(actor, actorNew);
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -12;
actorNew->spr.xrepeat = 40-RandomRange(30);
actorNew->spr.yrepeat = 40-RandomRange(30);
2021-12-30 17:21:02 +00:00
actorNew->opos = actor->opos;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
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->user.change.X = actor->user.change.X;
actorNew->user.change.Y = actor->user.change.Y;
actorNew->user.change.Z = actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actorNew, 20000);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
actorNew->user.Flags |= (SPR_UNDERWATER);
}
return false;
}
2021-11-01 08:39:44 +00:00
int DoPhosphorus(DSWActor* actor)
{
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
{
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 50000);
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20*2;
actor->user.change.Z += actor->user.Counter;
}
else
{
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20*2;
actor->user.change.Z += actor->user.Counter;
}
actor->user.coll = move_missile(actor, actor->user.change.X, actor->user.change.Y, actor->user.change.Z,
2022-08-20 21:30:18 +00:00
actor->user.int_ceiling_dist(), actor->user.int_floor_dist(), CLIPMASK_MISSILE, MISSILEMOVETICS*2);
2021-11-03 16:52:16 +00:00
MissileHitDiveArea(actor);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
2021-11-01 20:58:16 +00:00
SpawnBubble(actor);
{
2021-12-25 21:06:35 +00:00
switch (actor->user.coll.type)
{
2021-11-25 20:42:49 +00:00
case kHitVoid:
KillActor(actor);
return true;
case kHitSprite:
{
short wall_ang;
2021-12-25 21:06:35 +00:00
auto hitActor = actor->user.coll.actor();
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
{
2022-08-16 21:15:49 +00:00
wall_ang = NORM_ANGLE(hitActor->int_ang());
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000);
}
else
{
if ((hitActor->spr.extra & SPRX_BURNABLE))
{
2021-12-25 19:26:31 +00:00
if (!hitActor->hasU())
SpawnUser(hitActor, hitActor->spr.picnum, nullptr);
SpawnFireballExp(actor);
2021-12-25 19:26:31 +00:00
if (hitActor->hasU())
SpawnFireballFlames(actor, hitActor);
DoFlamesDamageTest(actor);
}
actor->user.change.X = actor->user.change.Y = 0;
KillActor(actor);
return true;
}
break;
}
case kHitWall:
{
short hit_wall, nw, wall_ang;
2021-12-31 14:41:39 +00:00
walltype* wph;
2021-12-25 21:06:35 +00:00
wph = actor->user.coll.hitWall;
if (wph->lotag == TAG_WALL_BREAK)
{
2022-08-16 21:17:01 +00:00
HitBreakWall(wph, actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), actor->user.ID);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
break;
}
2021-11-24 15:55:24 +00:00
wall_ang = NORM_ANGLE(getangle(wph->delta()) + 512);
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000);
break;
}
case kHitSector:
{
2020-09-09 18:32:24 +00:00
bool did_hit_wall;
2021-11-03 21:02:01 +00:00
if (SlopeBounce(actor, &did_hit_wall))
{
if (did_hit_wall)
{
// hit a wall
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 28000);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
// hit a sector
2022-08-20 20:49:30 +00:00
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);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 18000
actor->user.change.Z /= 6;
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
actor->user.change.X = actor->user.change.Y = 0;
SpawnFireballExp(actor);
KillActor(actor);
return true;
}
}
else
{
// hit a ceiling
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 22000
}
}
}
else
{
// hit floor
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
2021-12-27 18:34:06 +00:00
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);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // Was 18000
actor->user.change.Z /= 6;
}
else
{
actor->user.change.X = actor->user.change.Y = 0;
SpawnFireballExp(actor);
KillActor(actor);
return true;
}
}
else
// hit something above
{
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 22000
}
}
break;
}
}
}
// if you haven't bounced or your going slow do some puffs
if (!(actor->user.Flags & (SPR_BOUNCE | SPR_UNDERWATER)) && !(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
{
auto actorNew = SpawnActor(STAT_SKIP4, PUFF, s_PhosphorExp, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), 100);
2021-12-24 18:41:29 +00:00
actorNew->spr.hitag = LUMINOUS; // Always full brightness
2021-11-01 08:39:44 +00:00
SetOwner(actor, actorNew);
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -40;
actorNew->spr.xrepeat = 12 + RandomRange(10);
actorNew->spr.yrepeat = 12 + RandomRange(10);
2021-12-30 17:21:02 +00:00
actorNew->opos = actor->opos;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
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->user.change.X = actor->user.change.X;
actorNew->user.change.Y = actor->user.change.Y;
actorNew->user.change.Z = actor->user.change.Z;
2021-12-25 18:57:14 +00:00
actorNew->user.spal = actorNew->spr.pal = PALETTE_PLAYER3; // RED
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actorNew, 20000);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
actorNew->user.Flags |= (SPR_UNDERWATER);
}
return false;
}
2021-11-01 08:49:37 +00:00
int DoChemBomb(DSWActor* actor)
{
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
{
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 50000);
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20;
actor->user.change.Z += actor->user.Counter;
}
else
{
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20;
actor->user.change.Z += actor->user.Counter;
}
actor->user.coll = move_missile(actor, actor->user.change.X, actor->user.change.Y, actor->user.change.Z,
2022-08-20 21:30:18 +00:00
actor->user.int_ceiling_dist(), actor->user.int_floor_dist(), CLIPMASK_MISSILE, MISSILEMOVETICS);
2021-11-03 16:52:16 +00:00
MissileHitDiveArea(actor);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER) && (RANDOM_P2(1024 << 4) >> 4) < 256)
2021-11-01 20:58:16 +00:00
SpawnBubble(actor);
{
2021-12-25 21:06:35 +00:00
switch (actor->user.coll.type)
{
2021-11-25 20:42:49 +00:00
case kHitVoid:
KillActor(actor);
return true;
case kHitSprite:
{
short wall_ang;
if (!(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
PlaySound(DIGI_CHEMBOUNCE, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
auto hitActor = actor->user.coll.actor();
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
{
2022-08-16 21:15:49 +00:00
wall_ang = NORM_ANGLE(hitActor->int_ang());
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000);
}
else
{
// Canister pops when first smoke starts out
if (actor->user.WaitTics == CHEMTICS && !(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
{
PlaySound(DIGI_GASPOP, actor, v3df_dontpan | v3df_doppler);
PlaySound(DIGI_CHEMGAS, actor, v3df_dontpan | v3df_doppler);
}
actor->user.change.X = actor->user.change.Y = 0;
2021-12-25 21:06:35 +00:00
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
if (actor->user.WaitTics <= 0)
KillActor(actor);
return true;
}
break;
}
case kHitWall:
{
2021-12-25 21:06:35 +00:00
auto wph = actor->user.coll.hitWall;
if (wph->lotag == TAG_WALL_BREAK)
{
2022-08-16 21:17:01 +00:00
HitBreakWall(wph, actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), actor->user.ID);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
break;
}
if (!(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
PlaySound(DIGI_CHEMBOUNCE, actor, v3df_dontpan);
2021-11-24 15:55:24 +00:00
int wall_ang = NORM_ANGLE(getangle(wph->delta()) + 512);
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000);
break;
}
case kHitSector:
{
2020-09-09 18:32:24 +00:00
bool did_hit_wall;
2021-11-03 21:02:01 +00:00
if (SlopeBounce(actor, &did_hit_wall))
{
if (did_hit_wall)
{
// hit a wall
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 28000);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
// hit a sector
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
// hit a floor
if (!(actor->user.Flags & SPR_BOUNCE))
{
if (!(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
PlaySound(DIGI_CHEMBOUNCE, actor, v3df_dontpan);
actor->user.Flags |= (SPR_BOUNCE);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 18000
actor->user.change.Z /= 6;
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
// Canister pops when first smoke starts out
if (actor->user.WaitTics == CHEMTICS && !(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
{
PlaySound(DIGI_GASPOP, actor, v3df_dontpan | v3df_doppler);
PlaySound(DIGI_CHEMGAS, actor, v3df_dontpan | v3df_doppler);
}
2021-11-01 09:51:25 +00:00
SpawnRadiationCloud(actor);
actor->user.change.X = actor->user.change.Y = 0;
2021-12-25 21:06:35 +00:00
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
if (actor->user.WaitTics <= 0)
KillActor(actor);
return true;
}
}
else
{
// hit a ceiling
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 22000
}
}
}
else
{
// hit floor
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
2021-12-27 18:34:06 +00:00
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))
{
if (!(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
PlaySound(DIGI_CHEMBOUNCE, actor, v3df_dontpan);
actor->user.Flags |= (SPR_BOUNCE);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // Was 18000
actor->user.change.Z /= 6;
}
else
{
// Canister pops when first smoke starts out
if (actor->user.WaitTics == CHEMTICS && !(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
{
PlaySound(DIGI_GASPOP, actor, v3df_dontpan | v3df_doppler);
PlaySound(DIGI_CHEMGAS, actor, v3df_dontpan | v3df_doppler);
}
2021-11-01 09:51:25 +00:00
SpawnRadiationCloud(actor);
actor->user.change.X = actor->user.change.Y = 0;
2021-12-25 21:06:35 +00:00
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
if (actor->user.WaitTics <= 0)
KillActor(actor);
return true;
}
}
else
// hit something above
{
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 32000); // was 22000
}
}
break;
}
}
}
// if you haven't bounced or your going slow do some puffs
if (!(actor->user.Flags & (SPR_BOUNCE | SPR_UNDERWATER)) && !(actor->spr.cstat & CSTAT_SPRITE_INVISIBLE))
{
auto actorNew = SpawnActor(STAT_MISSILE, PUFF, s_Puff, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), 100);
2021-11-01 08:49:37 +00:00
SetOwner(actor, actorNew);
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -40;
actorNew->spr.xrepeat = 40;
actorNew->spr.yrepeat = 40;
2021-12-30 17:21:02 +00:00
actorNew->opos = actor->opos;
// !Frank - dont do translucent
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
// actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER|CSTAT_SPRITE_TRANSLUCENT);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
actorNew->user.change.X = actor->user.change.X;
actorNew->user.change.Y = actor->user.change.Y;
actorNew->user.change.Z = actor->user.change.Z;
2021-12-25 18:57:14 +00:00
actorNew->user.spal = actorNew->spr.pal = PALETTE_PLAYER6;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actorNew, 20000);
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
actorNew->user.Flags |= (SPR_UNDERWATER);
}
return false;
}
2021-11-01 08:49:37 +00:00
int DoCaltropsStick(DSWActor* actor)
{
2021-12-25 21:06:35 +00:00
actor->user.Counter = !actor->user.Counter;
2021-12-25 21:06:35 +00:00
if (actor->user.Counter)
DoFlamesDamageTest(actor);
return 0;
}
2021-11-01 09:36:41 +00:00
int DoCaltrops(DSWActor* actor)
{
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
{
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 50000);
2021-12-25 21:06:35 +00:00
actor->user.Counter += 20;
actor->user.change.Z += actor->user.Counter;
}
else
{
2021-12-25 21:06:35 +00:00
actor->user.Counter += 70;
actor->user.change.Z += actor->user.Counter;
}
actor->user.coll = move_missile(actor, actor->user.change.X, actor->user.change.Y, actor->user.change.Z,
2022-08-20 21:30:18 +00:00
actor->user.int_ceiling_dist(), actor->user.int_floor_dist(), CLIPMASK_MISSILE, MISSILEMOVETICS);
2021-11-03 16:52:16 +00:00
MissileHitDiveArea(actor);
{
2021-12-25 21:06:35 +00:00
switch (actor->user.coll.type)
{
2021-11-25 20:42:49 +00:00
case kHitVoid:
KillActor(actor);
return true;
case kHitSprite:
{
short wall_ang;
PlaySound(DIGI_CALTROPS, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
auto hitActor = actor->user.coll.actor();
if ((hitActor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_WALL))
{
2022-08-16 21:15:49 +00:00
wall_ang = NORM_ANGLE(hitActor->int_ang());
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 10000);
}
else
{
// fall to the ground
actor->user.change.X = actor->user.change.Y = 0;
}
break;
}
case kHitWall:
{
2021-12-25 21:06:35 +00:00
auto wph = actor->user.coll.hitWall;
if (wph->lotag == TAG_WALL_BREAK)
{
2022-08-16 21:17:01 +00:00
HitBreakWall(wph, actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), actor->user.ID);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
break;
}
PlaySound(DIGI_CALTROPS, actor, v3df_dontpan);
2021-11-24 15:55:24 +00:00
int wall_ang = NORM_ANGLE(getangle(wph->delta()) + 512);
2021-11-03 16:55:17 +00:00
WallBounce(actor, wall_ang);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000);
break;
}
case kHitSector:
{
2020-09-09 18:32:24 +00:00
bool did_hit_wall;
2021-11-03 21:02:01 +00:00
if (SlopeBounce(actor, &did_hit_wall))
{
if (did_hit_wall)
{
// hit a wall
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
// hit a sector
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
// hit a floor
if (!(actor->user.Flags & SPR_BOUNCE))
{
PlaySound(DIGI_CALTROPS, actor, v3df_dontpan);
actor->user.Flags |= (SPR_BOUNCE);
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000); // was 18000
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
}
else
{
actor->user.change.X = actor->user.change.Y = 0;
actor->spr.extra |= (SPRX_BREAKABLE);
actor->spr.cstat |= (CSTAT_SPRITE_BREAKABLE);
2021-11-01 09:36:41 +00:00
ChangeState(actor, s_CaltropsStick);
return true;
}
}
else
{
// hit a ceiling
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000); // was 22000
}
}
}
else
{
// hit floor
2022-08-20 20:49:30 +00:00
if (actor->spr.pos.Z > ((actor->user.hiz + actor->user.loz) * 0.5))
{
2021-12-27 18:34:06 +00:00
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))
{
PlaySound(DIGI_CALTROPS, actor, v3df_dontpan);
actor->user.Flags |= (SPR_BOUNCE);
2021-12-25 21:06:35 +00:00
actor->user.coll.setNone();
actor->user.Counter = 0;
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000); // Was 18000
}
else
{
actor->user.change.X = actor->user.change.Y = 0;
actor->spr.extra |= (SPRX_BREAKABLE);
actor->spr.cstat |= (CSTAT_SPRITE_BREAKABLE);
2021-11-01 09:36:41 +00:00
ChangeState(actor, s_CaltropsStick);
return true;
}
}
else
// hit something above
{
actor->user.change.Z = -actor->user.change.Z;
2021-11-01 12:08:41 +00:00
ScaleSpriteVector(actor, 1000); // was 22000
}
}
break;
}
}
}
return false;
}
/////////////////////////////
//
// Deadly green gas clouds
//
/////////////////////////////
2021-11-01 09:51:25 +00:00
int SpawnRadiationCloud(DSWActor* actor)
{
if (!MoveSkip4)
return false;
// This basically works like a MoveSkip8, if one existed
2021-12-25 21:06:35 +00:00
// actor->user.Counter2 = !actor->user.Counter2;
if (actor->user.ID == MUSHROOM_CLOUD || actor->user.ID == 3121)
{
2021-12-25 21:06:35 +00:00
if ((actor->user.Counter2++) > 16)
actor->user.Counter2 = 0;
if (actor->user.Counter2)
return false;
}
else
{
2021-12-25 21:06:35 +00:00
if ((actor->user.Counter2++) > 2)
actor->user.Counter2 = 0;
if (actor->user.Counter2)
return false;
}
2021-12-27 18:34:06 +00:00
if (actor->user.Flags & (SPR_UNDERWATER))
return -1;
auto actorNew = SpawnActor(STAT_MISSILE, RADIATION_CLOUD, s_RadiationCloud, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z - RANDOM_P2(Z(8)), actor->int_ang(), 0);
2021-11-01 09:51:25 +00:00
SetOwner(GetOwner(actor), actorNew);
2021-12-25 18:57:14 +00:00
actorNew->user.WaitTics = 1 * 120;
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -40;
actorNew->spr.xrepeat = 32;
actorNew->spr.yrepeat = 32;
actorNew->spr.clipdist = actor->spr.clipdist;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
2021-12-25 18:57:14 +00:00
actorNew->user.spal = actorNew->spr.pal = PALETTE_PLAYER6;
// Won't take floor palettes
2021-12-24 18:41:29 +00:00
actorNew->spr.hitag = SECTFU_DONT_COPY_PALETTE;
if (RANDOM_P2(1024) < 512)
actorNew->spr.cstat |= (CSTAT_SPRITE_XFLIP);
//if (RANDOM_P2(1024) < 512)
//actorNew->spr.cstat |= (CSTAT_SPRITE_YFLIP);
2022-08-21 17:08:22 +00:00
actorNew->spr.angle = RANDOM_ANGLE();
2021-12-24 18:41:29 +00:00
actorNew->spr.xvel = RANDOM_P2(32);
2021-12-25 18:57:14 +00:00
actorNew->user.Counter = 0;
actorNew->user.Counter2 = 0;
2021-12-25 21:06:35 +00:00
if (actor->user.ID == MUSHROOM_CLOUD || actor->user.ID == 3121)
{
2021-12-25 18:57:14 +00:00
actorNew->user.Radius = 2000;
2022-08-16 21:17:01 +00:00
actorNew->user.change.X = (MOVEx(actorNew->spr.xvel>>2, actorNew->int_ang()));
actorNew->user.change.Y = (MOVEy(actorNew->spr.xvel>>2, actorNew->int_ang()));
2021-12-24 18:41:29 +00:00
actorNew->spr.zvel = Z(1) + RANDOM_P2(Z(2));
}
else
{
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
2021-12-24 18:41:29 +00:00
actorNew->spr.zvel = Z(4) + RANDOM_P2(Z(4));
2021-12-25 18:57:14 +00:00
actorNew->user.Radius = 4000;
}
return false;
}
2021-11-01 09:51:25 +00:00
int DoRadiationCloud(DSWActor* actor)
{
2022-02-01 19:04:41 +00:00
actor->add_int_pos({ actor->user.change.X, actor->user.change.Y, -actor->spr.zvel });
2021-12-25 21:06:35 +00:00
if (actor->user.ID)
{
DoFlamesDamageTest(actor);
}
return false;
}
//////////////////////////////////////////////
//
// Inventory Chemical Bombs
//
//////////////////////////////////////////////
2021-12-31 14:59:11 +00:00
int PlayerInitChemBomb(PLAYER* pp)
{
DSWActor* plActor = pp->actor;
int nx, ny, nz;
short oclipdist;
PlaySound(DIGI_THROW, pp, v3df_dontpan | v3df_doppler);
2021-11-24 20:45:03 +00:00
if (!pp->insector())
return 0;
2022-08-20 10:11:15 +00:00
nx = pp->int_ppos().X;
ny = pp->int_ppos().Y;
nz = pp->int_ppos().Z + pp->bob_z + Z(8);
// Spawn a shot
// Inserting and setting up variables
auto actorNew = SpawnActor(STAT_MISSILE, CHEMBOMB, s_ChemBomb, pp->cursector,
nx, ny, nz, pp->angle.ang.Buildang(), CHEMBOMB_VELOCITY);
// don't throw it as far if crawling
2021-12-27 18:34:06 +00:00
if (pp->Flags & (PF_CRAWLING))
{
actorNew->spr.xvel -= (actorNew->spr.xvel >> 2);
}
2021-12-25 20:24:27 +00:00
// actorNew->user.RotNum = 5;
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
SetOwner(pp->actor, actorNew);
actorNew->spr.yrepeat = 32;
actorNew->spr.xrepeat = 32;
actorNew->spr.shade = -15;
actorNew->user.WeaponNum = plActor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 200;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK);
2021-12-27 18:34:06 +00:00
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
actorNew->user.Flags |= (SPR_UNDERWATER);
actorNew->spr.zvel = -pp->horizon.horiz.asq16() >> 9;
2021-12-24 22:21:41 +00:00
oclipdist = plActor->spr.clipdist;
plActor->spr.clipdist = 0;
actorNew->spr.clipdist = 0;
MissileSetPos(actorNew, DoChemBomb, 1000);
2021-12-24 22:21:41 +00:00
plActor->spr.clipdist = uint8_t(oclipdist);
actorNew->spr.clipdist = 80L >> 2;
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
// adjust xvel according to player velocity
actorNew->user.change.X += pp->vect.X >> 14;
actorNew->user.change.Y += pp->vect.Y >> 14;
// Smoke will come out for this many seconds
2021-12-25 20:24:27 +00:00
actorNew->user.WaitTics = CHEMTICS;
return 0;
}
2021-11-01 09:59:10 +00:00
int InitSpriteChemBomb(DSWActor* actor)
{
int nx, ny, nz;
PlaySound(DIGI_THROW, actor, v3df_dontpan | v3df_doppler);
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
nz = actor->int_pos().Z;
// Spawn a shot
// Inserting and setting up variables
auto actorNew = SpawnActor(STAT_MISSILE, CHEMBOMB, s_ChemBomb, actor->sector(),
2022-08-16 21:17:01 +00:00
nx, ny, nz, actor->int_ang(), CHEMBOMB_VELOCITY);
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
2021-11-01 09:59:10 +00:00
SetOwner(actor, actorNew);
actorNew->spr.yrepeat = 32;
actorNew->spr.xrepeat = 32;
actorNew->spr.shade = -15;
2021-12-25 21:06:35 +00:00
actorNew->user.WeaponNum = actor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 200;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
actorNew->spr.cstat |= (CSTAT_SPRITE_BLOCK);
actorNew->spr.zvel = short(-RandomRange(100) * HORIZ_MULT);
actorNew->spr.clipdist = 80L >> 2;
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
// Smoke will come out for this many seconds
2021-12-25 20:24:27 +00:00
actorNew->user.WaitTics = CHEMTICS;
return 0;
}
2021-11-01 10:30:28 +00:00
int InitChemBomb(DSWActor* actor)
{
int nx, ny, nz;
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
nz = actor->int_pos().Z;
// Spawn a shot
// Inserting and setting up variables
auto actorNew = SpawnActor(STAT_MISSILE, MUSHROOM_CLOUD, s_ChemBomb, actor->sector(),
2022-08-16 21:17:01 +00:00
nx, ny, nz, actor->int_ang(), CHEMBOMB_VELOCITY);
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
2021-11-01 10:30:28 +00:00
SetOwner(GetOwner(actor), actorNew);
actorNew->spr.yrepeat = 32;
actorNew->spr.xrepeat = 32;
actorNew->spr.shade = -15;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 200;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER | CSTAT_SPRITE_INVISIBLE); // Make nuke radiation
// invis.
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK);
2021-12-24 19:27:12 +00:00
if (SpriteInUnderwaterArea(actorNew))
actorNew->user.Flags |= (SPR_UNDERWATER);
actorNew->spr.zvel = short(-RandomRange(100) * HORIZ_MULT);
actorNew->spr.clipdist = 0;
2021-12-25 21:06:35 +00:00
if (actor->user.ID == MUSHROOM_CLOUD || actor->user.ID == 3121 || actor->user.ID == SUMO_RUN_R0) // 3121 == GRENADE_EXP
{
actorNew->user.change.X = 0;
actorNew->user.change.Y = 0;
actorNew->user.change.Z = 0;
actorNew->spr.xvel = actorNew->spr.yvel = actorNew->spr.zvel = 0;
// Smoke will come out for this many seconds
2021-12-25 20:24:27 +00:00
actorNew->user.WaitTics = 40*120;
}
else
{
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
// Smoke will come out for this many seconds
2021-12-25 20:24:27 +00:00
actorNew->user.WaitTics = 3*120;
}
return 0;
}
//////////////////////////////////////////////
//
// Inventory Flash Bombs
//
//////////////////////////////////////////////
2021-11-01 10:30:28 +00:00
2021-12-31 14:59:11 +00:00
int PlayerInitFlashBomb(PLAYER* pp)
{
unsigned int stat;
int dist, tx, ty, tmin;
short damage;
DSWActor* actor = pp->actor;
PlaySound(DIGI_GASPOP, pp, v3df_dontpan | v3df_doppler);
// Set it just a little to let player know what he just did
SetFadeAmt(pp, -30, 1); // White flash
for (stat = 0; stat < SIZ(StatDamageList); stat++)
{
2021-11-01 10:30:28 +00:00
SWStatIterator it(StatDamageList[stat]);
while (auto itActor = it.Next())
{
if (itActor == pp->actor)
break;
2022-08-21 14:03:46 +00:00
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
if (dist > 16384) // Flash radius
continue;
if (!(actor->spr.cstat & CSTAT_SPRITE_BLOCK))
continue;
2022-08-21 14:57:51 +00:00
if (!FAFcansee(itActor->spr.pos, itActor->sector(), actor->spr.pos.plusZ(-ActorSizeZ(actor)), actor->sector()))
continue;
damage = GetDamage(itActor, pp->actor, DMG_FLASHBOMB);
2021-12-25 19:26:31 +00:00
if (itActor->user.sop_parent)
{
break;
}
2021-12-25 19:26:31 +00:00
else if (itActor->user.PlayerP)
{
2021-12-25 19:26:31 +00:00
// if(itActor->user.PlayerP->NightVision)
// {
2021-12-25 19:26:31 +00:00
// SetFadeAmt(itActor->user.PlayerP, -200, 1); // Got him with night vision on!
// PlayerUpdateHealth(itActor->user.PlayerP, -15); // Hurt eyes
// }else
if (damage < -70)
{
int choosesnd = 0;
choosesnd = RandomRange(MAX_PAIN);
PlayerSound(PlayerLowHealthPainVocs[choosesnd],v3df_dontpan|v3df_doppler|v3df_follow,pp);
}
2021-12-25 19:26:31 +00:00
SetFadeAmt(itActor->user.PlayerP, damage, 1); // White flash
}
else
{
2021-11-05 22:19:57 +00:00
ActorPain(itActor);
2021-10-31 08:04:24 +00:00
SpawnFlashBombOnActor(itActor);
}
}
}
return 0;
}
int InitFlashBomb(DSWActor* actor)
{
2020-10-15 15:45:07 +00:00
int i;
unsigned int stat;
int dist, tx, ty, tmin;
short damage;
2021-12-31 14:59:11 +00:00
PLAYER* pp = Player + screenpeek;
PlaySound(DIGI_GASPOP, actor, v3df_dontpan | v3df_doppler);
for (stat = 0; stat < SIZ(StatDamageList); stat++)
{
2021-11-01 10:35:48 +00:00
SWStatIterator it(StatDamageList[stat]);
while (auto itActor = it.Next())
{
2022-08-21 14:03:46 +00:00
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
if (dist > 16384) // Flash radius
continue;
if (!(actor->spr.cstat & CSTAT_SPRITE_BLOCK))
continue;
2022-08-21 14:57:51 +00:00
if (!FAFcansee(itActor->spr.pos, itActor->sector(), actor->spr.pos.plusZ(-ActorSizeZ(actor)), actor->sector()))
continue;
damage = GetDamage(itActor, actor, DMG_FLASHBOMB);
2021-12-25 19:26:31 +00:00
if (itActor->user.sop_parent)
{
break;
}
2021-12-25 19:26:31 +00:00
else if (itActor->user.PlayerP)
{
if (damage < -70)
{
int choosesnd = 0;
choosesnd = RandomRange(MAX_PAIN);
PlayerSound(PlayerLowHealthPainVocs[choosesnd],v3df_dontpan|v3df_doppler|v3df_follow,pp);
}
2021-12-25 19:26:31 +00:00
SetFadeAmt(itActor->user.PlayerP, damage, 1); // White flash
}
else
{
2021-11-01 10:35:48 +00:00
if (itActor != actor)
{
2021-11-05 22:19:57 +00:00
ActorPain(itActor);
2021-10-31 08:04:24 +00:00
SpawnFlashBombOnActor(itActor);
}
}
}
}
return 0;
}
// This is a sneaky function to make actors look blinded by flashbomb while using flaming code
2021-10-31 08:04:24 +00:00
void SpawnFlashBombOnActor(DSWActor* actor)
{
2021-10-31 08:04:24 +00:00
if (!actor->hasU()) return;
// Forget about burnable sprites
if ((actor->spr.extra & SPRX_BURNABLE))
2021-10-31 08:04:24 +00:00
return;
2021-10-31 08:04:24 +00:00
if (actor != nullptr)
{
2021-12-25 21:06:35 +00:00
DSWActor* flameActor = actor->user.flameActor;
if (flameActor != nullptr)
{
int sizez = (int_ActorSizeZ(actor) * 5) >> 2;
if (flameActor->user.Counter >= GetRepeatFromHeight(flameActor, sizez))
{
// keep flame only slightly bigger than the enemy itself
flameActor->user.Counter = GetRepeatFromHeight(flameActor, sizez) * 2;
}
else
{
// increase max size
flameActor->user.Counter += GetRepeatFromHeight(flameActor, 8 << 8) * 2;
}
// 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
2021-10-31 08:04:24 +00:00
return;
}
}
auto actorNew = SpawnActor(STAT_MISSILE, FIREBALL_FLAMES, s_FireballFlames, actor->sector(),
2022-08-16 21:17:01 +00:00
actor->int_pos().X, actor->int_pos().Y, actor->int_pos().Z, actor->int_ang(), 0);
2022-10-01 12:33:56 +00:00
if (actor != nullptr)
2021-12-25 21:06:35 +00:00
actor->user.flameActor = actorNew;
2021-12-24 18:41:29 +00:00
actorNew->spr.xrepeat = 16;
actorNew->spr.yrepeat = 16;
2021-12-25 21:06:35 +00:00
if (actor->user.flameActor != nullptr)
{
actorNew->user.Counter = GetRepeatFromHeight(actorNew, int_ActorSizeZ(actor) >> 1) * 4;
}
else
2021-12-25 18:57:14 +00:00
actorNew->user.Counter = 0; // max flame size
2021-12-24 18:41:29 +00:00
actorNew->spr.shade = -40;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER | CSTAT_SPRITE_INVISIBLE);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
2021-12-25 18:57:14 +00:00
actorNew->user.Radius = 200;
2021-12-25 21:06:35 +00:00
if (actor->user.flameActor != nullptr)
{
2021-10-31 11:20:03 +00:00
SetAttach(actor, actorNew);
}
2021-10-31 08:04:24 +00:00
return;
}
//////////////////////////////////////////////
//
// Inventory Caltrops
//
//////////////////////////////////////////////
2021-11-01 10:35:48 +00:00
2021-12-31 14:59:11 +00:00
int PlayerInitCaltrops(PLAYER* pp)
{
DSWActor* plActor = pp->actor;
int nx, ny, nz;
short oclipdist;
PlaySound(DIGI_THROW, pp, v3df_dontpan | v3df_doppler);
2021-11-24 20:45:03 +00:00
if (!pp->insector())
return 0;
2022-08-20 10:11:15 +00:00
nx = pp->int_ppos().X;
ny = pp->int_ppos().Y;
nz = pp->int_ppos().Z + pp->bob_z + Z(8);
2021-12-25 20:23:46 +00:00
auto actorNew = SpawnActor(STAT_DEAD_ACTOR, CALTROPS, s_Caltrops, pp->cursector,
nx, ny, nz, pp->angle.ang.Buildang(), (CHEMBOMB_VELOCITY + RandomRange(CHEMBOMB_VELOCITY)) / 2);
// don't throw it as far if crawling
2021-12-27 18:34:06 +00:00
if (pp->Flags & (PF_CRAWLING))
{
2021-12-25 20:23:46 +00:00
actorNew->spr.xvel -= (actorNew->spr.xvel >> 2);
}
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
SetOwner(pp->actor, actorNew);
2021-12-25 20:23:46 +00:00
actorNew->spr.yrepeat = 64;
actorNew->spr.xrepeat = 64;
actorNew->spr.shade = -15;
actorNew->user.WeaponNum = plActor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 200;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
// spawnedActor->spr.cstat |= (CSTAT_SPRITE_BLOCK);
2021-12-27 18:34:06 +00:00
if (pp->Flags & (PF_DIVING) || SpriteInUnderwaterArea(actorNew))
actorNew->user.Flags |= (SPR_UNDERWATER);
// They go out at different angles
// spawnedActor->spr.ang = NORM_ANGLE(pp->angle.ang.Buildang() + (RandomRange(50) - 25));
2021-12-25 20:23:46 +00:00
actorNew->spr.zvel = -pp->horizon.horiz.asq16() >> 9;
2021-12-24 22:21:41 +00:00
oclipdist = plActor->spr.clipdist;
plActor->spr.clipdist = 0;
2021-12-25 20:23:46 +00:00
actorNew->spr.clipdist = 0;
2021-12-25 20:23:46 +00:00
MissileSetPos(actorNew, DoCaltrops, 1000);
2021-12-24 22:21:41 +00:00
plActor->spr.clipdist = uint8_t(oclipdist);
2021-12-25 20:23:46 +00:00
actorNew->spr.clipdist = 80L >> 2;
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
// adjust xvel according to player velocity
actorNew->user.change.X += pp->vect.X >> 14;
actorNew->user.change.Y += pp->vect.Y >> 14;
2021-12-25 20:23:46 +00:00
SetupSpriteForBreak(actorNew); // Put Caltrops in the break queue
return 0;
}
2021-11-01 10:50:50 +00:00
int InitCaltrops(DSWActor* actor)
{
int nx, ny, nz;
PlaySound(DIGI_THROW, actor, v3df_dontpan | v3df_doppler);
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
nz = actor->int_pos().Z;
// Spawn a shot
// Inserting and setting up variables
auto actorNew = SpawnActor(STAT_DEAD_ACTOR, CALTROPS, s_Caltrops, actor->sector(),
2022-08-16 21:17:01 +00:00
nx, ny, nz, actor->int_ang(), CHEMBOMB_VELOCITY / 2);
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
2021-12-25 20:23:46 +00:00
SetOwner(actor, actorNew);
actorNew->spr.yrepeat = 64;
actorNew->spr.xrepeat = 64;
actorNew->spr.shade = -15;
// !FRANK - clipbox must be <= weapon otherwise can clip thru walls
2021-12-25 20:23:46 +00:00
actorNew->spr.clipdist = actor->spr.clipdist;
2021-12-25 21:06:35 +00:00
actorNew->user.WeaponNum = actor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 200;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
2021-12-25 20:23:46 +00:00
actorNew->spr.zvel = short(-RandomRange(100) * HORIZ_MULT);
2021-12-24 19:13:26 +00:00
// spawnedActor->spr.clipdist = 80L>>2;
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
2021-12-25 20:23:46 +00:00
SetupSpriteForBreak(actorNew); // Put Caltrops in the break queue
return 0;
}
2021-11-01 10:50:50 +00:00
int InitPhosphorus(DSWActor* actor)
{
int nx, ny, nz;
short daang;
PlaySound(DIGI_FIREBALL1, actor, v3df_follow);
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
nz = actor->int_pos().Z;
daang = NORM_ANGLE(RandomRange(2048));
// Spawn a shot
// Inserting and setting up variables
auto actorNew = SpawnActor(STAT_SKIP4, FIREBALL1, s_Phosphorus, actor->sector(),
nx, ny, nz, daang, CHEMBOMB_VELOCITY/3);
actorNew->spr.hitag = LUMINOUS; // Always full brightness
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
// !Frank - don't do translucent
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
// actorNew->spr.cstat |= (CSTAT_SPRITE_TRANSLUCENT|CSTAT_SPRITE_YCENTER);
actorNew->spr.shade = -128;
actorNew->spr.yrepeat = 64;
actorNew->spr.xrepeat = 64;
actorNew->spr.shade = -15;
// !FRANK - clipbox must be <= weapon otherwise can clip thru walls
2021-12-24 15:05:27 +00:00
if (actor->spr.clipdist > 0)
actorNew->spr.clipdist = actor->spr.clipdist-1;
else
actorNew->spr.clipdist = actor->spr.clipdist;
2021-12-25 21:06:35 +00:00
actorNew->user.WeaponNum = actor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 600;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.zvel = short(-RandomRange(100) * HORIZ_MULT);
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = (actorNew->spr.zvel >> 1);
return 0;
}
2021-11-01 10:57:51 +00:00
int InitBloodSpray(DSWActor* actor, bool dogib, short velocity)
{
int nx, ny, nz;
short i, cnt, ang, vel, rnd;
if (dogib)
cnt = RandomRange(3)+1;
else
cnt = 1;
//if(dogib)
// {
rnd = RandomRange(1000);
if (rnd > 650)
PlaySound(DIGI_GIBS1, actor, v3df_none);
else if (rnd > 350)
PlaySound(DIGI_GIBS2, actor, v3df_none);
else
PlaySound(DIGI_GIBS3, actor, v3df_none);
// }
2022-08-16 21:17:01 +00:00
ang = actor->int_ang();
vel = velocity;
for (i=0; i<cnt; i++)
{
if (velocity == -1)
vel = 105+RandomRange(320);
else if (velocity == -2)
vel = 105+RandomRange(100);
if (dogib)
ang = NORM_ANGLE(ang + 512 + RandomRange(200));
else
ang = NORM_ANGLE(ang+1024+256 - RandomRange(256));
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
nz = int_ActorZOfTop(actor)-20;
// Spawn a shot
auto actorNew = SpawnActor(STAT_MISSILE, GOREDrip, s_BloodSprayChunk, actor->sector(),
nx, ny, nz, ang, vel*2);
actorNew->user.Flags |= (SPR_XFLIP_TOGGLE);
if (dogib)
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
else
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER | CSTAT_SPRITE_INVISIBLE);
actorNew->spr.shade = -12;
2021-11-01 10:57:51 +00:00
SetOwner(actor, actorNew);
actorNew->spr.yrepeat = 64-RandomRange(35);
actorNew->spr.xrepeat = 64-RandomRange(35);
actorNew->spr.shade = -15;
actorNew->spr.clipdist = actor->spr.clipdist;
2021-12-25 21:06:35 +00:00
actorNew->user.WeaponNum = actor->user.WeaponNum;
2021-12-25 20:24:27 +00:00
actorNew->user.Radius = 600;
actorNew->user.ceiling_dist = (3);
actorNew->user.floor_dist = (3);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.zvel = short((-10 - RandomRange(50)) * HORIZ_MULT);
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel >> 1;
if (!GlobalSkipZrange)
2021-11-03 16:47:13 +00:00
DoActorZrange(actorNew);
}
return 0;
}
2021-11-01 11:04:55 +00:00
int BloodSprayFall(DSWActor* actor)
{
actor->spr.pos.Z += 5.86;
return 0;
}
////////////////// DEATHFLAG! ////////////////////////////////////////////////////////////////
// Rules: Run to an enemy flag, run over it an it will stick to you.
// The goal is to run the enemy's flag back to your startpoint.
// If an enemy flag touches a friendly start sector, then the opposing team explodes and
// your team wins and the level restarts.
// Once you pick up a flag, you have 30 seconds to score, otherwise, the flag detonates
// an explosion, killing you and anyone in the vicinity, and you don't score.
//////////////////////////////////////////////////////////////////////////////////////////////
// Update the scoreboard for team color that just scored.
2021-11-01 11:04:55 +00:00
void DoFlagScore(int16_t pal)
{
2021-11-01 11:04:55 +00:00
SWStatIterator it(STAT_DEFAULT);
while (auto actor = it.Next())
{
2021-12-24 15:05:27 +00:00
if (actor->spr.picnum < 1900 || actor->spr.picnum > 1999)
continue;
2021-12-24 15:05:27 +00:00
if (actor->spr.pal == pal)
actor->spr.picnum++; // Increment the counter
2021-12-24 15:05:27 +00:00
if (actor->spr.picnum > 1999)
actor->spr.picnum = 1900; // Roll it over if you must
}
}
2021-11-01 11:04:55 +00:00
DSWActor* DoFlagRangeTest(DSWActor* actor, int range)
{
unsigned int stat;
int dist, tx, ty;
int tmin;
for (stat = 0; stat < SIZ(StatDamageList); stat++)
{
2021-11-01 11:04:55 +00:00
SWStatIterator it(StatDamageList[stat]);
while (auto itActor = it.Next())
{
2022-08-21 14:03:46 +00:00
DISTANCE(itActor->spr.pos, actor->spr.pos, dist, tx, ty, tmin);
if (dist > range)
continue;
2021-11-01 11:04:55 +00:00
if (actor == itActor)
continue;
if (!(itActor->spr.cstat & CSTAT_SPRITE_BLOCK))
continue;
if (!(itActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
continue;
2022-08-21 14:57:51 +00:00
if (!FAFcansee(itActor->spr.pos, itActor->sector(), actor->spr.pos, actor->sector()))
continue;
dist = FindDistance3D(actor->int_pos() - itActor->int_pos());
if (dist > range)
continue;
2021-11-01 11:04:55 +00:00
return itActor;
}
}
2021-11-01 11:04:55 +00:00
return nullptr;
}
2021-11-01 11:04:55 +00:00
int DoCarryFlag(DSWActor* actor)
{
const int FLAG_DETONATE_STATE = 99;
2021-12-25 21:06:35 +00:00
DSWActor* fown = actor->user.flagOwnerActor;
if (!fown) return 0;
2021-12-25 21:06:35 +00:00
DSWActor* attached = actor->user.attachActor;
2021-11-01 11:58:02 +00:00
// if no Owner then die
if (attached != nullptr)
{
DVector3 pos(attached->spr.pos.XY(), ActorZOfMiddle(attached));
SetActorZ(actor, pos);
2022-08-22 16:34:43 +00:00
actor->spr.angle = attached->spr.angle + DAngle270;
}
// not activated yet
if (!(actor->user.Flags & SPR_ACTIVE))
{
2021-12-25 21:06:35 +00:00
if ((actor->user.WaitTics -= (MISSILEMOVETICS * 2)) > 0)
return false;
// activate it
2021-12-25 21:06:35 +00:00
actor->user.WaitTics = SEC(30); // You have 30 seconds to get it to
// scorebox
2021-12-25 21:06:35 +00:00
actor->user.Counter2 = 0;
actor->user.Flags |= (SPR_ACTIVE);
}
// limit the number of times DoFlagRangeTest is called
2021-12-25 21:06:35 +00:00
actor->user.Counter++;
if (actor->user.Counter > 1)
actor->user.Counter = 0;
2021-12-25 21:06:35 +00:00
if (!actor->user.Counter)
{
// not already in detonate state
2021-12-25 21:06:35 +00:00
if (actor->user.Counter2 < FLAG_DETONATE_STATE)
{
if (!attached->hasU() || attached->user.Health <= 0)
{
2021-12-25 21:06:35 +00:00
actor->user.Counter2 = FLAG_DETONATE_STATE;
actor->user.WaitTics = SEC(1) / 2;
}
// if in score box, score.
if (attached->sector()->hitag == 9000 && attached->sector()->lotag == attached->spr.pal
2021-12-24 22:21:41 +00:00
&& attached->spr.pal != actor->spr.pal)
{
if (fown != nullptr)
{
2021-12-24 22:21:41 +00:00
if (fown->spr.lotag) // Trigger everything if there is a lotag
DoMatchEverything(nullptr, fown->spr.lotag, 1);
}
2021-12-24 22:21:41 +00:00
if (!TEST_BOOL1(fown))
{
2021-12-25 21:06:35 +00:00
PlaySound(DIGI_BIGITEM, actor->user.attachActor, v3df_none);
2021-12-24 22:21:41 +00:00
DoFlagScore(attached->spr.pal);
if (SP_TAG5(fown) > 0)
{
fown->user.filler++;
if (fown->user.filler >= SP_TAG5(fown))
{
fown->user.filler = 0;
DoMatchEverything(nullptr, SP_TAG6(fown), 1);
}
}
}
2021-10-30 20:44:16 +00:00
SetSuicide(actor); // Kill the flag, you scored!
}
}
else
{
// Time's up! Move directly to detonate state
2021-12-25 21:06:35 +00:00
actor->user.Counter2 = FLAG_DETONATE_STATE;
actor->user.WaitTics = SEC(1) / 2;
}
}
2021-12-25 21:06:35 +00:00
actor->user.WaitTics -= (MISSILEMOVETICS * 2);
2021-12-25 21:06:35 +00:00
switch (actor->user.Counter2)
{
case 0:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(30))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 1:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(20))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 2:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(10))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 3:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(5))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 4:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(4))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 5:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(3))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
}
break;
case 6:
2021-12-25 21:06:35 +00:00
if (actor->user.WaitTics < SEC(2))
{
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2 = FLAG_DETONATE_STATE;
}
break;
case FLAG_DETONATE_STATE:
// start frantic beeping
PlaySound(DIGI_MINEBEEP, actor, v3df_dontpan);
2021-12-25 21:06:35 +00:00
actor->user.Counter2++;
break;
case FLAG_DETONATE_STATE + 1:
SpawnGrenadeExp(actor);
2021-10-30 20:44:16 +00:00
SetSuicide(actor);
return false;
break;
}
return false;
}
2021-11-01 11:25:35 +00:00
int DoCarryFlagNoDet(DSWActor* actor)
{
2021-12-25 21:06:35 +00:00
DSWActor* attached = actor->user.attachActor;
DSWActor* fown = actor->user.flagOwnerActor;
if (!fown) return 0;
2021-12-25 21:06:35 +00:00
if (actor->user.flagOwnerActor != nullptr)
fown->user.WaitTics = 30 * 120; // Keep setting respawn tics so it won't respawn
2021-11-01 11:58:02 +00:00
// if no Owner then die
if (attached != nullptr)
{
2022-08-22 16:34:43 +00:00
SetActorZ(actor, DVector3(attached->spr.pos.XY(), ActorZOfMiddle(attached)));
actor->spr.angle = attached->spr.angle + DAngle270;
actor->spr.pos.Z = attached->spr.pos.Z - (ActorSizeZ(attached) * 0.5);
}
if (!attached->hasU() || attached->user.Health <= 0)
{
2021-12-25 21:06:35 +00:00
if (actor->user.flagOwnerActor != nullptr)
fown->user.WaitTics = 0; // Tell it to respawn
2021-10-30 20:44:16 +00:00
SetSuicide(actor);
return false;
}
// if in score box, score.
if (attached->sector()->hitag == 9000 && attached->sector()->lotag == attached->spr.pal
&& attached->spr.pal != actor->spr.pal)
{
2021-12-25 21:06:35 +00:00
if (actor->user.flagOwnerActor != nullptr)
{
2021-12-24 22:21:41 +00:00
if (fown->spr.lotag) // Trigger everything if there is a lotag
DoMatchEverything(nullptr, fown->spr.lotag, 1);
fown->user.WaitTics = 0; // Tell it to respawn
}
2021-12-24 22:21:41 +00:00
if (!TEST_BOOL1(fown))
{
2021-12-25 21:06:35 +00:00
PlaySound(DIGI_BIGITEM, actor->user.attachActor, v3df_none);
DoFlagScore(attached->spr.pal);
2021-12-24 22:21:41 +00:00
if (SP_TAG5(fown) > 0)
{
fown->user.filler++;
if (fown->user.filler >= SP_TAG5(fown))
{
fown->user.filler = 0;
DoMatchEverything(nullptr, SP_TAG6(fown), 1);
}
}
}
2021-10-30 20:44:16 +00:00
SetSuicide(actor); // Kill the flag, you scored!
}
return false;
}
2021-11-01 11:04:55 +00:00
int SetCarryFlag(DSWActor* actor)
{
// stuck
actor->user.Flags |= (SPR_BOUNCE);
// not yet active for 1 sec
2021-12-25 21:08:57 +00:00
actor->spr.cstat |= (CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
2021-12-25 21:06:35 +00:00
actor->user.Counter = 0;
2021-11-01 11:04:55 +00:00
change_actor_stat(actor, STAT_ITEM);
2021-12-24 15:05:27 +00:00
if (actor->spr.hitag == 1)
2021-11-01 11:04:55 +00:00
ChangeState(actor, s_CarryFlagNoDet);
else
2021-11-01 11:04:55 +00:00
ChangeState(actor, s_CarryFlag);
return false;
}
2021-11-01 11:04:55 +00:00
int DoFlag(DSWActor* actor)
{
2021-11-01 11:04:55 +00:00
auto hitActor = DoFlagRangeTest(actor, 1000);
2021-11-01 11:04:55 +00:00
if (hitActor)
{
2021-11-01 11:04:55 +00:00
SetCarryFlag(actor);
// check to see if sprite is player or enemy
if ((hitActor->spr.extra & SPRX_PLAYER_OR_ENEMY))
{
// attach weapon to sprite
2021-12-27 17:58:15 +00:00
actor->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
2021-10-31 11:20:03 +00:00
SetAttach(hitActor, actor);
actor->user.pos.Z = hitActor->spr.pos.Z - (int_ActorSizeZ(hitActor) >> 1) * zinttoworld;
}
}
return false;
}
2021-11-01 11:25:35 +00:00
int SpawnShell(DSWActor* actor, int ShellNum)
{
int nx, ny, nz;
2021-12-24 15:09:55 +00:00
short id=0,velocity=0;
2021-12-31 15:00:14 +00:00
STATE* p=nullptr;
extern STATE s_UziShellShrap[];
extern STATE s_ShotgunShellShrap[];
nx = actor->int_pos().X;
ny = actor->int_pos().Y;
2022-10-05 15:57:38 +00:00
nz = int_ActorZOfMiddle(actor);
switch (ShellNum)
{
case -2:
case -3:
id = UZI_SHELL;
p = s_UziShellShrap;
velocity = 1500 + RandomRange(1000);
break;
case -4:
id = SHOT_SHELL;
p = s_ShotgunShellShrap;
velocity = 2000 + RandomRange(1000);
break;
}
2022-08-16 21:17:01 +00:00
auto actorNew = SpawnActor(STAT_SKIP4, id, p, actor->sector(), nx, ny, nz, actor->int_ang(), 64);
actorNew->spr.zvel = -(velocity);
2021-12-25 21:06:35 +00:00
if (actor->user.PlayerP)
{
2022-02-01 19:19:46 +00:00
actorNew->add_int_z(int(-actor->user.PlayerP->horizon.horiz.asbuildf() * HORIZ_MULT * (1. / 3.)));
}
2021-12-25 20:24:27 +00:00
switch (actorNew->user.ID)
{
case UZI_SHELL:
actorNew->spr.pos.Z -= 13;
if (ShellNum == -3)
{
actorNew->spr.angle = actor->spr.angle;
HelpMissileLateral(actorNew,2500);
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() - 512));
HelpMissileLateral(actorNew,1000); // Was 1500
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() + 712));
}
else
{
actorNew->spr.angle = actor->spr.angle;
HelpMissileLateral(actorNew,2500);
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() + 512));
HelpMissileLateral(actorNew,1500);
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() - 128));
}
2022-08-16 21:26:39 +00:00
actorNew->add_int_ang((RANDOM_P2(128<<5)>>5) - (128 / 2));
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang()));
// Set the shell number
2021-12-25 20:24:27 +00:00
actorNew->user.ShellNum = ShellCount;
actorNew->spr.yrepeat = actorNew->spr.xrepeat = 13;
break;
case SHOT_SHELL:
actorNew->spr.pos.Z -= 13;
actorNew->spr.angle = actor->spr.angle;
HelpMissileLateral(actorNew,2500);
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() + 512));
HelpMissileLateral(actorNew,1300);
2022-08-16 21:23:23 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->spr.int_ang() - 128 - 64));
2022-08-16 21:26:39 +00:00
actorNew->add_int_ang((RANDOM_P2(128<<5)>>5) - (128 / 2));
2022-08-16 21:23:42 +00:00
actorNew->set_int_ang(NORM_ANGLE(actorNew->int_ang()));
// Set the shell number
2021-12-25 20:24:27 +00:00
actorNew->user.ShellNum = ShellCount;
actorNew->spr.yrepeat = actorNew->spr.xrepeat = 18;
break;
}
2021-11-01 11:25:35 +00:00
SetOwner(actor, actorNew);
actorNew->spr.shade = -15;
actorNew->user.ceiling_dist = (1);
actorNew->user.floor_dist = (1);
2021-12-25 20:24:27 +00:00
actorNew->user.Counter = 0;
actorNew->spr.cstat |= (CSTAT_SPRITE_YCENTER);
2021-12-27 17:58:15 +00:00
actorNew->spr.cstat &= ~(CSTAT_SPRITE_BLOCK | CSTAT_SPRITE_BLOCK_HITSCAN);
actorNew->user.Flags &= ~(SPR_BOUNCE|SPR_UNDERWATER); // Make em' bounce
2022-08-16 21:15:49 +00:00
actorNew->user.change.X = MOVEx(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Y = MOVEy(actorNew->spr.xvel, actorNew->int_ang());
actorNew->user.change.Z = actorNew->spr.zvel;
2021-12-25 20:24:27 +00:00
actorNew->user.jump_speed = 200;
actorNew->user.jump_speed += RandomRange(400);
actorNew->user.jump_speed = -actorNew->user.jump_speed;
DoBeginJump(actorNew);
2021-12-25 20:24:27 +00:00
actorNew->user.jump_grav = ACTOR_GRAVITY;
return 0;
}
#include "saveable.h"
static saveable_data saveable_jweapon_data[] =
{
SAVE_DATA(s_BloodSpray),
SAVE_DATA(s_PhosphorExp),
SAVE_DATA(s_NukeMushroom),
SAVE_DATA(s_RadiationCloud),
SAVE_DATA(s_ChemBomb),
SAVE_DATA(s_Caltrops),
SAVE_DATA(s_CaltropsStick),
SAVE_DATA(s_CarryFlag),
SAVE_DATA(s_CarryFlagNoDet),
SAVE_DATA(s_Flag),
SAVE_DATA(s_Phosphorus),
SAVE_DATA(s_BloodSprayChunk),
SAVE_DATA(s_BloodSprayDrip),
};
saveable_module saveable_jweapon =
{
// code
nullptr,0,
// data
saveable_jweapon_data,
SIZ(saveable_jweapon_data)
};
END_SW_NS