raze/source/rr/src/actors.cpp
2019-12-17 00:29:38 +01:00

9705 lines
359 KiB
C++

//-------------------------------------------------------------------------
/*
Copyright (C) 2016 EDuke32 developers and contributors
This file is part of EDuke32.
EDuke32 is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License version 2
as published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
//-------------------------------------------------------------------------
#include "ns.h" // Must come before everything else!
#define actors_c_
#include "duke3d.h"
BEGIN_RR_NS
#if KRANDDEBUG
# define ACTOR_STATIC
#else
# define ACTOR_STATIC static
#endif
#define DELETE_SPRITE_AND_CONTINUE(KX) do { A_DeleteSprite(KX); goto next_sprite; } while (0)
int32_t otherp;
int G_SetInterpolation(int32_t *const posptr)
{
if (g_interpolationCnt >= MAXINTERPOLATIONS)
return 1;
for (bssize_t i = 0; i < g_interpolationCnt; ++i)
if (curipos[i] == posptr)
return 0;
curipos[g_interpolationCnt] = posptr;
oldipos[g_interpolationCnt] = *posptr;
g_interpolationCnt++;
return 0;
}
void G_StopInterpolation(const int32_t * const posptr)
{
for (bssize_t i = 0; i < g_interpolationCnt; ++i)
if (curipos[i] == posptr)
{
g_interpolationCnt--;
oldipos[i] = oldipos[g_interpolationCnt];
bakipos[i] = bakipos[g_interpolationCnt];
curipos[i] = curipos[g_interpolationCnt];
}
}
void G_DoInterpolations(int smoothRatio)
{
if (g_interpolationLock++)
return;
int32_t ndelta = 0;
for (bssize_t i = 0, j = 0; i < g_interpolationCnt; ++i)
{
int32_t const odelta = ndelta;
bakipos[i] = *curipos[i];
ndelta = (*curipos[i]) - oldipos[i];
if (odelta != ndelta)
j = mulscale16(ndelta, smoothRatio);
*curipos[i] = oldipos[i] + j;
}
}
void G_ClearCameraView(DukePlayer_t *ps)
{
ps->newowner = -1;
ps->pos = ps->opos;
ps->q16ang = ps->oq16ang;
updatesector(ps->pos.x, ps->pos.y, &ps->cursectnum);
P_UpdateScreenPal(ps);
for (bssize_t SPRITES_OF(STAT_ACTOR, k))
if (sprite[k].picnum==CAMERA1)
sprite[k].yvel = 0;
}
// Manhattan distance between wall-point and sprite.
static FORCE_INLINE int32_t G_WallSpriteDist(uwalltype const * const wal, uspritetype const * const spr)
{
return klabs(wal->x - spr->x) + klabs(wal->y - spr->y);
}
void A_RadiusDamage(int spriteNum, int blastRadius, int dmg1, int dmg2, int dmg3, int dmg4)
{
uspritetype const *const pSprite = (uspritetype *)&sprite[spriteNum];
int32_t sectorCount = 0;
int32_t numSectors = 1;
int16_t sectorList[64] = { pSprite->sectnum };
if ((pSprite->picnum == RPG && pSprite->xrepeat < 11)
|| (RRRA && pSprite->picnum == RPG2 && pSprite->xrepeat < 11)
|| (!RR && pSprite->picnum == SHRINKSPARK))
goto SKIPWALLCHECK;
do
{
int const sectorNum = sectorList[sectorCount++];
int const startWall = sector[sectorNum].wallptr;
int const endWall = startWall + sector[sectorNum].wallnum;
int const w2 = wall[startWall].point2;
// Check if "hit" 1st or 3rd wall-point. This mainly makes sense
// for rectangular "ceiling light"-style sectors.
if (G_WallSpriteDist((uwalltype *)&wall[startWall], pSprite) < blastRadius ||
G_WallSpriteDist((uwalltype *)&wall[wall[w2].point2], pSprite) < blastRadius)
{
if (((sector[sectorNum].ceilingz-pSprite->z)>>8) < blastRadius)
Sect_DamageCeiling(sectorNum);
}
native_t w = startWall;
for (uwalltype const *pWall = (uwalltype *)&wall[startWall]; w < endWall; w++, pWall++)
{
if (G_WallSpriteDist(pWall, pSprite) >= blastRadius)
continue;
int const nextSector = pWall->nextsector;
if (nextSector >= 0)
{
native_t otherSector = 0;
for (; otherSector < numSectors; ++otherSector)
if (sectorList[otherSector] == nextSector)
break;
if (otherSector == numSectors)
{
if (numSectors == ARRAY_SIZE(sectorList))
goto SKIPWALLCHECK; // prevent oob access of 'sectorlist'
sectorList[numSectors++] = nextSector;
}
}
int16_t damageSector = (nextSector >= 0) ? wall[wall[w].nextwall].nextsector : sectorofwall(w);
vec3_t const vect = { (((pWall->x + wall[pWall->point2].x) >> 1) + pSprite->x) >> 1,
(((pWall->y + wall[pWall->point2].y) >> 1) + pSprite->y) >> 1, pSprite->z };
updatesector(vect.x, vect.y, &damageSector);
if (damageSector >= 0 && cansee(vect.x, vect.y, pSprite->z, damageSector, pSprite->x, pSprite->y, pSprite->z, pSprite->sectnum))
A_DamageWall(spriteNum, w, &vect, pSprite->picnum);
}
}
while (sectorCount < numSectors);
SKIPWALLCHECK:
// this is really weird
int32_t const zRand = (RR ? -(24<<8) : -ZOFFSET2) + (krand2()&(ZOFFSET5-1));
static const uint8_t statnumList [] ={
STAT_DEFAULT, STAT_ACTOR, STAT_STANDABLE,
STAT_PLAYER, STAT_FALLER, STAT_ZOMBIEACTOR, STAT_MISC
};
for (unsigned char stati : statnumList)
{
int32_t otherSprite = headspritestat[stati];
while (otherSprite >= 0)
{
int const nextOther = nextspritestat[otherSprite];
spritetype *const pOther = &sprite[otherSprite];
// DEFAULT, ZOMBIEACTOR, MISC
if (stati == STAT_DEFAULT || stati == STAT_ZOMBIEACTOR || stati == STAT_MISC || AFLAMABLE(pOther->picnum))
{
if ((!RR && pSprite->picnum != SHRINKSPARK) || (pOther->cstat&257))
{
if (dist(pSprite, pOther) < blastRadius)
{
if (A_CheckEnemySprite(pOther) && !cansee(pOther->x, pOther->y, pOther->z+zRand, pOther->sectnum, pSprite->x, pSprite->y, pSprite->z+zRand, pSprite->sectnum))
goto next_sprite;
A_DamageObject(otherSprite, spriteNum);
}
}
}
else if (pOther->extra >= 0 && (uspritetype *)pOther != pSprite && ((pOther->cstat & 257) ||
(!RR && pOther->picnum == TRIPBOMB) || pOther->picnum == QUEBALL || (RR && pOther->picnum == RRTILE3440) || pOther->picnum == STRIPEBALL || pOther->picnum == DUKELYINGDEAD ||
A_CheckEnemySprite(pOther)))
{
if ((!RR && pSprite->picnum == SHRINKSPARK && pOther->picnum != SHARK && (otherSprite == pSprite->owner || pOther->xrepeat < 24))
|| (pSprite->picnum == MORTER && otherSprite == pSprite->owner)
|| (RRRA && ((pSprite->picnum == CHEERBOMB && otherSprite == pSprite->owner) || (pOther->picnum == MINION && pOther->pal == 19))))
goto next_sprite;
int32_t const spriteDist = pOther->picnum == APLAYER
? FindDistance3D(pSprite->x - pOther->x, pSprite->y - pOther->y, pSprite->z - (pOther->z - PHEIGHT))
: dist(pSprite, pOther);
if (spriteDist >= blastRadius || !cansee(pOther->x, pOther->y, pOther->z - ZOFFSET3, pOther->sectnum, pSprite->x,
pSprite->y, pSprite->z - ZOFFSET4, pSprite->sectnum))
goto next_sprite;
actor_t & dmgActor = actor[otherSprite];
dmgActor.ang = getangle(pOther->x - pSprite->x, pOther->y - pSprite->y);
if (RRRA && pOther->extra > 0 && pSprite->picnum == RPG2)
dmgActor.picnum = RPG;
else if ((pOther->extra > 0 && pSprite->picnum == RPG)
|| (!RR && pSprite->picnum == SHRINKSPARK))
dmgActor.picnum = pSprite->picnum;
else dmgActor.picnum = RADIUSEXPLOSION;
if (RR || pSprite->picnum != SHRINKSPARK)
{
if (spriteDist < blastRadius/3)
{
if (dmg4 == dmg3) dmg4++;
dmgActor.extra = dmg3 + (krand2()%(dmg4-dmg3));
}
else if (spriteDist < (2*blastRadius)/3)
{
if (dmg3 == dmg2) dmg3++;
dmgActor.extra = dmg2 + (krand2()%(dmg3-dmg2));
}
else if (spriteDist < blastRadius)
{
if (dmg2 == dmg1) dmg2++;
dmgActor.extra = dmg1 + (krand2()%(dmg2-dmg1));
}
if (!A_CheckSpriteFlags(otherSprite, SFLAG_NODAMAGEPUSH))
{
if (pOther->xvel < 0) pOther->xvel = 0;
pOther->xvel += ((RR ? pOther->extra : pSprite->extra)<<2);
}
switch (DYNAMICTILEMAP(pOther->picnum))
{
case PODFEM1__STATIC:
case FEM1__STATIC:
case FEM2__STATIC:
case FEM3__STATIC:
case FEM4__STATIC:
case FEM5__STATIC:
case FEM6__STATIC:
case FEM7__STATIC:
case FEM8__STATIC:
case FEM9__STATIC:
case FEM10__STATIC:
case SPACEMARINE__STATIC: if (RR) break; fallthrough__;
case STATUE__STATIC:
case STATUEFLASH__STATIC:
case QUEBALL__STATIC:
case RRTILE3440__STATICRR:
case STRIPEBALL__STATIC: A_DamageObject(otherSprite, spriteNum);
default: break;
}
}
else if (pSprite->extra == 0) dmgActor.extra = 0;
if (pOther->picnum != RADIUSEXPLOSION &&
pSprite->owner >= 0 && sprite[pSprite->owner].statnum < MAXSTATUS)
{
if (pOther->picnum == APLAYER)
{
DukePlayer_t *pPlayer = g_player[P_GetP((uspritetype *)pOther)].ps;
if (pPlayer->newowner >= 0)
G_ClearCameraView(pPlayer);
}
dmgActor.owner = pSprite->owner;
}
}
next_sprite:
otherSprite = nextOther;
}
}
}
// Check whether sprite <s> is on/in a non-SE7 water sector.
// <othersectptr>: if not NULL, the sector on the other side.
int A_CheckNoSE7Water(uspritetype const * const pSprite, int sectNum, int sectLotag, int32_t *pOther)
{
if (sectLotag == ST_1_ABOVE_WATER || sectLotag == ST_2_UNDERWATER)
{
int const otherSect =
yax_getneighborsect(pSprite->x, pSprite->y, sectNum, sectLotag == ST_1_ABOVE_WATER ? YAX_FLOOR : YAX_CEILING);
int const otherLotag = (sectLotag == ST_1_ABOVE_WATER) ? ST_2_UNDERWATER : ST_1_ABOVE_WATER;
// If submerging, the lower sector MUST have lotag 2.
// If emerging, the upper sector MUST have lotag 1.
// This way, the x/y coordinates where above/below water
// changes can happen are the same.
if (otherSect >= 0 && sector[otherSect].lotag == otherLotag)
{
if (pOther)
*pOther = otherSect;
return 1;
}
}
return 0;
}
// Check whether to do a z position update of sprite <spritenum>.
// Returns:
// 0 if no.
// 1 if yes, but stayed inside [actor[].ceilingz+1, actor[].floorz].
// <0 if yes, but passed a TROR no-SE7 water boundary. -returnvalue-1 is the
// other-side sector number.
static int32_t A_CheckNeedZUpdate(int32_t spriteNum, int32_t zChange, int32_t *pZcoord)
{
uspritetype const *const pSprite = (uspritetype *)&sprite[spriteNum];
int const newZ = pSprite->z + (zChange >> 1);
*pZcoord = newZ;
if (newZ > actor[spriteNum].ceilingz && newZ <= actor[spriteNum].floorz)
return 1;
#ifdef YAX_ENABLE
int const sectNum = pSprite->sectnum;
int const sectLotag = sector[sectNum].lotag;
int32_t otherSect;
// Non-SE7 water.
// PROJECTILE_CHSECT
if ((zChange < 0 && sectLotag == ST_2_UNDERWATER) || (zChange > 0 && sectLotag == ST_1_ABOVE_WATER))
{
if (A_CheckNoSE7Water(pSprite, sprite[spriteNum].sectnum, sectLotag, &otherSect))
{
A_Spawn(spriteNum, WATERSPLASH2);
// NOTE: Don't tweak its z position afterwards like with
// SE7-induced projectile teleportation. It doesn't look good
// with TROR water.
actor[spriteNum].flags |= SFLAG_DIDNOSE7WATER;
return -otherSect-1;
}
}
#endif
return 0;
}
int32_t A_MoveSprite(int32_t spriteNum, vec3_t const * const change, uint32_t clipType)
{
spritetype *const pSprite = &sprite[spriteNum];
int const isEnemy = A_CheckEnemySprite(pSprite);
vec2_t const oldPos = *(vec2_t *)pSprite;
if (pSprite->statnum == STAT_MISC || (isEnemy && pSprite->xrepeat < 4))
{
pSprite->x += change->x;
pSprite->y += change->y;
pSprite->z += change->z;
if (isEnemy)
setsprite(spriteNum, (vec3_t *)pSprite);
return 0;
}
int32_t clipDist;
if (isEnemy)
{
if (RR)
clipDist = 192;
else if (pSprite->xrepeat > 60)
clipDist = 1024;
else if (pSprite->picnum == LIZMAN)
clipDist = 292;
else if (A_CheckSpriteFlags(spriteNum, SFLAG_BADGUY))
clipDist = pSprite->clipdist<<2;
else
clipDist = 192;
}
else
{
if (pSprite->statnum == STAT_PROJECTILE)
clipDist = 8;
else if (RR)
clipDist = 128;
else
clipDist = pSprite->clipdist<<2;
}
int16_t newSectnum = pSprite->sectnum;
int32_t newZ = pSprite->z - 2 * tilesiz[pSprite->picnum].y * pSprite->yrepeat;
int const oldZ = pSprite->z;
// Handle horizontal movement first.
pSprite->z = newZ;
int returnValue =
clipmove((vec3_t *)pSprite, &newSectnum, change->x << 13, change->y << 13, clipDist, ZOFFSET6, ZOFFSET6, clipType);
pSprite->z = oldZ;
if (isEnemy)
{
// Handle potential stayput condition (map-provided or hard-coded).
if (newSectnum < 0
|| ((actor[spriteNum].actorstayput >= 0 && actor[spriteNum].actorstayput != newSectnum)
|| (!RR && ((pSprite->picnum == BOSS2 && pSprite->pal == 0 && sector[newSectnum].lotag != ST_3)
|| ((pSprite->picnum == BOSS1 || pSprite->picnum == BOSS2) && sector[newSectnum].lotag == ST_1_ABOVE_WATER)
|| (sector[newSectnum].lotag == ST_1_ABOVE_WATER
&& (pSprite->picnum == LIZMAN || (pSprite->picnum == LIZTROOP && pSprite->zvel == 0)))))
))
{
*(vec2_t *) pSprite = oldPos;
if ((newSectnum >= 0 && sector[newSectnum].lotag == ST_1_ABOVE_WATER && (RR || pSprite->picnum == LIZMAN))
|| ((AC_COUNT(actor[spriteNum].t_data)&3) == 1 && (RR || pSprite->picnum != COMMANDER)))
pSprite->ang = krand2()&2047;
setsprite(spriteNum, (vec3_t *)pSprite);
if (newSectnum < 0)
newSectnum = 0;
return 16384+newSectnum;
}
if ((returnValue&49152) >= 32768 && actor[spriteNum].cgg==0)
pSprite->ang += 768;
}
if (newSectnum == -1)
{
newSectnum = pSprite->sectnum;
// OSD_Printf("%s:%d wtf\n",__FILE__,__LINE__);
}
else if (newSectnum != pSprite->sectnum)
{
changespritesect(spriteNum, newSectnum);
// A_GetZLimits(spritenum);
}
Bassert(newSectnum == pSprite->sectnum);
int const doZUpdate = A_CheckNeedZUpdate(spriteNum, change->z, &newZ);
// Update sprite's z positions and (for TROR) maybe the sector number.
if (doZUpdate)
{
pSprite->z = newZ;
#ifdef YAX_ENABLE
if (doZUpdate < 0)
{
// If we passed a TROR no-SE7 water boundary, signal to the outside
// that the ceiling/floor was not hit. However, this is not enough:
// later, code checks for (retval&49152)!=49152
// [i.e. not "was ceiling or floor hit", but "was no sprite hit"]
// and calls G_WeaponHitCeilingOrFloor() then, so we need to set
// actor[].flags |= SFLAG_DIDNOSE7WATER in A_CheckNeedZUpdate()
// previously.
// XXX: Why is this contrived data flow necessary? (If at all.)
changespritesect(spriteNum, -doZUpdate-1);
return 0;
}
if (yax_getbunch(newSectnum, (change->z>0))>=0
&& (SECTORFLD(newSectnum,stat, (change->z>0))&yax_waltosecmask(clipType))==0)
{
setspritez(spriteNum, (vec3_t *)pSprite);
}
#endif
}
else if (returnValue == 0)
returnValue = 16384+newSectnum;
return returnValue;
}
int32_t block_deletesprite = 0;
#ifdef POLYMER
static void A_DeleteLight(int32_t s)
{
if (actor[s].lightId >= 0)
polymer_deletelight(actor[s].lightId);
actor[s].lightId = -1;
actor[s].lightptr = NULL;
}
void G_Polymer_UnInit(void)
{
int32_t i;
for (i=0; i<MAXSPRITES; i++)
A_DeleteLight(i);
}
#endif
// deletesprite() game wrapper
void A_DeleteSprite(int spriteNum)
{
if (EDUKE32_PREDICT_FALSE(block_deletesprite))
{
OSD_Printf(OSD_ERROR "A_DeleteSprite(): tried to remove sprite %d in EVENT_EGS\n", spriteNum);
return;
}
#ifdef POLYMER
if (actor[spriteNum].lightptr != NULL && videoGetRenderMode() == REND_POLYMER)
A_DeleteLight(spriteNum);
#endif
// AMBIENT_SFX_PLAYING
if (sprite[spriteNum].picnum == MUSICANDSFX && actor[spriteNum].t_data[0] == 1)
S_StopEnvSound(sprite[spriteNum].lotag, spriteNum);
// NetAlloc
//if (Net_IsRelevantSprite(spriteNum))
//{
// Net_DeleteSprite(spriteNum);
// return;
//}
deletesprite(spriteNum);
}
void A_AddToDeleteQueue(int spriteNum)
{
if (g_deleteQueueSize == 0)
{
A_DeleteSprite(spriteNum);
return;
}
if (SpriteDeletionQueue[g_spriteDeleteQueuePos] >= 0)
sprite[SpriteDeletionQueue[g_spriteDeleteQueuePos]].xrepeat = 0;
SpriteDeletionQueue[g_spriteDeleteQueuePos] = spriteNum;
g_spriteDeleteQueuePos = (g_spriteDeleteQueuePos+1)%g_deleteQueueSize;
}
void A_SpawnMultiple(int spriteNum, int tileNum, int spawnCnt)
{
spritetype *pSprite = &sprite[spriteNum];
for (; spawnCnt>0; spawnCnt--)
{
int32_t const r1 = krand2(), r2 = krand2();
int const j = A_InsertSprite(pSprite->sectnum, pSprite->x, pSprite->y, pSprite->z - (r2 % (47 << 8)), tileNum, -32, 8,
8, r1 & 2047, 0, 0, spriteNum, 5);
//A_Spawn(-1, j);
sprite[j].cstat = krand2()&12;
}
}
void A_DoGuts(int spriteNum, int tileNum, int spawnCnt)
{
uspritetype const *const pSprite = (uspritetype *)&sprite[spriteNum];
vec2_t repeat = { 32, 32 };
if (A_CheckEnemySprite(pSprite) && pSprite->xrepeat < 16)
repeat.x = repeat.y = 8;
int gutZ = pSprite->z - ZOFFSET3;
int floorz = getflorzofslope(pSprite->sectnum, pSprite->x, pSprite->y);
if (gutZ > (floorz-ZOFFSET3))
gutZ = floorz-ZOFFSET3;
if (!RR && pSprite->picnum == COMMANDER)
gutZ -= (24<<8);
uint8_t pal = 0;
if (A_CheckEnemySprite(pSprite) && pSprite->pal == 6)
pal = 6;
else if (RRRA && pSprite->picnum == MINION && (pSprite->pal == 8 || pSprite->pal == 19))
pal = pSprite->pal;
if (RR) repeat.x >>= 1, repeat.y >>= 1;
for (bssize_t j=spawnCnt; j>0; j--)
{
int32_t const ang = krand2() & 2047;
int32_t const r1 = krand2(), r2 = krand2(), r3 = krand2(), r4 = krand2(), r5 = krand2();
int const i = A_InsertSprite(pSprite->sectnum, pSprite->x + (r5 & 255) - 128,
pSprite->y + (r4 & 255) - 128, gutZ - (r3 & 8191), tileNum, -32, repeat.x,
repeat.y, ang, 48 + (r2 & 31), -512 - (r1 & 2047), spriteNum, 5);
if (!RR && PN(i) == JIBS2)
{
sprite[i].xrepeat >>= 2;
sprite[i].yrepeat >>= 2;
}
sprite[i].pal = pal;
}
}
void A_DoGutsDir(int spriteNum, int tileNum, int spawnCnt)
{
uspritetype const * const s = (uspritetype *)&sprite[spriteNum];
vec2_t repeat = { 32, 32 };
if (A_CheckEnemySprite(s) && s->xrepeat < 16)
repeat.x = repeat.y = 8;
int gutZ = s->z-ZOFFSET3;
int floorZ = getflorzofslope(s->sectnum,s->x,s->y);
if (gutZ > (floorZ-ZOFFSET3))
gutZ = floorZ-ZOFFSET3;
if (!RR && s->picnum == COMMANDER)
gutZ -= (24<<8);
for (bssize_t j=spawnCnt; j>0; j--)
{
int32_t const ang = krand2() & 2047;
int32_t const r1 = krand2(), r2 = krand2();
A_InsertSprite(s->sectnum, s->x, s->y, gutZ, tileNum, -32, repeat.x, repeat.y, ang,
256 + (r2 & 127), -512 - (r1 & 2047), spriteNum, 5);
}
}
static int32_t G_ToggleWallInterpolation(int32_t wallNum, int32_t setInterpolation)
{
if (setInterpolation)
{
return G_SetInterpolation(&wall[wallNum].x) || G_SetInterpolation(&wall[wallNum].y);
}
else
{
G_StopInterpolation(&wall[wallNum].x);
G_StopInterpolation(&wall[wallNum].y);
return 0;
}
}
void Sect_ToggleInterpolation(int sectNum, int setInterpolation)
{
for (bssize_t j = sector[sectNum].wallptr, endwall = sector[sectNum].wallptr + sector[sectNum].wallnum; j < endwall; j++)
{
G_ToggleWallInterpolation(j, setInterpolation);
int const nextWall = wall[j].nextwall;
if (nextWall >= 0)
{
G_ToggleWallInterpolation(nextWall, setInterpolation);
G_ToggleWallInterpolation(wall[nextWall].point2, setInterpolation);
}
}
}
void A_MoveSector(int spriteNum)
{
// T1,T2 and T3 are used for all the sector moving stuff!!!
spritetype *const pSprite = &sprite[spriteNum];
int const rotateAngle = T3(spriteNum);
int originIdx = T2(spriteNum);
pSprite->x += (pSprite->xvel * (sintable[(pSprite->ang + 512) & 2047])) >> 14;
pSprite->y += (pSprite->xvel * (sintable[pSprite->ang & 2047])) >> 14;
int const endWall = sector[pSprite->sectnum].wallptr + sector[pSprite->sectnum].wallnum;
for (bssize_t wallNum = sector[pSprite->sectnum].wallptr; wallNum < endWall; wallNum++)
{
vec2_t const origin = g_origins[originIdx];
vec2_t result;
rotatepoint(zerovec, origin, rotateAngle & 2047, &result);
dragpoint(wallNum, pSprite->x + result.x, pSprite->y + result.y, 0);
originIdx++;
}
}
// NOTE: T5 is AC_ACTION_ID
# define LIGHTRAD_PICOFS(i) (T5(i) ? *(apScript + T5(i)) + (*(apScript + T5(i) + 2)) * AC_CURFRAME(actor[i].t_data) : 0)
// this is the same crap as in game.c's tspr manipulation. puke.
// XXX: may access tilesizy out-of-bounds by bad user code.
#define LIGHTRAD(spriteNum, s) (s->yrepeat * tilesiz[s->picnum + LIGHTRAD_PICOFS(spriteNum)].y)
#define LIGHTRAD2(spriteNum, s) ((s->yrepeat + ((rand() % s->yrepeat)>>2)) * tilesiz[s->picnum + LIGHTRAD_PICOFS(spriteNum)].y)
void G_AddGameLight(int lightRadius, int spriteNum, int zOffset, int lightRange, int lightColor, int lightPrio)
{
#ifdef POLYMER
spritetype *s = &sprite[spriteNum];
if (videoGetRenderMode() != REND_POLYMER || pr_lighting != 1)
return;
if (actor[spriteNum].lightptr == NULL)
{
#pragma pack(push, 1)
_prlight mylight;
#pragma pack(pop)
Bmemset(&mylight, 0, sizeof(mylight));
mylight.sector = s->sectnum;
mylight.x = s->x;
mylight.y = s->y;
mylight.z = s->z - zOffset;
mylight.color[0] = lightColor & 255;
mylight.color[1] = (lightColor >> 8) & 255;
mylight.color[2] = (lightColor >> 16) & 255;
mylight.radius = lightRadius;
actor[spriteNum].lightmaxrange = mylight.range = lightRange;
mylight.priority = lightPrio;
mylight.tilenum = 0;
mylight.publicflags.emitshadow = 1;
mylight.publicflags.negative = 0;
actor[spriteNum].lightId = polymer_addlight(&mylight);
if (actor[spriteNum].lightId >= 0)
actor[spriteNum].lightptr = &prlights[actor[spriteNum].lightId];
return;
}
s->z -= zOffset;
if (lightRange<actor[spriteNum].lightmaxrange>> 1)
actor[spriteNum].lightmaxrange = 0;
if (lightRange > actor[spriteNum].lightmaxrange || lightPrio != actor[spriteNum].lightptr->priority ||
Bmemcmp(&sprite[spriteNum], actor[spriteNum].lightptr, sizeof(int32_t) * 3))
{
if (lightRange > actor[spriteNum].lightmaxrange)
actor[spriteNum].lightmaxrange = lightRange;
Bmemcpy(actor[spriteNum].lightptr, &sprite[spriteNum], sizeof(int32_t) * 3);
actor[spriteNum].lightptr->sector = s->sectnum;
actor[spriteNum].lightptr->flags.invalidate = 1;
}
actor[spriteNum].lightptr->priority = lightPrio;
actor[spriteNum].lightptr->range = lightRange;
actor[spriteNum].lightptr->color[0] = lightColor & 255;
actor[spriteNum].lightptr->color[1] = (lightColor >> 8) & 255;
actor[spriteNum].lightptr->color[2] = (lightColor >> 16) & 255;
s->z += zOffset;
#else
UNREFERENCED_PARAMETER(lightRadius);
UNREFERENCED_PARAMETER(spriteNum);
UNREFERENCED_PARAMETER(zOffset);
UNREFERENCED_PARAMETER(lightRange);
UNREFERENCED_PARAMETER(lightColor);
UNREFERENCED_PARAMETER(lightPrio);
#endif
}
int g_canSeePlayer = 0;
ACTOR_STATIC int G_WakeUp(spritetype *const pSprite, int const playerNum)
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (!pPlayer->make_noise)
return 0;
int const radius = pPlayer->noise_radius;
if (pSprite->pal == 30 || pSprite->pal == 32 || pSprite->pal == 33 || (RRRA && pSprite->pal == 8))
return 0;
return (pPlayer->noise_x - radius < pSprite->x && pPlayer->noise_x + radius > pSprite->x
&& pPlayer->noise_y - radius < pSprite->y && pPlayer->noise_y + radius > pSprite->y);
}
// sleeping monsters, etc
ACTOR_STATIC void G_MoveZombieActors(void)
{
int spriteNum = headspritestat[STAT_ZOMBIEACTOR], canSeePlayer;
if (RR)
canSeePlayer = g_canSeePlayer;
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
int32_t playerDist;
spritetype *const pSprite = &sprite[spriteNum];
int const playerNum = A_FindPlayer(pSprite, &playerDist);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (sprite[pPlayer->i].extra > 0)
{
if (playerDist < 30000)
{
actor[spriteNum].timetosleep++;
if (actor[spriteNum].timetosleep >= (playerDist>>8))
{
if (A_CheckEnemySprite(pSprite))
{
vec3_t p = { pPlayer->opos.x + 64 - (krand2() & 127),
pPlayer->opos.y + 64 - (krand2() & 127),
0 };
int16_t pSectnum = pPlayer->cursectnum;
updatesector(p.x, p.y, &pSectnum);
if (pSectnum == -1)
{
spriteNum = nextSprite;
continue;
}
vec3_t s = { pSprite->x + 64 - (krand2() & 127),
pSprite->y + 64 - (krand2() & 127),
0 };
int16_t sectNum = pSprite->sectnum;
updatesector(s.x, s.y, &sectNum);
//if (sectNum == -1)
//{
// spriteNum = nextSprite;
// continue;
//}
canSeePlayer = 0;
if (!RR || pSprite->pal == 33 || A_CheckSpriteFlags(spriteNum, SFLAG_NOCANSEECHECK)
|| (RRRA && pSprite->picnum == MINION && pSprite->pal == 8)
|| (sintable[(pSprite->ang+512)&2047]*(p.x-s.x)+sintable[pSprite->ang&2047]*(p.y-s.y) >= 0))
{
p.z = pPlayer->opos.z - (krand2() % ZOFFSET5);
s.z = pSprite->z - (krand2() % (52 << 8));
canSeePlayer = cansee(s.x, s.y, s.z, pSprite->sectnum, p.x, p.y, p.z, pPlayer->cursectnum);
}
}
else
{
int32_t const r1 = krand2(), r2 = krand2();
canSeePlayer = cansee(pSprite->x, pSprite->y, pSprite->z - ((r2 & 31) << 8), pSprite->sectnum, pPlayer->opos.x,
pPlayer->opos.y, pPlayer->opos.z - ((r1 & 31) << 8), pPlayer->cursectnum);
}
if (canSeePlayer)
{
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case CANWITHSOMETHING2__STATIC:
case CANWITHSOMETHING3__STATIC:
case CANWITHSOMETHING4__STATIC:
case TRIPBOMB__STATIC:
if (RR) goto default_case;
fallthrough__;
case RUBBERCAN__STATIC:
case EXPLODINGBARREL__STATIC:
case WOODENHORSE__STATIC:
case HORSEONSIDE__STATIC:
case CANWITHSOMETHING__STATIC:
case FIREBARREL__STATIC:
case FIREVASE__STATIC:
case NUKEBARREL__STATIC:
case NUKEBARRELDENTED__STATIC:
case NUKEBARRELLEAKED__STATIC:
pSprite->shade = (sector[pSprite->sectnum].ceilingstat & 1)
? sector[pSprite->sectnum].ceilingshade
: sector[pSprite->sectnum].floorshade;
actor[spriteNum].timetosleep = 0;
changespritestat(spriteNum, STAT_STANDABLE);
break;
default:
default_case:
if (A_CheckSpriteFlags(spriteNum, SFLAG_USEACTIVATOR) && sector[sprite[spriteNum].sectnum].lotag & 16384)
break;
actor[spriteNum].timetosleep = 0;
A_PlayAlertSound(spriteNum);
changespritestat(spriteNum, STAT_ACTOR);
break;
}
}
else
actor[spriteNum].timetosleep = 0;
}
}
if ((!RR || !canSeePlayer) && A_CheckEnemySprite(pSprite))
{
pSprite->shade = (sector[pSprite->sectnum].ceilingstat & 1)
? sector[pSprite->sectnum].ceilingshade
: sector[pSprite->sectnum].floorshade;
if (RR && G_WakeUp(pSprite, playerNum))
{
actor[spriteNum].timetosleep = 0;
A_PlayAlertSound(spriteNum);
changespritestat(spriteNum, STAT_ACTOR);
}
}
}
spriteNum = nextSprite;
}
}
// stupid name, but it's what the function does.
static FORCE_INLINE int G_FindExplosionInSector(int const sectNum)
{
if (RR)
{
for (bssize_t SPRITES_OF(STAT_MISC, i))
if (PN(i) == EXPLOSION2 || (PN(i) == EXPLOSION3 && sectNum == SECT(i)))
return i;
}
else
{
for (bssize_t SPRITES_OF(STAT_MISC, i))
if (PN(i) == EXPLOSION2 && sectNum == SECT(i))
return i;
}
return -1;
}
static FORCE_INLINE void P_Nudge(int playerNum, int spriteNum, int shiftLeft)
{
g_player[playerNum].ps->vel.x += actor[spriteNum].extra * (sintable[(actor[spriteNum].ang + 512) & 2047]) << shiftLeft;
g_player[playerNum].ps->vel.y += actor[spriteNum].extra * (sintable[actor[spriteNum].ang & 2047]) << shiftLeft;
}
int A_IncurDamage(int const spriteNum)
{
spritetype *const pSprite = &sprite[spriteNum];
actor_t *const pActor = &actor[spriteNum];
// dmg->picnum check: safety, since it might have been set to <0 from CON.
if (pActor->extra < 0 || pSprite->extra < 0 || pActor->picnum < 0)
{
pActor->extra = -1;
return -1;
}
if (pSprite->picnum == APLAYER)
{
if (ud.god && (RR || pActor->picnum != SHRINKSPARK))
return -1;
int const playerNum = P_GetP(pSprite);
if (pActor->owner >= 0 && (sprite[pActor->owner].picnum == APLAYER))
{
if (
(ud.ffire == 0) &&
((g_gametypeFlags[ud.coop] & GAMETYPE_PLAYERSFRIENDLY) ||
((g_gametypeFlags[ud.coop] & GAMETYPE_TDM) && g_player[playerNum].ps->team == g_player[P_Get(pActor->owner)].ps->team))
)
{
return -1;
}
}
pSprite->extra -= pActor->extra;
if (pActor->owner >= 0 && pSprite->extra <= 0 && pActor->picnum != FREEZEBLAST)
{
int const damageOwner = pActor->owner;
pSprite->extra = 0;
g_player[playerNum].ps->wackedbyactor = damageOwner;
if (sprite[damageOwner].picnum == APLAYER && playerNum != P_Get(damageOwner))
g_player[playerNum].ps->frag_ps = P_Get(damageOwner);
pActor->owner = g_player[playerNum].ps->i;
}
switch (DYNAMICTILEMAP(pActor->picnum))
{
case RPG2__STATICRR:
if (!RRRA) goto default_case;
fallthrough__;
case TRIPBOMBSPRITE__STATIC:
if (!RR) goto default_case;
fallthrough__;
case RADIUSEXPLOSION__STATIC:
case SEENINE__STATIC:
case RPG__STATIC:
case HYDRENT__STATIC:
case HEAVYHBOMB__STATIC:
case OOZFILTER__STATIC:
case EXPLODINGBARREL__STATIC:
P_Nudge(playerNum, spriteNum, 2);
break;
default:
default_case:
P_Nudge(playerNum, spriteNum, 1);
break;
}
pActor->extra = -1;
return pActor->picnum;
}
if (pActor->extra == 0 && (RR || pActor->picnum == SHRINKSPARK) && pSprite->xrepeat < 24)
return -1;
pSprite->extra -= pActor->extra;
if (pSprite->picnum != RECON && pSprite->owner >= 0 && sprite[pSprite->owner].statnum < MAXSTATUS)
pSprite->owner = pActor->owner;
pActor->extra = -1;
return pActor->picnum;
}
void A_MoveCyclers(void)
{
for (bssize_t i=g_cyclerCnt-1; i>=0; i--)
{
int16_t *const pCycler = g_cyclers[i];
int const sectNum = pCycler[0];
int spriteShade = pCycler[2];
int const floorShade = pCycler[3];
int sectorShade = clamp(floorShade + (sintable[pCycler[1] & 2047] >> 10), spriteShade, floorShade);
pCycler[1] += sector[sectNum].extra;
if (pCycler[5]) // angle 1536...
{
walltype *pWall = &wall[sector[sectNum].wallptr];
for (bssize_t wallsLeft = sector[sectNum].wallnum; wallsLeft > 0; wallsLeft--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->shade = sectorShade;
if ((pWall->cstat&2) && pWall->nextwall >= 0)
wall[pWall->nextwall].shade = sectorShade;
}
}
sector[sectNum].floorshade = sector[sectNum].ceilingshade = sectorShade;
}
}
}
void A_MoveDummyPlayers(void)
{
int spriteNum = headspritestat[STAT_DUMMYPLAYER];
while (spriteNum >= 0)
{
int const playerNum = P_Get(OW(spriteNum));
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
int const nextSprite = nextspritestat[spriteNum];
int const playerSectnum = pPlayer->cursectnum;
if ((!RR && pPlayer->on_crane >= 0) || (playerSectnum >= 0 && sector[playerSectnum].lotag != ST_1_ABOVE_WATER) || sprite[pPlayer->i].extra <= 0)
{
pPlayer->dummyplayersprite = -1;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
if (pPlayer->on_ground && pPlayer->on_warping_sector == 1 && playerSectnum >= 0 && sector[playerSectnum].lotag == ST_1_ABOVE_WATER)
{
CS(spriteNum) = 257;
SZ(spriteNum) = sector[SECT(spriteNum)].ceilingz+(27<<8);
SA(spriteNum) = fix16_to_int(pPlayer->q16ang);
if (T1(spriteNum) == 8)
T1(spriteNum) = 0;
else T1(spriteNum)++;
}
else
{
if (sector[SECT(spriteNum)].lotag != ST_2_UNDERWATER) SZ(spriteNum) = sector[SECT(spriteNum)].floorz;
CS(spriteNum) = 32768;
}
}
SX(spriteNum) += (pPlayer->pos.x-pPlayer->opos.x);
SY(spriteNum) += (pPlayer->pos.y-pPlayer->opos.y);
setsprite(spriteNum, (vec3_t *)&sprite[spriteNum]);
next_sprite:
spriteNum = nextSprite;
}
}
static int P_Submerge(int, int, DukePlayer_t *, int, int);
static int P_Emerge(int, int, DukePlayer_t *, int, int);
static void P_FinishWaterChange(int, DukePlayer_t *, int, int, int);
ACTOR_STATIC void G_MovePlayers(void)
{
int spriteNum = headspritestat[STAT_PLAYER];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite = &sprite[spriteNum];
DukePlayer_t *const pPlayer = g_player[P_GetP(pSprite)].ps;
if (pSprite->owner >= 0)
{
if (pPlayer->newowner >= 0) //Looking thru the camera
{
pSprite->x = pPlayer->opos.x;
pSprite->y = pPlayer->opos.y;
pSprite->z = pPlayer->opos.z + PHEIGHT;
actor[spriteNum].bpos.z = pSprite->z;
pSprite->ang = fix16_to_int(pPlayer->oq16ang);
setsprite(spriteNum, (vec3_t *)pSprite);
}
else
{
int32_t otherPlayerDist;
#ifdef YAX_ENABLE
// TROR water submerge/emerge
int const playerSectnum = pSprite->sectnum;
int const sectorLotag = sector[playerSectnum].lotag;
int32_t otherSector;
if (A_CheckNoSE7Water((uspritetype const *)pSprite, playerSectnum, sectorLotag, &otherSector))
{
// NOTE: Compare with G_MoveTransports().
pPlayer->on_warping_sector = 1;
if ((sectorLotag == ST_1_ABOVE_WATER ?
P_Submerge(spriteNum, P_GetP(pSprite), pPlayer, playerSectnum, otherSector) :
P_Emerge(spriteNum, P_GetP(pSprite), pPlayer, playerSectnum, otherSector)) == 1)
P_FinishWaterChange(spriteNum, pPlayer, sectorLotag, -1, otherSector);
}
#endif
if (g_netServer || ud.multimode > 1)
otherp = P_FindOtherPlayer(P_GetP(pSprite), &otherPlayerDist);
else
{
otherp = P_GetP(pSprite);
otherPlayerDist = 0;
}
if (G_HaveActor(sprite[spriteNum].picnum))
A_Execute(spriteNum, P_GetP(pSprite), otherPlayerDist);
if (g_netServer || ud.multimode > 1)
{
if (sprite[g_player[otherp].ps->i].extra > 0)
{
if (pSprite->yrepeat > 32 && sprite[g_player[otherp].ps->i].yrepeat < 32)
{
if (otherPlayerDist < 1400 && pPlayer->knee_incs == 0)
{
pPlayer->knee_incs = 1;
pPlayer->weapon_pos = -1;
pPlayer->actorsqu = g_player[otherp].ps->i;
}
}
}
}
if (ud.god)
{
pSprite->extra = pPlayer->max_player_health;
pSprite->cstat = 257;
if (!RR)
pPlayer->inv_amount[GET_JETPACK] = 1599;
}
if (pSprite->extra > 0)
{
actor[spriteNum].owner = spriteNum;
if (ud.god == 0)
if (G_CheckForSpaceCeiling(pSprite->sectnum) || G_CheckForSpaceFloor(pSprite->sectnum))
P_QuickKill(pPlayer);
}
else
{
pPlayer->pos.x = pSprite->x;
pPlayer->pos.y = pSprite->y;
pPlayer->pos.z = pSprite->z-(20<<8);
pPlayer->newowner = -1;
if (pPlayer->wackedbyactor >= 0 && sprite[pPlayer->wackedbyactor].statnum < MAXSTATUS)
{
pPlayer->q16ang += fix16_to_int(G_GetAngleDelta(pPlayer->q16ang,
getangle(sprite[pPlayer->wackedbyactor].x - pPlayer->pos.x,
sprite[pPlayer->wackedbyactor].y - pPlayer->pos.y))
>> 1);
pPlayer->q16ang &= 0x7FFFFFF;
}
}
pSprite->ang = fix16_to_int(pPlayer->q16ang);
}
}
else
{
if (pPlayer->holoduke_on == -1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
Bmemcpy(&actor[spriteNum].bpos, pSprite, sizeof(vec3_t));
pSprite->cstat = 0;
if (pSprite->xrepeat < 42)
{
pSprite->xrepeat += 4;
pSprite->cstat |= 2;
}
else pSprite->xrepeat = 42;
if (pSprite->yrepeat < 36)
pSprite->yrepeat += 4;
else
{
pSprite->yrepeat = 36;
if (sector[pSprite->sectnum].lotag != ST_2_UNDERWATER)
A_Fall(spriteNum);
if (pSprite->zvel == 0 && sector[pSprite->sectnum].lotag == ST_1_ABOVE_WATER)
pSprite->z += ZOFFSET5;
}
if (pSprite->extra < 8)
{
pSprite->xvel = 128;
pSprite->ang = fix16_to_int(pPlayer->q16ang);
pSprite->extra++;
A_SetSprite(spriteNum,CLIPMASK0);
}
else
{
pSprite->ang = 2047-fix16_to_int(pPlayer->q16ang);
setsprite(spriteNum,(vec3_t *)pSprite);
}
}
pSprite->shade =
logapproach(pSprite->shade, (sector[pSprite->sectnum].ceilingstat & 1) ? sector[pSprite->sectnum].ceilingshade
: sector[pSprite->sectnum].floorshade);
next_sprite:
spriteNum = nextSprite;
}
}
ACTOR_STATIC void G_MoveFX(void)
{
int spriteNum = headspritestat[STAT_FX];
while (spriteNum >= 0)
{
spritetype *const pSprite = &sprite[spriteNum];
int const nextSprite = nextspritestat[spriteNum];
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case RESPAWN__STATIC:
if (pSprite->extra == 66)
{
int32_t j = A_Spawn(spriteNum,SHT(spriteNum));
if (RRRA)
{
sprite[j].pal = pSprite->pal;
if (sprite[j].picnum == MAMA)
{
switch (sprite[j].pal)
{
case 30:
sprite[j].xrepeat = sprite[j].yrepeat = 26;
sprite[j].clipdist = 75;
break;
case 31:
sprite[j].xrepeat = sprite[j].yrepeat = 36;
sprite[j].clipdist = 100;
break;
default:
sprite[j].xrepeat = sprite[j].yrepeat = 50;
sprite[j].clipdist = 100;
break;
}
}
if (sprite[j].pal == 8)
sprite[j].cstat |= 2;
else if (sprite[j].pal == 6)
{
pSprite->extra = 66-13;
sprite[j].pal = 0;
break;
}
}
// sprite[j].pal = sprite[i].pal;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if (pSprite->extra > (66-13))
sprite[spriteNum].extra++;
break;
case MUSICANDSFX__STATIC:
{
int32_t const spriteHitag = (uint16_t)pSprite->hitag;
DukePlayer_t *const pPlayer = g_player[screenpeek].ps;
if (T2(spriteNum) != (int)SoundEnabled())
{
// If sound playback was toggled, restart.
T2(spriteNum) = SoundEnabled();
T1(spriteNum) = 0;
}
if (pSprite->lotag >= 1000 && pSprite->lotag < 2000)
{
int32_t playerDist = ldist(&sprite[pPlayer->i], pSprite);
#ifdef SPLITSCREEN_MOD_HACKS
if (g_fakeMultiMode==2)
{
// HACK for splitscreen mod
int32_t otherdist = ldist(&sprite[g_player[1].ps->i],pSprite);
playerDist = min(playerDist, otherdist);
}
#endif
if (playerDist < spriteHitag && T1(spriteNum) == 0)
{
FX_SetReverb(pSprite->lotag - 1000);
T1(spriteNum) = 1;
}
else if (playerDist >= spriteHitag && T1(spriteNum) == 1)
{
FX_SetReverb(0);
T1(spriteNum) = 0;
}
}
else if (pSprite->lotag < 999 && (unsigned)sector[pSprite->sectnum].lotag < 9 && // ST_9_SLIDING_ST_DOOR
snd_ambience && sector[SECT(spriteNum)].floorz != sector[SECT(spriteNum)].ceilingz)
{
auto flags = S_GetUserFlags(pSprite->lotag);
if (flags & SF_MSFX)
{
int playerDist = dist(&sprite[pPlayer->i], pSprite);
#ifdef SPLITSCREEN_MOD_HACKS
if (g_fakeMultiMode==2)
{
// HACK for splitscreen mod
int32_t otherdist = dist(&sprite[g_player[1].ps->i],pSprite);
playerDist = min(playerDist, otherdist);
}
#endif
if (playerDist < spriteHitag && T1(spriteNum) == 0)// && FX_VoiceAvailable(g_sounds[pSprite->lotag].pr-1))
{
// Start playing an ambience sound.
#if 0 // let the sound system handle this internally.
if (g_numEnvSoundsPlaying == snd_numvoices)
{
int32_t j;
for (SPRITES_OF(STAT_FX, j))
if (j != spriteNum && S_IsAmbientSFX(j) && actor[j].t_data[0] == 1 &&
dist(&sprite[j], &sprite[pPlayer->i]) > playerDist)
{
S_StopEnvSound(sprite[j].lotag,j);
break;
}
if (j == -1)
goto next_sprite;
}
#endif
A_PlaySound(pSprite->lotag, spriteNum, CHAN_AUTO, CHANF_LOOP);
T1(spriteNum) = 1; // AMBIENT_SFX_PLAYING
}
else if (playerDist >= spriteHitag && T1(spriteNum) == 1)
{
// Stop playing ambience sound because we're out of its range.
// T1 will be reset in sounds.c: CLEAR_SOUND_T0
// T1 = 0;
S_StopEnvSound(pSprite->lotag,spriteNum);
}
}
if ((flags & (SF_GLOBAL | SF_DTAG)) == SF_GLOBAL)
{
// Randomly playing global sounds (flyby of planes, screams, ...)
if (T5(spriteNum) > 0)
T5(spriteNum)--;
else
{
for (int TRAVERSE_CONNECT(playerNum))
if (playerNum == myconnectindex && g_player[playerNum].ps->cursectnum == pSprite->sectnum)
{
S_PlaySound(pSprite->lotag + (unsigned)g_globalRandom % (pSprite->hitag+1));
T5(spriteNum) = GAMETICSPERSEC*40 + g_globalRandom%(GAMETICSPERSEC*40);
}
}
}
}
break;
}
}
next_sprite:
spriteNum = nextSprite;
}
}
ACTOR_STATIC void G_MoveFallers(void)
{
int spriteNum = headspritestat[STAT_FALLER];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite = &sprite[spriteNum];
int const sectNum = pSprite->sectnum;
if (T1(spriteNum) == 0)
{
const int16_t oextra = pSprite->extra;
int j;
pSprite->z -= ZOFFSET2;
T2(spriteNum) = pSprite->ang;
if ((j = A_IncurDamage(spriteNum)) >= 0)
{
if ((!RR && j == FIREEXT) || j == RPG || (RRRA && j == RPG2) || j == RADIUSEXPLOSION || j == SEENINE || j == OOZFILTER)
{
if (pSprite->extra <= 0)
{
T1(spriteNum) = 1;
for (bssize_t SPRITES_OF(STAT_FALLER, j))
{
if (sprite[j].hitag == SHT(spriteNum))
{
actor[j].t_data[0] = 1;
sprite[j].cstat &= (65535-64);
if (sprite[j].picnum == CEILINGSTEAM || sprite[j].picnum == STEAM)
sprite[j].cstat |= 32768;
}
}
}
}
else
{
actor[spriteNum].extra = 0;
pSprite->extra = oextra;
}
}
pSprite->ang = T2(spriteNum);
pSprite->z += ZOFFSET2;
}
else if (T1(spriteNum) == 1)
{
if ((int16_t)pSprite->lotag > 0)
{
pSprite->lotag-=3;
if (RR)
{
pSprite->xvel = (64+krand2())&127;
pSprite->zvel = -(1024+(krand2()&1023));
}
else if ((int16_t)pSprite->lotag <= 0)
{
pSprite->xvel = (32+krand2())&63;
pSprite->zvel = -(1024+(krand2()&1023));
}
}
else
{
int32_t spriteGravity = g_spriteGravity;
if (pSprite->xvel > 0)
{
pSprite->xvel -= RR ? 2 : 8;
A_SetSprite(spriteNum,CLIPMASK0);
}
if (EDUKE32_PREDICT_FALSE(G_CheckForSpaceFloor(pSprite->sectnum)))
spriteGravity = 0;
else if (EDUKE32_PREDICT_FALSE(G_CheckForSpaceCeiling(pSprite->sectnum)))
spriteGravity = g_spriteGravity / 6;
if (pSprite->z < (sector[sectNum].floorz-ZOFFSET))
{
pSprite->zvel += spriteGravity;
if (pSprite->zvel > 6144)
pSprite->zvel = 6144;
pSprite->z += pSprite->zvel;
}
if ((sector[sectNum].floorz-pSprite->z) < ZOFFSET2)
{
for (size_t x = 0, x_end = 1+(krand2()&7); x < x_end; ++x)
RANDOMSCRAP(pSprite, spriteNum);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
}
next_sprite:
spriteNum = nextSprite;
}
}
ACTOR_STATIC void G_MoveStandables(void)
{
int spriteNum = headspritestat[STAT_STANDABLE], j, switchPic;
while (spriteNum >= 0)
{
const int nextSprite = nextspritestat[spriteNum];
int32_t *const pData = &actor[spriteNum].t_data[0];
spritetype *const pSprite = &sprite[spriteNum];
const int sectNum = pSprite->sectnum;
if (sectNum < 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
Bmemcpy(&actor[spriteNum].bpos, pSprite, sizeof(vec3_t));
if (PN(spriteNum) >= CRANE && PN(spriteNum) <= CRANE+3)
{
int32_t nextj;
//t[0] = state
//t[1] = checking sector number
if (pSprite->xvel) A_GetZLimits(spriteNum);
if (pData[0] == 0) //Waiting to check the sector
{
for (SPRITES_OF_SECT_SAFE(pData[1], j, nextj))
{
switch (sprite[j].statnum)
{
case STAT_ACTOR:
case STAT_ZOMBIEACTOR:
case STAT_STANDABLE:
case STAT_PLAYER:
{
vec3_t vect = { g_origins[pData[4]+1].x, g_origins[pData[4]+1].y, sprite[j].z };
pSprite->ang = getangle(vect.x-pSprite->x, vect.y-pSprite->y);
setsprite(j, &vect);
pData[0]++;
goto next_sprite;
}
}
}
}
else if (pData[0]==1)
{
if (pSprite->xvel < 184)
{
pSprite->picnum = CRANE+1;
pSprite->xvel += 8;
}
A_SetSprite(spriteNum,CLIPMASK0);
if (sectNum == pData[1])
pData[0]++;
}
else if (pData[0]==2 || pData[0]==7)
{
pSprite->z += (1024+512);
if (pData[0]==2)
{
if (sector[sectNum].floorz - pSprite->z < (64<<8))
if (pSprite->picnum > CRANE) pSprite->picnum--;
if (sector[sectNum].floorz - pSprite->z < 4096+1024)
pData[0]++;
}
if (pData[0]==7)
{
if (sector[sectNum].floorz - pSprite->z < (64<<8))
{
if (pSprite->picnum > CRANE) pSprite->picnum--;
else
{
if (pSprite->owner==-2)
{
int32_t p = A_FindPlayer(pSprite, NULL);
A_PlaySound(RR ? 390 : DUKE_GRUNT,g_player[p].ps->i);
if (g_player[p].ps->on_crane == spriteNum)
g_player[p].ps->on_crane = -1;
}
pData[0]++;
pSprite->owner = -1;
}
}
}
}
else if (pData[0]==3)
{
pSprite->picnum++;
if (pSprite->picnum == CRANE+2)
{
int32_t p = G_CheckPlayerInSector(pData[1]);
if (p >= 0 && g_player[p].ps->on_ground)
{
pSprite->owner = -2;
g_player[p].ps->on_crane = spriteNum;
A_PlaySound(RR ? 390 : DUKE_GRUNT,g_player[p].ps->i);
g_player[p].ps->q16ang = fix16_from_int(pSprite->ang+1024);
}
else
{
for (SPRITES_OF_SECT(pData[1], j))
{
switch (sprite[j].statnum)
{
case STAT_ACTOR:
case STAT_STANDABLE:
pSprite->owner = j;
break;
}
}
}
pData[0]++;//Grabbed the sprite
pData[2]=0;
goto next_sprite;
}
}
else if (pData[0]==4) //Delay before going up
{
pData[2]++;
if (pData[2] > 10)
pData[0]++;
}
else if (pData[0]==5 || pData[0] == 8)
{
if (pData[0]==8 && pSprite->picnum < (CRANE+2))
if ((sector[sectNum].floorz-pSprite->z) > 8192)
pSprite->picnum++;
if (pSprite->z < g_origins[pData[4]+2].x)
{
pData[0]++;
pSprite->xvel = 0;
}
else
pSprite->z -= (1024+512);
}
else if (pData[0]==6)
{
if (pSprite->xvel < 192)
pSprite->xvel += 8;
pSprite->ang = getangle(g_origins[pData[4]].x - pSprite->x, g_origins[pData[4]].y - pSprite->y);
A_SetSprite(spriteNum,CLIPMASK0);
if (((pSprite->x-g_origins[pData[4]].x)*(pSprite->x-g_origins[pData[4]].x)+(pSprite->y-g_origins[pData[4]].y)*(pSprite->y-g_origins[pData[4]].y)) < (128*128))
pData[0]++;
}
else if (pData[0]==9)
pData[0] = 0;
{
vec3_t vect;
Bmemcpy(&vect,pSprite,sizeof(vec3_t));
vect.z -= (34<<8);
setsprite(g_origins[pData[4]+2].y, &vect);
}
if (pSprite->owner != -1)
{
int32_t p = A_FindPlayer(pSprite, NULL);
if (A_IncurDamage(spriteNum) >= 0)
{
if (pSprite->owner == -2)
if (g_player[p].ps->on_crane == spriteNum)
g_player[p].ps->on_crane = -1;
pSprite->owner = -1;
pSprite->picnum = CRANE;
goto next_sprite;
}
if (pSprite->owner >= 0)
{
setsprite(pSprite->owner,(vec3_t *)pSprite);
Bmemcpy(&actor[pSprite->owner].bpos, pSprite, sizeof(vec3_t));
pSprite->zvel = 0;
}
else if (pSprite->owner == -2)
{
DukePlayer_t *const ps = g_player[p].ps;
ps->opos.x = ps->pos.x = pSprite->x-(sintable[(fix16_to_int(ps->q16ang)+512)&2047]>>6);
ps->opos.y = ps->pos.y = pSprite->y-(sintable[fix16_to_int(ps->q16ang)&2047]>>6);
ps->opos.z = ps->pos.z = pSprite->z+(2<<8);
setsprite(ps->i, (vec3_t *)ps);
ps->cursectnum = sprite[ps->i].sectnum;
}
}
goto next_sprite;
}
else if (PN(spriteNum) >= WATERFOUNTAIN && PN(spriteNum) <= WATERFOUNTAIN+3)
{
if (pData[0] > 0)
{
if (pData[0] < 20)
{
pData[0]++;
pSprite->picnum++;
if (pSprite->picnum == (WATERFOUNTAIN+3))
pSprite->picnum = WATERFOUNTAIN+1;
}
else
{
int32_t playerDist;
A_FindPlayer(pSprite,&playerDist);
if (playerDist > 512)
{
pData[0] = 0;
pSprite->picnum = WATERFOUNTAIN;
}
else pData[0] = 1;
}
}
goto next_sprite;
}
else if (AFLAMABLE(pSprite->picnum))
{
if (T1(spriteNum) == 1)
{
if ((++T2(spriteNum)&3) > 0) goto next_sprite;
if (pSprite->picnum == TIRE && T2(spriteNum) == 32)
{
pSprite->cstat = 0;
j = A_Spawn(spriteNum,BLOODPOOL);
sprite[j].shade = 127;
}
else
{
if (pSprite->shade < 64) pSprite->shade++;
else DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
j = pSprite->xrepeat-(krand2()&7);
if (j < 10)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
pSprite->xrepeat = j;
j = pSprite->yrepeat-(krand2()&7);
if (j < 4)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
pSprite->yrepeat = j;
}
if (!RR && pSprite->picnum == BOX)
{
A_Fall(spriteNum);
actor[spriteNum].ceilingz = sector[pSprite->sectnum].ceilingz;
}
goto next_sprite;
}
else if (!RR && pSprite->picnum == TRIPBOMB)
{
if (T3(spriteNum) > 0)
{
T3(spriteNum)--;
if (T3(spriteNum) == 8)
{
for (j=0; j<5; j++)
RANDOMSCRAP(pSprite, spriteNum);
int const dmg = pSprite->extra;
A_RadiusDamage(spriteNum, g_tripbombRadius, dmg>>2, dmg>>1, dmg-(dmg>>2), dmg);
j = A_Spawn(spriteNum,EXPLOSION2);
A_PlaySound(LASERTRIP_EXPLODE,j);
sprite[j].ang = pSprite->ang;
sprite[j].xvel = 348;
A_SetSprite(j,CLIPMASK0);
for (SPRITES_OF(STAT_MISC, j))
{
if (sprite[j].picnum == LASERLINE && pSprite->hitag == sprite[j].hitag)
sprite[j].xrepeat = sprite[j].yrepeat = 0;
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
}
else
{
int const oldExtra = pSprite->extra;
int const oldAng = pSprite->ang;
pSprite->extra = 1;
if (A_IncurDamage(spriteNum) >= 0)
{
T3(spriteNum) = 16;
}
pSprite->extra = oldExtra;
pSprite->ang = oldAng;
}
if (T1(spriteNum) < 32)
{
int32_t playerDist;
A_FindPlayer(pSprite, &playerDist);
if (playerDist > 768 || T1(spriteNum) > 16) T1(spriteNum)++;
}
if (T1(spriteNum) == 32)
{
int16_t hitSprite;
int const oldAng = pSprite->ang;
pSprite->ang = T6(spriteNum);
T4(spriteNum) = pSprite->x;
T5(spriteNum) = pSprite->y;
pSprite->x += sintable[(T6(spriteNum)+512)&2047]>>9;
pSprite->y += sintable[(T6(spriteNum))&2047]>>9;
pSprite->z -= (3<<8);
setsprite(spriteNum,(vec3_t *)pSprite);
int hitDist = A_CheckHitSprite(spriteNum, &hitSprite);
actor[spriteNum].lastv.x = hitDist;
pSprite->ang = oldAng;
// we're on a trip wire
//int16_t cursectnum;
while (hitDist > 0)
{
j = A_Spawn(spriteNum,LASERLINE);
setsprite(j,(vec3_t *)&sprite[j]);
sprite[j].hitag = pSprite->hitag;
actor[j].t_data[1] = sprite[j].z;
pSprite->x += sintable[(T6(spriteNum)+512)&2047]>>4;
pSprite->y += sintable[(T6(spriteNum))&2047]>>4;
if (hitDist < 1024)
{
sprite[j].xrepeat = hitDist>>5;
break;
}
hitDist -= 1024;
//cursectnum = pSprite->sectnum;
//updatesector(pSprite->x, pSprite->y, &cursectnum);
//if (cursectnum < 0)
// break;
}
T1(spriteNum)++;
pSprite->x = T4(spriteNum);
pSprite->y = T5(spriteNum);
pSprite->z += (3<<8);
setsprite(spriteNum,(vec3_t *)pSprite);
T4(spriteNum) = T3(spriteNum) = 0;
if (hitSprite >= 0)
{
T3(spriteNum) = 13;
A_PlaySound(LASERTRIP_ARMING,spriteNum);
}
}
if (T1(spriteNum) == 33)
{
T2(spriteNum)++;
T4(spriteNum) = pSprite->x;
T5(spriteNum) = pSprite->y;
pSprite->x += sintable[(T6(spriteNum)+512)&2047]>>9;
pSprite->y += sintable[(T6(spriteNum))&2047]>>9;
pSprite->z -= (3<<8);
setsprite(spriteNum, (vec3_t *) pSprite);
int hitDist = A_CheckHitSprite(spriteNum, NULL);
pSprite->x = T4(spriteNum);
pSprite->y = T5(spriteNum);
pSprite->z += (3<<8);
setsprite(spriteNum, (vec3_t *) pSprite);
// if( Actor[i].lastvx != x && lTripBombControl & TRIPBOMB_TRIPWIRE)
if (actor[spriteNum].lastv.x != hitDist)
{
T3(spriteNum) = 13;
A_PlaySound(LASERTRIP_ARMING, spriteNum);
}
}
goto next_sprite;
}
else if (pSprite->picnum >= CRACK1 && pSprite->picnum <= CRACK4)
{
if (pSprite->hitag)
{
pData[0] = pSprite->cstat;
pData[1] = pSprite->ang;
int const dmgTile = A_IncurDamage(spriteNum);
if (dmgTile < 0)
goto crack_default;
switch (DYNAMICTILEMAP(dmgTile))
{
case RPG2__STATICRR:
if (!RRRA) goto crack_default;
fallthrough__;
case FIREEXT__STATIC:
if (RR) goto crack_default;
fallthrough__;
case RPG__STATIC:
case RADIUSEXPLOSION__STATIC:
case SEENINE__STATIC:
case OOZFILTER__STATIC:
for (SPRITES_OF(STAT_STANDABLE, j))
{
if (pSprite->hitag == sprite[j].hitag &&
(sprite[j].picnum == OOZFILTER || sprite[j].picnum == SEENINE))
if (sprite[j].shade != -32)
sprite[j].shade = -32;
}
goto DETONATE;
crack_default:
default:
pSprite->cstat = pData[0];
pSprite->ang = pData[1];
pSprite->extra = 0;
goto next_sprite;
}
}
goto next_sprite;
}
else if (!RR && pSprite->picnum == FIREEXT)
{
if (A_IncurDamage(spriteNum) < 0)
goto next_sprite;
for (bsize_t k=0; k<16; k++)
{
int32_t const r1 = krand2(), r2 = krand2(), r3 = krand2(), r4 = krand2(), r5 = krand2();
j = A_InsertSprite(SECT(spriteNum), SX(spriteNum), SY(spriteNum), SZ(spriteNum) - (r5 % (48 << 8)),
SCRAP3 + (r4 & 3), -8, 48, 48, r3 & 2047, (r2 & 63) + 64,
-(r1 & 4095) - (sprite[spriteNum].zvel >> 2), spriteNum, 5);
sprite[j].pal = 2;
}
j = A_Spawn(spriteNum,EXPLOSION2);
A_PlaySound(PIPEBOMB_EXPLODE,j);
A_PlaySound(GLASS_HEAVYBREAK,j);
if ((int16_t)pSprite->hitag > 0)
{
for (SPRITES_OF(STAT_STANDABLE, j))
{
// XXX: This block seems to be CODEDUP'd a lot of times.
if (pSprite->hitag == sprite[j].hitag && (sprite[j].picnum == OOZFILTER || sprite[j].picnum == SEENINE))
if (sprite[j].shade != -32)
sprite[j].shade = -32;
}
int const dmg = pSprite->extra;
A_RadiusDamage(spriteNum, g_pipebombRadius,dmg>>2, dmg-(dmg>>1),dmg-(dmg>>2), dmg);
j = A_Spawn(spriteNum,EXPLOSION2);
A_PlaySound(PIPEBOMB_EXPLODE,j);
goto DETONATE;
}
else
{
A_RadiusDamage(spriteNum,g_seenineRadius,10,15,20,25);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
}
else if (pSprite->picnum == OOZFILTER || pSprite->picnum == SEENINE || pSprite->picnum == SEENINEDEAD || pSprite->picnum == SEENINEDEAD+1)
{
if (pSprite->shade != -32 && pSprite->shade != -33)
{
if (pSprite->xrepeat)
j = (A_IncurDamage(spriteNum) >= 0);
else
j = 0;
if (j || pSprite->shade == -31)
{
if (j) pSprite->lotag = 0;
pData[3] = 1;
for (SPRITES_OF(STAT_STANDABLE, j))
{
if (pSprite->hitag == sprite[j].hitag && (sprite[j].picnum == SEENINE || sprite[j].picnum == OOZFILTER))
sprite[j].shade = -32;
}
}
}
else
{
if (pSprite->shade == -32)
{
if ((int16_t)pSprite->lotag > 0)
{
pSprite->lotag -= 3;
if ((int16_t)pSprite->lotag <= 0)
pSprite->lotag = -99;
}
else
pSprite->shade = -33;
}
else
{
if (pSprite->xrepeat > 0)
{
T3(spriteNum)++;
if (T3(spriteNum) == 3)
{
if (pSprite->picnum == OOZFILTER)
{
T3(spriteNum) = 0;
goto DETONATE;
}
if (pSprite->picnum != (SEENINEDEAD+1))
{
T3(spriteNum) = 0;
if (pSprite->picnum == SEENINEDEAD)
pSprite->picnum++;
else if (pSprite->picnum == SEENINE)
pSprite->picnum = SEENINEDEAD;
}
else goto DETONATE;
}
goto next_sprite;
}
DETONATE:
g_earthquakeTime = 16;
for (SPRITES_OF(STAT_EFFECTOR, j))
{
if (pSprite->hitag == sprite[j].hitag)
{
if (sprite[j].lotag == SE_13_EXPLOSIVE)
{
if (actor[j].t_data[2] == 0)
actor[j].t_data[2] = 1;
}
else if (sprite[j].lotag == SE_8_UP_OPEN_DOOR_LIGHTS)
actor[j].t_data[4] = 1;
else if (sprite[j].lotag == SE_18_INCREMENTAL_SECTOR_RISE_FALL)
{
if (actor[j].t_data[0] == 0)
actor[j].t_data[0] = 1;
}
else if (sprite[j].lotag == SE_21_DROP_FLOOR)
actor[j].t_data[0] = 1;
}
}
pSprite->z -= ZOFFSET5;
if ((pData[3] == 1 && pSprite->xrepeat) || (int16_t)pSprite->lotag == -99)
{
int const newSprite = A_Spawn(spriteNum,EXPLOSION2);
int const dmg = pSprite->extra;
A_RadiusDamage(spriteNum,g_seenineRadius,dmg>>2, dmg-(dmg>>1),dmg-(dmg>>2), dmg);
A_PlaySound(PIPEBOMB_EXPLODE, newSprite);
}
if (pSprite->xrepeat)
for (bsize_t x=0; x<8; x++)
RANDOMSCRAP(pSprite, spriteNum);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
goto next_sprite;
}
else if (pSprite->picnum == MASTERSWITCH)
{
if (pSprite->yvel == 1)
{
if ((int16_t)--pSprite->hitag <= 0)
{
G_OperateSectors(sectNum,spriteNum);
for (SPRITES_OF_SECT(sectNum, j))
{
if (sprite[j].statnum == STAT_EFFECTOR)
{
switch (sprite[j].lotag)
{
case SE_2_EARTHQUAKE:
case SE_21_DROP_FLOOR:
case SE_31_FLOOR_RISE_FALL:
case SE_32_CEILING_RISE_FALL:
case SE_36_PROJ_SHOOTER:
actor[j].t_data[0] = 1;
break;
case SE_3_RANDOM_LIGHTS_AFTER_SHOT_OUT:
actor[j].t_data[4] = 1;
break;
}
}
else if (sprite[j].statnum == STAT_STANDABLE)
{
switch (DYNAMICTILEMAP(sprite[j].picnum))
{
case SEENINE__STATIC:
case OOZFILTER__STATIC:
sprite[j].shade = -31;
break;
}
}
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
goto next_sprite;
}
else
{
switchPic = pSprite->picnum;
if (!RR && switchPic > SIDEBOLT1 && switchPic <= SIDEBOLT1 + 3)
switchPic = SIDEBOLT1;
else if (switchPic > BOLT1 && switchPic <= BOLT1 + 3)
switchPic = BOLT1;
switch (DYNAMICTILEMAP(switchPic))
{
case VIEWSCREEN__STATIC:
case VIEWSCREEN2__STATIC:
if (RR) goto next_sprite;
if (pSprite->xrepeat == 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
{
int32_t playerDist;
int const p = A_FindPlayer(pSprite, &playerDist);
const DukePlayer_t *const ps = g_player[p].ps;
if (dist(&sprite[ps->i], pSprite) < VIEWSCREEN_ACTIVE_DISTANCE)
{
#if 0
if (sprite[i].yvel == 1) // VIEWSCREEN_YVEL
g_curViewscreen = i;
#endif
}
else if (g_curViewscreen == spriteNum /*&& T1 == 1*/)
{
g_curViewscreen = -1;
sprite[spriteNum].yvel = 0; // VIEWSCREEN_YVEL
T1(spriteNum) = 0;
}
}
goto next_sprite;
case TRASH__STATIC:
if (pSprite->xvel == 0)
pSprite->xvel = 1;
if (A_SetSprite(spriteNum, CLIPMASK0))
{
A_Fall(spriteNum);
if (krand2() & 1)
pSprite->zvel -= 256;
if ((pSprite->xvel) < 48)
pSprite->xvel += (krand2() & 3);
}
else
DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
case SIDEBOLT1__STATIC:
// case SIDEBOLT1+1:
// case SIDEBOLT1+2:
// case SIDEBOLT1+3:
{
if (RR) goto next_sprite;
int32_t playerDist;
A_FindPlayer(pSprite, &playerDist);
if (playerDist > 20480)
goto next_sprite;
CLEAR_THE_BOLT2:
if (pData[2])
{
pData[2]--;
goto next_sprite;
}
if ((pSprite->xrepeat | pSprite->yrepeat) == 0)
{
pSprite->xrepeat = pData[0];
pSprite->yrepeat = pData[1];
}
if ((krand2() & 8) == 0)
{
pData[0] = pSprite->xrepeat;
pData[1] = pSprite->yrepeat;
pData[2] = g_globalRandom & 4;
pSprite->xrepeat = pSprite->yrepeat = 0;
goto CLEAR_THE_BOLT2;
}
pSprite->picnum++;
#if 0
// NOTE: Um, this 'l' was assigned to last at the beginning of this function.
// SIDEBOLT1 never gets translucent as a consequence, unlike BOLT1.
if (randomRepeat & 1)
pSprite->cstat ^= 2;
#endif
if ((krand2() & 1) && sector[sectNum].floorpicnum == HURTRAIL)
A_PlaySound(SHORT_CIRCUIT, spriteNum);
if (pSprite->picnum == SIDEBOLT1 + 4)
pSprite->picnum = SIDEBOLT1;
goto next_sprite;
}
case BOLT1__STATIC:
// case BOLT1+1:
// case BOLT1+2:
// case BOLT1+3:
{
int32_t playerDist;
A_FindPlayer(pSprite, &playerDist);
if (playerDist > 20480)
goto next_sprite;
if (pData[3] == 0)
pData[3] = sector[sectNum].floorshade;
CLEAR_THE_BOLT:
if (pData[2])
{
pData[2]--;
sector[sectNum].floorshade = 20;
sector[sectNum].ceilingshade = 20;
goto next_sprite;
}
if ((pSprite->xrepeat | pSprite->yrepeat) == 0)
{
pSprite->xrepeat = pData[0];
pSprite->yrepeat = pData[1];
}
else if ((krand2() & 8) == 0)
{
pData[0] = pSprite->xrepeat;
pData[1] = pSprite->yrepeat;
pData[2] = g_globalRandom & 4;
pSprite->xrepeat = pSprite->yrepeat = 0;
goto CLEAR_THE_BOLT;
}
pSprite->picnum++;
int const randomRepeat = g_globalRandom & 7;
pSprite->xrepeat = randomRepeat + 8;
if (randomRepeat & 1)
pSprite->cstat ^= 2;
if (pSprite->picnum == (BOLT1 + 1)
&& (RR ? (krand2() & 1) != 0 : (krand2() & 7) == 0) && sector[sectNum].floorpicnum == HURTRAIL)
A_PlaySound(SHORT_CIRCUIT, spriteNum);
if (pSprite->picnum == BOLT1 + 4)
pSprite->picnum = BOLT1;
if (pSprite->picnum & 1)
{
sector[sectNum].floorshade = 0;
sector[sectNum].ceilingshade = 0;
}
else
{
sector[sectNum].floorshade = 20;
sector[sectNum].ceilingshade = 20;
}
goto next_sprite;
}
case WATERDRIP__STATIC:
if (pData[1])
{
if (--pData[1] == 0)
pSprite->cstat &= 32767;
}
else
{
A_Fall(spriteNum);
A_SetSprite(spriteNum, CLIPMASK0);
if (pSprite->xvel > 0)
pSprite->xvel -= 2;
if (pSprite->zvel == 0)
{
pSprite->cstat |= 32768;
if (pSprite->pal != 2 && (RR || pSprite->hitag == 0))
A_PlaySound(SOMETHING_DRIPPING, spriteNum);
if (sprite[pSprite->owner].picnum != WATERDRIP)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
actor[spriteNum].bpos.z = pSprite->z = pData[0];
pData[1] = 48 + (krand2() & 31);
}
}
}
goto next_sprite;
case DOORSHOCK__STATIC:
pSprite->yrepeat = (klabs(sector[sectNum].ceilingz - sector[sectNum].floorz) >> 9) + 4;
pSprite->xrepeat = 16;
pSprite->z = sector[sectNum].floorz;
goto next_sprite;
case TOUCHPLATE__STATIC:
if (pData[1] == 1 && (int16_t)pSprite->hitag >= 0) // Move the sector floor
{
int const floorZ = sector[sectNum].floorz;
if (pData[3] == 1)
{
if (floorZ >= pData[2])
{
sector[sectNum].floorz = floorZ;
pData[1] = 0;
}
else
{
sector[sectNum].floorz += sector[sectNum].extra;
int const playerNum = G_CheckPlayerInSector(sectNum);
if (playerNum >= 0)
g_player[playerNum].ps->pos.z += sector[sectNum].extra;
}
}
else
{
if (floorZ <= pSprite->z)
{
sector[sectNum].floorz = pSprite->z;
pData[1] = 0;
}
else
{
int32_t p;
sector[sectNum].floorz -= sector[sectNum].extra;
p = G_CheckPlayerInSector(sectNum);
if (p >= 0)
g_player[p].ps->pos.z -= sector[sectNum].extra;
}
}
goto next_sprite;
}
if (pData[5] == 1)
goto next_sprite;
{
int32_t p = G_CheckPlayerInSector(sectNum);
if (p >= 0 && (g_player[p].ps->on_ground || pSprite->ang == 512))
{
if (pData[0] == 0 && !G_CheckActivatorMotion(pSprite->lotag))
{
pData[0] = 1;
pData[1] = 1;
pData[3] = !pData[3];
G_OperateMasterSwitches(pSprite->lotag);
G_OperateActivators(pSprite->lotag, p);
if ((int16_t)pSprite->hitag > 0)
{
pSprite->hitag--;
if (pSprite->hitag == 0)
pData[5] = 1;
}
}
}
else
pData[0] = 0;
}
if (pData[1] == 1)
{
for (SPRITES_OF(STAT_STANDABLE, j))
{
if (j != spriteNum && sprite[j].picnum == TOUCHPLATE && sprite[j].lotag == pSprite->lotag)
{
actor[j].t_data[1] = 1;
actor[j].t_data[3] = pData[3];
}
}
}
goto next_sprite;
case CANWITHSOMETHING2__STATIC:
case CANWITHSOMETHING3__STATIC:
case CANWITHSOMETHING4__STATIC:
if (RR) goto next_sprite;
fallthrough__;
case CANWITHSOMETHING__STATIC:
A_Fall(spriteNum);
if (A_IncurDamage(spriteNum) >= 0)
{
A_PlaySound(VENT_BUST, spriteNum);
for (j = 9; j >= 0; j--) RANDOMSCRAP(pSprite, spriteNum);
if (pSprite->lotag)
A_Spawn(spriteNum, pSprite->lotag);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
case FLOORFLAME__STATIC:
if (RR) goto next_sprite;
fallthrough__;
case EXPLODINGBARREL__STATIC:
case WOODENHORSE__STATIC:
case HORSEONSIDE__STATIC:
case FIREBARREL__STATIC:
case FIREVASE__STATIC:
case NUKEBARREL__STATIC:
case NUKEBARRELDENTED__STATIC:
case NUKEBARRELLEAKED__STATIC:
case TOILETWATER__STATIC:
case RUBBERCAN__STATIC:
case STEAM__STATIC:
case CEILINGSTEAM__STATIC:
case WATERBUBBLEMAKER__STATIC:
if (!G_HaveActor(sprite[spriteNum].picnum))
goto next_sprite;
{
int32_t playerDist;
int const playerNum = A_FindPlayer(pSprite, &playerDist);
A_Execute(spriteNum, playerNum, playerDist);
}
goto next_sprite;
}
}
next_sprite:
spriteNum = nextSprite;
}
}
ACTOR_STATIC void A_DoProjectileBounce(int const spriteNum)
{
spritetype * const pSprite = &sprite[spriteNum];
int32_t const hitSectnum = pSprite->sectnum;
int const firstWall = sector[hitSectnum].wallptr;
int const secondWall = wall[firstWall].point2;
int const wallAngle = getangle(wall[secondWall].x - wall[firstWall].x, wall[secondWall].y - wall[firstWall].y);
vec3_t vect = { mulscale10(pSprite->xvel, sintable[(pSprite->ang + 512) & 2047]),
mulscale10(pSprite->xvel, sintable[pSprite->ang & 2047]), pSprite->zvel };
int k = (pSprite->z<(actor[spriteNum].floorz + actor[spriteNum].ceilingz)>> 1) ? sector[hitSectnum].ceilingheinum
: sector[hitSectnum].floorheinum;
vec3_t const da = { mulscale14(k, sintable[(wallAngle)&2047]),
mulscale14(k, sintable[(wallAngle + 1536) & 2047]), 4096 };
k = vect.x * da.x + vect.y * da.y + vect.z * da.z;
int l = da.x * da.x + da.y * da.y + da.z * da.z;
if ((klabs(k) >> 14) < l)
{
k = divscale17(k, l);
vect.x -= mulscale16(da.x, k);
vect.y -= mulscale16(da.y, k);
vect.z -= mulscale16(da.z, k);
}
pSprite->zvel = vect.z;
pSprite->xvel = ksqrt(dmulscale8(vect.x, vect.x, vect.y, vect.y));
pSprite->ang = getangle(vect.x, vect.y);
}
ACTOR_STATIC void P_HandleBeingSpitOn(DukePlayer_t * const ps)
{
ps->q16horiz += F16(32);
ps->return_to_center = 8;
if (ps->loogcnt)
return;
if (!A_CheckSoundPlaying(ps->i, DUKE_LONGTERM_PAIN))
A_PlaySound(DUKE_LONGTERM_PAIN,ps->i);
int j = 3+(krand2()&3);
ps->numloogs = j;
ps->loogcnt = 24*4;
for (bssize_t x=0; x < j; x++)
{
ps->loogiex[x] = krand2()%xdim;
ps->loogiey[x] = krand2()%ydim;
}
}
static void G_WeaponHitCeilingOrFloor(int32_t i, spritetype *s, int *j)
{
if (actor[i].flags & SFLAG_DIDNOSE7WATER)
{
actor[i].flags &= ~SFLAG_DIDNOSE7WATER;
return;
}
if (s->z < actor[i].ceilingz)
{
*j = 16384|s->sectnum;
s->zvel = -1;
}
else if (s->z > actor[i].floorz + (RR ? 0 : (ZOFFSET2*(sector[s->sectnum].lotag == ST_1_ABOVE_WATER))))
{
*j = 16384|s->sectnum;
if (sector[s->sectnum].lotag != ST_1_ABOVE_WATER)
s->zvel = 1;
}
}
static void Proj_BounceOffWall(spritetype *s, int j)
{
int k = getangle(
wall[wall[j].point2].x-wall[j].x,
wall[wall[j].point2].y-wall[j].y);
s->ang = ((k<<1) - s->ang)&2047;
}
#define PROJ_DECAYVELOCITY(s) s->xvel >>= 1, s->zvel >>= 1
// Maybe damage a ceiling or floor as the consequence of projectile impact.
// Returns 1 if sprite <s> should be killed.
// NOTE: Compare with Proj_MaybeDamageCF2() in sector.c
static int Proj_MaybeDamageCF(int spriteNum)
{
uspritetype const * const s = (uspritetype const *)&sprite[spriteNum];
if (s->zvel < 0)
{
if ((sector[s->sectnum].ceilingstat&1) && sector[s->sectnum].ceilingpal == 0)
return 1;
Sect_DamageCeiling(s->sectnum);
}
return 0;
}
ACTOR_STATIC void G_MoveWeapons(void)
{
int spriteNum = headspritestat[STAT_PROJECTILE];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite = &sprite[spriteNum];
if (pSprite->sectnum < 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
actor[spriteNum].bpos = *(vec3_t *)pSprite;
// hard coded projectiles
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case KNEE__STATIC:
if (RR) goto next_sprite;
fallthrough__;
case RADIUSEXPLOSION__STATIC: DELETE_SPRITE_AND_CONTINUE(spriteNum);
case TONGUE__STATIC:
T1(spriteNum) = sintable[T2(spriteNum)&2047] >> 9;
T2(spriteNum) += 32;
if (T2(spriteNum) > 2047)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (sprite[pSprite->owner].statnum == MAXSTATUS && !A_CheckEnemySprite(&sprite[pSprite->owner]))
DELETE_SPRITE_AND_CONTINUE(spriteNum);
pSprite->ang = sprite[pSprite->owner].ang;
pSprite->x = sprite[pSprite->owner].x;
pSprite->y = sprite[pSprite->owner].y;
if (sprite[pSprite->owner].picnum == APLAYER)
pSprite->z = sprite[pSprite->owner].z - (34<<8);
for (int i = 0; i < T1(spriteNum); i++) {
int const newSprite = A_InsertSprite(pSprite->sectnum,
pSprite->x+((i*sintable[(pSprite->ang+512)&2047])>>9),
pSprite->y+((i*sintable[pSprite->ang])>>9),
pSprite->z+((i*ksgn(pSprite->zvel))*klabs(pSprite->zvel/12)), TONGUE, -40+i*2,
8, 8, 0, 0, 0, spriteNum, STAT_MISC);
sprite[newSprite].cstat = 48;
sprite[newSprite].pal = 8;
}
{
int const i = max(T1(spriteNum), 0);
int const newSprite = A_InsertSprite(pSprite->sectnum,
pSprite->x+((i*sintable[(pSprite->ang+512)&2047])>>9),
pSprite->y+((i*sintable[pSprite->ang])>>9),
pSprite->z+((i*ksgn(pSprite->zvel))*klabs(pSprite->zvel/12)), INNERJAW, -40,
32, 32, 0, 0, 0, spriteNum, STAT_MISC);
sprite[newSprite].cstat = 128;
if (T2(spriteNum) > 512 && T2(spriteNum) < 1024)
sprite[newSprite].picnum = INNERJAW+1;
}
goto next_sprite;
case FREEZEBLAST__STATIC:
if (pSprite->yvel < 1 || pSprite->extra < 2 || (pSprite->xvel | pSprite->zvel) == 0)
{
int const newSprite = A_Spawn(spriteNum, TRANSPORTERSTAR);
sprite[newSprite].pal = 1;
sprite[newSprite].xrepeat = 32;
sprite[newSprite].yrepeat = 32;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
fallthrough__;
case RPG2__STATICRR:
case RRTILE1790__STATICRR:
if (!RRRA && pSprite->picnum != FREEZEBLAST) goto next_sprite;
fallthrough__;
case SHRINKSPARK__STATIC:
case RPG__STATIC:
case FIRELASER__STATIC:
case SPIT__STATIC:
case COOLEXPLOSION1__STATIC:
case OWHIP__STATICRR:
case UWHIP__STATICRR:
{
if (!RR && pSprite->picnum == COOLEXPLOSION1)
if (!S_CheckSoundPlaying(spriteNum, WIERDSHOT_FLY))
A_PlaySound(WIERDSHOT_FLY, spriteNum);
int spriteXvel = pSprite->xvel;
int spriteZvel = pSprite->zvel;
if ((pSprite->picnum == RPG || (RRRA && pSprite->picnum == RPG2)) && sector[pSprite->sectnum].lotag == ST_2_UNDERWATER)
{
spriteXvel >>= 1;
spriteZvel >>= 1;
}
vec3_t davect = *(vec3_t *) pSprite;
A_GetZLimits(spriteNum);
if (pSprite->picnum == RPG && actor[spriteNum].picnum != BOSS2 && pSprite->xrepeat >= 10
&& sector[pSprite->sectnum].lotag != ST_2_UNDERWATER)
{
int const newSprite = A_Spawn(spriteNum, SMALLSMOKE);
sprite[newSprite].z += (1 << 8);
}
if (RRRA)
{
if (pSprite->picnum == RPG2)
{
pSprite->hitag++;
if (actor[spriteNum].picnum != BOSS2 && pSprite->xrepeat >= 10
&& sector[pSprite->sectnum].lotag != ST_2_UNDERWATER)
{
int const newSprite = A_Spawn(spriteNum, SMALLSMOKE);
sprite[newSprite].z += (1 << 8);
if ((krand2() & 15) == 2)
A_Spawn(spriteNum, MONEY);
}
if (sprite[pSprite->lotag].extra <= 0)
pSprite->lotag = 0;
else if (pSprite->lotag != 0 && pSprite->hitag > 5)
{
spritetype *pTarget = &sprite[pSprite->lotag];
int const angle = getangle(pTarget->x - pSprite->x, pTarget->y - pSprite->y);
int const angleDiff = angle - pSprite->ang;
int const angleDiffAbs = klabs(angleDiff);
if (angleDiff < 100)
{
if (angleDiffAbs > 1023)
pSprite->ang += 51;
else
pSprite->ang -= 51;
}
else if (angleDiff > 100)
{
if (angleDiffAbs > 1023)
pSprite->ang -= 51;
else
pSprite->ang += 51;
}
else
pSprite->ang = angle;
if (pSprite->hitag > 180 && pSprite->zvel <= 0)
pSprite->zvel += 200;
}
}
else if (pSprite->picnum == RRTILE1790)
{
if (pSprite->extra)
{
pSprite->zvel = -(pSprite->extra * 250);
pSprite->extra--;
}
else
A_Fall(spriteNum);
if (pSprite->xrepeat >= 10 && sector[pSprite->sectnum].lotag != ST_2_UNDERWATER)
{
int const newSprite = A_Spawn(spriteNum, SMALLSMOKE);
sprite[newSprite].z += (1 << 8);
}
}
}
vec3_t const tmpvect = { (spriteXvel * (sintable[(pSprite->ang + 512) & 2047])) >> 14,
(spriteXvel * (sintable[pSprite->ang & 2047])) >> 14, spriteZvel };
int moveSprite = A_MoveSprite(spriteNum, &tmpvect, CLIPMASK1);
if ((pSprite->picnum == RPG || (RRRA && (pSprite->picnum == RPG2 || pSprite->picnum == RRTILE1790)))
&& (unsigned) pSprite->yvel < MAXSPRITES) // RPG_YVEL
if (FindDistance2D(pSprite->x - sprite[pSprite->yvel].x, pSprite->y - sprite[pSprite->yvel].y) < 256)
moveSprite = 49152 | pSprite->yvel;
//actor[spriteNum].movflag = moveSprite;
if (pSprite->sectnum < 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
//if (RR && g_sectorExtra[pSprite->sectnum] == 800)
// DELETE_SPRITE_AND_CONTINUE(spriteNum);
if ((moveSprite & 49152) != 49152 && pSprite->picnum != FREEZEBLAST)
G_WeaponHitCeilingOrFloor(spriteNum, pSprite, &moveSprite);
if (pSprite->picnum == FIRELASER)
{
for (bssize_t k = -3; k < 2; k++)
{
int const newSprite
= A_InsertSprite(pSprite->sectnum, pSprite->x + ((k * sintable[(pSprite->ang + 512) & 2047]) >> 9),
pSprite->y + ((k * sintable[pSprite->ang & 2047]) >> 9),
pSprite->z + ((k * ksgn(pSprite->zvel)) * klabs(pSprite->zvel / 24)), FIRELASER, -40 + (k << 2),
pSprite->xrepeat, pSprite->yrepeat, 0, 0, 0, pSprite->owner, 5);
sprite[newSprite].cstat = 128;
sprite[newSprite].pal = pSprite->pal;
}
}
else if (pSprite->picnum == SPIT)
if (pSprite->zvel < 6144)
pSprite->zvel += g_spriteGravity - 112;
if (moveSprite != 0)
{
if (!RR && pSprite->picnum == COOLEXPLOSION1)
{
if ((moveSprite & 49152) == 49152 && sprite[moveSprite & (MAXSPRITES - 1)].picnum != APLAYER)
goto COOLEXPLOSION;
pSprite->xvel = 0;
pSprite->zvel = 0;
}
switch (moveSprite & 49152)
{
case 49152:
moveSprite &= (MAXSPRITES - 1);
if (RRRA && sprite[moveSprite].picnum == MINION && (pSprite->picnum == RPG || pSprite->picnum == RPG2)
&& sprite[moveSprite].pal == 19)
{
int const newSprite = A_Spawn(spriteNum, EXPLOSION2);
A_PlaySound(RPG_EXPLODE, newSprite);
sprite[newSprite].x = davect.x;
sprite[newSprite].y = davect.y;
sprite[newSprite].z = davect.z;
goto next_sprite;
}
if (!RRRA && pSprite->picnum == FREEZEBLAST && sprite[moveSprite].pal == 1)
if (A_CheckEnemySprite(&sprite[moveSprite]) || sprite[moveSprite].picnum == APLAYER)
{
int const newSprite = A_Spawn(spriteNum, TRANSPORTERSTAR);
sprite[newSprite].pal = 1;
sprite[newSprite].xrepeat = 32;
sprite[newSprite].yrepeat = 32;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
A_DamageObject(moveSprite, spriteNum);
if (sprite[moveSprite].picnum == APLAYER)
{
int const playerNum = P_Get(moveSprite);
A_PlaySound(PISTOL_BODYHIT, moveSprite);
if (pSprite->picnum == SPIT)
{
if (RRRA && sprite[pSprite->owner].picnum == MAMA)
{
A_DoGuts(spriteNum, RABBITJIBA, 2);
A_DoGuts(spriteNum, RABBITJIBB, 2);
A_DoGuts(spriteNum, RABBITJIBC, 2);
}
P_HandleBeingSpitOn(g_player[playerNum].ps);
}
}
break;
case 32768:
moveSprite &= (MAXWALLS - 1);
if (RRRA && sprite[pSprite->owner].picnum == MAMA)
{
A_DoGuts(spriteNum, RABBITJIBA, 2);
A_DoGuts(spriteNum, RABBITJIBB, 2);
A_DoGuts(spriteNum, RABBITJIBC, 2);
}
if (pSprite->picnum != RPG && (!RRRA || pSprite->picnum != RPG2) && pSprite->picnum != FREEZEBLAST && pSprite->picnum != SPIT
&& (!RR || pSprite->picnum != SHRINKSPARK) && (wall[moveSprite].overpicnum == MIRROR || wall[moveSprite].picnum == MIRROR))
{
Proj_BounceOffWall(pSprite, moveSprite);
pSprite->owner = spriteNum;
A_Spawn(spriteNum, TRANSPORTERSTAR);
goto next_sprite;
}
else
{
setsprite(spriteNum, &davect);
A_DamageWall(spriteNum, moveSprite, (vec3_t *)pSprite, pSprite->picnum);
if (!RRRA && pSprite->picnum == FREEZEBLAST)
{
if (wall[moveSprite].overpicnum != MIRROR && wall[moveSprite].picnum != MIRROR)
{
pSprite->extra >>= 1;
if (RR)
{
if (pSprite->xrepeat > 8)
pSprite->xrepeat--;
if (pSprite->yrepeat > 8)
pSprite->yrepeat--;
}
pSprite->yvel--;
}
Proj_BounceOffWall(pSprite, moveSprite);
goto next_sprite;
}
if (RR && pSprite->picnum == SHRINKSPARK)
{
if (wall[moveSprite].picnum >= RRTILE3643 && wall[moveSprite].picnum < RRTILE3643+3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (pSprite->extra <= 0)
{
pSprite->x += sintable[(pSprite->ang+512)&2047]>>7;
pSprite->y += sintable[pSprite->ang&2047]>>7;
if (!RRRA || (sprite[pSprite->owner].picnum != CHEER && sprite[pSprite->owner].picnum != CHEERSTAYPUT))
{
int const newSprite = A_Spawn(spriteNum, CIRCLESTUCK);
sprite[newSprite].xrepeat = 8;
sprite[newSprite].yrepeat = 8;
sprite[newSprite].cstat = 16;
sprite[newSprite].ang = (sprite[newSprite].ang+512)&2047;
sprite[newSprite].clipdist = mulscale7(pSprite->xrepeat, tilesiz[pSprite->picnum].x);
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (wall[moveSprite].overpicnum != MIRROR && wall[moveSprite].picnum != MIRROR)
{
pSprite->extra -= 20;
pSprite->yvel--;
}
Proj_BounceOffWall(pSprite, moveSprite);
goto next_sprite;
}
}
break;
case 16384:
setsprite(spriteNum, &davect);
if (RRRA && sprite[pSprite->owner].picnum == MAMA)
{
A_DoGuts(spriteNum, RABBITJIBA, 2);
A_DoGuts(spriteNum, RABBITJIBB, 2);
A_DoGuts(spriteNum, RABBITJIBC, 2);
}
if (Proj_MaybeDamageCF(spriteNum))
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (!RRRA && pSprite->picnum == FREEZEBLAST)
{
A_DoProjectileBounce(spriteNum);
A_SetSprite(spriteNum, CLIPMASK1);
pSprite->extra >>= 1;
if (pSprite->xrepeat > 8)
pSprite->xrepeat -= 2;
if (pSprite->yrepeat > 8)
pSprite->yrepeat -= 2;
pSprite->yvel--;
goto next_sprite;
}
break;
default: break;
}
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case COOLEXPLOSION1__STATIC:
if (RR) goto default_case;
fallthrough__;
case SPIT__STATIC:
case FREEZEBLAST__STATIC:
case FIRELASER__STATIC: break;
case RPG__STATIC:
case RPG2__STATICRR:
case RRTILE1790__STATICRR:
{
if (!RRRA && (pSprite->picnum == RPG2 || pSprite->picnum == RRTILE1790))
break;
int const newSprite = A_Spawn(spriteNum, EXPLOSION2);
if (pSprite->picnum == RPG2)
{
pSprite->extra = 150;
A_PlaySound(247, newSprite);
}
else if (pSprite->picnum == RRTILE1790)
{
pSprite->extra = 160;
A_PlaySound(RPG_EXPLODE, newSprite);
}
else
A_PlaySound(RPG_EXPLODE, newSprite);
Bmemcpy(&sprite[newSprite], &davect, sizeof(vec3_t));
if (pSprite->xrepeat < 10)
{
sprite[newSprite].xrepeat = 6;
sprite[newSprite].yrepeat = 6;
}
else if ((moveSprite & 49152) == 16384)
{
if (!RR && pSprite->zvel > 0)
A_Spawn(spriteNum, EXPLOSION2BOT);
else
{
sprite[newSprite].cstat |= 8;
sprite[newSprite].z += (48 << 8);
}
}
if (pSprite->xrepeat >= 10)
{
int const x = pSprite->extra;
A_RadiusDamage(spriteNum, g_rpgRadius, x >> 2, x >> 1, x - (x >> 2), x);
}
else
{
int const x = pSprite->extra + (g_globalRandom & 3);
A_RadiusDamage(spriteNum, (g_rpgRadius >> 1), x >> 2, x >> 1, x - (x >> 2), x);
}
break;
}
case SHRINKSPARK__STATIC:
if (RR)
break;
A_Spawn(spriteNum, SHRINKEREXPLOSION);
A_PlaySound(SHRINKER_HIT, spriteNum);
A_RadiusDamage(spriteNum, g_shrinkerRadius, 0, 0, 0, 0);
break;
default:
default_case:
{
int const newSprite = A_Spawn(spriteNum, EXPLOSION2);
sprite[newSprite].xrepeat = sprite[newSprite].yrepeat = pSprite->xrepeat >> 1;
if ((moveSprite & 49152) == 16384)
{
if (pSprite->zvel < 0)
{
sprite[newSprite].cstat |= 8;
sprite[newSprite].z += (72 << 8);
}
}
break;
}
}
if (RR || pSprite->picnum != COOLEXPLOSION1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (!RR && pSprite->picnum == COOLEXPLOSION1)
{
COOLEXPLOSION:
pSprite->shade++;
if (pSprite->shade >= 40)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if ((pSprite->picnum == RPG || (RRRA && pSprite->picnum == RPG2)) && sector[pSprite->sectnum].lotag == ST_2_UNDERWATER && pSprite->xrepeat >= 10 && rnd(140))
A_Spawn(spriteNum, WATERBUBBLE);
goto next_sprite;
}
case SHOTSPARK1__STATIC:
{
if (!G_HaveActor(sprite[spriteNum].picnum))
goto next_sprite;
int32_t playerDist;
int const playerNum = A_FindPlayer(pSprite, &playerDist);
A_Execute(spriteNum, playerNum, playerDist);
goto next_sprite;
}
}
next_sprite:
spriteNum = nextSprite;
}
}
static int P_Submerge(int const spriteNum, int const playerNum, DukePlayer_t * const pPlayer, int const sectNum, int const otherSect)
{
if ((!RR && pPlayer->on_ground && pPlayer->pos.z > sector[sectNum].floorz - ZOFFSET2
&& (TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_CROUCH) || pPlayer->vel.z > 2048))
|| (RR && pPlayer->pos.z > (sector[sectNum].floorz-(6<<8))) || pPlayer->on_motorcycle)
// if( onfloorz && sectlotag == 1 && ps->pos.z > (sector[sect].floorz-(6<<8)) )
{
if (pPlayer->on_boat) return 0;
if (screenpeek == playerNum)
{
FX_StopAllSounds();
S_ClearSoundLocks();
}
if (RR || sprite[pPlayer->i].extra > 0)
A_PlaySound(DUKE_UNDERWATER, spriteNum);
pPlayer->opos.z = pPlayer->pos.z = sector[otherSect].ceilingz + (7<<8);
if (!RR)
{
pPlayer->vel.x = 4096-(krand2()&8192);
pPlayer->vel.y = 4096-(krand2()&8192);
}
if (pPlayer->on_motorcycle)
pPlayer->moto_underwater = 1;
return 1;
}
return 0;
}
static int P_Emerge(int const spriteNum, int const playerNum, DukePlayer_t * const pPlayer, int const sectNum, int const otherSect)
{
// r1449-:
if (pPlayer->pos.z < (sector[sectNum].ceilingz+(6<<8)))
// r1450+, breaks submergible slime in bobsp2:
// if (onfloorz && sectlotag == 2 && ps->pos.z <= sector[sect].ceilingz /*&& ps->vel.z == 0*/)
{
if (RR && sprite[pPlayer->i].extra <= 0) return 1;
if (screenpeek == playerNum)
{
FX_StopAllSounds();
S_ClearSoundLocks();
}
A_PlaySound(DUKE_GASP, spriteNum);
pPlayer->opos.z = pPlayer->pos.z = sector[otherSect].floorz - (7<<8);
//pPlayer->vel.z = 0;
// ps->vel.z += 1024;
if (!RR)
{
pPlayer->jumping_toggle = 1;
pPlayer->jumping_counter = 0;
}
return 1;
}
return 0;
}
static void P_FinishWaterChange(int const playerNum, DukePlayer_t * const pPlayer, int const sectLotag, int const spriteOwner, int const newSector)
{
/*pPlayer->bobpos.x = */pPlayer->opos.x = pPlayer->pos.x;
/*pPlayer->bobpos.y = */pPlayer->opos.y = pPlayer->pos.y;
if (spriteOwner < 0 || sprite[spriteOwner].owner != spriteOwner)
pPlayer->transporter_hold = -2;
pPlayer->cursectnum = newSector;
changespritesect(playerNum, newSector);
if (!RR)
{
vec3_t vect = pPlayer->pos;
vect.z += PHEIGHT;
setsprite(pPlayer->i, &vect);
}
P_UpdateScreenPal(pPlayer);
if ((krand2()&255) < 32)
A_Spawn(RR ? pPlayer->i : playerNum, WATERSPLASH2);
if (!RR && sectLotag == ST_1_ABOVE_WATER)
{
for (bssize_t l = 0; l < 9; l++)
sprite[A_Spawn(pPlayer->i, WATERBUBBLE)].z += krand2() & 16383;
}
}
// Check prevention of teleportation *when alive*. For example, commanders and
// octabrains would be transported by SE7 (both water and normal) only if dead.
static int A_CheckNonTeleporting(int const spriteNum)
{
int const tileNum = sprite[spriteNum].picnum;
if (RRRA)
{
return !!(tileNum == SHARK || tileNum == CHEERBOAT || tileNum == HULKBOAT
|| tileNum == MINIONBOAT || tileNum == UFO1);
}
else if (RR)
{
return !!(tileNum == SHARK || tileNum == UFO1 || tileNum == UFO2
|| tileNum == UFO3 || tileNum == UFO4 || tileNum == UFO5);
}
return !!(tileNum == SHARK || tileNum == COMMANDER || tileNum == OCTABRAIN
|| (tileNum >= GREENSLIME && tileNum <= GREENSLIME + 7));
}
ACTOR_STATIC void G_MoveTransports(void)
{
int spriteNum = headspritestat[STAT_TRANSPORT];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (OW(spriteNum) == spriteNum)
{
spriteNum = nextSprite;
continue;
}
int const sectNum = SECT(spriteNum);
int const sectLotag = sector[sectNum].lotag;
int const onFloor = T5(spriteNum); // ONFLOORZ
if (T1(spriteNum) > 0)
T1(spriteNum)--;
int sectSprite = headspritesect[sectNum];
while (sectSprite >= 0)
{
int const nextSectSprite = nextspritesect[sectSprite];
switch (sprite[sectSprite].statnum)
{
case STAT_PLAYER:
if (sprite[sectSprite].owner != -1)
{
int const playerNum = P_Get(sectSprite);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
pPlayer->on_warping_sector = 1;
if (pPlayer->transporter_hold == 0 && pPlayer->jumping_counter == 0)
{
if (pPlayer->on_ground && sectLotag == 0 && onFloor && pPlayer->jetpack_on == 0)
{
if (RR || sprite[spriteNum].pal == 0)
{
A_Spawn(spriteNum, TRANSPORTERBEAM);
A_PlaySound(TELEPORTER, spriteNum);
}
for (int TRAVERSE_CONNECT(otherPlayer))
{
if (g_player[otherPlayer].ps->cursectnum == sprite[OW(spriteNum)].sectnum)
{
g_player[otherPlayer].ps->frag_ps = playerNum;
sprite[g_player[otherPlayer].ps->i].extra = 0;
}
}
pPlayer->q16ang = fix16_from_int(sprite[OW(spriteNum)].ang);
if (sprite[OW(spriteNum)].owner != OW(spriteNum))
{
T1(spriteNum) = 13;
actor[OW(spriteNum)].t_data[0] = 13;
pPlayer->transporter_hold = 13;
}
pPlayer->pos = *(vec3_t *)&sprite[OW(spriteNum)];
pPlayer->pos.z -= PHEIGHT-(RR ? (4<<8) : 0);
pPlayer->opos = pPlayer->pos;
pPlayer->bobpos = *(vec2_t *)&pPlayer->pos;
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
pPlayer->cursectnum = sprite[sectSprite].sectnum;
if (RR || sprite[spriteNum].pal == 0)
{
int const newSprite = A_Spawn(OW(spriteNum), TRANSPORTERBEAM);
A_PlaySound(TELEPORTER, newSprite);
}
break;
}
}
else if (RR || !(sectLotag == ST_1_ABOVE_WATER && pPlayer->on_ground == 1))
break;
if (onFloor == 0 && klabs(SZ(spriteNum) - pPlayer->pos.z) < 6144)
if (!pPlayer->jetpack_on || TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_JUMP)
|| (TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_CROUCH) ^ pPlayer->crouch_toggle))
{
pPlayer->pos.x += sprite[OW(spriteNum)].x - SX(spriteNum);
pPlayer->pos.y += sprite[OW(spriteNum)].y - SY(spriteNum);
pPlayer->pos.z = (pPlayer->jetpack_on && (TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_JUMP)
|| pPlayer->jetpack_on < 11))
? sprite[OW(spriteNum)].z - 6144
: sprite[OW(spriteNum)].z + 6144;
if (!RR)
actor[pPlayer->i].bpos = pPlayer->pos;
pPlayer->opos = pPlayer->pos;
//pPlayer->bobpos = *(vec2_t *)&pPlayer->pos;
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
pPlayer->cursectnum = sprite[OW(spriteNum)].sectnum;
break;
}
int doWater = 0;
if (RRRA)
{
if (onFloor)
{
if (sectLotag == 160 && pPlayer->pos.z > (sector[sectNum].floorz-(48<<8)))
{
doWater = 2;
pPlayer->opos.z = pPlayer->pos.z = sector[sprite[OW(spriteNum)].sectnum].ceilingz+(7<<8);
}
else if (sectLotag == 161 && pPlayer->pos.z < (sector[sectNum].ceilingz+(6<<8)) && sprite[pPlayer->i].extra > 0)
{
doWater = 2;
pPlayer->opos.z = pPlayer->pos.z = sector[sprite[OW(spriteNum)].sectnum].floorz-(49<<8);
}
}
}
if (onFloor)
{
if (sectLotag == ST_1_ABOVE_WATER)
doWater = P_Submerge(sectSprite, playerNum, pPlayer, sectNum, sprite[OW(spriteNum)].sectnum);
else if (sectLotag == ST_2_UNDERWATER)
doWater = P_Emerge(sectSprite, playerNum, pPlayer, sectNum, sprite[OW(spriteNum)].sectnum);
}
if (doWater == 1)
{
pPlayer->pos.x += sprite[OW(spriteNum)].x - SX(spriteNum);
pPlayer->pos.y += sprite[OW(spriteNum)].y - SY(spriteNum);
P_FinishWaterChange(sectSprite, pPlayer, sectLotag, OW(spriteNum), sprite[OW(spriteNum)].sectnum);
}
else if (doWater == 2)
{
pPlayer->pos.x += sprite[OW(spriteNum)].x - SX(spriteNum);
pPlayer->pos.y += sprite[OW(spriteNum)].y - SY(spriteNum);
pPlayer->opos.x = pPlayer->pos.x;
pPlayer->opos.y = pPlayer->pos.y;
if (sprite[OW(spriteNum)].owner != OW(spriteNum))
pPlayer->transporter_hold = -2;
pPlayer->cursectnum = sprite[OW(spriteNum)].sectnum;
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
}
}
break;
////////// Non-player teleportation //////////
case STAT_ACTOR:
if ((RR || sprite[sectSprite].extra > 0) && A_CheckNonTeleporting(sectSprite))
goto JBOLT;
fallthrough__;
case STAT_PROJECTILE:
case STAT_MISC:
case STAT_FALLER:
case STAT_DUMMYPLAYER:
{
if (RR && sprite[sectSprite].statnum == STAT_FALLER) break;
//if ((totalclock & UINT8_MAX) != actor[sectSprite].lasttransport)
{
int const zvel = sprite[sectSprite].zvel;
int const absZvel = klabs(zvel);
int doWarp = 0;
int warpDir;
int absZdiff;
if (zvel >= 0)
warpDir = 2;
else
warpDir = 1;
if (absZvel != 0)
{
if (sectLotag == ST_2_UNDERWATER && sprite[sectSprite].z < (sector[sectNum].ceilingz + absZvel))
doWarp = 1;
if (sectLotag == ST_1_ABOVE_WATER && sprite[sectSprite].z > (sector[sectNum].floorz - absZvel))
if (!RRRA || (sprite[sectSprite].picnum != CHEERBOAT && sprite[sectSprite].picnum != HULKBOAT && sprite[sectSprite].picnum != MINIONBOAT))
doWarp = 1;
}
if (RRRA)
{
if (absZvel != 0 && sectLotag == 161 && sprite[sectSprite].z < (sector[sectNum].ceilingz + absZvel) && warpDir == 1)
{
doWarp = 1;
absZdiff = absZvel - klabs(sprite[sectSprite].z-sector[sectNum].ceilingz);
}
else if (sectLotag == 161 && sprite[sectSprite].z < (sector[sectNum].ceilingz + 1000) && warpDir == 1)
{
doWarp = 1;
absZdiff = 1;
}
if (absZvel != 0 && sectLotag == 160 && sprite[sectSprite].z > (sector[sectNum].floorz - absZvel) && warpDir == 2)
{
doWarp = 1;
absZdiff = absZvel - klabs(sector[sectNum].floorz-sprite[sectSprite].z);
}
else if (sectLotag == 160 && sprite[sectSprite].z > (sector[sectNum].floorz - 1000) && warpDir == 2)
{
doWarp = 1;
absZdiff = 1;
}
}
if (sectLotag == 0 && (onFloor || klabs(sprite[sectSprite].z - SZ(spriteNum)) < 4096))
{
if (sprite[OW(spriteNum)].owner != OW(spriteNum) && onFloor && T1(spriteNum) > 0
&& sprite[sectSprite].statnum != STAT_MISC)
{
T1(spriteNum)++;
goto next_sprite;
}
doWarp = 1;
}
if (doWarp)
{
switch (DYNAMICTILEMAP(sprite[sectSprite].picnum))
{
case TRIPBOMB__STATIC:
case BURNING2__STATIC:
case FIRE2__STATIC:
case TOILETWATER__STATIC:
case LASERLINE__STATIC:
if (RR) goto default_case;
fallthrough__;
case TRIPBOMBSPRITE__STATIC:
if (!RR && sprite[sectSprite].picnum == TRIPBOMBSPRITE)
goto default_case;
fallthrough__;
case TRANSPORTERSTAR__STATIC:
case TRANSPORTERBEAM__STATIC:
case BULLETHOLE__STATIC:
case WATERSPLASH2__STATIC:
case BURNING__STATIC:
case FIRE__STATIC:
case MUD__STATICRR: goto JBOLT;
case PLAYERONWATER__STATIC:
if (sectLotag == ST_2_UNDERWATER)
{
sprite[sectSprite].cstat &= 32767;
break;
}
fallthrough__;
default:
default_case:
if (sprite[sectSprite].statnum == STAT_MISC && !(sectLotag == ST_1_ABOVE_WATER || sectLotag == ST_2_UNDERWATER || (RRRA && (sectLotag == 160 || sectLotag == 161))))
break;
fallthrough__;
case WATERBUBBLE__STATIC:
if (RR)
if( rnd(192) && sprite[sectSprite].picnum == WATERBUBBLE)
break;
if (sectLotag > 0)
{
// Water SE7 teleportation.
int const osect = sprite[OW(spriteNum)].sectnum;
Bassert(sectLotag == ST_1_ABOVE_WATER || sectLotag == ST_2_UNDERWATER || (RRRA && (sectLotag == 160 || sectLotag == 161)));
int const newSprite = A_Spawn(sectSprite, WATERSPLASH2);
if (sectLotag == ST_1_ABOVE_WATER && sprite[sectSprite].statnum == STAT_PROJECTILE)
{
sprite[newSprite].xvel = sprite[sectSprite].xvel >> 1;
sprite[newSprite].ang = sprite[sectSprite].ang;
A_SetSprite(newSprite, CLIPMASK0);
}
actor[sectSprite].lasttransport = ((int32_t) totalclock & UINT8_MAX);
if (sectLotag == ST_1_ABOVE_WATER || sectLotag == ST_2_UNDERWATER)
{
sprite[sectSprite].x += sprite[OW(spriteNum)].x - SX(spriteNum);
sprite[sectSprite].y += sprite[OW(spriteNum)].y - SY(spriteNum);
sprite[sectSprite].z = (sectLotag == ST_1_ABOVE_WATER) ? sector[osect].ceilingz+absZvel : sector[osect].floorz-absZvel;
actor[sectSprite].bpos = *(vec3_t *)&sprite[sectSprite];
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
}
else
{
sprite[sectSprite].x += sprite[OW(spriteNum)].x - SX(spriteNum);
sprite[sectSprite].y += sprite[OW(spriteNum)].y - SY(spriteNum);
sprite[sectSprite].z = (sectLotag == 160) ? sector[osect].ceilingz+absZdiff : sector[osect].floorz-absZdiff;
actor[sectSprite].bpos = *(vec3_t *)&sprite[sectSprite];
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
vec3_t const vect = {
(sprite[sectSprite].xvel*sintable[(sprite[sectSprite].ang+512)&2047])>>14,
(sprite[sectSprite].xvel*sintable[sprite[sectSprite].ang])>>14,
0
};
A_MoveSprite(sectSprite, &vect, CLIPMASK1);
}
}
else if (Bassert(sectLotag == 0), 1)
{
// Non-water SE7 teleportation.
if (onFloor)
{
if ((!RR && sprite[sectSprite].statnum == STAT_PROJECTILE)
|| (G_CheckPlayerInSector(sectNum) == -1
&& G_CheckPlayerInSector(sprite[OW(spriteNum)].sectnum) == -1))
{
sprite[sectSprite].x += (sprite[OW(spriteNum)].x - SX(spriteNum));
sprite[sectSprite].y += (sprite[OW(spriteNum)].y - SY(spriteNum));
sprite[sectSprite].z -= SZ(spriteNum) - sector[sprite[OW(spriteNum)].sectnum].floorz;
sprite[sectSprite].ang = sprite[OW(spriteNum)].ang;
actor[sectSprite].bpos = *(vec3_t *)&sprite[sectSprite];
if (RR || sprite[spriteNum].pal == 0)
{
int newSprite = A_Spawn(spriteNum, TRANSPORTERBEAM);
A_PlaySound(TELEPORTER, newSprite);
newSprite = A_Spawn(OW(spriteNum), TRANSPORTERBEAM);
A_PlaySound(TELEPORTER, newSprite);
}
if (sprite[OW(spriteNum)].owner != OW(spriteNum))
{
T1(spriteNum) = 13;
actor[OW(spriteNum)].t_data[0] = 13;
}
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
}
}
else
{
sprite[sectSprite].x += (sprite[OW(spriteNum)].x - SX(spriteNum));
sprite[sectSprite].y += (sprite[OW(spriteNum)].y - SY(spriteNum));
sprite[sectSprite].z = sprite[OW(spriteNum)].z + 4096;
actor[sectSprite].bpos = *(vec3_t *)&sprite[sectSprite];
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
}
}
break;
} // switch (DYNAMICTILEMAP(sprite[j].picnum))
} // if (doWarp)
} // if (totalclock > actor[j].lasttransport)
break;
} // five cases
} // switch (sprite[j].statnum)
JBOLT:
sectSprite = nextSectSprite;
}
next_sprite:
spriteNum = nextSprite;
}
}
static int A_FindLocator(int const tag, int const sectNum)
{
for (bssize_t SPRITES_OF(STAT_LOCATOR, spriteNum))
{
if ((sectNum == -1 || sectNum == SECT(spriteNum)) && tag == SLT(spriteNum))
return spriteNum;
}
return -1;
}
ACTOR_STATIC int A_PinSectorResetDown(int16_t const sectNum)
{
if (GetAnimationGoal(&sector[sectNum].ceilingz) == -1)
{
int const newZ = sector[sectNum].floorz;
SetAnimation(sectNum,&sector[sectNum].ceilingz,newZ,64);
return 1;
}
return 0;
}
ACTOR_STATIC int A_PinSectorResetUp(int16_t const sectNum)
{
if (GetAnimationGoal(&sector[sectNum].ceilingz) == -1)
{
int const newZ = sector[nextsectorneighborz(sectNum, sector[sectNum].ceilingz,-1,-1)].ceilingz;
SetAnimation(sectNum,&sector[sectNum].ceilingz,newZ,64);
return 1;
}
return 0;
}
ACTOR_STATIC void A_BallReturn(int16_t const pinSprite)
{
int spriteNum = headspritestat[105];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].picnum == RRTILE281)
if (sprite[pinSprite].sectnum == sprite[spriteNum].sectnum)
{
int otherSprite = headspritestat[105];
while (otherSprite >= 0)
{
int const otherSpriteNext = nextspritestat[otherSprite];
if (sprite[otherSprite].picnum == RRTILE282)
if (sprite[spriteNum].hitag == sprite[otherSprite].hitag)
A_Spawn(otherSprite, BOWLINGBALLSPRITE);
if (sprite[otherSprite].picnum == RRTILE280)
if (sprite[spriteNum].hitag == sprite[otherSprite].hitag)
if (sprite[otherSprite].lotag == 0)
{
sprite[otherSprite].lotag = 100;
sprite[otherSprite].extra++;
A_PinSectorResetDown(sprite[otherSprite].sectnum);
}
otherSprite = otherSpriteNext;
}
}
spriteNum = nextSprite;
}
}
ACTOR_STATIC int A_CheckPins(int16_t const sectNum)
{
uint8_t pins[10];
int pinCount = 0;
int tag;
Bmemset(pins, 0, sizeof(pins));
int spriteNum = headspritesect[sectNum];
while (spriteNum >= 0)
{
if (sprite[spriteNum].picnum == RRTILE3440)
{
pinCount++;
pins[sprite[spriteNum].lotag] = 1;
}
if (sprite[spriteNum].picnum == RRTILE280)
{
tag = sprite[spriteNum].hitag;
}
spriteNum = nextspritesect[spriteNum];
}
if (tag != 0)
{
int const tileNumber = LANEPICS + tag + 1;
TileFiles.tileMakeWritable(tileNumber);
tileCopySection(LANEPICS+1, 0, 0, 128, 64, tileNumber, 0, 0);
for (int pin = 0; pin < 10; pin++)
{
if (pins[pin] == 1)
{
int x, y;
switch (pin)
{
case 0:
x = 64;
y = 48;
break;
case 1:
x = 56;
y = 40;
break;
case 2:
x = 72;
y = 40;
break;
case 3:
x = 48;
y = 32;
break;
case 4:
x = 64;
y = 32;
break;
case 5:
x = 80;
y = 32;
break;
case 6:
x = 40;
y = 24;
break;
case 7:
x = 56;
y = 24;
break;
case 8:
x = 72;
y = 24;
break;
case 9:
x = 88;
y = 24;
break;
}
tileCopySection(LANEPICS, 0, 0, 8, 8, tileNumber, x - 4, y - 10);
}
}
}
return pinCount;
}
ACTOR_STATIC void A_ResetPins(int16_t sect)
{
int step = 0;
int tag = 0;
int spriteNum = headspritesect[sect];
while (spriteNum >= 0)
{
if (spriteNum > MAXSECTORSV7 || step > 1000)
break;
int const nextSprite = headspritesect[spriteNum];
if (sprite[spriteNum].picnum == RRTILE3440)
deletesprite(spriteNum);
spriteNum = nextSprite;
step++;
}
spriteNum = headspritesect[sect];
while (spriteNum >= 0)
{
int const nextSprite = nextspritesect[spriteNum];
if (sprite[spriteNum].picnum == RRTILE283)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3440);
sprite[newSprite].lotag = sprite[spriteNum].lotag;
krand2();
sprite[newSprite].clipdist = 48;
sprite[newSprite].ang -= ((krand2()&32)-(krand2()&64))&2047;
}
if (sprite[spriteNum].picnum == RRTILE280)
tag = sprite[spriteNum].hitag;
spriteNum = nextSprite;
}
if (tag != 0)
{
int const tileNumber = LANEPICS + tag + 1;
TileFiles.tileMakeWritable(tileNumber);
tileCopySection(LANEPICS+1, 0, 0, 128, 64, tileNumber, 0, 0);
for (int pin = 0; pin < 10; pin++)
{
int x, y;
switch (pin)
{
case 0:
x = 64;
y = 48;
break;
case 1:
x = 56;
y = 40;
break;
case 2:
x = 72;
y = 40;
break;
case 3:
x = 48;
y = 32;
break;
case 4:
x = 64;
y = 32;
break;
case 5:
x = 80;
y = 32;
break;
case 6:
x = 40;
y = 24;
break;
case 7:
x = 56;
y = 24;
break;
case 8:
x = 72;
y = 24;
break;
case 9:
x = 88;
y = 24;
break;
}
tileCopySection(LANEPICS, 0, 0, 8, 8, tileNumber, x - 4, y - 10);
}
}
}
void A_ResetLanePics(void)
{
for (int tag = 1; tag <= 4; tag++)
{
int const tileNumber = LANEPICS + tag + 1;
TileFiles.tileMakeWritable(tileNumber);
tileCopySection(LANEPICS + 1, 0, 0, 128, 64, tileNumber, 0, 0);
for (int pin = 0; pin < 10; pin++)
{
int x, y;
switch (pin)
{
case 0:
x = 64;
y = 48;
break;
case 1:
x = 56;
y = 40;
break;
case 2:
x = 72;
y = 40;
break;
case 3:
x = 48;
y = 32;
break;
case 4:
x = 64;
y = 32;
break;
case 5:
x = 80;
y = 32;
break;
case 6:
x = 40;
y = 24;
break;
case 7:
x = 56;
y = 24;
break;
case 8:
x = 72;
y = 24;
break;
case 9:
x = 88;
y = 24;
break;
}
tileCopySection(LANEPICS, 0, 0, 8, 8, tileNumber, x - 4, y - 10);
}
}
}
ACTOR_STATIC void G_MoveActors(void)
{
int spriteNum;
if (g_jailDoorCnt)
G_DoJailDoor();
if (g_mineCartCnt)
G_MoveMineCart();
int bBoom = 0;
if (RRRA)
{
int spriteNum = headspritestat[117];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].hitag > 2)
sprite[spriteNum].hitag = 0;
if ((sprite[spriteNum].picnum == RRTILE8488 || sprite[spriteNum].picnum == RRTILE8490) && sprite[spriteNum].hitag != 2)
{
sprite[spriteNum].hitag = 2;
sprite[spriteNum].extra = -100;
}
if (sprite[spriteNum].hitag == 0)
{
sprite[spriteNum].extra++;
if (sprite[spriteNum].extra >= 30)
sprite[spriteNum].hitag = 1;
}
else if (sprite[spriteNum].hitag == 1)
{
sprite[spriteNum].extra--;
if (sprite[spriteNum].extra <= -30)
sprite[spriteNum].hitag = 0;
}
else if (sprite[spriteNum].hitag == 2)
{
sprite[spriteNum].extra--;
if (sprite[spriteNum].extra <= -104)
{
A_Spawn(spriteNum, sprite[spriteNum].lotag);
deletesprite(spriteNum);
}
}
vec3_t const vect = { 0, 0, sprite[spriteNum].extra * 2};
A_MoveSprite(spriteNum, &vect, CLIPMASK0);
spriteNum = nextSprite;
}
spriteNum = headspritestat[118];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].hitag > 1)
sprite[spriteNum].hitag = 0;
if (sprite[spriteNum].hitag == 0)
{
sprite[spriteNum].extra++;
if (sprite[spriteNum].extra >= 20)
sprite[spriteNum].hitag = 1;
}
else if (sprite[spriteNum].hitag == 1)
{
sprite[spriteNum].extra--;
if (sprite[spriteNum].extra <= -20)
sprite[spriteNum].hitag = 0;
}
vec3_t const vect = { 0, 0, sprite[spriteNum].extra };
A_MoveSprite(spriteNum, &vect, CLIPMASK0);
spriteNum = nextSprite;
}
if (g_player[myconnectindex].ps->level_end_timer > 0)
{
if (--g_player[myconnectindex].ps->level_end_timer == 0)
{
for (bssize_t TRAVERSE_CONNECT(playerNum))
g_player[playerNum].ps->gm = MODE_EOL;
ud.eog = 1;
ud.level_number++;
if (ud.level_number > 6)
ud.level_number = 0;
m_level_number = ud.level_number;
}
}
if (g_changeEnemySize > 0)
{
for (bssize_t enemySprite = 0; enemySprite < MAXSPRITES; enemySprite++)
{
switch (DYNAMICTILEMAP(sprite[enemySprite].picnum))
{
case BILLYCOCK__STATICRR:
case BILLYRAY__STATICRR:
case BILLYRAYSTAYPUT__STATICRR:
case BRAYSNIPER__STATICRR:
case DOGRUN__STATICRR:
case LTH__STATICRR:
case HULKJUMP__STATICRR:
case HULK__STATICRR:
case HULKSTAYPUT__STATICRR:
case HEN__STATICRR:
case DRONE__STATIC:
case PIG__STATICRR:
case MINION__STATICRR:
case MINIONSTAYPUT__STATICRR:
case UFO1__STATICRR:
case UFO2__STATICRR:
case UFO3__STATICRR:
case UFO4__STATICRR:
case UFO5__STATICRR:
case COOT__STATICRR:
case COOTSTAYPUT__STATICRR:
case VIXEN__STATICRR:
case BIKERB__STATICRR:
case BIKERBV2__STATICRR:
case BIKER__STATICRR:
case MAKEOUT__STATICRR:
case CHEERB__STATICRR:
case CHEER__STATICRR:
case CHEERSTAYPUT__STATICRR:
case COOTPLAY__STATICRR:
case BILLYPLAY__STATICRR:
case MINIONBOAT__STATICRR:
case HULKBOAT__STATICRR:
case CHEERBOAT__STATICRR:
case RABBIT__STATICRR:
case MAMA__STATICRR:
if (g_changeEnemySize == 3)
{
sprite[enemySprite].xrepeat = sprite[enemySprite].xrepeat << 1;
sprite[enemySprite].yrepeat = sprite[enemySprite].yrepeat << 1;
sprite[enemySprite].clipdist = mulscale7(sprite[enemySprite].xrepeat, tilesiz[sprite[enemySprite].picnum].x);
}
else if (g_changeEnemySize == 2)
{
sprite[enemySprite].xrepeat = sprite[enemySprite].xrepeat >> 1;
sprite[enemySprite].yrepeat = sprite[enemySprite].yrepeat >> 1;
sprite[enemySprite].clipdist = mulscale7(sprite[enemySprite].xrepeat, tilesiz[sprite[enemySprite].picnum].x);
}
break;
}
}
g_changeEnemySize = 0;
}
spriteNum = headspritestat[121];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
sprite[spriteNum].extra++;
if (sprite[spriteNum].extra < 100)
{
if (sprite[spriteNum].extra == 90)
{
sprite[spriteNum].picnum--;
if (sprite[spriteNum].picnum < PIG + 7)
sprite[spriteNum].picnum = PIG + 7;
sprite[spriteNum].extra = 1;
}
vec3_t const vect = { 0, 0, -300 };
A_MoveSprite(spriteNum, &vect, CLIPMASK0);
if (sector[sprite[spriteNum].sectnum].ceilingz + (4 << 8) > sprite[spriteNum].z)
{
sprite[spriteNum].picnum = 0;
sprite[spriteNum].extra = 100;
}
}
else if (sprite[spriteNum].extra == 200)
{
vec3_t const pos = { sprite[spriteNum].x, sprite[spriteNum].y, sector[sprite[spriteNum].sectnum].floorz - 10 };
setsprite(spriteNum, &pos);
sprite[spriteNum].extra = 1;
sprite[spriteNum].picnum = PIG + 11;
A_Spawn(spriteNum, TRANSPORTERSTAR);
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[119];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].hitag > 0)
{
if (sprite[spriteNum].extra == 0)
{
sprite[spriteNum].hitag--;
sprite[spriteNum].extra = 150;
A_Spawn(spriteNum, RABBIT);
}
else
sprite[spriteNum].extra--;
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[116];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].extra)
{
if (sprite[spriteNum].extra == sprite[spriteNum].lotag)
S_PlaySound(183);
sprite[spriteNum].extra--;
vec3_t const vect = {
(sprite[spriteNum].hitag*sintable[(sprite[spriteNum].ang + 512) & 2047]) >> 14,
(sprite[spriteNum].hitag*sintable[sprite[spriteNum].ang & 2047]) >> 14,
sprite[spriteNum].hitag << 1
};
if (A_MoveSprite(spriteNum, &vect, CLIPMASK0) > 0)
{
A_PlaySound(PIPEBOMB_EXPLODE, spriteNum);
deletesprite(spriteNum);
}
if (sprite[spriteNum].extra == 0)
{
S_PlaySound(215);
deletesprite(spriteNum);
g_earthquakeTime = 32;
P_PalFrom(g_player[myconnectindex].ps, 48, 32, 32, 32);
}
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[115];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].extra)
{
if (sprite[spriteNum].picnum != RRTILE8162)
sprite[spriteNum].picnum = RRTILE8162;
sprite[spriteNum].extra--;
if (sprite[spriteNum].extra == 0)
{
int const fortune = krand2()&127;
if (fortune < 96)
{
sprite[spriteNum].picnum = RRTILE8162 + 3;
}
else if (fortune < 112)
{
if (g_slotWin & 1)
{
sprite[spriteNum].picnum = RRTILE8162 + 3;
}
else
{
sprite[spriteNum].picnum = RRTILE8162 + 2;
A_Spawn(spriteNum, BATTERYAMMO);
g_slotWin |= 1;
A_PlaySound(52, spriteNum);
}
}
else if (fortune < 120)
{
if (g_slotWin & 2)
{
sprite[spriteNum].picnum = RRTILE8162 + 3;
}
else
{
sprite[spriteNum].picnum = RRTILE8162 + 6;
A_Spawn(spriteNum, HEAVYHBOMB);
g_slotWin |= 2;
A_PlaySound(52, spriteNum);
}
}
else if (fortune < 126)
{
if (g_slotWin & 4)
{
sprite[spriteNum].picnum = RRTILE8162 + 3;
}
else
{
sprite[spriteNum].picnum = RRTILE8162 + 5;
A_Spawn(spriteNum, SIXPAK);
g_slotWin |= 4;
A_PlaySound(52, spriteNum);
}
}
else
{
if (g_slotWin & 8)
{
sprite[spriteNum].picnum = RRTILE8162 + 3;
}
else
{
sprite[spriteNum].picnum = RRTILE8162 + 4;
A_Spawn(spriteNum, ATOMICHEALTH);
g_slotWin |= 8;
A_PlaySound(52, spriteNum);
}
}
}
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[122];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].extra)
{
if (sprite[spriteNum].picnum != RRTILE8589)
sprite[spriteNum].picnum = RRTILE8589;
sprite[spriteNum].extra--;
if (sprite[spriteNum].extra == 0)
{
int const fortune = krand2()&127;
if (fortune < 96)
{
sprite[spriteNum].picnum = RRTILE8589 + 4;
}
else if (fortune < 112)
{
if (g_slotWin & 1)
{
sprite[spriteNum].picnum = RRTILE8589 + 4;
}
else
{
sprite[spriteNum].picnum = RRTILE8589 + 5;
A_Spawn(spriteNum, BATTERYAMMO);
g_slotWin |= 1;
A_PlaySound(342, spriteNum);
}
}
else if (fortune < 120)
{
if (g_slotWin & 2)
{
sprite[spriteNum].picnum = RRTILE8589 + 4;
}
else
{
sprite[spriteNum].picnum = RRTILE8589 + 6;
A_Spawn(spriteNum, HEAVYHBOMB);
g_slotWin |= 2;
A_PlaySound(342, spriteNum);
}
}
else if (fortune < 126)
{
if (g_slotWin & 4)
{
sprite[spriteNum].picnum = RRTILE8589 + 4;
}
else
{
sprite[spriteNum].picnum = RRTILE8589 + 2;
A_Spawn(spriteNum, SIXPAK);
g_slotWin |= 4;
A_PlaySound(342, spriteNum);
}
}
else
{
if (g_slotWin & 8)
{
sprite[spriteNum].picnum = RRTILE8589 + 4;
}
else
{
sprite[spriteNum].picnum = RRTILE8589 + 3;
A_Spawn(spriteNum, ATOMICHEALTH);
g_slotWin |= 8;
A_PlaySound(342, spriteNum);
}
}
}
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[123];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].lotag == 5)
if (!S_CheckSoundPlaying(spriteNum, 330))
A_PlaySound(330, spriteNum);
spriteNum = nextSprite;
}
}
if (RR)
{
spriteNum = headspritestat[107];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].hitag == 100)
{
sprite[spriteNum].z += (4 << 8);
if (sprite[spriteNum].z >= sector[sprite[spriteNum].sectnum].floorz + 15168)
sprite[spriteNum].z = sector[sprite[spriteNum].sectnum].floorz + 15168;
}
if (sprite[spriteNum].picnum == LUMBERBLADE)
{
sprite[spriteNum].extra++;
if (sprite[spriteNum].extra == 192)
{
sprite[spriteNum].hitag = 0;
sprite[spriteNum].z = sector[sprite[spriteNum].sectnum].floorz - 15168;
sprite[spriteNum].extra = 0;
sprite[spriteNum].picnum = RRTILE3410;
int otherSprite = headspritestat[0];
while (otherSprite >= 0)
{
int const nextOtherSprite = nextspritestat[otherSprite];
if (sprite[otherSprite].picnum == DIPSWITCH3 + 1)
if (sprite[otherSprite].hitag == 999)
sprite[otherSprite].picnum = DIPSWITCH3;
otherSprite = nextOtherSprite;
}
}
}
spriteNum = nextSprite;
}
if (g_chickenPlant)
{
spriteNum = headspritestat[106];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
switch (DYNAMICTILEMAP(sprite[spriteNum].picnum))
{
case RRTILE285__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3190);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 128;
}
break;
case RRTILE286__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3192);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 256;
}
break;
case RRTILE287__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
A_SpawnMultiple(spriteNum, MONEY, (krand2()&3)+4);
sprite[spriteNum].lotag = 84;
}
break;
case RRTILE288__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3132);
sprite[spriteNum].lotag = 96;
if (!RRRA)
A_PlaySound(472, newSprite);
}
break;
case RRTILE289__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3120);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 448;
}
break;
case RRTILE290__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3122);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 64;
}
break;
case RRTILE291__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3123);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 512;
}
break;
case RRTILE292__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
int const newSprite = A_Spawn(spriteNum, RRTILE3124);
sprite[newSprite].ang = sprite[spriteNum].ang;
sprite[spriteNum].lotag = 224;
}
break;
case RRTILE293__STATICRR:
sprite[spriteNum].lotag--;
if (sprite[spriteNum].lotag < 0)
{
A_DoGuts(spriteNum, JIBS1, 1);
A_DoGuts(spriteNum, JIBS2, 1);
A_DoGuts(spriteNum, JIBS3, 1);
A_DoGuts(spriteNum, JIBS4, 1);
sprite[spriteNum].lotag = 256;
}
break;
}
spriteNum = nextSprite;
}
}
spriteNum = headspritestat[105];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
if (sprite[spriteNum].picnum == RRTILE280)
if (sprite[spriteNum].lotag == 100)
{
if (A_PinSectorResetUp(SECT(spriteNum)))
{
sprite[spriteNum].lotag = 0;
if (sprite[spriteNum].extra == 1)
{
if (!A_CheckPins(SECT(spriteNum)))
{
sprite[spriteNum].extra = 2;
}
}
if (sprite[spriteNum].extra == 2)
{
sprite[spriteNum].extra = 0;
A_ResetPins(SECT(spriteNum));
}
}
}
spriteNum = nextSprite;
}
spriteNum = headspritestat[108];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite= &sprite[spriteNum];
if (pSprite->picnum == RRTILE296)
{
int playerDist;
DukePlayer_t *const pPlayer = g_player[A_FindPlayer(pSprite, &playerDist)].ps;
if (playerDist < 2047)
{
int otherSprite = headspritestat[108];
while (otherSprite >= 0)
{
int const nextOtherSprite = nextspritestat[otherSprite];
if (sprite[otherSprite].picnum == RRTILE297)
{
pPlayer->q16ang = F16(sprite[otherSprite].ang);
pPlayer->pos = *(vec3_t*)&sprite[otherSprite];
pPlayer->pos.z -= (36<<8);
pPlayer->opos = pPlayer->pos;
pPlayer->bobpos = *(vec2_t*)&pPlayer->pos;
changespritesect(pPlayer->i, sprite[otherSprite].sectnum);
pPlayer->cursectnum = sprite[otherSprite].sectnum;
A_PlaySound(TELEPORTER, otherSprite);
A_DeleteSprite(otherSprite);
}
otherSprite = nextOtherSprite;
}
}
}
spriteNum = nextSprite;
}
}
spriteNum = headspritestat[STAT_ACTOR];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite = &sprite[spriteNum];
int const sectNum = pSprite->sectnum;
int32_t *const pData = actor[spriteNum].t_data;
int deleteAfterExecute = 0;
int switchPic;
if (pSprite->xrepeat == 0 || sectNum < 0 || sectNum >= MAXSECTORS)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
Bmemcpy(&actor[spriteNum].bpos, pSprite, sizeof(vec3_t));
switchPic = pSprite->picnum;
if (!RR && pSprite->picnum > GREENSLIME && pSprite->picnum <= GREENSLIME+7)
switchPic = GREENSLIME;
if (RR && (switchPic == RRTILE3440 + 1 || switchPic == HENSTAND + 1))
switchPic--;
switch (DYNAMICTILEMAP(switchPic))
{
case DUCK__STATIC:
case TARGET__STATIC:
if (RR) break;
if (pSprite->cstat&32)
{
pData[0]++;
if (pData[0] > 60)
{
pData[0] = 0;
pSprite->cstat = 128+257+16;
pSprite->extra = 1;
}
}
else
{
if (A_IncurDamage(spriteNum) >= 0)
{
int doEffects = 1;
pSprite->cstat = 32+128;
for (bssize_t SPRITES_OF(STAT_ACTOR, actorNum))
{
if ((sprite[actorNum].lotag == pSprite->lotag && sprite[actorNum].picnum == pSprite->picnum)
&& ((sprite[actorNum].hitag != 0) ^ ((sprite[actorNum].cstat & 32) != 0)))
{
doEffects = 0;
break;
}
}
if (doEffects == 1)
{
G_OperateActivators(pSprite->lotag, -1);
G_OperateForceFields(spriteNum, pSprite->lotag);
G_OperateMasterSwitches(pSprite->lotag);
}
}
}
goto next_sprite;
case RESPAWNMARKERRED__STATIC:
case RESPAWNMARKERYELLOW__STATIC:
case RESPAWNMARKERGREEN__STATIC:
if (++T1(spriteNum) > g_itemRespawnTime)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (T1(spriteNum) >= (g_itemRespawnTime>>1) && T1(spriteNum) < ((g_itemRespawnTime>>1)+(g_itemRespawnTime>>2)))
PN(spriteNum) = RESPAWNMARKERYELLOW;
else if (T1(spriteNum) > ((g_itemRespawnTime>>1)+(g_itemRespawnTime>>2)))
PN(spriteNum) = RESPAWNMARKERGREEN;
A_Fall(spriteNum);
break;
case HELECOPT__STATIC:
case DUKECAR__STATIC:
if (RR) break;
pSprite->z += pSprite->zvel;
pData[0]++;
if (pData[0] == 4)
A_PlaySound(WAR_AMBIENCE2,spriteNum);
if (pData[0] > (GAMETICSPERSEC*8))
{
g_earthquakeTime = 16;
S_PlaySound(RPG_EXPLODE);
for (bssize_t j = 0; j < 32; j++)
RANDOMSCRAP(pSprite, spriteNum);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if ((pData[0]&3) == 0)
A_Spawn(spriteNum,EXPLOSION2);
A_SetSprite(spriteNum,CLIPMASK0);
break;
case RAT__STATIC:
A_Fall(spriteNum);
if (A_SetSprite(spriteNum, CLIPMASK0))
{
if (!RRRA && (krand2()&255) < 3) A_PlaySound(RATTY,spriteNum);
pSprite->ang += (krand2()&31)-15+(sintable[(pData[0]<<8)&2047]>>11);
}
else
{
T1(spriteNum)++;
if (T1(spriteNum) > 1)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else pSprite->ang = (krand2()&2047);
}
if (pSprite->xvel < 128)
pSprite->xvel+=2;
pSprite->ang += (krand2()&3)-6;
break;
case RRTILE3190__STATICRR:
case RRTILE3191__STATICRR:
case RRTILE3192__STATICRR:
{
if (!g_chickenPlant) DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (sector[SECT(spriteNum)].lotag == 903)
A_Fall(spriteNum);
vec3_t const vect = {
(pSprite->xvel*(sintable[(pSprite->ang + 512) & 2047])) >> 14,
(pSprite->xvel*(sintable[pSprite->ang & 2047])) >> 14,
pSprite->zvel
};
int const moveSprite = A_MoveSprite(spriteNum, &vect, CLIPMASK0);
switch (sector[SECT(spriteNum)].lotag)
{
case 901:
sprite[spriteNum].picnum = RRTILE3191;
break;
case 902:
sprite[spriteNum].picnum = RRTILE3192;
break;
case 903:
if (SZ(spriteNum) >= sector[SECT(spriteNum)].floorz - ZOFFSET3) { DELETE_SPRITE_AND_CONTINUE(spriteNum); }
break;
case 904:
DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
}
if (moveSprite & 49152)
{
if ((moveSprite & 49152) == 32768) { DELETE_SPRITE_AND_CONTINUE(spriteNum); }
else if ((moveSprite & 49152) == 49152) { DELETE_SPRITE_AND_CONTINUE(spriteNum); }
}
break;
}
case RRTILE3120__STATICRR:
case RRTILE3122__STATICRR:
case RRTILE3123__STATICRR:
case RRTILE3124__STATICRR:
{
if (!g_chickenPlant) DELETE_SPRITE_AND_CONTINUE(spriteNum);
A_Fall(spriteNum);
vec3_t const vect = {
(pSprite->xvel*(sintable[(pSprite->ang + 512) & 2047])) >> 14,
(pSprite->xvel*(sintable[pSprite->ang & 2047])) >> 14,
pSprite->zvel
};
int const moveSprite = A_MoveSprite(spriteNum, &vect, CLIPMASK0);
if (moveSprite & 49152)
{
if ((moveSprite & 49152) == 32768) { DELETE_SPRITE_AND_CONTINUE(spriteNum); }
else if ((moveSprite & 49152) == 49152) { DELETE_SPRITE_AND_CONTINUE(spriteNum); }
}
if (sector[pSprite->sectnum].lotag == 903)
{
if (SZ(spriteNum) >= sector[SECT(spriteNum)].floorz - (4 << 8))
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
else if (sector[pSprite->sectnum].lotag == 904)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
break;
}
case RRTILE3132__STATICRR:
{
if (!g_chickenPlant) DELETE_SPRITE_AND_CONTINUE(spriteNum);
A_Fall(spriteNum);
vec3_t const vect = {
(pSprite->xvel*(sintable[(pSprite->ang + 512) & 2047])) >> 14,
(pSprite->xvel*(sintable[pSprite->ang & 2047])) >> 14,
pSprite->zvel
};
A_MoveSprite(spriteNum, &vect, CLIPMASK0);
if (pSprite->z >= sector[pSprite->sectnum].floorz - (8 << 8))
{
if (sector[pSprite->sectnum].lotag == 1)
{
int const newSprite = A_Spawn(spriteNum, WATERSPLASH2);
sprite[newSprite].z = sector[sprite[newSprite].sectnum].floorz;
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
break;
}
case BOWLINGBALL__STATICRR:
if (pSprite->xvel)
{
if (!A_CheckSoundPlaying(spriteNum, 356))
A_PlaySound(356, spriteNum);
}
else
{
A_Spawn(spriteNum, BOWLINGBALLSPRITE);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (sector[pSprite->sectnum].lotag == 900)
{
S_StopEnvSound(356, spriteNum);
}
fallthrough__;
case RRTILE3440__STATICRR:
case HENSTAND__STATICRR:
if (pSprite->picnum == HENSTAND || pSprite->picnum == HENSTAND + 1)
{
if (--pSprite->lotag == 0)
{
A_Spawn(spriteNum, HEN);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
if (sector[pSprite->sectnum].lotag == 900)
pSprite->xvel = 0;
if (pSprite->xvel)
{
A_Fall(spriteNum);
vec3_t const vect = {
(pSprite->xvel*(sintable[(pSprite->ang + 512) & 2047])) >> 14,
(pSprite->xvel*(sintable[pSprite->ang & 2047])) >> 14,
pSprite->zvel
};
int moveSprite = A_MoveSprite(spriteNum, &vect, CLIPMASK0);
if (moveSprite & 49152)
{
if ((moveSprite & 49152) == 32768)
{
moveSprite &= (MAXWALLS - 1);
Proj_BounceOffWall(pSprite, moveSprite);
}
else if ((moveSprite & 49152) == 49152)
{
moveSprite &= (MAXSPRITES - 1);
A_DamageObject(spriteNum, moveSprite);
if (sprite[moveSprite].picnum == HEN)
{
int const newSprite = A_Spawn(moveSprite, HENSTAND);
A_DeleteSprite(moveSprite);
sprite[newSprite].xvel = 32;
sprite[newSprite].lotag = 40;
sprite[newSprite].ang = pSprite->ang;
}
}
}
if (--pSprite->xvel < 0) pSprite->xvel = 0;
pSprite->cstat = 257;
if (pSprite->picnum == RRTILE3440)
{
pSprite->cstat |= 4 & pSprite->xvel;
pSprite->cstat |= 8 & pSprite->xvel;
if (krand2() & 1)
pSprite->picnum = RRTILE3440 + 1;
}
else if (pSprite->picnum == HENSTAND)
{
pSprite->cstat |= 4 & pSprite->xvel;
pSprite->cstat |= 8 & pSprite->xvel;
if (krand2() & 1)
pSprite->picnum = HENSTAND + 1;
if (!pSprite->xvel)
deleteAfterExecute = 1;
}
if (pSprite->picnum == RRTILE3440 || (pSprite->picnum == RRTILE3440 + 1 && !pSprite->xvel))
deleteAfterExecute = 1;
}
else if (sector[pSprite->sectnum].lotag == 900)
{
if (pSprite->picnum == BOWLINGBALL)
A_BallReturn(spriteNum);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
break;
case QUEBALL__STATIC:
case STRIPEBALL__STATIC:
if (pSprite->xvel)
{
for (bssize_t SPRITES_OF(STAT_DEFAULT, hitObject))
if (sprite[hitObject].picnum == POCKET && ldist(&sprite[hitObject],pSprite) < 52)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
int hitObject = clipmove((vec3_t *)pSprite, &pSprite->sectnum,
(((pSprite->xvel * (sintable[(pSprite->ang + 512) & 2047])) >> 14) * TICSPERFRAME) << 11,
(((pSprite->xvel * (sintable[pSprite->ang & 2047])) >> 14) * TICSPERFRAME) << 11, 24L, ZOFFSET6,
ZOFFSET6, CLIPMASK1);
if (hitObject & 49152)
{
if ((hitObject & 49152) == 32768)
{
hitObject &= (MAXWALLS - 1);
Proj_BounceOffWall(pSprite, hitObject);
}
else if ((hitObject & 49152) == 49152)
{
hitObject &= (MAXSPRITES - 1);
A_DamageObject(spriteNum, hitObject);
}
}
if (--pSprite->xvel < 0)
pSprite->xvel = 0;
if (pSprite->picnum == STRIPEBALL)
{
pSprite->cstat = 257;
pSprite->cstat |= (4 & pSprite->xvel) | (8 & pSprite->xvel);
}
}
else
{
int32_t playerDist;
int const playerNum = A_FindPlayer(pSprite,&playerDist);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
// I'm 50/50 on this being either a typo or a stupid hack
if (playerDist < 1596)
{
int const angDiff = G_GetAngleDelta(fix16_to_int(pPlayer->q16ang),getangle(pSprite->x-pPlayer->pos.x,pSprite->y-pPlayer->pos.y));
if (angDiff > -64 && angDiff < 64 && TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_OPEN)
&& pPlayer->toggle_key_flag == 1)
{
int ballSprite;
for (SPRITES_OF(STAT_ACTOR, ballSprite))
{
if (sprite[ballSprite].picnum == QUEBALL || sprite[ballSprite].picnum == STRIPEBALL)
{
int const angDiff2 = G_GetAngleDelta(
fix16_to_int(pPlayer->q16ang), getangle(sprite[ballSprite].x - pPlayer->pos.x, sprite[ballSprite].y - pPlayer->pos.y));
if (angDiff2 > -64 && angDiff2 < 64)
{
int32_t ballDist;
A_FindPlayer(&sprite[ballSprite], &ballDist);
if (playerDist > ballDist)
break;
}
}
}
if (ballSprite == -1)
{
pSprite->xvel = (pSprite->pal == 12) ? 164 : 140;
pSprite->ang = fix16_to_int(pPlayer->q16ang);
pPlayer->toggle_key_flag = 2;
}
}
}
if (playerDist < 512 && pSprite->sectnum == pPlayer->cursectnum)
{
pSprite->ang = getangle(pSprite->x-pPlayer->pos.x,pSprite->y-pPlayer->pos.y);
pSprite->xvel = 48;
}
}
break;
case FORCESPHERE__STATIC:
if (pSprite->yvel == 0)
{
pSprite->yvel = 1;
for (bssize_t l = 512; l < (2048 - 512); l += 128)
{
for (bssize_t j = 0; j < 2048; j += 128)
{
int const newSprite = A_Spawn(spriteNum, FORCESPHERE);
sprite[newSprite].cstat = 257 + 128;
sprite[newSprite].clipdist = 64;
sprite[newSprite].ang = j;
sprite[newSprite].zvel = sintable[l & 2047] >> 5;
sprite[newSprite].xvel = sintable[(l + 512) & 2047] >> 9;
sprite[newSprite].owner = spriteNum;
}
}
}
if (pData[3] > 0)
{
if (pSprite->zvel < 6144)
pSprite->zvel += 192;
pSprite->z += pSprite->zvel;
if (pSprite->z > sector[sectNum].floorz)
pSprite->z = sector[sectNum].floorz;
if (--pData[3] == 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if (pData[2] > 10)
{
for (bssize_t SPRITES_OF(STAT_MISC, miscSprite))
{
if (sprite[miscSprite].owner == spriteNum && sprite[miscSprite].picnum == FORCESPHERE)
actor[miscSprite].t_data[1] = 1 + (krand2() & 63);
}
pData[3] = 64;
}
goto next_sprite;
case RECON__STATIC:
case UFO1__STATICRR:
case UFO2__STATICRR:
case UFO3__STATICRR:
case UFO4__STATICRR:
case UFO5__STATICRR:
{
int playerNum;
DukePlayer_t *pPlayer;
A_GetZLimits(spriteNum);
pSprite->shade += (sector[pSprite->sectnum].ceilingstat & 1) ? (sector[pSprite->sectnum].ceilingshade - pSprite->shade) >> 1
: (sector[pSprite->sectnum].floorshade - pSprite->shade) >> 1;
if (pSprite->z < sector[sectNum].ceilingz + ZOFFSET5)
pSprite->z = sector[sectNum].ceilingz + ZOFFSET5;
#if 0 //def POLYMER
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].sector = s->sectnum;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].x = s->x;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].y = s->y;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].z = s->z + 10248;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].range = 8192;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].angle = s->ang;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].horiz = 100;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].radius = 256;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].faderadius = 200;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].color[0] = 255;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].color[1] = 255;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].color[2] = 255;
gamelights[gamelightcount&(PR_MAXLIGHTS-1)].priority = PR_LIGHT_PRIO_MAX_GAME;
if (gamelightcount < PR_MAXLIGHTS)
gamelightcount++;
#endif
if (!g_netServer && ud.multimode < 2)
{
if (g_noEnemies == 1)
{
pSprite->cstat = 32768;
goto next_sprite;
}
else if (g_noEnemies == 2) pSprite->cstat = 257;
}
if (A_IncurDamage(spriteNum) >= 0)
{
if (pSprite->extra < 0 && pData[0] != -1)
{
pData[0] = -1;
pSprite->extra = 0;
}
if (!RR)
A_PlaySound(RECO_PAIN,spriteNum);
RANDOMSCRAP(pSprite, spriteNum);
}
if (pData[0] == -1)
{
pSprite->z += 1024;
pData[2]++;
if ((pData[2]&3) == 0)
A_Spawn(spriteNum,EXPLOSION2);
A_GetZLimits(spriteNum);
pSprite->ang += 96;
pSprite->xvel = 128;
if (A_SetSprite(spriteNum, CLIPMASK0) != 1 || pSprite->z > actor[spriteNum].floorz)
{
for (bssize_t l = 0; l < 16; l++)
RANDOMSCRAP(pSprite, spriteNum);
//int const newSprite = A_Spawn(spriteNum, EXPLOSION2);
A_PlaySound(LASERTRIP_EXPLODE, spriteNum);
if (RR)
{
if (RRRA && g_ufoSpawnMinion)
A_Spawn(spriteNum, MINION);
else if (pSprite->picnum == UFO1)
A_Spawn(spriteNum, HEN);
else if (pSprite->picnum == UFO2)
A_Spawn(spriteNum, COOT);
else if (pSprite->picnum == UFO3)
A_Spawn(spriteNum, COW);
else if (pSprite->picnum == UFO4)
A_Spawn(spriteNum, PIG);
else if (pSprite->picnum == UFO5)
A_Spawn(spriteNum, BILLYRAY);
}
else
A_Spawn(spriteNum, PIGCOP);
P_AddKills(g_player[myconnectindex].ps, 1);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
}
else
{
if (pSprite->z > actor[spriteNum].floorz-(48<<8))
pSprite->z = actor[spriteNum].floorz-(48<<8);
}
int32_t playerDist;
playerNum = A_FindPlayer(pSprite, &playerDist);
pPlayer = g_player[playerNum].ps;
int const spriteOwner = pSprite->owner;
// 3 = findplayerz, 4 = shoot
if (pData[0] >= 4)
{
if ((++pData[2] & 15) == 0)
{
int const saveAng = pSprite->ang;
pSprite->ang = actor[spriteNum].tempang;
if (!RR)
A_PlaySound(RECO_ATTACK, spriteNum);
A_Shoot(spriteNum, FIRELASER);
pSprite->ang = saveAng;
}
if (pData[2] > (GAMETICSPERSEC * 3)
|| !cansee(pSprite->x, pSprite->y, pSprite->z - ZOFFSET2, pSprite->sectnum, pPlayer->pos.x, pPlayer->pos.y,
pPlayer->pos.z, pPlayer->cursectnum))
{
pData[0] = 0;
pData[2] = 0;
}
else actor[spriteNum].tempang += G_GetAngleDelta(actor[spriteNum].tempang,
getangle(pPlayer->pos.x - pSprite->x,
pPlayer->pos.y - pSprite->y)) / 3;
}
else if (pData[0] == 2 || pData[0] == 3)
{
pData[3] = 0;
pSprite->xvel = (pSprite->xvel > 0) ? pSprite->xvel - 16 : 0;
if (pData[0] == 2)
{
int const zDiff = pPlayer->pos.z - pSprite->z;
if (klabs(zDiff) < (48 << 8))
pData[0] = 3;
else
pSprite->z += ksgn(pPlayer->pos.z - pSprite->z) << (RR ? 8 : 10);
}
else
{
pData[2]++;
if (pData[2] > (GAMETICSPERSEC*3) ||
!cansee(pSprite->x,pSprite->y,pSprite->z-ZOFFSET2,pSprite->sectnum, pPlayer->pos.x,pPlayer->pos.y,pPlayer->pos.z,pPlayer->cursectnum))
{
pData[0] = 1;
pData[2] = 0;
}
else if ((pData[2]&15) == 0)
{
if (!RR)
A_PlaySound(RECO_ATTACK,spriteNum);
A_Shoot(spriteNum,FIRELASER);
}
}
pSprite->ang += G_GetAngleDelta(pSprite->ang, getangle(pPlayer->pos.x - pSprite->x, pPlayer->pos.y - pSprite->y)) >> 2;
}
if (pData[0] != 2 && pData[0] != 3)
{
int newAngle;
int locatorDist = ldist(&sprite[spriteOwner], pSprite);
if (locatorDist <= 1524)
{
newAngle = pSprite->ang;
pSprite->xvel >>= 1;
}
else newAngle = getangle(sprite[spriteOwner].x - pSprite->x, sprite[spriteOwner].y - pSprite->y);
if (pData[0] == 1 || pData[0] == 4) // Found a locator and going with it
{
locatorDist = dist(&sprite[spriteOwner], pSprite);
if (locatorDist <= 1524)
{
pData[0] = (pData[0] == 1) ? 0 : 5;
}
else
{
// Control speed here
if (pSprite->xvel < 256) pSprite->xvel += 32;
}
if (pData[0] < 2) pData[2]++;
if (playerDist < 6144 && pData[0] < 2 && pData[2] > (GAMETICSPERSEC*4))
{
pData[0] = 2+(krand2()&2);
pData[2] = 0;
actor[spriteNum].tempang = pSprite->ang;
}
}
int locatorSprite = pSprite->owner;
if (pData[0] == 0 || pData[0] == 5)
{
pData[0] = (pData[0] == 0) ? 1 : 4;
pSprite->owner = A_FindLocator(pSprite->hitag, -1);
locatorSprite = pSprite->owner;
if (locatorSprite == -1)
{
locatorSprite = actor[spriteNum].t_data[5];
pSprite->hitag = locatorSprite;
pSprite->owner = A_FindLocator(locatorSprite, -1);
locatorSprite = pSprite->owner;
if (locatorSprite == -1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else pSprite->hitag++;
}
// RECON_T4
pData[3] = G_GetAngleDelta(pSprite->ang,newAngle);
pSprite->ang += pData[3]>>3;
if (pSprite->z < sprite[locatorSprite].z)
pSprite->z += 1024;
else
pSprite->z -= 1024;
}
int sndNum = RR ? 457 : RECO_ROAM;
if (!A_CheckSoundPlaying(spriteNum,sndNum))
A_PlaySound(sndNum,spriteNum);
A_SetSprite(spriteNum,CLIPMASK0);
goto next_sprite;
}
case OOZ2__STATIC:
if (RR) break;
fallthrough__;
case OOZ__STATIC:
{
A_GetZLimits(spriteNum);
int const yrepeat = min((actor[spriteNum].floorz - actor[spriteNum].ceilingz) >> 9, 255);
int const xrepeat = clamp(25 - (yrepeat >> 1), 8, 48);
pSprite->yrepeat = yrepeat;
pSprite->xrepeat = xrepeat;
pSprite->z = actor[spriteNum].floorz;
goto next_sprite;
}
case GREENSLIME__STATIC:
{
if (RR) break;
// #ifndef VOLUMEONE
if (!g_netServer && ud.multimode < 2)
{
if (g_noEnemies == 1)
{
pSprite->cstat = 32768;
goto next_sprite;
}
else if (g_noEnemies == 2) pSprite->cstat = 257;
}
// #endif
pData[1]+=128;
if (sector[sectNum].floorstat&1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
int32_t playerDist;
int const playerNum = A_FindPlayer(pSprite, &playerDist);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (playerDist > 20480)
{
if (++actor[spriteNum].timetosleep > SLEEPTIME)
{
actor[spriteNum].timetosleep = 0;
changespritestat(spriteNum, STAT_ZOMBIEACTOR);
goto next_sprite;
}
}
if (pData[0] == -5) // FROZEN
{
pData[3]++;
if (pData[3] > 280)
{
pSprite->pal = 0;
pData[0] = 0;
goto next_sprite;
}
A_Fall(spriteNum);
pSprite->cstat = 257;
pSprite->picnum = GREENSLIME + 2;
pSprite->extra = 1;
pSprite->pal = 1;
int const damageTile = A_IncurDamage(spriteNum);
if (damageTile >= 0)
{
if (damageTile == FREEZEBLAST)
goto next_sprite;
P_AddKills(pPlayer, 1);
for (bssize_t j = 16; j >= 0; --j)
{
int32_t const r1 = krand2(), r2 = krand2(), r3 = krand2();
int32_t newSprite = A_InsertSprite(SECT(spriteNum), SX(spriteNum), SY(spriteNum), SZ(spriteNum),
GLASSPIECES + (j % 3), -32, 36, 36, r3 & 2047, 32 + (r2 & 63),
1024 - (r1 & 1023), spriteNum, 5);
sprite[newSprite].pal = 1;
}
A_PlaySound(GLASS_BREAKING, spriteNum);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if (playerDist < 1024 && pPlayer->quick_kick == 0)
{
int const angDiff = G_GetAngleDelta(fix16_to_int(pPlayer->q16ang), getangle(SX(spriteNum) - pPlayer->pos.x,
SY(spriteNum) - pPlayer->pos.y));
if (angDiff > -128 && angDiff < 128)
pPlayer->quick_kick = 14;
}
goto next_sprite;
}
pSprite->cstat = (playerDist < 1596) ? 0 : 257;
if (pData[0] == -4) //On the player
{
if (sprite[pPlayer->i].extra < 1)
{
pData[0] = 0;
goto next_sprite;
}
setsprite(spriteNum,(vec3_t *)pSprite);
pSprite->ang = fix16_to_int(pPlayer->q16ang);
if ((TEST_SYNC_KEY(g_player[playerNum].inputBits->bits, SK_FIRE) || (pPlayer->quick_kick > 0)) && sprite[pPlayer->i].extra > 0)
if (pPlayer->quick_kick > 0 ||
(pPlayer->curr_weapon != HANDREMOTE_WEAPON && pPlayer->curr_weapon != HANDBOMB_WEAPON &&
pPlayer->curr_weapon != TRIPBOMB_WEAPON && pPlayer->ammo_amount[pPlayer->curr_weapon] >= 0))
{
for (bssize_t x = 0; x < 8; ++x)
{
int32_t const r1 = krand2(), r2 = krand2(), r3 = krand2(), r4 = krand2();
int const j
= A_InsertSprite(sectNum, pSprite->x, pSprite->y, pSprite->z - ZOFFSET3, SCRAP3 + (r4 & 3), -8, 48, 48,
r3 & 2047, (r2 & 63) + 64, -(r1 & 4095) - (pSprite->zvel >> 2), spriteNum, 5);
sprite[j].pal = 6;
}
A_PlaySound(SLIM_DYING,spriteNum);
A_PlaySound(SQUISHED,spriteNum);
if ((krand2()&255) < 32)
{
int const j = A_Spawn(spriteNum,BLOODPOOL);
sprite[j].pal = 0;
}
P_AddKills(pPlayer, 1);
pData[0] = -3;
if (pPlayer->somethingonplayer == spriteNum)
pPlayer->somethingonplayer = -1;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
pSprite->z = pPlayer->pos.z + pPlayer->pyoff - pData[2] + ZOFFSET3 + (fix16_to_int(F16(100) - pPlayer->q16horiz) << 4);
if (pData[2] > 512)
pData[2] -= 128;
if (pData[2] < 348)
pData[2] += 128;
if (pPlayer->newowner >= 0)
G_ClearCameraView(pPlayer);
if (pData[3] > 0)
{
static const char slimeFrames[] = { 5, 5, 6, 6, 7, 7, 6, 5 };
pSprite->picnum = GREENSLIME + slimeFrames[pData[3]];
if (pData[3] == 5)
{
sprite[pPlayer->i].extra += -(5 + (krand2() & 3));
A_PlaySound(SLIM_ATTACK, spriteNum);
}
if (pData[3] < 7)
pData[3]++;
else
pData[3] = 0;
}
else
{
pSprite->picnum = GREENSLIME + 5;
if (rnd(32))
pData[3] = 1;
}
pSprite->xrepeat = 20 + (sintable[pData[1] & 2047] >> 13);
pSprite->yrepeat = 15 + (sintable[pData[1] & 2047] >> 13);
pSprite->x = pPlayer->pos.x + (sintable[(fix16_to_int(pPlayer->q16ang) + 512) & 2047] >> 7);
pSprite->y = pPlayer->pos.y + (sintable[fix16_to_int(pPlayer->q16ang) & 2047] >> 7);
goto next_sprite;
}
else if (pSprite->xvel < 64 && playerDist < 768)
{
if (pPlayer->somethingonplayer == -1)
{
pPlayer->somethingonplayer = spriteNum;
if (pData[0] == 3 || pData[0] == 2) // Falling downward
pData[2] = (12 << 8);
else
pData[2] = -(13 << 8); // Climbing up player
pData[0] = -4;
}
}
int const damageTile = A_IncurDamage(spriteNum);
if (damageTile >= 0)
{
A_PlaySound(SLIM_DYING,spriteNum);
P_AddKills(pPlayer, 1);
if (pPlayer->somethingonplayer == spriteNum)
pPlayer->somethingonplayer = -1;
if (damageTile == FREEZEBLAST)
{
A_PlaySound(SOMETHINGFROZE, spriteNum);
pData[0] = -5;
pData[3] = 0;
goto next_sprite;
}
if ((krand2()&255) < 32)
{
int const j = A_Spawn(spriteNum,BLOODPOOL);
sprite[j].pal = 0;
}
for (bssize_t x=0; x<8; x++)
{
int32_t const r1 = krand2(), r2 = krand2(), r3 = krand2(), r4 = krand2();
int const j = A_InsertSprite(sectNum, pSprite->x, pSprite->y, pSprite->z - ZOFFSET3, SCRAP3 + (r4 & 3), -8,
48, 48, r3 & 2047, (r2 & 63) + 64, -(r1 & 4095) - (pSprite->zvel >> 2),
spriteNum, 5);
sprite[j].pal = 6;
}
pData[0] = -3;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
// All weap
if (pData[0] == -1) //Shrinking down
{
A_Fall(spriteNum);
pSprite->cstat &= 65535-8;
pSprite->picnum = GREENSLIME+4;
// if(s->yrepeat > 62)
// A_DoGuts(s,JIBS6,5,myconnectindex);
if (pSprite->xrepeat > 32) pSprite->xrepeat -= krand2()&7;
if (pSprite->yrepeat > 16) pSprite->yrepeat -= krand2()&7;
else
{
pSprite->xrepeat = 40;
pSprite->yrepeat = 16;
pData[5] = -1;
pData[0] = 0;
}
goto next_sprite;
}
else if (pData[0] != -2) A_GetZLimits(spriteNum);
if (pData[0] == -2) //On top of somebody
{
A_Fall(spriteNum);
sprite[pData[5]].xvel = 0;
int const ang = sprite[pData[5]].ang;
pSprite->x = sprite[pData[5]].x + (sintable[(ang + 512) & 2047] >> 11);
pSprite->y = sprite[pData[5]].y + (sintable[ang & 2047] >> 11);
pSprite->z = sprite[pData[5]].z;
pSprite->picnum = GREENSLIME + 2 + (g_globalRandom & 1);
if (pSprite->yrepeat < 64)
pSprite->yrepeat += 2;
else
{
if (pSprite->xrepeat < 32)
pSprite->xrepeat += 4;
else
{
pData[0] = -1;
playerDist = ldist(pSprite, &sprite[pData[5]]);
if (playerDist < 768)
sprite[pData[5]].xrepeat = 0;
}
}
goto next_sprite;
}
//Check randomly to see of there is an actor near
if (rnd(32))
{
for (bssize_t SPRITES_OF_SECT(sectNum, j))
{
if (A_CheckSpriteFlags(j, SFLAG_GREENSLIMEFOOD))
{
if (ldist(pSprite, &sprite[j]) < 768 && (klabs(pSprite->z - sprite[j].z) < 8192)) // Gulp them
{
pData[5] = j;
pData[0] = -2;
pData[1] = 0;
goto next_sprite;
}
}
}
}
//Moving on the ground or ceiling
if (pData[0] == 0 || pData[0] == 2)
{
pSprite->picnum = GREENSLIME;
if ((krand2()&511) == 0)
A_PlaySound(SLIM_ROAM,spriteNum);
if (pData[0]==2)
{
pSprite->zvel = 0;
pSprite->cstat &= (65535-8);
if ((sector[sectNum].ceilingstat&1) || (actor[spriteNum].ceilingz+6144) < pSprite->z)
{
pSprite->z += 2048;
pData[0] = 3;
goto next_sprite;
}
}
else
{
pSprite->cstat |= 8;
A_Fall(spriteNum);
}
if (everyothertime&1) A_SetSprite(spriteNum,CLIPMASK0);
if (pSprite->xvel > 96)
{
pSprite->xvel -= 2;
goto next_sprite;
}
else
{
if (pSprite->xvel < 32) pSprite->xvel += 4;
pSprite->xvel = 64 - (sintable[(pData[1]+512)&2047]>>9);
pSprite->ang += G_GetAngleDelta(pSprite->ang,
getangle(pPlayer->pos.x-pSprite->x,pPlayer->pos.y-pSprite->y))>>3;
// TJR
}
pSprite->xrepeat = 36 + (sintable[(pData[1]+512)&2047]>>11);
pSprite->yrepeat = 16 + (sintable[pData[1]&2047]>>13);
if (rnd(4) && (sector[sectNum].ceilingstat&1) == 0 &&
klabs(actor[spriteNum].floorz-actor[spriteNum].ceilingz)
< (192<<8))
{
pSprite->zvel = 0;
pData[0]++;
}
}
if (pData[0]==1)
{
pSprite->picnum = GREENSLIME;
if (pSprite->yrepeat < 40) pSprite->yrepeat+=8;
if (pSprite->xrepeat > 8) pSprite->xrepeat-=4;
if (pSprite->zvel > -(2048+1024))
pSprite->zvel -= 348;
pSprite->z += pSprite->zvel;
if (pSprite->z < actor[spriteNum].ceilingz+4096)
{
pSprite->z = actor[spriteNum].ceilingz+4096;
pSprite->xvel = 0;
pData[0] = 2;
}
}
if (pData[0]==3)
{
pSprite->picnum = GREENSLIME+1;
A_Fall(spriteNum);
if (pSprite->z > actor[spriteNum].floorz-ZOFFSET3)
{
pSprite->yrepeat-=4;
pSprite->xrepeat+=2;
}
else
{
if (pSprite->yrepeat < (40-4)) pSprite->yrepeat+=8;
if (pSprite->xrepeat > 8) pSprite->xrepeat-=4;
}
if (pSprite->z > actor[spriteNum].floorz-2048)
{
pSprite->z = actor[spriteNum].floorz-2048;
pData[0] = 0;
pSprite->xvel = 0;
}
}
goto next_sprite;
}
case EMPTYBIKE__STATICRR:
if(!RRRA) break;
A_Fall(spriteNum);
A_GetZLimits(spriteNum);
if (sector[sectNum].lotag == ST_1_ABOVE_WATER)
{
vec3_t const pos = { pSprite->x, pSprite->y, actor[spriteNum].floorz + ZOFFSET2 };
setsprite(spriteNum, &pos);
}
break;
case EMPTYBOAT__STATICRR:
if (!RRRA) break;
A_Fall(spriteNum);
A_GetZLimits(spriteNum);
break;
case TRIPBOMBSPRITE__STATIC:
if (!RR) break;
if (!RRRA || (sector[sectNum].lotag != ST_1_ABOVE_WATER && sector[sectNum].lotag != 160))
if (pSprite->xvel)
{
vec3_t const vect = {
(pSprite->xvel*(sintable[(pSprite->ang + 512) & 2047])) >> 14,
(pSprite->xvel*(sintable[pSprite->ang & 2047])) >> 14,
pSprite->zvel
};
A_MoveSprite(spriteNum, &vect, CLIPMASK0);
pSprite->xvel--;
}
break;
case BOUNCEMINE__STATIC:
if (RR) break;
fallthrough__;
case MORTER__STATIC:
if (!RR)
{
int const j = A_Spawn(spriteNum, FRAMEEFFECT1);
actor[j].t_data[0] = 3;
}
fallthrough__;
case HEAVYHBOMB__STATIC:
case CHEERBOMB__STATICRR:
{
int playerNum;
DukePlayer_t *pPlayer;
int detonatePlayer;
if ((pSprite->cstat&32768))
{
if (--pData[2] <= 0)
{
A_PlaySound(TELEPORTER, spriteNum);
A_Spawn(spriteNum, TRANSPORTERSTAR);
pSprite->cstat = 257;
}
goto next_sprite;
}
int32_t playerDist;
playerNum = A_FindPlayer(pSprite, &playerDist);
pPlayer = g_player[playerNum].ps;
if (playerDist < 1220)
pSprite->cstat &= ~257;
else
pSprite->cstat |= 257;
if (pData[3] == 0)
{
if (A_IncurDamage(spriteNum) >= 0)
{
pData[3] = 1;
pData[4] = 0;
detonatePlayer = 0;
pSprite->xvel = 0;
goto DETONATEB;
}
}
if (RR || pSprite->picnum != BOUNCEMINE)
{
A_Fall(spriteNum);
if (sector[sectNum].lotag != ST_1_ABOVE_WATER && (!RRRA || sector[sectNum].lotag != 160) && pSprite->z >= actor[spriteNum].floorz-(ZOFFSET) && pSprite->yvel < 3)
{
if (pSprite->yvel > 0 || (pSprite->yvel == 0 && actor[spriteNum].floorz == sector[sectNum].floorz))
{
if (RRRA && pSprite->picnum == CHEERBOMB)
{
pData[3] = 1;
pData[4] = 1;
detonatePlayer = 0;
goto DETONATEB;
}
else
A_PlaySound(PIPEBOMB_BOUNCE,spriteNum);
}
pSprite->zvel = -((4-pSprite->yvel)<<8);
if (sector[pSprite->sectnum].lotag == ST_2_UNDERWATER)
pSprite->zvel >>= 2;
pSprite->yvel++;
}
if (RR)
{
if ((!RRRA || pSprite->picnum != CHEERBOMB) && pSprite->z < actor[spriteNum].ceilingz+(16<<8) && sector[sectNum].lotag != ST_2_UNDERWATER )
{
pSprite->z = actor[spriteNum].ceilingz+(16<<8);
pSprite->zvel = 0;
}
}
else
{
if (pSprite->z < actor[spriteNum].ceilingz) // && sector[sect].lotag != ST_2_UNDERWATER )
{
pSprite->z = actor[spriteNum].ceilingz+(3<<8);
pSprite->zvel = 0;
}
}
}
// can't initialize this because of the goto above
vec3_t tmpvect;
tmpvect.x = (pSprite->xvel * (sintable[(pSprite->ang + 512) & 2047])) >> 14;
tmpvect.y = (pSprite->xvel * (sintable[pSprite->ang & 2047])) >> 14;
tmpvect.z = pSprite->zvel;
int moveSprite;
moveSprite = A_MoveSprite(spriteNum, &tmpvect, CLIPMASK0);
//actor[spriteNum].movflag = moveSprite;
if (sector[SECT(spriteNum)].lotag == ST_1_ABOVE_WATER && pSprite->zvel == 0)
{
pSprite->z += ZOFFSET5;
if (pData[5] == 0)
{
pData[5] = 1;
A_Spawn(spriteNum,WATERSPLASH2);
if (RRRA && pSprite->picnum == MORTER)
pSprite->xvel = 0;
}
}
else pData[5] = 0;
if (pData[3] == 0 && ((!RR && pSprite->picnum == BOUNCEMINE) || pSprite->picnum == MORTER || (RRRA && pSprite->picnum == CHEERBOMB))
&& (moveSprite || playerDist < 844))
{
pData[3] = 1;
pData[4] = 0;
detonatePlayer = 0;
pSprite->xvel = 0;
goto DETONATEB;
}
if (sprite[pSprite->owner].picnum == APLAYER)
detonatePlayer = P_Get(pSprite->owner);
else detonatePlayer = -1;
if (pSprite->xvel > 0)
{
pSprite->xvel -= 5;
if (sector[sectNum].lotag == ST_2_UNDERWATER)
pSprite->xvel -= 10;
if (pSprite->xvel < 0)
pSprite->xvel = 0;
if (pSprite->xvel&8) pSprite->cstat ^= 4;
}
if ((moveSprite&49152) == 32768)
{
vec3_t davect = *(vec3_t *)pSprite;
moveSprite &= (MAXWALLS - 1);
A_DamageWall(spriteNum, moveSprite, &davect, pSprite->picnum);
if (RRRA && pSprite->picnum == CHEERBOMB)
{
pData[3] = 1;
pData[4] = 0;
detonatePlayer = 0;
pSprite->xvel = 0;
goto DETONATEB;
}
Proj_BounceOffWall(pSprite, moveSprite);
pSprite->xvel >>= 1;
}
DETONATEB:
if (!NAM)
bBoom = 0;
if ((detonatePlayer >= 0 && g_player[detonatePlayer].ps->hbomb_on == 0) || pData[3] == 1)
bBoom = 1;
if (NAM && pSprite->picnum == HEAVYHBOMB)
{
pSprite->extra--;
if (pSprite->extra <= 0)
bBoom = 1;
}
if (bBoom)
{
pData[4]++;
if (pData[4] == 2)
{
int const x = pSprite->extra;
int radius = 0;
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case TRIPBOMBSPRITE__STATIC: if(RR) radius = g_tripbombRadius; break;
case HEAVYHBOMB__STATIC: radius = g_pipebombRadius; break;
case HBOMBAMMO__STATIC: if (RR) radius = g_pipebombRadius; break;
case MORTER__STATIC: radius = g_morterRadius; break;
case BOUNCEMINE__STATIC: if (!RR) radius = g_bouncemineRadius; break;
case CHEERBOMB__STATICRR: if (RRRA) radius = g_morterRadius; break;
}
if (!RR || sector[pSprite->sectnum].lotag != 800)
{
A_RadiusDamage(spriteNum, radius, x >> 2, x >> 1, x - (x >> 2), x);
int const j = A_Spawn(spriteNum, EXPLOSION2);
A_PlaySound(PIPEBOMB_EXPLODE, j);
if (RRRA && pSprite->picnum == CHEERBOMB)
A_Spawn(spriteNum, BURNING);
if (!RR && pSprite->zvel == 0)
A_Spawn(spriteNum,EXPLOSION2BOT);
for (bssize_t x = 0; x < 8; ++x)
RANDOMSCRAP(pSprite, spriteNum);
}
}
if (pSprite->yrepeat)
{
pSprite->yrepeat = 0;
goto next_sprite;
}
if (pData[4] > 20)
{
if (RR || pSprite->owner != spriteNum || ud.respawn_items == 0)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
pData[4] = g_itemRespawnTime;
A_Spawn(spriteNum,RESPAWNMARKERRED);
pSprite->cstat = 32768;
pSprite->yrepeat = 9;
goto next_sprite;
}
}
if (RRRA && pSprite->picnum == CHEERBOMB)
{
A_Spawn(spriteNum, BURNING);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
else if (pSprite->picnum == HEAVYHBOMB && playerDist < 788 && pData[0] > 7 && pSprite->xvel == 0)
{
if (cansee(pSprite->x, pSprite->y, pSprite->z - ZOFFSET3, pSprite->sectnum,
pPlayer->pos.x, pPlayer->pos.y, pPlayer->pos.z, pPlayer->cursectnum))
{
if (pPlayer->ammo_amount[HANDBOMB_WEAPON] < pPlayer->max_ammo_amount[HANDBOMB_WEAPON] && (!RR || pSprite->pal == 0))
{
if ((g_gametypeFlags[ud.coop] & GAMETYPE_WEAPSTAY) && (RR || pSprite->owner == spriteNum))
{
for (bssize_t j = 0; j < pPlayer->weapreccnt; j++)
{
if (pPlayer->weaprecs[j] == (RR ? spriteNum : pSprite->picnum))
goto next_sprite;
}
if (pPlayer->weapreccnt < MAX_WEAPON_RECS-1)
pPlayer->weaprecs[pPlayer->weapreccnt++] = (RR ? spriteNum : pSprite->picnum);
}
P_AddAmmo(pPlayer, HANDBOMB_WEAPON, 1);
if (RR)
P_AddAmmo(pPlayer, RPG_WEAPON, 1);
A_PlaySound(DUKE_GET, pPlayer->i);
if ((pPlayer->gotweapon & (1<<HANDBOMB_WEAPON)) == 0 || pSprite->owner == pPlayer->i)
P_AddWeapon(pPlayer, HANDBOMB_WEAPON);
if (sprite[pSprite->owner].picnum != APLAYER)
P_PalFrom(pPlayer, 32, 0, 32, 0);
if ((RR && actor[pSprite->owner].picnum != HEAVYHBOMB) || (!RR && pSprite->owner != spriteNum)
|| ud.respawn_items == 0)
{
if ((!RR || (pSprite->picnum == HEAVYHBOMB && sprite[pSprite->owner].picnum != APLAYER)) && (RR || pSprite->owner == spriteNum) && (g_gametypeFlags[ud.coop] & GAMETYPE_WEAPSTAY))
goto next_sprite;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
pData[2] = g_itemRespawnTime;
A_Spawn(spriteNum, RESPAWNMARKERRED);
pSprite->cstat = 32768;
}
}
}
}
if (pData[0] < 8)
pData[0]++;
goto next_sprite;
}
case REACTORBURNT__STATIC:
case REACTOR2BURNT__STATIC:
goto next_sprite;
case REACTOR__STATIC:
case REACTOR2__STATIC:
{
if (pData[4] == 1)
{
for (bssize_t SPRITES_OF_SECT(sectNum, j))
{
switch (DYNAMICTILEMAP(sprite[j].picnum))
{
case SECTOREFFECTOR__STATIC:
if (sprite[j].lotag == 1)
{
sprite[j].lotag = 65535u;
sprite[j].hitag = 65535u;
}
break;
case REACTOR__STATIC:
sprite[j].picnum = REACTORBURNT;
break;
case REACTOR2__STATIC:
sprite[j].picnum = REACTOR2BURNT;
break;
case REACTORSPARK__STATIC:
case REACTOR2SPARK__STATIC:
sprite[j].cstat = 32768;
break;
}
}
goto next_sprite;
}
if (pData[1] >= 20)
{
pData[4] = 1;
goto next_sprite;
}
int32_t playerDist;
int playerNum = A_FindPlayer(pSprite, &playerDist);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (++pData[2] == 4)
pData[2] = 0;
if (playerDist < 4096)
{
if ((krand2() & 255) < 16)
{
if (!A_CheckSoundPlaying(pPlayer->i, DUKE_LONGTERM_PAIN))
A_PlaySound(DUKE_LONGTERM_PAIN, pPlayer->i);
A_PlaySound(SHORT_CIRCUIT, spriteNum);
sprite[pPlayer->i].extra--;
P_PalFrom(pPlayer, 32, 32, 0, 0);
}
pData[0] += 128;
if (pData[3] == 0)
pData[3] = 1;
}
else pData[3] = 0;
if (pData[1])
{
pData[1]++;
pData[4] = pSprite->z;
pSprite->z = sector[sectNum].floorz - (krand2() % (sector[sectNum].floorz - sector[sectNum].ceilingz));
switch (pData[1])
{
case 3:
// Turn on all of those flashing sectoreffector.
A_RadiusDamage(spriteNum, 4096, g_impactDamage << 2, g_impactDamage << 2, g_impactDamage << 2, g_impactDamage << 2);
for (bssize_t SPRITES_OF(STAT_STANDABLE, j))
{
if (sprite[j].picnum == MASTERSWITCH && sprite[j].hitag == pSprite->hitag && sprite[j].yvel == 0)
sprite[j].yvel = 1;
}
break;
case 4:
case 7:
case 10:
case 15:
for (bssize_t SPRITES_OF_SECT(sectNum, j))
{
if (j != spriteNum)
{
A_DeleteSprite(j);
break;
}
}
break;
}
for (bssize_t x = 0; x < 16; x++)
RANDOMSCRAP(pSprite, spriteNum);
pSprite->z = pData[4];
pData[4] = 0;
}
else if (A_IncurDamage(spriteNum) >= 0)
{
for (bssize_t x = 0; x < 32; x++)
RANDOMSCRAP(pSprite, spriteNum);
if (pSprite->extra < 0)
pData[1] = 1;
}
goto next_sprite;
}
case CAMERA1__STATIC:
if (pData[0] == 0)
{
pData[1]+=8;
if (g_damageCameras)
{
if (A_IncurDamage(spriteNum) >= 0)
{
pData[0] = 1; // static
pSprite->cstat = 32768;
for (bssize_t x = 0; x < 5; x++)
RANDOMSCRAP(pSprite, spriteNum);
goto next_sprite;
}
}
if (pSprite->hitag > 0)
{
if (pData[1] < pSprite->hitag) pSprite->ang += 8;
else if (pData[1] < pSprite->hitag * 3) pSprite->ang -= 8;
else if (pData[1] < (pSprite->hitag << 2)) pSprite->ang += 8;
else
{
pData[1] = 8;
pSprite->ang += 16;
}
}
}
goto next_sprite;
}
if (!g_netServer && ud.multimode < 2 && A_CheckEnemySprite(pSprite))
{
if (g_noEnemies == 1)
{
pSprite->cstat = 32768;
goto next_sprite;
}
else if (g_noEnemies == 2)
{
pSprite->cstat = 257;
}
}
if (G_HaveActor(sprite[spriteNum].picnum))
{
int32_t playerDist;
int playerNum = A_FindPlayer(pSprite, &playerDist);
A_Execute(spriteNum, playerNum, playerDist);
}
if (deleteAfterExecute)
A_DeleteSprite(spriteNum);
next_sprite:
spriteNum = nextSprite;
}
}
ACTOR_STATIC void G_MoveMisc(void) // STATNUM 5
{
int spriteNum = headspritestat[STAT_MISC];
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
int32_t playerDist;
int32_t *const pData = actor[spriteNum].t_data;
spritetype *const pSprite = &sprite[spriteNum];
int sectNum = pSprite->sectnum; // XXX: not const
int switchPic;
if (sectNum < 0 || pSprite->xrepeat == 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
Bmemcpy(&actor[spriteNum].bpos, pSprite, sizeof(vec3_t));
switchPic = pSprite->picnum;
if (!RR && pSprite->picnum > NUKEBUTTON && pSprite->picnum <= NUKEBUTTON+3)
switchPic = NUKEBUTTON;
if (pSprite->picnum > GLASSPIECES && pSprite->picnum <= GLASSPIECES+2)
switchPic = GLASSPIECES;
if (pSprite->picnum == INNERJAW+1)
switchPic--;
if ((pSprite->picnum == MONEY+1) || (!RR && (pSprite->picnum == MAIL+1 || pSprite->picnum == PAPER+1)))
{
actor[spriteNum].floorz = pSprite->z = getflorzofslope(pSprite->sectnum,pSprite->x,pSprite->y);
if (RR && sector[pSprite->sectnum].lotag == 800)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else switch (DYNAMICTILEMAP(switchPic))
{
//case APLAYER__STATIC: pSprite->cstat = 32768; goto next_sprite;
case SHOTGUNSPRITE__STATIC:
if (!RR) break;
if (sector[pSprite->sectnum].lotag == 800 && pSprite->z >= sector[pSprite->sectnum].floorz - ZOFFSET3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
case NEON1__STATIC:
case NEON2__STATIC:
case NEON3__STATIC:
case NEON4__STATIC:
case NEON5__STATIC:
case NEON6__STATIC:
pSprite->shade = ((tabledivide32_noinline(g_globalRandom, pSprite->lotag + 1) & 31) > 4) ? -127 : 127;
goto next_sprite;
case BLOODSPLAT1__STATIC:
case BLOODSPLAT2__STATIC:
case BLOODSPLAT3__STATIC:
case BLOODSPLAT4__STATIC:
if (pData[0] == 7 * GAMETICSPERSEC)
goto next_sprite;
pSprite->z += 16 + (krand2() & 15);
if ((++pData[0] % 9) == 0)
pSprite->yrepeat++;
goto next_sprite;
case NUKEBUTTON__STATIC:
// case NUKEBUTTON+1:
// case NUKEBUTTON+2:
// case NUKEBUTTON+3:
if (RR) break;
if (pData[0])
{
pData[0]++;
if (pData[0] == 8)
pSprite->picnum = NUKEBUTTON + 1;
else if (pData[0] == 16)
{
pSprite->picnum = NUKEBUTTON + 2;
g_player[P_Get(pSprite->owner)].ps->fist_incs = 1;
}
if (g_player[P_Get(pSprite->owner)].ps->fist_incs == GAMETICSPERSEC)
pSprite->picnum = NUKEBUTTON + 3;
}
goto next_sprite;
case FORCESPHERE__STATIC:
{
int forceRepeat = pSprite->xrepeat;
if (pData[1] > 0)
{
pData[1]--;
if (pData[1] == 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (actor[pSprite->owner].t_data[1] == 0)
{
if (pData[0] < 64)
{
pData[0]++;
forceRepeat += 3;
}
}
else if (pData[0] > 64)
{
pData[0]--;
forceRepeat -= 3;
}
*(vec3_t *)pSprite = *(vec3_t *)&sprite[pSprite->owner];
pSprite->ang += actor[pSprite->owner].t_data[0];
forceRepeat = clamp2(forceRepeat, 1, 64);
pSprite->xrepeat = forceRepeat;
pSprite->yrepeat = forceRepeat;
pSprite->shade = (forceRepeat >> 1) - 48;
for (bsize_t j = pData[0]; j > 0; j--)
A_SetSprite(spriteNum, CLIPMASK0);
goto next_sprite;
}
case MUD__STATICRR:
pData[0]++;
if (pData[0] == 1)
{
if (sector[sectNum].floorpicnum != RRTILE3073)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (!S_CheckSoundPlaying(spriteNum,ITEM_SPLASH))
A_PlaySound(ITEM_SPLASH,spriteNum);
}
if (pData[0] == 3)
{
pData[0] = 0;
pData[1]++; // WATERSPLASH_T2
}
if (pData[1] == 5)
A_DeleteSprite(spriteNum);
goto next_sprite;
case WATERSPLASH2__STATIC:
pData[0]++;
if (pData[0] == 1)
{
if (sector[sectNum].lotag != ST_1_ABOVE_WATER && sector[sectNum].lotag != ST_2_UNDERWATER)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
/*
else
{
l = getflorzofslope(sect,s->x,s->y)-s->z;
if( l > ZOFFSET2 ) KILLIT(i);
}
else
*/
if (!S_CheckSoundPlaying(spriteNum,ITEM_SPLASH))
A_PlaySound(ITEM_SPLASH,spriteNum);
}
if (pData[0] == 3)
{
pData[0] = 0;
pData[1]++; // WATERSPLASH_T2
}
if (pData[1] == 5)
A_DeleteSprite(spriteNum);
goto next_sprite;
case FRAMEEFFECT1__STATIC:
if (pSprite->owner >= 0)
{
pData[0]++;
if (pData[0] > 7)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else if (pData[0] > 4)
pSprite->cstat |= 512 + 2;
else if (pData[0] > 2)
pSprite->cstat |= 2;
pSprite->xoffset = sprite[pSprite->owner].xoffset;
pSprite->yoffset = sprite[pSprite->owner].yoffset;
}
goto next_sprite;
case INNERJAW__STATIC:
{
// case INNERJAW+1:
int32_t playerDist, playerNum = A_FindPlayer(pSprite,&playerDist);
if (playerDist < 512)
{
P_PalFrom(g_player[playerNum].ps, 32, 32,0,0);
sprite[g_player[playerNum].ps->i].extra -= 4;
}
}
fallthrough__;
case COOLEXPLOSION1__STATIC:
if (!RR && switchPic == COOLEXPLOSION1)
break;
fallthrough__;
case OWHIP__STATICRR:
case UWHIP__STATICRR:
case FIRELASER__STATIC:
if (pSprite->extra != 999)
pSprite->extra = 999;
else DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
case TONGUE__STATIC:
DELETE_SPRITE_AND_CONTINUE(spriteNum);
case MONEY__STATIC:
case MAIL__STATIC:
case PAPER__STATIC:
{
if (RR && (switchPic == MAIL || switchPic == PAPER))
break;
pSprite->xvel = (krand2()&7)+(sintable[T1(spriteNum)&2047]>>9);
T1(spriteNum) += (krand2()&63);
if ((T1(spriteNum)&2047) > 512 && (T1(spriteNum)&2047) < 1596)
{
if (sector[sectNum].lotag == ST_2_UNDERWATER)
{
if (pSprite->zvel < 64)
pSprite->zvel += (g_spriteGravity>>5)+(krand2()&7);
}
else if (pSprite->zvel < 144)
pSprite->zvel += (g_spriteGravity>>5)+(krand2()&7);
}
A_SetSprite(spriteNum, CLIPMASK0);
if ((krand2()&3) == 0)
setsprite(spriteNum, (vec3_t *) pSprite);
if (pSprite->sectnum == -1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
int const floorZ = getflorzofslope(pSprite->sectnum, pSprite->x, pSprite->y);
if (pSprite->z > floorZ)
{
pSprite->z = floorZ;
A_AddToDeleteQueue(spriteNum);
PN(spriteNum)++;
for (bssize_t SPRITES_OF(STAT_MISC, j))
{
if (sprite[j].picnum == BLOODPOOL && ldist(pSprite, &sprite[j]) < 348)
{
pSprite->pal = 2;
break;
}
}
}
if (RR && sector[pSprite->sectnum].lotag == 800 && pSprite->z >= sector[pSprite->sectnum].floorz - ZOFFSET3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
}
case HEADJIB1__STATIC:
case ARMJIB1__STATIC:
case LEGJIB1__STATIC:
case LIZMANHEAD1__STATIC:
case LIZMANARM1__STATIC:
case LIZMANLEG1__STATIC:
if (RR) break;
goto jib_code;
case RRTILE2460__STATICRR:
case RRTILE2465__STATICRR:
case BIKEJIBA__STATICRR:
case BIKEJIBB__STATICRR:
case BIKEJIBC__STATICRR:
case BIKERJIBA__STATICRR:
case BIKERJIBB__STATICRR:
case BIKERJIBC__STATICRR:
case BIKERJIBD__STATICRR:
case CHEERJIBA__STATICRR:
case CHEERJIBB__STATICRR:
case CHEERJIBC__STATICRR:
case CHEERJIBD__STATICRR:
case FBOATJIBA__STATICRR:
case FBOATJIBB__STATICRR:
case RABBITJIBA__STATICRR:
case RABBITJIBB__STATICRR:
case RABBITJIBC__STATICRR:
case MAMAJIBA__STATICRR:
case MAMAJIBB__STATICRR:
if (!RRRA) break;
goto jib_code;
case JIBS1__STATIC:
case JIBS2__STATIC:
case JIBS3__STATIC:
case JIBS4__STATIC:
case JIBS5__STATIC:
case JIBS6__STATIC:
case DUKETORSO__STATIC:
case DUKEGUN__STATIC:
case DUKELEG__STATIC:
case BILLYJIBA__STATICRR:
case BILLYJIBB__STATICRR:
case HULKJIBA__STATICRR:
case HULKJIBB__STATICRR:
case HULKJIBC__STATICRR:
case MINJIBA__STATICRR:
case MINJIBB__STATICRR:
case MINJIBC__STATICRR:
case COOTJIBA__STATICRR:
case COOTJIBB__STATICRR:
case COOTJIBC__STATICRR:
jib_code:
{
pSprite->xvel = (pSprite->xvel > 0) ? pSprite->xvel - 1 : 0;
if (!RR)
{
if (pData[5] < (30*10))
pData[5]++;
else
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (pSprite->zvel > 1024 && pSprite->zvel < 1280)
{
setsprite(spriteNum, (vec3_t *) pSprite);
sectNum = pSprite->sectnum;
}
if (RR)
setsprite(spriteNum, (vec3_t * ) pSprite);
int32_t floorZ, ceilZ;
getzsofslope(sectNum, pSprite->x, pSprite->y, &ceilZ, &floorZ);
if (ceilZ == floorZ || sectNum < 0 || sectNum >= MAXSECTORS)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (pSprite->z < floorZ-(2<<8))
{
if (pData[1] < 2) pData[1]++;
else if (sector[sectNum].lotag != ST_2_UNDERWATER)
{
pData[1] = 0;
if (pSprite->picnum == DUKELEG || pSprite->picnum == DUKETORSO || pSprite->picnum == DUKEGUN)
{
pData[0] = (pData[0] > 6) ? 0 : pData[0] + 1;
}
else
{
pData[0] = (pData[0] > 2) ? 0 : pData[0] + 1;
}
}
if (pSprite->zvel < 6144)
{
if (sector[sectNum].lotag == ST_2_UNDERWATER)
{
if (pSprite->zvel < 1024)
pSprite->zvel += 48;
else pSprite->zvel = 1024;
}
else pSprite->zvel += g_spriteGravity-50;
}
pSprite->x += (pSprite->xvel*sintable[(pSprite->ang+512)&2047])>>14;
pSprite->y += (pSprite->xvel*sintable[pSprite->ang&2047])>>14;
pSprite->z += pSprite->zvel;
if (RR && pSprite->z >= sector[pSprite->sectnum].floorz)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
if (RRRA || (pSprite->picnum == RRTILE2465 || pSprite->picnum == RRTILE2560))
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (pData[2] == 0)
{
if (pSprite->sectnum == -1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if ((sector[pSprite->sectnum].floorstat&2))
DELETE_SPRITE_AND_CONTINUE(spriteNum);
pData[2]++;
}
floorZ = getflorzofslope(pSprite->sectnum, pSprite->x, pSprite->y);
pSprite->z = floorZ - (2 << 8);
pSprite->xvel = 0;
if (pSprite->picnum == JIBS6)
{
pData[1]++;
if ((pData[1]&3) == 0 && pData[0] < 7)
pData[0]++;
if (pData[1] > 20)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
pSprite->picnum = JIBS6;
pData[0] = 0;
pData[1] = 0;
}
}
if (RR && sector[pSprite->sectnum].lotag == 800 && pSprite->z >= sector[pSprite->sectnum].floorz - ZOFFSET3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
goto next_sprite;
}
case PUKE__STATIC:
if (RR) break;
fallthrough__;
case BLOODPOOL__STATIC:
{
if (pData[0] == 0)
{
pData[0] = 1;
if (sector[sectNum].floorstat&2)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else A_AddToDeleteQueue(spriteNum);
}
A_Fall(spriteNum);
int32_t playerDist;
int const playerNum = A_FindPlayer(pSprite, &playerDist);
pSprite->z = actor[spriteNum].floorz - ZOFFSET;
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (pData[2] < 32)
{
pData[2]++;
if (actor[spriteNum].picnum == TIRE)
{
if (pSprite->xrepeat < 64 && pSprite->yrepeat < 64)
{
pSprite->xrepeat += krand2()&3;
pSprite->yrepeat += krand2()&3;
}
}
else
{
if (pSprite->xrepeat < 32 && pSprite->yrepeat < 32)
{
pSprite->xrepeat += krand2()&3;
pSprite->yrepeat += krand2()&3;
}
}
}
if (playerDist < 844 && pSprite->xrepeat > 6 && pSprite->yrepeat > 6)
{
if (pSprite->pal == 0 && (krand2()&255) < 16 && (RR || pSprite->picnum != PUKE))
{
if (pPlayer->inv_amount[GET_BOOTS] > 0)
pPlayer->inv_amount[GET_BOOTS]--;
else
{
if (!A_CheckSoundPlaying(pPlayer->i,DUKE_LONGTERM_PAIN))
A_PlaySound(DUKE_LONGTERM_PAIN,pPlayer->i);
sprite[pPlayer->i].extra --;
P_PalFrom(pPlayer, 32, 16,0,0);
}
}
if (pData[1] == 1) goto next_sprite;
pData[1] = 1;
pPlayer->footprintcount = (actor[spriteNum].picnum == TIRE) ? 10 : 3;
pPlayer->footprintpal = pSprite->pal;
pPlayer->footprintshade = pSprite->shade;
if (pData[2] == 32)
{
pSprite->xrepeat -= 6;
pSprite->yrepeat -= 6;
}
}
else pData[1] = 0;
if (RR && sector[pSprite->sectnum].lotag == 800 && pSprite->z >= sector[pSprite->sectnum].floorz - ZOFFSET3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
goto next_sprite;
}
case BURNING2__STATIC:
case FECES__STATIC:
case SHRINKEREXPLOSION__STATIC:
case EXPLOSION2BOT__STATIC:
case LASERSITE__STATIC:
if (RR) break;
fallthrough__;
case BURNING__STATIC:
case WATERBUBBLE__STATIC:
case SMALLSMOKE__STATIC:
case EXPLOSION2__STATIC:
case EXPLOSION3__STATICRR:
case BLOOD__STATIC:
case FORCERIPPLE__STATIC:
case TRANSPORTERSTAR__STATIC:
case TRANSPORTERBEAM__STATIC:
{
if (!G_HaveActor(sprite[spriteNum].picnum))
goto next_sprite;
int const playerNum = A_FindPlayer(pSprite, &playerDist);
A_Execute(spriteNum, playerNum, playerDist);
goto next_sprite;
}
case SHELL__STATIC:
case SHOTGUNSHELL__STATIC:
A_SetSprite(spriteNum,CLIPMASK0);
if (sectNum < 0 || (!RR && (sector[sectNum].floorz + (24<<8)) < pSprite->z))
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (sector[sectNum].lotag == ST_2_UNDERWATER)
{
pData[1]++;
if (pData[1] > 8)
{
pData[1] = 0;
pData[0]++;
pData[0] &= 3;
}
if (pSprite->zvel < 128) pSprite->zvel += (g_spriteGravity/13); // 8
else pSprite->zvel -= 64;
if (pSprite->xvel > 0)
pSprite->xvel -= 4;
else pSprite->xvel = 0;
}
else
{
pData[1]++;
if (pData[1] > 3)
{
pData[1] = 0;
pData[0]++;
pData[0] &= 3;
}
if (pSprite->zvel < 512) pSprite->zvel += (g_spriteGravity/3); // 52;
if (pSprite->xvel > 0)
pSprite->xvel --;
else
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
case GLASSPIECES__STATIC:
case POPCORN__STATICRR:
// case GLASSPIECES+1:
// case GLASSPIECES+2:
A_Fall(spriteNum);
if (pSprite->zvel > 4096) pSprite->zvel = 4096;
if (sectNum < 0)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (pSprite->z == actor[spriteNum].floorz-(ZOFFSET) && pData[0] < 3)
{
pSprite->zvel = -((3-pData[0])<<8)-(krand2()&511);
if (sector[sectNum].lotag == ST_2_UNDERWATER)
pSprite->zvel >>= 1;
pSprite->xrepeat >>= 1;
pSprite->yrepeat >>= 1;
if (rnd(96))
setsprite(spriteNum,(vec3_t *)pSprite);
pData[0]++;//Number of bounces
}
else if (pData[0] == 3)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
if (pSprite->xvel > 0)
{
pSprite->xvel -= 2;
pSprite->cstat = ((pSprite->xvel&3)<<2);
}
else pSprite->xvel = 0;
A_SetSprite(spriteNum,CLIPMASK0);
goto next_sprite;
}
if (PN(spriteNum) >= SCRAP6 && PN(spriteNum) <= SCRAP5+3)
{
if (pSprite->xvel > 0)
pSprite->xvel--;
else pSprite->xvel = 0;
if (pSprite->zvel > 1024 && pSprite->zvel < 1280)
{
setsprite(spriteNum,(vec3_t *)pSprite);
sectNum = pSprite->sectnum;
}
if (pSprite->z < sector[sectNum].floorz-(2<<8))
{
if (pData[1] < 1) pData[1]++;
else
{
pData[1] = 0;
if (pSprite->picnum < SCRAP6 + 8)
pData[0] = (pData[0] > 6) ? 0 : pData[0] + 1;
else
pData[0] = (pData[0] > 2) ? 0 : pData[0] + 1;
}
if (pSprite->zvel < 4096)
pSprite->zvel += g_spriteGravity - 50;
pSprite->x += (pSprite->xvel*sintable[(pSprite->ang+512)&2047])>>14;
pSprite->y += (pSprite->xvel*sintable[pSprite->ang&2047])>>14;
pSprite->z += pSprite->zvel;
}
else
{
if (pSprite->picnum == SCRAP1 && pSprite->yvel > 0 && pSprite->yvel < MAXUSERTILES)
{
int32_t j = A_Spawn(spriteNum, pSprite->yvel);
setsprite(j,(vec3_t *)pSprite);
A_GetZLimits(j);
sprite[j].hitag = sprite[j].lotag = 0;
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
goto next_sprite;
}
next_sprite:
spriteNum = nextSprite;
}
}
// i: SE spritenum
static void HandleSE31(int spriteNum, int setFloorZ, int spriteZ, int SEdir, int zDifference)
{
const spritetype *pSprite = &sprite[spriteNum];
sectortype *const pSector = &sector[sprite[spriteNum].sectnum];
int32_t *const pData = actor[spriteNum].t_data;
if (klabs(pSector->floorz - spriteZ) < SP(spriteNum))
{
if (setFloorZ)
pSector->floorz = spriteZ;
pData[2] = SEdir;
pData[0] = 0;
if (!RR)
pData[3] = pSprite->hitag;
A_CallSound(pSprite->sectnum, spriteNum);
}
else
{
int const zChange = ksgn(zDifference) * SP(spriteNum);
pSector->floorz += zChange;
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, j))
{
if (sprite[j].picnum == APLAYER && sprite[j].owner >= 0)
{
int const playerNum = P_Get(j);
if (g_player[playerNum].ps->on_ground == 1)
g_player[playerNum].ps->pos.z += zChange;
}
if (sprite[j].zvel == 0 && sprite[j].statnum != STAT_EFFECTOR && (RR || sprite[j].statnum != STAT_PROJECTILE))
{
sprite[j].z += zChange;
actor[j].bpos.z = sprite[j].z;
actor[j].floorz = pSector->floorz;
}
}
}
}
// s: SE sprite
static void MaybeTrainKillPlayer(const spritetype *pSprite, int const setOPos)
{
for (bssize_t TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (sprite[pPlayer->i].extra > 0)
{
int16_t playerSectnum = pPlayer->cursectnum;
updatesector(pPlayer->pos.x, pPlayer->pos.y, &playerSectnum);
if ((playerSectnum == -1 && !ud.noclip) || (pPlayer->cursectnum != pSprite->sectnum && playerSectnum == pSprite->sectnum))
{
*(vec2_t *)pPlayer = *(vec2_t const *)pSprite;
if (setOPos)
*(vec2_t *)&pPlayer->opos = *(vec2_t *)pPlayer;
pPlayer->cursectnum = pSprite->sectnum;
setsprite(pPlayer->i, (vec3_t const *)pSprite);
P_QuickKill(pPlayer);
}
}
}
}
// i: SE spritenum
static void MaybeTrainKillEnemies(int const spriteNum, int const gutSpawnCnt)
{
int findSprite = headspritesect[sprite[OW(spriteNum)].sectnum];
do
{
int const nextSprite = nextspritesect[findSprite];
if (sprite[findSprite].statnum == STAT_ACTOR && A_CheckEnemySprite(&sprite[findSprite]))
{
int16_t sectNum = sprite[findSprite].sectnum;
updatesector(sprite[findSprite].x,sprite[findSprite].y,&sectNum);
if (sprite[findSprite].extra >= 0 && sectNum == sprite[spriteNum].sectnum)
{
A_DoGutsDir(findSprite, JIBS6, gutSpawnCnt);
A_PlaySound(SQUISHED, findSprite);
A_DeleteSprite(findSprite);
}
}
findSprite = nextSprite;
}
while (findSprite >= 0);
}
ACTOR_STATIC void G_MoveEffectors(void) //STATNUM 3
{
int32_t q = 0, j, k, l, m, x;
int spriteNum = headspritestat[STAT_EFFECTOR];
for (native_t TRAVERSE_CONNECT(playerNum))
{
vec2_t & fric = g_player[playerNum].ps->fric;
fric.x = fric.y = 0;
}
while (spriteNum >= 0)
{
int const nextSprite = nextspritestat[spriteNum];
spritetype *const pSprite = &sprite[spriteNum];
int32_t playerDist;
int playerNum = A_FindPlayer(pSprite, &playerDist);
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
sectortype *const pSector = &sector[pSprite->sectnum];
int const spriteLotag = pSprite->lotag;
int const spriteHitag = pSprite->hitag;
int32_t *const pData = &actor[spriteNum].t_data[0];
switch (spriteLotag)
{
case SE_0_ROTATING_SECTOR:
{
int32_t zchange = 0;
j = pSprite->owner;
if ((uint16_t)sprite[j].lotag == UINT16_MAX)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
q = pSector->extra>>3;
l = 0;
if (pSector->lotag == ST_30_ROTATE_RISE_BRIDGE)
{
q >>= 2;
if (sprite[spriteNum].extra == 1)
{
if (actor[spriteNum].tempang < 256)
{
actor[spriteNum].tempang += 4;
if (actor[spriteNum].tempang >= 256)
A_CallSound(pSprite->sectnum,spriteNum);
if (pSprite->clipdist) l = 1;
else l = -1;
}
else actor[spriteNum].tempang = 256;
if (pSector->floorz > pSprite->z) //z's are touching
{
pSector->floorz -= 512;
zchange = -512;
if (pSector->floorz < pSprite->z)
pSector->floorz = pSprite->z;
}
else if (pSector->floorz < pSprite->z) //z's are touching
{
pSector->floorz += 512;
zchange = 512;
if (pSector->floorz > pSprite->z)
pSector->floorz = pSprite->z;
}
}
else if (sprite[spriteNum].extra == 3)
{
if (actor[spriteNum].tempang > 0)
{
actor[spriteNum].tempang -= 4;
if (actor[spriteNum].tempang <= 0)
A_CallSound(pSprite->sectnum,spriteNum);
if (pSprite->clipdist) l = -1;
else l = 1;
}
else actor[spriteNum].tempang = 0;
if (pSector->floorz > T4(spriteNum)) //z's are touching
{
pSector->floorz -= 512;
zchange = -512;
if (pSector->floorz < T4(spriteNum))
pSector->floorz = T4(spriteNum);
}
else if (pSector->floorz < T4(spriteNum)) //z's are touching
{
pSector->floorz += 512;
zchange = 512;
if (pSector->floorz > T4(spriteNum))
pSector->floorz = T4(spriteNum);
}
}
}
else
{
if (actor[j].t_data[0] == 0) break;
if (actor[j].t_data[0] == 2) DELETE_SPRITE_AND_CONTINUE(spriteNum);
l = (sprite[j].ang > 1024) ? -1 : 1;
if (pData[3] == 0)
pData[3] = ldist(pSprite,&sprite[j]);
pSprite->xvel = pData[3];
pSprite->x = sprite[j].x;
pSprite->y = sprite[j].y;
}
pSprite->ang += (l*q);
pData[2] += (l*q);
if (l && (pSector->floorstat&64))
{
for (TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (pPlayer->cursectnum == pSprite->sectnum && pPlayer->on_ground == 1)
{
pPlayer->q16ang += fix16_from_int(l*q);
pPlayer->q16ang &= 0x7FFFFFF;
pPlayer->pos.z += zchange;
vec2_t r;
rotatepoint(*(vec2_t *)&sprite[j],*(vec2_t *)&pPlayer->pos,(q*l),&r);
pPlayer->bobpos.x += r.x-pPlayer->pos.x;
pPlayer->bobpos.y += r.y-pPlayer->pos.y;
*(vec2_t *)&pPlayer->pos = r;
if (sprite[pPlayer->i].extra <= 0)
*(vec2_t *)&sprite[pPlayer->i] = r;
}
}
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, p))
{
// KEEPINSYNC1
if (sprite[p].statnum != STAT_EFFECTOR && sprite[p].statnum != STAT_PROJECTILE)
if (RR || sprite[p].picnum != LASERLINE)
{
if (sprite[p].picnum == APLAYER && sprite[p].owner >= 0)
continue;
sprite[p].ang += (l*q);
sprite[p].ang &= 2047;
sprite[p].z += zchange;
rotatepoint(*(vec2_t *)&sprite[j], *(vec2_t *)&sprite[p], (q * l), (vec2_t *)&sprite[p].x);
}
}
}
A_MoveSector(spriteNum);
}
break;
case SE_1_PIVOT: //Nothing for now used as the pivot
if (pSprite->owner == -1) //Init
{
pSprite->owner = spriteNum;
for (SPRITES_OF(STAT_EFFECTOR, j))
{
if (sprite[j].lotag == SE_19_EXPLOSION_LOWERS_CEILING && sprite[j].hitag == spriteHitag)
{
pData[0] = 0;
break;
}
}
}
break;
case SE_6_SUBWAY:
k = pSector->extra;
if (pData[4] > 0)
{
pData[4]--;
if (pData[4] >= (k-(k>>3)))
pSprite->xvel -= (k>>5);
if (pData[4] > ((k>>1)-1) && pData[4] < (k-(k>>3)))
pSprite->xvel = 0;
if (pData[4] < (k>>1))
pSprite->xvel += (k>>5);
if (pData[4] < ((k>>1)-(k>>3)))
{
pData[4] = 0;
pSprite->xvel = k;
if (RR && (!RRRA || g_lastLevel) && g_hulkSpawn)
{
g_hulkSpawn--;
int newSprite = A_Spawn(spriteNum, HULK);
sprite[newSprite].z = sector[sprite[newSprite].sectnum].ceilingz;
sprite[newSprite].pal = 33;
if (!g_hulkSpawn)
{
newSprite = A_InsertSprite(pSprite->sectnum, pSprite->x, pSprite->y,
sector[pSprite->sectnum].ceilingz + 119428, RRTILE3677, -8, 16, 16, 0, 0, 0, spriteNum, STAT_MISC);
sprite[newSprite].cstat = 514;
sprite[newSprite].pal = 7;
sprite[newSprite].xrepeat = 80;
sprite[newSprite].yrepeat = 255;
newSprite = A_Spawn(spriteNum, RRTILE296);
sprite[newSprite].cstat = 0;
sprite[newSprite].cstat |= 32768;
sprite[newSprite].z = sector[pSprite->sectnum].floorz - 6144;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
}
}
else
{
pSprite->xvel = k;
if (RR)
{
int otherSprite = headspritesect[pSprite->sectnum];
while (otherSprite >= 0)
{
int const nextOtherSprite = nextspritesect[otherSprite];
if (sprite[otherSprite].picnum == UFOBEAM)
if (g_ufoSpawn)
if (++g_ufoCnt == 64)
{
g_ufoCnt = 0;
g_ufoSpawn--;
int ufoTile = UFO1;
switch (krand2()&3)
{
case 0:
ufoTile = UFO1;
break;
case 1:
ufoTile = UFO2;
break;
case 2:
ufoTile = UFO3;
break;
case 3:
ufoTile = UFO4;
break;
}
if (RRRA)
ufoTile = UFO1;
int const newSprite = A_Spawn(spriteNum, ufoTile);
sprite[newSprite].z = sector[sprite[newSprite].sectnum].ceilingz;
}
otherSprite = nextOtherSprite;
}
}
}
for (SPRITES_OF(STAT_EFFECTOR, j))
{
if (sprite[j].lotag == SE_14_SUBWAY_CAR && spriteHitag == sprite[j].hitag && actor[j].t_data[0] == pData[0])
{
sprite[j].xvel = pSprite->xvel;
// if( t[4] == 1 )
{
if (actor[j].t_data[5] == 0)
actor[j].t_data[5] = dist(&sprite[j],pSprite);
x = ksgn(dist(&sprite[j],pSprite)-actor[j].t_data[5]);
if (sprite[j].extra)
x = -x;
pSprite->xvel += x;
}
actor[j].t_data[4] = pData[4];
}
}
x = 0; // XXX: This assignment is dead?
fallthrough__;
case SE_14_SUBWAY_CAR:
if (pSprite->owner==-1)
pSprite->owner = A_FindLocator((int16_t)pData[3],(int16_t)pData[0]);
if (pSprite->owner == -1)
{
// debugging subway cars (mapping-wise) is freakin annoying
// let's at least have a helpful message...
Bsprintf(tempbuf,"Could not find any locators in sector %d"
" for SE# 6 or 14 with hitag %d.\n", (int)pData[0], (int)pData[3]);
G_GameExit(tempbuf);
}
j = ldist(&sprite[pSprite->owner],pSprite);
if (j < 1024L)
{
if (spriteLotag==SE_6_SUBWAY)
if (sprite[pSprite->owner].hitag&1)
pData[4]=pSector->extra; //Slow it down
pData[3]++;
pSprite->owner = A_FindLocator(pData[3],pData[0]);
if (pSprite->owner==-1)
{
pData[3]=0;
pSprite->owner = A_FindLocator(0,pData[0]);
}
}
if (pSprite->xvel)
{
#ifdef YAX_ENABLE
int32_t firstrun = 1;
#endif
x = getangle(sprite[pSprite->owner].x-pSprite->x,sprite[pSprite->owner].y-pSprite->y);
q = G_GetAngleDelta(pSprite->ang,x)>>3;
pData[2] += q;
pSprite->ang += q;
if (pSprite->xvel == pSector->extra)
{
if (RR)
{
if (!S_CheckSoundPlaying(spriteNum,actor[spriteNum].lastv.x))
A_PlaySound(actor[spriteNum].lastv.x,spriteNum);
}
if (!RR && (pSector->floorstat&1) == 0 && (pSector->ceilingstat&1) == 0)
{
if (!S_CheckSoundPlaying(spriteNum,actor[spriteNum].lastv.x))
A_PlaySound(actor[spriteNum].lastv.x,spriteNum);
}
else if (ud.monsters_off == 0 && pSector->floorpal == 0 && (pSector->floorstat&1) && rnd(8))
{
if (playerDist < 20480)
{
j = pSprite->ang;
pSprite->ang = getangle(pSprite->x-g_player[playerNum].ps->pos.x,pSprite->y-g_player[playerNum].ps->pos.y);
A_Shoot(spriteNum,RPG);
pSprite->ang = j;
}
}
}
if (pSprite->xvel <= 64 && (RR || ((pSector->floorstat&1) == 0 && (pSector->ceilingstat&1) == 0)))
S_StopEnvSound(actor[spriteNum].lastv.x,spriteNum);
if ((pSector->floorz-pSector->ceilingz) < (108<<8))
{
if (ud.noclip == 0 && pSprite->xvel >= 192)
MaybeTrainKillPlayer(pSprite, 0);
}
m = (pSprite->xvel*sintable[(pSprite->ang+512)&2047])>>14;
x = (pSprite->xvel*sintable[pSprite->ang&2047])>>14;
for (TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
// might happen when squished into void space
if (pPlayer->cursectnum < 0)
break;
if (sector[pPlayer->cursectnum].lotag != ST_2_UNDERWATER)
{
if (g_playerSpawnPoints[playerNum].sect == pSprite->sectnum)
{
g_playerSpawnPoints[playerNum].pos.x += m;
g_playerSpawnPoints[playerNum].pos.y += x;
}
if (pSprite->sectnum == sprite[pPlayer->i].sectnum
#ifdef YAX_ENABLE
|| (pData[9]>=0 && pData[9] == sprite[pPlayer->i].sectnum)
#endif
)
{
rotatepoint(*(vec2_t *)pSprite, *(vec2_t *)&pPlayer->pos, q, (vec2_t *)&pPlayer->pos);
pPlayer->pos.x += m;
pPlayer->pos.y += x;
pPlayer->bobpos.x += m;
pPlayer->bobpos.y += x;
pPlayer->q16ang += fix16_from_int(q);
pPlayer->q16ang &= 0x7FFFFFF;
if (g_netServer || numplayers > 1)
{
pPlayer->opos.x = pPlayer->pos.x;
pPlayer->opos.y = pPlayer->pos.y;
}
if (sprite[pPlayer->i].extra <= 0)
{
sprite[pPlayer->i].x = pPlayer->pos.x;
sprite[pPlayer->i].y = pPlayer->pos.y;
}
}
}
}
// NOTE: special loop handling
j = headspritesect[pSprite->sectnum];
while (j >= 0)
{
// KEEPINSYNC2
// XXX: underwater check?
if (sprite[j].statnum != STAT_PLAYER && sector[sprite[j].sectnum].lotag != ST_2_UNDERWATER &&
(sprite[j].picnum != SECTOREFFECTOR || (sprite[j].lotag == SE_49_POINT_LIGHT||sprite[j].lotag == SE_50_SPOT_LIGHT))
&& sprite[j].picnum != LOCATORS)
{
rotatepoint(*(vec2_t *)pSprite,*(vec2_t *)&sprite[j],q,(vec2_t *)&sprite[j].x);
sprite[j].x+= m;
sprite[j].y+= x;
sprite[j].ang+=q;
if (g_netServer || numplayers > 1)
{
actor[j].bpos.x = sprite[j].x;
actor[j].bpos.y = sprite[j].y;
}
}
j = nextspritesect[j];
#ifdef YAX_ENABLE
if (j < 0)
{
if (pData[9]>=0 && firstrun)
{
firstrun = 0;
j = headspritesect[pData[9]];
}
}
#endif
}
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
if ((pSector->floorz-pSector->ceilingz) < (108<<8))
{
if (ud.noclip == 0 && pSprite->xvel >= 192)
MaybeTrainKillPlayer(pSprite, 1);
MaybeTrainKillEnemies(spriteNum, 72);
}
}
break;
case SE_30_TWO_WAY_TRAIN:
if (pSprite->owner == -1)
{
pData[3] = !pData[3];
pSprite->owner = A_FindLocator(pData[3],pData[0]);
}
else
{
if (pData[4] == 1) // Starting to go
{
if (ldist(&sprite[pSprite->owner],pSprite) < (2048-128))
pData[4] = 2;
else
{
if (pSprite->xvel == 0)
G_OperateActivators(pSprite->hitag+(!pData[3]),-1);
if (pSprite->xvel < 256)
pSprite->xvel += 16;
}
}
if (pData[4] == 2)
{
l = FindDistance2D(sprite[pSprite->owner].x-pSprite->x,sprite[pSprite->owner].y-pSprite->y);
if (l <= 128)
pSprite->xvel = 0;
if (pSprite->xvel > 0)
pSprite->xvel -= 16;
else
{
pSprite->xvel = 0;
G_OperateActivators(pSprite->hitag+(int16_t)pData[3],-1);
pSprite->owner = -1;
pSprite->ang += 1024;
pData[4] = 0;
G_OperateForceFields(spriteNum,pSprite->hitag);
for (SPRITES_OF_SECT(pSprite->sectnum, j))
{
if (sprite[j].picnum != SECTOREFFECTOR && sprite[j].picnum != LOCATORS)
{
actor[j].bpos.x = sprite[j].x;
actor[j].bpos.y = sprite[j].y;
}
}
}
}
}
if (pSprite->xvel)
{
l = (pSprite->xvel*sintable[(pSprite->ang+512)&2047])>>14;
x = (pSprite->xvel*sintable[pSprite->ang&2047])>>14;
if ((pSector->floorz-pSector->ceilingz) < (108<<8))
if (ud.noclip == 0)
MaybeTrainKillPlayer(pSprite, 0);
for (int TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (sprite[pPlayer->i].sectnum == pSprite->sectnum)
{
pPlayer->pos.x += l;
pPlayer->pos.y += x;
if (g_netServer || numplayers > 1)
{
pPlayer->opos.x = pPlayer->pos.x;
pPlayer->opos.y = pPlayer->pos.y;
}
pPlayer->bobpos.x += l;
pPlayer->bobpos.y += x;
}
if (g_playerSpawnPoints[playerNum].sect == pSprite->sectnum)
{
g_playerSpawnPoints[playerNum].pos.x += l;
g_playerSpawnPoints[playerNum].pos.y += x;
}
}
for (SPRITES_OF_SECT(pSprite->sectnum, j))
{
// TODO: replace some checks for SE 49/50 with statnum LIGHT instead?
if ((sprite[j].picnum != SECTOREFFECTOR || sprite[j].lotag==SE_49_POINT_LIGHT || sprite[j].lotag==SE_50_SPOT_LIGHT)
&& sprite[j].picnum != LOCATORS)
{
if (numplayers < 2 && !g_netServer)
{
actor[j].bpos.x = sprite[j].x;
actor[j].bpos.y = sprite[j].y;
}
sprite[j].x += l;
sprite[j].y += x;
if (g_netServer || numplayers > 1)
{
actor[j].bpos.x = sprite[j].x;
actor[j].bpos.y = sprite[j].y;
}
}
}
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
if (pSector->floorz-pSector->ceilingz < (108<<8))
{
if (ud.noclip == 0)
MaybeTrainKillPlayer(pSprite, 1);
MaybeTrainKillEnemies(spriteNum, 24);
}
}
break;
case SE_2_EARTHQUAKE://Quakes
if (pData[4] > 0 && pData[0] == 0)
{
if (pData[4] < spriteHitag)
pData[4]++;
else pData[0] = 1;
}
if (pData[0] > 0)
{
pData[0]++;
pSprite->xvel = 3;
if (pData[0] > 96)
{
pData[0] = -1; //Stop the quake
pData[4] = -1;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else
{
if ((pData[0]&31) == 8)
{
g_earthquakeTime = 48;
A_PlaySound(EARTHQUAKE,g_player[screenpeek].ps->i);
}
pSector->floorheinum = (klabs(pSector->floorheinum - pData[5]) < 8)
? pData[5]
: pSector->floorheinum + (ksgn(pData[5] - pSector->floorheinum) << 4);
}
vec2_t const vect = { (pSprite->xvel * sintable[(pSprite->ang + 512) & 2047]) >> 14,
(pSprite->xvel * sintable[pSprite->ang & 2047]) >> 14 };
for (TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (pPlayer->cursectnum == pSprite->sectnum && pPlayer->on_ground)
{
pPlayer->pos.x += vect.x;
pPlayer->pos.y += vect.y;
pPlayer->bobpos.x += vect.x;
pPlayer->bobpos.y += vect.y;
}
}
for (bssize_t nextSprite, SPRITES_OF_SECT_SAFE(pSprite->sectnum, sectSprite, nextSprite))
{
if (sprite[sectSprite].picnum != SECTOREFFECTOR)
{
sprite[sectSprite].x+=vect.x;
sprite[sectSprite].y+=vect.y;
setsprite(sectSprite,(vec3_t *)&sprite[sectSprite]);
}
}
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
}
break;
//Flashing sector lights after reactor EXPLOSION2
case SE_3_RANDOM_LIGHTS_AFTER_SHOT_OUT:
{
if (pData[4] == 0) break;
// if(t[5] > 0) { t[5]--; break; }
if ((tabledivide32_noinline(g_globalRandom, spriteHitag+1)&31) < 4 && !pData[2])
{
// t[5] = 4+(g_globalRandom&7);
pSector->ceilingpal = pSprite->owner >> 8;
pSector->floorpal = pSprite->owner & 0xff;
pData[0] = pSprite->shade + (g_globalRandom & 15);
}
else
{
// t[5] = 4+(g_globalRandom&3);
pSector->ceilingpal = pSprite->pal;
pSector->floorpal = pSprite->pal;
pData[0] = pData[3];
}
pSector->ceilingshade = pData[0];
pSector->floorshade = pData[0];
walltype *pWall = &wall[pSector->wallptr];
for (x=pSector->wallnum; x > 0; x--,pWall++)
{
if (pWall->hitag != 1)
{
pWall->shade = pData[0];
if ((pWall->cstat & 2) && pWall->nextwall >= 0)
wall[pWall->nextwall].shade = pWall->shade;
}
}
break;
}
case SE_4_RANDOM_LIGHTS:
{
// See A_Spawn():
// s->owner: original ((ceilingpal<<8) | floorpal)
// t[2]: original floor shade
// t[3]: max wall shade
int lightFlag;
if ((tabledivide32_noinline(g_globalRandom, spriteHitag+1)&31) < 4)
{
pData[1] = pSprite->shade + (g_globalRandom & 15); // Got really bright
pData[0] = pSprite->shade + (g_globalRandom & 15);
pSector->ceilingpal = pSprite->owner >> 8;
pSector->floorpal = pSprite->owner & 0xff;
lightFlag = 1;
}
else
{
pData[1] = pData[2];
pData[0] = pData[3];
pSector->ceilingpal = pSprite->pal;
pSector->floorpal = pSprite->pal;
lightFlag = 0;
}
pSector->floorshade = pData[1];
pSector->ceilingshade = pData[1];
walltype *pWall = &wall[pSector->wallptr];
for (x=pSector->wallnum; x > 0; x--,pWall++)
{
if (lightFlag) pWall->pal = (pSprite->owner&0xff);
else pWall->pal = pSprite->pal;
if (pWall->hitag != 1)
{
pWall->shade = pData[0];
if ((pWall->cstat&2) && pWall->nextwall >= 0)
wall[pWall->nextwall].shade = pWall->shade;
}
}
for (bssize_t SPRITES_OF_SECT(SECT(spriteNum), sectSprite))
{
if (sprite[sectSprite].cstat&16)
sprite[sectSprite].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
if (pData[4])
DELETE_SPRITE_AND_CONTINUE(spriteNum);
break;
}
//BOSS
case SE_5:
{
if (playerDist < 8192)
{
int const saveAng = pSprite->ang;
pSprite->ang = getangle(pSprite->x - pPlayer->pos.x, pSprite->y - pPlayer->pos.y);
A_Shoot(spriteNum, FIRELASER);
pSprite->ang = saveAng;
}
if (pSprite->owner==-1) //Start search
{
pData[4] = 0;
int closestLocatorDist = INT32_MAX;
int closestLocator = pSprite->owner;
//Find the shortest dist
do
{
pSprite->owner = A_FindLocator((int16_t)pData[4], -1); // t[0] hold sectnum
if (pSprite->owner == -1)
break;
int const locatorDist = ldist(&sprite[pPlayer->i],&sprite[pSprite->owner]);
if (closestLocatorDist > locatorDist)
{
closestLocator = pSprite->owner;
closestLocatorDist = locatorDist;
}
pData[4]++;
}
while (1);
pSprite->owner = closestLocator;
pSprite->zvel = ksgn(sprite[closestLocator].z - pSprite->z) << 4;
}
if (ldist(&sprite[pSprite->owner],pSprite) < 1024)
{
pSprite->owner = -1;
goto next_sprite;
}
else pSprite->xvel=256;
int const angInc = G_GetAngleDelta(pSprite->ang, getangle(sprite[pSprite->owner].x-pSprite->x,
sprite[pSprite->owner].y-pSprite->y))>>3;
pSprite->ang += angInc;
if (rnd(32))
{
pData[2] += angInc;
pSector->ceilingshade = 127;
}
else
{
pData[2] += G_GetAngleDelta(pData[2] + 512, getangle(pPlayer->pos.x - pSprite->x, pPlayer->pos.y - pSprite->y)) >> 2;
pSector->ceilingshade = 0;
}
if (A_IncurDamage(spriteNum) >= 0)
{
if (++pData[3] == 5)
{
pSprite->zvel += 1024;
P_DoQuote(QUOTE_WASTED, g_player[myconnectindex].ps);
}
}
pSprite->z += pSprite->zvel;
pSector->ceilingz += pSprite->zvel;
sector[pData[0]].ceilingz += pSprite->zvel;
A_MoveSector(spriteNum);
setsprite(spriteNum, (vec3_t *)pSprite);
break;
}
case SE_8_UP_OPEN_DOOR_LIGHTS:
case SE_9_DOWN_OPEN_DOOR_LIGHTS:
{
// work only if its moving
int animGoal = -1;
if (actor[spriteNum].t_data[4])
{
if (++actor[spriteNum].t_data[4] > 8)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
animGoal = 1;
}
else animGoal = GetAnimationGoal(&pSector->ceilingz);
if (animGoal >= 0)
{
int shadeInc = ((pSector->lotag & 0x8000u) || actor[spriteNum].t_data[4]) ? -pData[3] : pData[3];
if (spriteLotag == SE_9_DOWN_OPEN_DOOR_LIGHTS)
shadeInc = -shadeInc;
for (bssize_t SPRITES_OF(STAT_EFFECTOR, sectorEffector))
{
if (sprite[sectorEffector].lotag == spriteLotag && sprite[sectorEffector].hitag == spriteHitag)
{
int const sectNum = sprite[sectorEffector].sectnum;
int const spriteShade = sprite[sectorEffector].shade;
walltype *pWall = &wall[sector[sectNum].wallptr];
for (bsize_t l=sector[sectNum].wallnum; l>0; l--, pWall++)
{
if (pWall->hitag == 1)
continue;
pWall->shade += shadeInc;
if (pWall->shade < spriteShade)
pWall->shade = spriteShade;
else if (pWall->shade > actor[sectorEffector].t_data[2])
pWall->shade = actor[sectorEffector].t_data[2];
if (pWall->nextwall >= 0 && wall[pWall->nextwall].hitag != 1)
wall[pWall->nextwall].shade = pWall->shade;
}
sector[sectNum].floorshade += shadeInc;
sector[sectNum].ceilingshade += shadeInc;
if (sector[sectNum].floorshade < spriteShade)
sector[sectNum].floorshade = spriteShade;
else if (sector[sectNum].floorshade > actor[sectorEffector].t_data[0])
sector[sectNum].floorshade = actor[sectorEffector].t_data[0];
if (sector[sectNum].ceilingshade < spriteShade)
sector[sectNum].ceilingshade = spriteShade;
else if (sector[sectNum].ceilingshade > actor[sectorEffector].t_data[1])
sector[sectNum].ceilingshade = actor[sectorEffector].t_data[1];
if (RR && sector[sectNum].hitag == 1)
sector[sectNum].ceilingshade = actor[sectorEffector].t_data[1];
}
}
}
break;
}
case SE_10_DOOR_AUTO_CLOSE:
// XXX: 32791, what the hell?
if ((pSector->lotag&0xff) == ST_27_STRETCH_BRIDGE || (pSector->floorz > pSector->ceilingz && (pSector->lotag&0xff) != ST_23_SWINGING_DOOR) || pSector->lotag == (int16_t)32791u)
{
j = 1;
if ((pSector->lotag&0xff) != ST_27_STRETCH_BRIDGE)
for (bssize_t TRAVERSE_CONNECT(playerNum))
if (pSector->lotag != ST_30_ROTATE_RISE_BRIDGE && pSector->lotag != ST_31_TWO_WAY_TRAIN && pSector->lotag != 0
&& pSprite->sectnum == sprite[g_player[playerNum].ps->i].sectnum)
j = 0;
if (j == 1)
{
if (pData[0] > spriteHitag)
switch (sector[pSprite->sectnum].lotag)
{
case ST_20_CEILING_DOOR:
case ST_21_FLOOR_DOOR:
case ST_22_SPLITTING_DOOR:
case ST_26_SPLITTING_ST_DOOR:
if (!RR && GetAnimationGoal(&sector[pSprite->sectnum].ceilingz) >= 0)
break;
fallthrough__;
default:
G_ActivateBySector(pSprite->sectnum,spriteNum);
pData[0] = 0;
break;
}
else pData[0]++;
}
}
else pData[0]=0;
break;
case SE_11_SWINGING_DOOR: //Swingdoor
if (pData[5] > 0)
{
pData[5]--;
break;
}
if (pData[4])
{
int const endWall = pSector->wallptr+pSector->wallnum;
l = (SP(spriteNum) >> 3) * pData[3];
for (j=pSector->wallptr; j<endWall; j++)
{
for (SPRITES_OF(STAT_ACTOR, k))
{
if (sprite[k].extra > 0 && A_CheckEnemySprite(&sprite[k])
&& clipinsidebox((vec2_t *)&sprite[k], j, 256) == 1)
goto next_sprite;
}
for (SPRITES_OF(STAT_PLAYER, k))
{
if (sprite[k].owner >= 0 && clipinsidebox((vec2_t *)&sprite[k], j, 144) == 1)
{
pData[5] = 8; // Delay
pData[2] -= l;
pData[4] -= l;
A_MoveSector(spriteNum);
setsprite(spriteNum, (vec3_t *)pSprite);
goto next_sprite;
}
}
}
pData[2] += l;
pData[4] += l;
A_MoveSector(spriteNum);
setsprite(spriteNum, (vec3_t *)pSprite);
if (pData[4] <= -511 || pData[4] >= 512)
{
pData[4] = 0;
pData[2] &= 0xffffff00;
A_MoveSector(spriteNum);
setsprite(spriteNum, (vec3_t *) pSprite);
break;
}
}
break;
case SE_12_LIGHT_SWITCH:
if (pData[0] == 3 || pData[3] == 1) //Lights going off
{
pSector->floorpal = 0;
pSector->ceilingpal = 0;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->shade = pData[1];
pWall->pal = 0;
}
}
pSector->floorshade = pData[1];
pSector->ceilingshade = pData[2];
pData[0] = 0;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat & 16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
if (pData[3] == 1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (pData[0] == 1) //Lights flickering on
{
if (pSector->floorshade > pSprite->shade)
{
pSector->floorpal = pSprite->pal;
pSector->ceilingpal = pSprite->pal;
pSector->floorshade -= 2;
pSector->ceilingshade -= 2;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->pal = pSprite->pal;
pWall->shade -= 2;
}
}
}
else pData[0] = 2;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat&16)
{
if (sprite[j].cstat & 16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
}
}
break;
case 47:
if (!RRRA)
break;
if (pData[0] == 3 || pData[3] == 1) //Lights going off
{
pSector->floorpal = 0;
pSector->ceilingpal = 0;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->shade = pData[1];
pWall->pal = 0;
}
}
pSector->floorshade = pData[1];
pSector->ceilingshade = pData[2];
pData[0] = 0;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat & 16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
if (pData[3] == 1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (pData[0] == 1) //Lights flickering on
{
if (pSector->floorshade > pSprite->shade)
{
pSector->floorpal = pSprite->pal;
pSector->floorshade -= 2;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->pal = pSprite->pal;
pWall->shade -= 2;
}
}
}
else pData[0] = 2;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat&16)
{
if (sprite[j].cstat & 16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
}
}
break;
case 48:
if (!RRRA)
break;
if (pData[0] == 3 || pData[3] == 1) //Lights going off
{
pSector->floorpal = 0;
pSector->ceilingpal = 0;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->shade = pData[1];
pWall->pal = 0;
}
}
pSector->floorshade = pData[1];
pSector->ceilingshade = pData[2];
pData[0] = 0;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat & 16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
if (pData[3] == 1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (pData[0] == 1) //Lights flickering on
{
if (pSector->ceilingshade > pSprite->shade)
{
pSector->ceilingpal = pSprite->pal;
pSector->ceilingshade -= 2;
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
{
if (pWall->hitag != 1)
{
pWall->pal = pSprite->pal;
pWall->shade -= 2;
}
}
}
else pData[0] = 2;
for (SPRITES_OF_SECT(SECT(spriteNum), j))
{
if (sprite[j].cstat&16)
sprite[j].shade = (pSector->ceilingstat & 1) ? pSector->ceilingshade : pSector->floorshade;
}
}
break;
case SE_13_EXPLOSIVE:
if (pData[2])
{
// t[0]: ceiling z
// t[1]: floor z
// s->owner: 1 if affect ceiling, 0 if affect floor
// t[3]: 1 if ceiling was parallaxed at premap, 0 else
j = (SP(spriteNum)<<5)|1;
if (pSprite->ang == 512)
{
if (pSprite->owner)
{
pSector->ceilingz = (klabs(pData[0] - pSector->ceilingz) >= j)
? pSector->ceilingz + ksgn(pData[0] - pSector->ceilingz) * j
: pData[0];
}
else
{
pSector->floorz = (klabs(pData[1] - pSector->floorz) >= j)
? pSector->floorz + ksgn(pData[1] - pSector->floorz) * j
: pData[1];
}
}
else
{
pSector->floorz = (klabs(pData[1] - pSector->floorz) >= j)
? pSector->floorz + ksgn(pData[1] - pSector->floorz) * j
: pData[1];
pSector->ceilingz = /*(klabs(pData[0] - pSector->ceilingz) >= j)
? pSector->ceilingz + ksgn(pData[0] - pSector->ceilingz) * j
: */pData[0];
}
#ifdef YAX_ENABLE
if (pSprite->ang == 512)
{
int16_t cf=!pSprite->owner, bn=yax_getbunch(pSector-sector, cf);
int32_t jj, daz=SECTORFLD(pSector-sector,z, cf);
if (bn >= 0)
{
for (SECTORS_OF_BUNCH(bn, cf, jj))
{
SECTORFLD(jj,z, cf) = daz;
SECTORFLD(jj,stat, cf) &= ~(128+256 + 512+2048);
}
for (SECTORS_OF_BUNCH(bn, !cf, jj))
{
SECTORFLD(jj,z, !cf) = daz;
SECTORFLD(jj,stat, !cf) &= ~(128+256 + 512+2048);
}
}
}
#endif
if (pData[3] == 1)
{
//Change the shades
pData[3]++;
pSector->ceilingstat ^= 1;
if (pSprite->ang == 512)
{
walltype *pWall = &wall[pSector->wallptr];
for (j = pSector->wallnum; j > 0; j--, pWall++)
pWall->shade = pSprite->shade;
pSector->floorshade = pSprite->shade;
if (g_player[0].ps->one_parallax_sectnum >= 0)
{
pSector->ceilingpicnum = sector[g_player[0].ps->one_parallax_sectnum].ceilingpicnum;
pSector->ceilingshade = sector[g_player[0].ps->one_parallax_sectnum].ceilingshade;
}
}
}
if (++pData[2] > 256)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
if (pData[2] == 4 && pSprite->ang != 512)
for (x=0; x<7; x++) RANDOMSCRAP(pSprite, spriteNum);
break;
case SE_15_SLIDING_DOOR:
if (pData[4])
{
pSprite->xvel = 16;
if (pData[4] == 1) //Opening
{
if (pData[3] >= (SP(spriteNum)>>3))
{
pData[4] = 0; //Turn off the sliders
A_CallSound(pSprite->sectnum,spriteNum);
break;
}
pData[3]++;
}
else if (pData[4] == 2)
{
if (pData[3]<1)
{
pData[4] = 0;
A_CallSound(pSprite->sectnum,spriteNum);
break;
}
pData[3]--;
}
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
}
break;
case SE_16_REACTOR: //Reactor
pData[2]+=32;
if (pSector->floorz < pSector->ceilingz)
pSprite->shade = 0;
else if (pSector->ceilingz < pData[3])
{
//The following code check to see if
//there is any other sprites in the sector.
//If there isn't, then kill this sectoreffector
//itself.....
for (SPRITES_OF_SECT(pSprite->sectnum, j))
{
if (sprite[j].picnum == REACTOR || sprite[j].picnum == REACTOR2)
break;
}
if (j == -1)
DELETE_SPRITE_AND_CONTINUE(spriteNum);
pSprite->shade = 1;
}
pSector->ceilingz = (pSprite->shade)
? pSector->ceilingz + 1024
: pSector->ceilingz - 512;
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
break;
case SE_17_WARP_ELEVATOR:
{
int32_t nextk;
q = pData[0]*(SP(spriteNum)<<2);
pSector->ceilingz += q;
pSector->floorz += q;
for (SPRITES_OF_SECT(pSprite->sectnum, j))
{
if (sprite[j].statnum == STAT_PLAYER && sprite[j].owner >= 0)
{
int const warpPlayer = P_Get(j);
DukePlayer_t *const pPlayer = g_player[warpPlayer].ps;
if (numplayers < 2 && !g_netServer)
pPlayer->opos.z = pPlayer->pos.z;
pPlayer->pos.z += q;
pPlayer->truefz += q;
pPlayer->truecz += q;
if (g_netServer || numplayers > 1)
pPlayer->opos.z = pPlayer->pos.z;
}
if (sprite[j].statnum != STAT_EFFECTOR)
{
actor[j].bpos.z = sprite[j].z;
sprite[j].z += q;
}
actor[j].floorz = pSector->floorz;
actor[j].ceilingz = pSector->ceilingz;
}
if (pData[0]) //If in motion
{
if (klabs(pSector->floorz-pData[2]) <= SP(spriteNum))
{
G_ActivateWarpElevators(spriteNum,0);
break;
}
// If we still see the opening, we can't yet teleport.
if (pData[0]==-1)
{
if (pSector->floorz > pData[3])
break;
}
else if (pSector->ceilingz < pData[4]) break;
if (pData[1] == 0) break;
pData[1] = 0;
for (SPRITES_OF(STAT_EFFECTOR, j))
{
if (spriteNum != j && sprite[j].lotag == SE_17_WARP_ELEVATOR)
if (pSector->hitag-pData[0] == sector[sprite[j].sectnum].hitag
&& spriteHitag == sprite[j].hitag)
break;
}
if (j == -1) break;
for (SPRITES_OF_SECT_SAFE(pSprite->sectnum, k, nextk))
{
if (sprite[k].statnum == STAT_PLAYER && sprite[k].owner >= 0)
{
int const warpPlayer = P_Get(k);
DukePlayer_t *const pPlayer = g_player[warpPlayer].ps;
pPlayer->pos.x += sprite[j].x - pSprite->x;
pPlayer->pos.y += sprite[j].y - pSprite->y;
pPlayer->pos.z = sector[sprite[j].sectnum].floorz - (pSector->floorz - pPlayer->pos.z);
actor[k].floorz = sector[sprite[j].sectnum].floorz;
actor[k].ceilingz = sector[sprite[j].sectnum].ceilingz;
*(vec2_t *)&pPlayer->opos = *(vec2_t *)pPlayer;
*(vec2_t *)&pPlayer->bobpos = *(vec2_t *)pPlayer;
pPlayer->opos.z = pPlayer->pos.z;
pPlayer->truefz = actor[k].floorz;
pPlayer->truecz = actor[k].ceilingz;
pPlayer->bobcounter = 0;
changespritesect(k, sprite[j].sectnum);
pPlayer->cursectnum = sprite[j].sectnum;
}
else if (sprite[k].statnum != STAT_EFFECTOR)
{
sprite[k].x += sprite[j].x-pSprite->x;
sprite[k].y += sprite[j].y-pSprite->y;
sprite[k].z = sector[sprite[j].sectnum].floorz - (pSector->floorz - sprite[k].z);
Bmemcpy(&actor[k].bpos, &sprite[k], sizeof(vec3_t));
changespritesect(k,sprite[j].sectnum);
setsprite(k,(vec3_t *)&sprite[k]);
actor[k].floorz = sector[sprite[j].sectnum].floorz;
actor[k].ceilingz = sector[sprite[j].sectnum].ceilingz;
}
}
}
break;
}
case SE_18_INCREMENTAL_SECTOR_RISE_FALL:
if (pData[0])
{
if (pSprite->pal)
{
if (pSprite->ang == 512)
{
pSector->ceilingz -= pSector->extra;
if (pSector->ceilingz <= pData[1])
{
pSector->ceilingz = pData[1];
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
else
{
pSector->floorz += pSector->extra;
if (!RR)
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, sectSprite))
{
if (sprite[sectSprite].picnum == APLAYER && sprite[sectSprite].owner >= 0 && g_player[P_Get(sectSprite)].ps->on_ground == 1)
g_player[P_Get(sectSprite)].ps->pos.z += pSector->extra;
if (sprite[sectSprite].zvel == 0 && sprite[sectSprite].statnum != STAT_EFFECTOR && sprite[sectSprite].statnum != STAT_PROJECTILE)
{
actor[sectSprite].bpos.z = sprite[sectSprite].z += pSector->extra;
actor[sectSprite].floorz = pSector->floorz;
}
}
if (pSector->floorz >= pData[1])
{
pSector->floorz = pData[1];
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
}
else
{
if (pSprite->ang == 512)
{
pSector->ceilingz += pSector->extra;
if (pSector->ceilingz >= pSprite->z)
{
pSector->ceilingz = pSprite->z;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
else
{
pSector->floorz -= pSector->extra;
if (!RR)
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, sectSprite))
{
if (sprite[sectSprite].picnum == APLAYER && sprite[sectSprite].owner >= 0 &&g_player[P_Get(sectSprite)].ps->on_ground == 1)
g_player[P_Get(sectSprite)].ps->pos.z -= pSector->extra;
if (sprite[sectSprite].zvel == 0 && sprite[sectSprite].statnum != STAT_EFFECTOR && sprite[sectSprite].statnum != STAT_PROJECTILE)
{
actor[sectSprite].bpos.z = sprite[sectSprite].z -= pSector->extra;
actor[sectSprite].floorz = pSector->floorz;
}
}
if (pSector->floorz <= pSprite->z)
{
pSector->floorz = pSprite->z;
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
}
if (++pData[2] >= pSprite->hitag)
{
pData[2] = 0;
pData[0] = 0;
}
}
break;
case SE_19_EXPLOSION_LOWERS_CEILING: //Battlestar galactia shields
if (pData[0])
{
if (pData[0] == 1)
{
pData[0]++;
x = pSector->wallptr;
q = x+pSector->wallnum;
for (j=x; j<q; j++)
{
if (wall[j].overpicnum == BIGFORCE)
{
wall[j].cstat &= (128+32+8+4+2);
wall[j].overpicnum = 0;
if (wall[j].nextwall >= 0)
{
wall[wall[j].nextwall].overpicnum = 0;
wall[wall[j].nextwall].cstat &= (128+32+8+4+2);
}
}
}
}
if (pSector->ceilingz < pSector->floorz)
pSector->ceilingz += SP(spriteNum);
else
{
pSector->ceilingz = pSector->floorz;
for (SPRITES_OF(STAT_EFFECTOR, j))
{
if (sprite[j].lotag == SE_0_ROTATING_SECTOR && sprite[j].hitag==spriteHitag)
{
sectortype *const pSector = &sector[sprite[j].sectnum];
int const ownerSector = sprite[sprite[j].owner].sectnum;
pSector->ceilingpal = sector[ownerSector].floorpal;
pSector->floorpal = pSector->ceilingpal;
pSector->ceilingshade = sector[ownerSector].floorshade;
pSector->floorshade = pSector->ceilingshade;
actor[sprite[j].owner].t_data[0] = 2;
}
}
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
else //Not hit yet
{
if (G_FindExplosionInSector(pSprite->sectnum) >= 0)
{
P_DoQuote(QUOTE_UNLOCKED, g_player[myconnectindex].ps);
for (SPRITES_OF(STAT_EFFECTOR, l))
{
switch (sprite[l].lotag & 0x7fff)
{
case SE_0_ROTATING_SECTOR:
if (sprite[l].hitag == spriteHitag)
{
int const spriteOwner = sprite[l].owner;
int const sectNum = sprite[l].sectnum;
sector[sectNum].ceilingshade = sprite[spriteOwner].shade;
sector[sectNum].floorshade = sector[sectNum].ceilingshade;
sector[sectNum].ceilingpal = sprite[spriteOwner].pal;
sector[sectNum].floorpal = sector[sectNum].ceilingpal;
}
break;
case SE_1_PIVOT:
case SE_12_LIGHT_SWITCH:
// case SE_18_INCREMENTAL_SECTOR_RISE_FALL:
case SE_19_EXPLOSION_LOWERS_CEILING:
if (spriteHitag == sprite[l].hitag)
if (actor[l].t_data[0] == 0)
{
actor[l].t_data[0] = 1; // Shut them all on
sprite[l].owner = spriteNum;
}
break;
}
}
}
}
break;
case SE_20_STRETCH_BRIDGE: //Extend-o-bridge
if (pData[0] == 0) break;
pSprite->xvel = (pData[0] == 1) ? 8 : -8;
if (pSprite->xvel) //Moving
{
vec2_t const vect = { (pSprite->xvel * sintable[(pSprite->ang + 512) & 2047]) >> 14,
(pSprite->xvel * sintable[pSprite->ang & 2047]) >> 14 };
pData[3] += pSprite->xvel;
pSprite->x += vect.x;
pSprite->y += vect.y;
if (pData[3] <= 0 || (pData[3] >> 6) >= (SP(spriteNum) >> 6))
{
pSprite->x -= vect.x;
pSprite->y -= vect.y;
pData[0] = 0;
A_CallSound(pSprite->sectnum, spriteNum);
break;
}
for (bssize_t nextSprite, SPRITES_OF_SECT_SAFE(pSprite->sectnum, sectSprite, nextSprite))
{
if (sprite[sectSprite].statnum != STAT_EFFECTOR && sprite[sectSprite].zvel == 0)
{
sprite[sectSprite].x += vect.x;
sprite[sectSprite].y += vect.y;
setsprite(sectSprite, (vec3_t *)&sprite[sectSprite]);
if (sector[sprite[sectSprite].sectnum].floorstat & 2 && sprite[sectSprite].statnum == STAT_ZOMBIEACTOR)
A_Fall(sectSprite);
}
}
dragpoint((int16_t)pData[1], wall[pData[1]].x + vect.x, wall[pData[1]].y + vect.y, 0);
dragpoint((int16_t)pData[2], wall[pData[2]].x + vect.x, wall[pData[2]].y + vect.y, 0);
for (bssize_t TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (pPlayer->cursectnum == pSprite->sectnum && pPlayer->on_ground)
{
pPlayer->pos.x += vect.x;
pPlayer->pos.y += vect.y;
pPlayer->opos.x = pPlayer->pos.x;
pPlayer->opos.y = pPlayer->pos.y;
pPlayer->pos.z += PHEIGHT;
setsprite(pPlayer->i, (vec3_t *)pPlayer);
pPlayer->pos.z -= PHEIGHT;
}
}
pSector->floorxpanning -= vect.x >> 3;
pSector->floorypanning -= vect.y >> 3;
pSector->ceilingxpanning -= vect.x >> 3;
pSector->ceilingypanning -= vect.y >> 3;
}
break;
case SE_21_DROP_FLOOR: // Cascading effect
{
if (pData[0] == 0) break;
int32_t *zptr = (pSprite->ang == 1536) ? &pSector->ceilingz : &pSector->floorz;
if (pData[0] == 1) //Decide if the s->sectnum should go up or down
{
pSprite->zvel = ksgn(pSprite->z-*zptr) * (SP(spriteNum)<<4);
pData[0]++;
}
if (pSector->extra == 0)
{
*zptr += pSprite->zvel;
if (klabs(*zptr-pSprite->z) < 1024)
{
*zptr = pSprite->z;
DELETE_SPRITE_AND_CONTINUE(spriteNum); //All done // SE_21_KILLIT, see sector.c
}
}
else pSector->extra--;
break;
}
case SE_22_TEETH_DOOR:
if (pData[1])
{
if (GetAnimationGoal(&sector[pData[0]].ceilingz) >= 0)
pSector->ceilingz += pSector->extra*9;
else pData[1] = 0;
}
break;
case 156:
if (!RRRA) break;
fallthrough__;
case SE_24_CONVEYOR:
case SE_34:
{
if (pData[4])
break;
vec2_t const vect = { (SP(spriteNum) * sintable[(pSprite->ang + 512) & 2047]) >> 18,
(SP(spriteNum) * sintable[pSprite->ang & 2047]) >> 18 };
k = 0;
for (bssize_t nextSprite, SPRITES_OF_SECT_SAFE(pSprite->sectnum, sectSprite, nextSprite))
{
if (sprite[sectSprite].zvel < 0)
continue;
switch (sprite[sectSprite].statnum)
{
case STAT_MISC:
switch (DYNAMICTILEMAP(sprite[sectSprite].picnum))
{
case PUKE__STATIC:
case FOOTPRINTS4__STATIC:
case BLOODSPLAT1__STATIC:
case BLOODSPLAT2__STATIC:
case BLOODSPLAT3__STATIC:
case BLOODSPLAT4__STATIC:
if (RR) break;
fallthrough__;
case BULLETHOLE__STATIC:
if (RR && sprite[sectSprite].picnum == BULLETHOLE) continue;
fallthrough__;
case BLOODPOOL__STATIC:
case FOOTPRINTS__STATIC:
case FOOTPRINTS2__STATIC:
case FOOTPRINTS3__STATIC: sprite[sectSprite].xrepeat = sprite[sectSprite].yrepeat = 0; continue;
case LASERLINE__STATIC: if (RR) break; continue;
}
fallthrough__;
case STAT_STANDABLE:
if (!RR && sprite[sectSprite].picnum == TRIPBOMB)
break;
fallthrough__;
case STAT_ACTOR:
case STAT_DEFAULT:
if (sprite[sectSprite].picnum == BOLT1
|| sprite[sectSprite].picnum == BOLT1 + 1
|| sprite[sectSprite].picnum == BOLT1 + 2
|| sprite[sectSprite].picnum == BOLT1 + 3
|| (!RR && (sprite[sectSprite].picnum == SIDEBOLT1
|| sprite[sectSprite].picnum == SIDEBOLT1 + 1
|| sprite[sectSprite].picnum == SIDEBOLT1 + 2
|| sprite[sectSprite].picnum == SIDEBOLT1 + 3))
|| A_CheckSwitchTile(sectSprite))
break;
if (!(sprite[sectSprite].picnum >= CRANE && sprite[sectSprite].picnum <= CRANE + 3))
{
if (sprite[sectSprite].z > actor[sectSprite].floorz - ZOFFSET2)
{
actor[sectSprite].bpos.x = sprite[sectSprite].x;
actor[sectSprite].bpos.y = sprite[sectSprite].y;
sprite[sectSprite].x += vect.x >> (RR ? 1 : 2);
sprite[sectSprite].y += vect.y >> (RR ? 1 : 2);
setsprite(sectSprite, (vec3_t *)&sprite[sectSprite]);
if (sector[sprite[sectSprite].sectnum].floorstat & 2)
if (sprite[sectSprite].statnum == STAT_ZOMBIEACTOR)
A_Fall(sectSprite);
}
}
break;
}
}
for (bssize_t TRAVERSE_CONNECT(playerNum))
{
DukePlayer_t *const pPlayer = g_player[playerNum].ps;
if (pPlayer->cursectnum == pSprite->sectnum && pPlayer->on_ground)
{
if (klabs(pPlayer->pos.z - pPlayer->truefz) < PHEIGHT + (9 << 8))
{
pPlayer->fric.x += vect.x << 3;
pPlayer->fric.y += vect.y << 3;
}
}
}
if (!RRRA || spriteLotag != 156)
pSector->floorxpanning += SP(spriteNum)>>7;
break;
}
case SE_35:
if (pSector->ceilingz > pSprite->z)
{
for (j = 0; j < 8; j++)
{
pSprite->ang += krand2()&511;
k = A_Spawn(spriteNum, SMALLSMOKE);
sprite[k].xvel = 96+(krand2()&127);
A_SetSprite(k, CLIPMASK0);
setsprite(k, (vec3_t *) &sprite[k]);
if (rnd(16))
A_Spawn(spriteNum, EXPLOSION2);
}
}
switch (pData[0])
{
case 0:
pSector->ceilingz += pSprite->yvel;
if (pSector->ceilingz > pSector->floorz)
pSector->floorz = pSector->ceilingz;
if (pSector->ceilingz > pSprite->z+ZOFFSET5)
pData[0]++;
break;
case 1:
pSector->ceilingz-=(pSprite->yvel<<2);
if (pSector->ceilingz < pData[4])
{
pSector->ceilingz = pData[4];
pData[0] = 0;
}
break;
}
break;
case SE_25_PISTON: //PISTONS
if (pData[4] == 0) break;
if (pSector->floorz <= pSector->ceilingz)
pSprite->shade = 0;
else if (pSector->ceilingz <= pData[RR?4:3])
pSprite->shade = 1;
if (pSprite->shade)
{
pSector->ceilingz += SP(spriteNum)<<4;
if (pSector->ceilingz > pSector->floorz)
{
pSector->ceilingz = pSector->floorz;
if (RRRA && g_pistonSound)
A_PlaySound(371, spriteNum);
}
}
else
{
pSector->ceilingz -= SP(spriteNum)<<4;
if (pSector->ceilingz < pData[RR?4:3])
{
pSector->ceilingz = pData[RR?4:3];
if (RRRA && g_pistonSound)
A_PlaySound(167, spriteNum);
}
}
break;
case SE_26:
{
int32_t p, nextj;
pSprite->xvel = 32;
l = (pSprite->xvel*sintable[(pSprite->ang+512)&2047])>>14;
x = (pSprite->xvel*sintable[pSprite->ang&2047])>>14;
pSprite->shade++;
if (pSprite->shade > 7)
{
pSprite->x = pData[3];
pSprite->y = pData[4];
pSector->floorz -= ((pSprite->zvel*pSprite->shade)-pSprite->zvel);
pSprite->shade = 0;
}
else
pSector->floorz += pSprite->zvel;
for (SPRITES_OF_SECT_SAFE(pSprite->sectnum, j, nextj))
{
if (sprite[j].statnum != STAT_EFFECTOR && sprite[j].statnum != STAT_PLAYER )
{
actor[j].bpos.x = sprite[j].x;
actor[j].bpos.y = sprite[j].y;
sprite[j].x += l;
sprite[j].y += x;
sprite[j].z += pSprite->zvel;
setsprite(j, (vec3_t *)&sprite[j]);
}
}
for (TRAVERSE_CONNECT(p))
{
DukePlayer_t *const pPlayer = g_player[p].ps;
if (pSprite->sectnum == sprite[pPlayer->i].sectnum && pPlayer->on_ground)
{
pPlayer->fric.x += l << 5;
pPlayer->fric.y += x << 5;
pPlayer->pos.z += pSprite->zvel;
}
}
A_MoveSector(spriteNum);
setsprite(spriteNum,(vec3_t *)pSprite);
break;
}
case SE_27_DEMO_CAM:
{
if (ud.recstat == 0 || !cl_democams) break;
actor[spriteNum].tempang = pSprite->ang;
int const p = A_FindPlayer(pSprite,&x);
DukePlayer_t * const ps = g_player[p].ps;
if (sprite[ps->i].extra > 0 && myconnectindex == screenpeek)
{
if (pData[0] < 0)
{
ud.camerasprite = spriteNum;
pData[0]++;
}
else if (ud.recstat == 2 && ps->newowner == -1)
{
if (cansee(pSprite->x,pSprite->y,pSprite->z,SECT(spriteNum),ps->pos.x,ps->pos.y,ps->pos.z,ps->cursectnum))
{
if (x < (int32_t)((unsigned)spriteHitag))
{
ud.camerasprite = spriteNum;
pData[0] = 999;
pSprite->ang += G_GetAngleDelta(pSprite->ang,getangle(ps->pos.x-pSprite->x,ps->pos.y-pSprite->y))>>3;
SP(spriteNum) = 100+((pSprite->z-ps->pos.z)/257);
}
else if (pData[0] == 999)
{
if (ud.camerasprite == spriteNum)
pData[0] = 0;
else pData[0] = -10;
ud.camerasprite = spriteNum;
}
}
else
{
pSprite->ang = getangle(ps->pos.x-pSprite->x,ps->pos.y-pSprite->y);
if (pData[0] == 999)
{
if (ud.camerasprite == spriteNum)
pData[0] = 0;
else pData[0] = -20;
ud.camerasprite = spriteNum;
}
}
}
}
break;
}
case SE_28_LIGHTNING:
{
if (RR)
break;
if (pData[5] > 0)
{
pData[5]--;
break;
}
if (T1(spriteNum) == 0)
{
A_FindPlayer(pSprite,&x);
if (x > 15500)
break;
T1(spriteNum) = 1;
T2(spriteNum) = 64 + (krand2()&511);
T3(spriteNum) = 0;
}
else
{
T3(spriteNum)++;
if (T3(spriteNum) > T2(spriteNum))
{
T1(spriteNum) = 0;
g_player[screenpeek].ps->visibility = ud.const_visibility;
break;
}
else if (T3(spriteNum) == (T2(spriteNum)>>1))
A_PlaySound(THUNDER,spriteNum);
else if (T3(spriteNum) == (T2(spriteNum)>>3))
A_PlaySound(LIGHTNING_SLAP,spriteNum);
else if (T3(spriteNum) == (T2(spriteNum)>>2))
{
for (SPRITES_OF(STAT_DEFAULT, j))
if (sprite[j].picnum == NATURALLIGHTNING && sprite[j].hitag == pSprite->hitag)
sprite[j].cstat |= 32768;
}
else if (T3(spriteNum) > (T2(spriteNum)>>3) && T3(spriteNum) < (T2(spriteNum)>>2))
{
if (cansee(pSprite->x,pSprite->y,pSprite->z,pSprite->sectnum,g_player[screenpeek].ps->pos.x,g_player[screenpeek].ps->pos.y,g_player[screenpeek].ps->pos.z,g_player[screenpeek].ps->cursectnum))
j = 1;
else j = 0;
if (rnd(192) && (T3(spriteNum)&1))
{
if (j)
g_player[screenpeek].ps->visibility = 0;
}
else if (j)
g_player[screenpeek].ps->visibility = ud.const_visibility;
for (SPRITES_OF(STAT_DEFAULT, j))
{
if (sprite[j].picnum == NATURALLIGHTNING && sprite[j].hitag == pSprite->hitag)
{
if (rnd(32) && (T3(spriteNum)&1))
{
int32_t p;
DukePlayer_t *ps;
sprite[j].cstat &= 32767;
A_Spawn(j,SMALLSMOKE);
p = A_FindPlayer(pSprite, NULL);
ps = g_player[p].ps;
x = ldist(&sprite[ps->i], &sprite[j]);
if (x < 768)
{
if (!A_CheckSoundPlaying(ps->i,DUKE_LONGTERM_PAIN))
A_PlaySound(DUKE_LONGTERM_PAIN,ps->i);
A_PlaySound(SHORT_CIRCUIT,ps->i);
sprite[ps->i].extra -= 8+(krand2()&7);
P_PalFrom(ps, 32, 16,0,0);
}
break;
}
else sprite[j].cstat |= 32768;
}
}
}
}
break;
}
case SE_29_WAVES:
pSprite->hitag += 64;
l = mulscale12((int32_t)pSprite->yvel,sintable[pSprite->hitag&2047]);
pSector->floorz = pSprite->z + l;
break;
case SE_31_FLOOR_RISE_FALL: // True Drop Floor
if (pData[0] == 1)
{
// Choose dir
if (!RR && pData[3] > 0)
{
pData[3]--;
break;
}
if (pData[2] == 1) // Retract
{
if (SA(spriteNum) != 1536)
HandleSE31(spriteNum, 1, pSprite->z, 0, pSprite->z-pSector->floorz);
else
HandleSE31(spriteNum, 1, pData[1], 0, pData[1]-pSector->floorz);
Yax_SetBunchZs(pSector-sector, YAX_FLOOR, pSector->floorz);
break;
}
if ((pSprite->ang&2047) == 1536)
HandleSE31(spriteNum, 0, pSprite->z, 1, pSprite->z-pSector->floorz);
else
HandleSE31(spriteNum, 0, pData[1], 1, pData[1]-pSprite->z);
Yax_SetBunchZs(pSector-sector, YAX_FLOOR, pSector->floorz);
}
break;
case SE_32_CEILING_RISE_FALL: // True Drop Ceiling
if (pData[0] == 1)
{
// Choose dir
if (pData[2] == 1) // Retract
{
if (SA(spriteNum) != 1536)
{
if (klabs(pSector->ceilingz - pSprite->z) < (SP(spriteNum)<<1))
{
pSector->ceilingz = pSprite->z;
A_CallSound(pSprite->sectnum,spriteNum);
pData[2] = 0;
pData[0] = 0;
}
else pSector->ceilingz += ksgn(pSprite->z-pSector->ceilingz)*SP(spriteNum);
}
else
{
if (klabs(pSector->ceilingz - pData[1]) < (SP(spriteNum)<<1))
{
pSector->ceilingz = pData[1];
A_CallSound(pSprite->sectnum,spriteNum);
pData[2] = 0;
pData[0] = 0;
}
else pSector->ceilingz += ksgn(pData[1]-pSector->ceilingz)*SP(spriteNum);
}
Yax_SetBunchZs(pSector-sector, YAX_CEILING, pSector->ceilingz);
break;
}
if ((pSprite->ang&2047) == 1536)
{
if (klabs(pSector->ceilingz-pSprite->z) < (SP(spriteNum)<<1))
{
pData[0] = 0;
pData[2] = !pData[2];
A_CallSound(pSprite->sectnum,spriteNum);
pSector->ceilingz = pSprite->z;
}
else pSector->ceilingz += ksgn(pSprite->z-pSector->ceilingz)*SP(spriteNum);
}
else
{
if (klabs(pSector->ceilingz-pData[1]) < (SP(spriteNum)<<1))
{
pData[0] = 0;
pData[2] = !pData[2];
A_CallSound(pSprite->sectnum,spriteNum);
}
else pSector->ceilingz -= ksgn(pSprite->z-pData[1])*SP(spriteNum);
}
Yax_SetBunchZs(pSector-sector, YAX_CEILING, pSector->ceilingz);
}
break;
case SE_33_QUAKE_DEBRIS:
if (g_earthquakeTime > 0 && (krand2()&7) == 0)
RANDOMSCRAP(pSprite, spriteNum);
break;
case SE_36_PROJ_SHOOTER:
if (pData[0])
{
if (pData[0] == 1)
A_Shoot(spriteNum,pSector->extra);
else if (pData[0] == GAMETICSPERSEC*5)
pData[0] = 0;
pData[0]++;
}
break;
case 128: //SE to control glass breakage
{
walltype *pWall = &wall[pData[2]];
if (pWall->cstat|32)
{
pWall->cstat &= (255-32);
pWall->cstat |= 16;
if (pWall->nextwall >= 0)
{
wall[pWall->nextwall].cstat &= (255-32);
wall[pWall->nextwall].cstat |= 16;
}
}
else break;
pWall->overpicnum++;
if (pWall->nextwall >= 0)
wall[pWall->nextwall].overpicnum++;
if (pData[0] < pData[1]) pData[0]++;
else
{
pWall->cstat &= (128+32+8+4+2);
if (pWall->nextwall >= 0)
wall[pWall->nextwall].cstat &= (128+32+8+4+2);
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
}
break;
case SE_130:
if (pData[0] > 80)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else pData[0]++;
x = pSector->floorz-pSector->ceilingz;
if (rnd(64))
{
k = A_Spawn(spriteNum,EXPLOSION2);
sprite[k].xrepeat = sprite[k].yrepeat = 2+(krand2()&7);
sprite[k].z = pSector->floorz-(krand2()%x);
sprite[k].ang += 256-(krand2()%511);
sprite[k].xvel = krand2()&127;
A_SetSprite(k,CLIPMASK0);
}
break;
case SE_131:
if (pData[0] > 40)
{
DELETE_SPRITE_AND_CONTINUE(spriteNum);
}
else pData[0]++;
x = pSector->floorz-pSector->ceilingz;
if (rnd(32))
{
k = A_Spawn(spriteNum,EXPLOSION2);
sprite[k].xrepeat = sprite[k].yrepeat = 2+(krand2()&3);
sprite[k].z = pSector->floorz-(krand2()%x);
sprite[k].ang += 256-(krand2()%511);
sprite[k].xvel = krand2()&127;
A_SetSprite(k,CLIPMASK0);
}
break;
case SE_49_POINT_LIGHT:
case SE_50_SPOT_LIGHT:
changespritestat(spriteNum, STAT_LIGHT);
break;
}
next_sprite:
spriteNum = nextSprite;
}
//Sloped sin-wave floors!
for (SPRITES_OF(STAT_EFFECTOR, spriteNum))
{
const spritetype *s = &sprite[spriteNum];
if (s->lotag == SE_29_WAVES)
{
usectortype const *const sc = (usectortype *)&sector[s->sectnum];
if (sc->wallnum == 4)
{
walltype *const pWall = &wall[sc->wallptr+2];
if (pWall->nextsector >= 0)
alignflorslope(s->sectnum, pWall->x,pWall->y, sector[pWall->nextsector].floorz);
}
}
}
}
static void G_DoEffectorLights(void) // STATNUM 14
{
#ifdef POLYMER
int32_t i;
for (SPRITES_OF(STAT_LIGHT, i))
{
switch (sprite[i].lotag)
{
case SE_49_POINT_LIGHT:
{
if (!A_CheckSpriteFlags(i, SFLAG_NOLIGHT) && videoGetRenderMode() == REND_POLYMER &&
!(A_CheckSpriteFlags(i, SFLAG_USEACTIVATOR) && sector[sprite[i].sectnum].lotag & 16384))
{
if (actor[i].lightptr == NULL)
{
#pragma pack(push,1)
_prlight mylight;
#pragma pack(pop)
mylight.sector = SECT(i);
Bmemcpy(&mylight, &sprite[i], sizeof(int32_t) * 3);
mylight.range = SHT(i);
mylight.color[0] = sprite[i].xvel;
mylight.color[1] = sprite[i].yvel;
mylight.color[2] = sprite[i].zvel;
mylight.radius = 0;
mylight.angle = SA(i);
mylight.horiz = SH(i);
mylight.minshade = sprite[i].xoffset;
mylight.maxshade = sprite[i].yoffset;
mylight.tilenum = 0;
mylight.publicflags.emitshadow = 0;
mylight.publicflags.negative = !!(CS(i) & 128);
if (CS(i) & 2)
{
if (CS(i) & 512)
mylight.priority = PR_LIGHT_PRIO_LOW;
else
mylight.priority = PR_LIGHT_PRIO_HIGH;
}
else
mylight.priority = PR_LIGHT_PRIO_MAX;
actor[i].lightId = polymer_addlight(&mylight);
if (actor[i].lightId >= 0)
actor[i].lightptr = &prlights[actor[i].lightId];
break;
}
if (Bmemcmp(&sprite[i], actor[i].lightptr, sizeof(int32_t) * 3))
{
Bmemcpy(actor[i].lightptr, &sprite[i], sizeof(int32_t) * 3);
actor[i].lightptr->sector = sprite[i].sectnum;
actor[i].lightptr->flags.invalidate = 1;
}
if (SHT(i) != actor[i].lightptr->range)
{
actor[i].lightptr->range = SHT(i);
actor[i].lightptr->flags.invalidate = 1;
}
if ((sprite[i].xvel != actor[i].lightptr->color[0]) ||
(sprite[i].yvel != actor[i].lightptr->color[1]) ||
(sprite[i].zvel != actor[i].lightptr->color[2]))
{
actor[i].lightptr->color[0] = sprite[i].xvel;
actor[i].lightptr->color[1] = sprite[i].yvel;
actor[i].lightptr->color[2] = sprite[i].zvel;
}
if ((int)!!(CS(i) & 128) != actor[i].lightptr->publicflags.negative) {
actor[i].lightptr->publicflags.negative = !!(CS(i) & 128);
}
}
break;
}
case SE_50_SPOT_LIGHT:
{
if (!A_CheckSpriteFlags(i, SFLAG_NOLIGHT) && videoGetRenderMode() == REND_POLYMER &&
!(A_CheckSpriteFlags(i, SFLAG_USEACTIVATOR) && sector[sprite[i].sectnum].lotag & 16384))
{
if (actor[i].lightptr == NULL)
{
#pragma pack(push,1)
_prlight mylight;
#pragma pack(pop)
mylight.sector = SECT(i);
Bmemcpy(&mylight, &sprite[i], sizeof(int32_t) * 3);
mylight.range = SHT(i);
mylight.color[0] = sprite[i].xvel;
mylight.color[1] = sprite[i].yvel;
mylight.color[2] = sprite[i].zvel;
mylight.radius = (256-(SS(i)+128))<<1;
mylight.faderadius = (int16_t)(mylight.radius * 0.75f);
mylight.angle = SA(i);
mylight.horiz = SH(i);
mylight.minshade = sprite[i].xoffset;
mylight.maxshade = sprite[i].yoffset;
mylight.tilenum = actor[i].picnum;
mylight.publicflags.emitshadow = !(CS(i) & 64);
mylight.publicflags.negative = !!(CS(i) & 128);
if (CS(i) & 2)
{
if (CS(i) & 512)
mylight.priority = PR_LIGHT_PRIO_LOW;
else
mylight.priority = PR_LIGHT_PRIO_HIGH;
}
else
mylight.priority = PR_LIGHT_PRIO_MAX;
actor[i].lightId = polymer_addlight(&mylight);
if (actor[i].lightId >= 0)
{
actor[i].lightptr = &prlights[actor[i].lightId];
// Hack in case polymer_addlight tweaked the horiz value
if (actor[i].lightptr->horiz != SH(i))
SH(i) = actor[i].lightptr->horiz;
}
break;
}
if (Bmemcmp(&sprite[i], actor[i].lightptr, sizeof(int32_t) * 3))
{
Bmemcpy(actor[i].lightptr, &sprite[i], sizeof(int32_t) * 3);
actor[i].lightptr->sector = sprite[i].sectnum;
actor[i].lightptr->flags.invalidate = 1;
}
if (SHT(i) != actor[i].lightptr->range)
{
actor[i].lightptr->range = SHT(i);
actor[i].lightptr->flags.invalidate = 1;
}
if ((sprite[i].xvel != actor[i].lightptr->color[0]) ||
(sprite[i].yvel != actor[i].lightptr->color[1]) ||
(sprite[i].zvel != actor[i].lightptr->color[2]))
{
actor[i].lightptr->color[0] = sprite[i].xvel;
actor[i].lightptr->color[1] = sprite[i].yvel;
actor[i].lightptr->color[2] = sprite[i].zvel;
}
if (((256-(SS(i)+128))<<1) != actor[i].lightptr->radius)
{
actor[i].lightptr->radius = (256-(SS(i)+128))<<1;
actor[i].lightptr->faderadius = (int16_t)(actor[i].lightptr->radius * 0.75f);
actor[i].lightptr->flags.invalidate = 1;
}
if (SA(i) != actor[i].lightptr->angle)
{
actor[i].lightptr->angle = SA(i);
actor[i].lightptr->flags.invalidate = 1;
}
if (SH(i) != actor[i].lightptr->horiz)
{
actor[i].lightptr->horiz = SH(i);
actor[i].lightptr->flags.invalidate = 1;
}
if ((int)!(CS(i) & 64) != actor[i].lightptr->publicflags.emitshadow) {
actor[i].lightptr->publicflags.emitshadow = !(CS(i) & 64);
}
if ((int)!!(CS(i) & 128) != actor[i].lightptr->publicflags.negative) {
actor[i].lightptr->publicflags.negative = !!(CS(i) & 128);
}
actor[i].lightptr->tilenum = actor[i].picnum;
}
break;
}
}
}
#endif // POLYMER
}
#ifdef POLYMER
static void A_DoLight(int spriteNum)
{
spritetype *const pSprite = &sprite[spriteNum];
int savedFires = 0;
if (((sector[pSprite->sectnum].floorz - sector[pSprite->sectnum].ceilingz) < 16) || pSprite->z > sector[pSprite->sectnum].floorz || pSprite->z > actor[spriteNum].floorz ||
(pSprite->picnum != SECTOREFFECTOR && ((pSprite->cstat & 32768) || pSprite->yrepeat < 4)) ||
A_CheckSpriteFlags(spriteNum, SFLAG_NOLIGHT) || (A_CheckSpriteFlags(spriteNum, SFLAG_USEACTIVATOR) && sector[pSprite->sectnum].lotag & 16384))
{
if (actor[spriteNum].lightptr != NULL)
A_DeleteLight(spriteNum);
}
else
{
if (actor[spriteNum].lightptr != NULL && actor[spriteNum].lightcount)
{
if (!(--actor[spriteNum].lightcount))
A_DeleteLight(spriteNum);
}
if (pr_lighting != 1)
return;
for (bsize_t ii=0; ii<2; ii++)
{
if (pSprite->picnum <= 0) // oob safety
break;
switch (DYNAMICTILEMAP(pSprite->picnum-1+ii))
{
case DIPSWITCH__STATIC:
case DIPSWITCH2__STATIC:
case DIPSWITCH3__STATIC:
case PULLSWITCH__STATIC:
case SLOTDOOR__STATIC:
case LIGHTSWITCH__STATIC:
case SPACELIGHTSWITCH__STATIC:
case SPACEDOORSWITCH__STATIC:
case FRANKENSTINESWITCH__STATIC:
case POWERSWITCH1__STATIC:
case LOCKSWITCH1__STATIC:
case POWERSWITCH2__STATIC:
case TECHSWITCH__STATIC:
case ACCESSSWITCH__STATIC:
case ACCESSSWITCH2__STATIC:
{
if ((pSprite->cstat & 32768) || A_CheckSpriteFlags(spriteNum, SFLAG_NOLIGHT))
{
if (actor[spriteNum].lightptr != NULL)
A_DeleteLight(spriteNum);
break;
}
vec2_t const d = { sintable[(pSprite->ang+512)&2047]>>7, sintable[(pSprite->ang)&2047]>>7 };
pSprite->x += d.x;
pSprite->y += d.y;
int16_t sectnum = pSprite->sectnum;
updatesector(pSprite->x, pSprite->y, &sectnum);
if ((unsigned) sectnum >= MAXSECTORS || pSprite->z > sector[sectnum].floorz || pSprite->z < sector[sectnum].ceilingz)
goto POOP;
G_AddGameLight(0, spriteNum, (pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1, 512-ii*128,
ii==0 ? (172+(200<<8)+(104<<16)) : 216+(52<<8)+(20<<16), PR_LIGHT_PRIO_LOW);
POOP:
pSprite->x -= d.x;
pSprite->y -= d.y;
}
break;
}
}
switch (DYNAMICTILEMAP(pSprite->picnum))
{
case ATOMICHEALTH__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD2(spriteNum, pSprite), 128+(128<<8)+(255<<16),PR_LIGHT_PRIO_HIGH_GAME);
break;
case FIRE__STATIC:
case FIRE2__STATIC:
case BURNING__STATIC:
case BURNING2__STATIC:
{
uint32_t color;
int32_t jj;
static int32_t savedfires[32][4]; // sectnum x y z
/*
if (Actor[i].floorz - Actor[i].ceilingz < 128) break;
if (s->z > Actor[i].floorz+2048) break;
*/
switch (pSprite->pal)
{
case 1: color = 128+(128<<8)+(255<<16); break;
case 2: color = 255+(48<<8)+(48<<16); break;
case 8: color = 48+(255<<8)+(48<<16); break;
default: color = 240+(160<<8)+(80<<16); break;
}
for (jj=savedFires-1; jj>=0; jj--)
if (savedfires[jj][0]==pSprite->sectnum && savedfires[jj][1]==(pSprite->x>>3) &&
savedfires[jj][2]==(pSprite->y>>3) && savedfires[jj][3]==(pSprite->z>>7))
break;
if (jj==-1 && savedFires<32)
{
jj = savedFires;
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD2(spriteNum, pSprite), color, PR_LIGHT_PRIO_HIGH_GAME);
savedfires[jj][0] = pSprite->sectnum;
savedfires[jj][1] = pSprite->x>>3;
savedfires[jj][2] = pSprite->y>>3;
savedfires[jj][3] = pSprite->z>>7;
savedFires++;
}
}
break;
case OOZFILTER__STATIC:
if (pSprite->xrepeat > 4)
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 4096, 176+(252<<8)+(120<<16),PR_LIGHT_PRIO_HIGH_GAME);
break;
case FLOORFLAME__STATIC:
case FIREBARREL__STATIC:
case FIREVASE__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<2), LIGHTRAD2(spriteNum, pSprite)>>1, 255+(95<<8),PR_LIGHT_PRIO_HIGH_GAME);
break;
case EXPLOSION2__STATIC:
if (!actor[spriteNum].lightcount)
{
// XXX: This block gets CODEDUP'd too much.
int32_t x = ((sintable[(pSprite->ang+512)&2047])>>6);
int32_t y = ((sintable[(pSprite->ang)&2047])>>6);
pSprite->x -= x;
pSprite->y -= y;
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD(spriteNum, pSprite), 240+(160<<8)+(80<<16),
pSprite->yrepeat > 32 ? PR_LIGHT_PRIO_HIGH_GAME : PR_LIGHT_PRIO_LOW_GAME);
pSprite->x += x;
pSprite->y += y;
}
break;
case FORCERIPPLE__STATIC:
case TRANSPORTERBEAM__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD(spriteNum, pSprite), 80+(80<<8)+(255<<16),PR_LIGHT_PRIO_LOW_GAME);
break;
case GROWSPARK__STATIC:
{
int32_t x = ((sintable[(pSprite->ang+512)&2047])>>6);
int32_t y = ((sintable[(pSprite->ang)&2047])>>6);
pSprite->x -= x;
pSprite->y -= y;
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 1024, 216+(52<<8)+(20<<16),PR_LIGHT_PRIO_HIGH_GAME);
pSprite->x += x;
pSprite->y += y;
}
break;
case SHRINKEREXPLOSION__STATIC:
{
int32_t x = ((sintable[(pSprite->ang+512)&2047])>>6);
int32_t y = ((sintable[(pSprite->ang)&2047])>>6);
pSprite->x -= x;
pSprite->y -= y;
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 2048, 176+(252<<8)+(120<<16),PR_LIGHT_PRIO_HIGH_GAME);
pSprite->x += x;
pSprite->y += y;
}
break;
case FREEZEBLAST__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD(spriteNum, pSprite)<<2, 72+(88<<8)+(140<<16),PR_LIGHT_PRIO_HIGH_GAME);
break;
case COOLEXPLOSION1__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD(spriteNum, pSprite)<<2, 128+(0<<8)+(255<<16),PR_LIGHT_PRIO_HIGH_GAME);
break;
case SHRINKSPARK__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), LIGHTRAD(spriteNum, pSprite), 176+(252<<8)+(120<<16),PR_LIGHT_PRIO_HIGH_GAME);
break;
case FIRELASER__STATIC:
if (pSprite->statnum == STAT_PROJECTILE)
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 64 * pSprite->yrepeat, 255+(95<<8),PR_LIGHT_PRIO_LOW_GAME);
break;
case RPG__STATIC:
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 128 * pSprite->yrepeat, 255+(95<<8),PR_LIGHT_PRIO_LOW_GAME);
break;
case SHOTSPARK1__STATIC:
if (actor[spriteNum].t_data[2] == 0) // check for first frame of action
{
int32_t x = ((sintable[(pSprite->ang+512)&2047])>>7);
int32_t y = ((sintable[(pSprite->ang)&2047])>>7);
pSprite->x -= x;
pSprite->y -= y;
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat*tilesiz[pSprite->picnum].y)<<1), 8 * pSprite->yrepeat, 240+(160<<8)+(80<<16),PR_LIGHT_PRIO_LOW_GAME);
actor[spriteNum].lightcount = 1;
pSprite->x += x;
pSprite->y += y;
}
break;
}
}
}
#endif // POLYMER
void A_PlayAlertSound(int spriteNum)
{
if (RR)
{
if (sprite[spriteNum].extra > 0)
{
switch (DYNAMICTILEMAP(PN(spriteNum)))
{
case COOT__STATICRR: if (!RRRA || (krand2()&3) == 2) A_PlaySound(PRED_RECOG, spriteNum); break;
case LTH__STATICRR: break;
case BILLYCOCK__STATICRR:
case BILLYRAY__STATICRR:
case BRAYSNIPER__STATICRR: A_PlaySound(PIG_RECOG, spriteNum); break;
case DOGRUN__STATICRR:
case HULK__STATICRR:
case HEN__STATICRR:
case DRONE__STATICRR:
case PIG__STATICRR:
case RECON__STATICRR:
case MINION__STATICRR:
case COW__STATICRR:
case VIXEN__STATICRR:
case RABBIT__STATICRR: break;
}
}
return;
}
if (sprite[spriteNum].extra > 0)
{
switch (DYNAMICTILEMAP(PN(spriteNum)))
{
case LIZTROOPONTOILET__STATIC:
case LIZTROOPJUSTSIT__STATIC:
case LIZTROOPSHOOT__STATIC:
case LIZTROOPJETPACK__STATIC:
case LIZTROOPDUCKING__STATIC:
case LIZTROOPRUNNING__STATIC:
case LIZTROOP__STATIC: A_PlaySound(PRED_RECOG, spriteNum); break;
case LIZMAN__STATIC:
case LIZMANSPITTING__STATIC:
case LIZMANFEEDING__STATIC:
case LIZMANJUMP__STATIC: A_PlaySound(CAPT_RECOG, spriteNum); break;
case PIGCOP__STATIC:
case PIGCOPDIVE__STATIC: A_PlaySound(PIG_RECOG, spriteNum); break;
case RECON__STATIC: A_PlaySound(RECO_RECOG, spriteNum); break;
case DRONE__STATIC: A_PlaySound(DRON_RECOG, spriteNum); break;
case COMMANDER__STATIC:
case COMMANDERSTAYPUT__STATIC: A_PlaySound(COMM_RECOG, spriteNum); break;
case ORGANTIC__STATIC: A_PlaySound(TURR_RECOG, spriteNum); break;
case OCTABRAIN__STATIC:
case OCTABRAINSTAYPUT__STATIC: A_PlaySound(OCTA_RECOG, spriteNum); break;
case BOSS1__STATIC: S_PlaySound(BOS1_RECOG); break;
case BOSS2__STATIC: S_PlaySound((sprite[spriteNum].pal == 1) ? BOS2_RECOG : WHIPYOURASS); break;
case BOSS3__STATIC: S_PlaySound((sprite[spriteNum].pal == 1) ? BOS3_RECOG : RIPHEADNECK); break;
case BOSS4__STATIC:
case BOSS4STAYPUT__STATIC: if (sprite[spriteNum].pal == 1) S_PlaySound(BOS4_RECOG); S_PlaySound(BOSS4_FIRSTSEE); break;
case GREENSLIME__STATIC: A_PlaySound(SLIM_RECOG, spriteNum); break;
}
}
}
int A_CheckSwitchTile(int spriteNum)
{
// picnum 0 would oob in the switch below,
if (PN(spriteNum) <= 0)
return 0;
// MULTISWITCH has 4 states so deal with it separately,
// ACCESSSWITCH and ACCESSSWITCH2 are only active in one state so deal with
// them separately.
if ((PN(spriteNum) >= MULTISWITCH && PN(spriteNum) <= MULTISWITCH + 3) || (PN(spriteNum) == ACCESSSWITCH || PN(spriteNum) == ACCESSSWITCH2))
return 1;
if (RRRA && PN(spriteNum) >= MULTISWITCH2 && PN(spriteNum) <= MULTISWITCH2 + 3)
return 1;
// Loop to catch both states of switches.
for (bssize_t j=1; j>=0; j--)
{
switch (DYNAMICTILEMAP(PN(spriteNum)-j))
{
case RRTILE8464__STATICRR:
if (RRRA) return 1;
break;
case NUKEBUTTON__STATIC:
if (RR) return 1;
break;
case HANDPRINTSWITCH__STATIC:
case ALIENSWITCH__STATIC:
case MULTISWITCH__STATIC:
case PULLSWITCH__STATIC:
case HANDSWITCH__STATIC:
case SLOTDOOR__STATIC:
case LIGHTSWITCH__STATIC:
case SPACELIGHTSWITCH__STATIC:
case SPACEDOORSWITCH__STATIC:
case FRANKENSTINESWITCH__STATIC:
case LIGHTSWITCH2__STATIC:
case POWERSWITCH1__STATIC:
case LOCKSWITCH1__STATIC:
case POWERSWITCH2__STATIC:
case DIPSWITCH__STATIC:
case DIPSWITCH2__STATIC:
case TECHSWITCH__STATIC:
case DIPSWITCH3__STATIC:
return 1;
}
}
return 0;
}
void G_RefreshLights(void)
{
#ifdef POLYMER
if (Numsprites && videoGetRenderMode() == REND_POLYMER)
{
int statNum = 0;
do
{
int spriteNum = headspritestat[statNum++];
while (spriteNum >= 0)
{
A_DoLight(spriteNum);
spriteNum = nextspritestat[spriteNum];
}
}
while (statNum < MAXSTATUS);
}
#endif
}
void G_MoveWorld(void)
{
extern double g_moveActorsTime, g_moveWorldTime;
const double worldTime = timerGetHiTicks();
G_MoveZombieActors(); //ST 2
G_MoveWeapons(); //ST 4
G_MoveTransports(); //ST 9
G_MovePlayers(); //ST 10
G_MoveFallers(); //ST 12
G_MoveMisc(); //ST 5
const double actorsTime = timerGetHiTicks();
G_MoveActors(); //ST 1
g_moveActorsTime = (1-0.033)*g_moveActorsTime + 0.033*(timerGetHiTicks()-actorsTime);
// XXX: Has to be before effectors, in particular movers?
// TODO: lights in moving sectors ought to be interpolated
G_DoEffectorLights();
G_MoveEffectors(); //ST 3
G_MoveStandables(); //ST 6
G_RefreshLights();
G_DoSectorAnimations();
G_MoveFX(); //ST 11
if (RR && numplayers < 2 && g_thunderOn)
G_Thunder();
g_moveWorldTime = (1-0.033)*g_moveWorldTime + 0.033*(timerGetHiTicks()-worldTime);
}
END_RR_NS