mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-22 08:22:38 +00:00
52cfb7fb92
Sadly this uncovered a few more places where script events are allowed too broad access to the game state.
8568 lines
312 KiB
C++
8568 lines
312 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_DUKE_NS
|
|
|
|
|
|
#if KRANDDEBUG
|
|
# define ACTOR_STATIC
|
|
#else
|
|
# define ACTOR_STATIC static
|
|
#endif
|
|
|
|
uint8_t g_radiusDmgStatnums[(MAXSTATUS+7)>>3];
|
|
|
|
#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;
|
|
}
|
|
|
|
void A_RadiusDamageObject_Internal(int const spriteNum, int const otherSprite, int const blastRadius, int spriteDist,
|
|
int const zOffset, int const dmg1, int dmg2, int dmg3, int dmg4)
|
|
{
|
|
auto const pSprite = (uspriteptr_t)&sprite[spriteNum];
|
|
auto const pOther = &sprite[otherSprite];
|
|
|
|
// DEFAULT, ZOMBIEACTOR, MISC
|
|
if (pOther->statnum == STAT_DEFAULT || pOther->statnum == STAT_ZOMBIEACTOR || pOther->statnum == STAT_MISC || AFLAMABLE(pOther->picnum))
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (pSprite->picnum != SHRINKSPARK || (pOther->cstat&257))
|
|
#endif
|
|
{
|
|
if (A_CheckEnemySprite(pOther) && !cansee(pOther->x, pOther->y, pOther->z+zOffset, pOther->sectnum, pSprite->x, pSprite->y, pSprite->z+zOffset, pSprite->sectnum))
|
|
return;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
A_DamageObject_Duke3D(otherSprite, spriteNum);
|
|
else
|
|
#endif
|
|
A_DamageObject_Generic(otherSprite, spriteNum);
|
|
}
|
|
}
|
|
else if (pOther->extra >= 0 && (uspriteptr_t)pOther != pSprite && ((pOther->cstat & 257) ||
|
|
#ifndef EDUKE32_STANDALONE
|
|
pOther->picnum == TRIPBOMB || pOther->picnum == QUEBALL || pOther->picnum == STRIPEBALL || pOther->picnum == DUKELYINGDEAD ||
|
|
#endif
|
|
A_CheckEnemySprite(pOther)))
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
if ((pSprite->picnum == SHRINKSPARK && pOther->picnum != SHARK && (otherSprite == pSprite->owner || pOther->xrepeat < 24))
|
|
|| (pSprite->picnum == MORTER && otherSprite == pSprite->owner))
|
|
return;
|
|
#endif
|
|
if (pOther->picnum == APLAYER)
|
|
spriteDist = FindDistance3D(pSprite->x - pOther->x, pSprite->y - pOther->y, pSprite->z - (pOther->z - PHEIGHT));
|
|
|
|
if (spriteDist >= blastRadius || !cansee(pOther->x, pOther->y, pOther->z - ZOFFSET3, pOther->sectnum,
|
|
pSprite->x, pSprite->y, pSprite->z - ZOFFSET4, pSprite->sectnum))
|
|
return;
|
|
|
|
if (A_CheckSpriteFlags(otherSprite, SFLAG_DAMAGEEVENT))
|
|
if (VM_OnEventWithReturn(EVENT_DAMAGESPRITE, spriteNum, -1, otherSprite) < 0)
|
|
return;
|
|
|
|
auto &dmgActor = actor[otherSprite];
|
|
|
|
dmgActor.ang = getangle(pOther->x - pSprite->x, pOther->y - pSprite->y);
|
|
|
|
if ((pOther->extra > 0 && ((A_CheckSpriteFlags(spriteNum, SFLAG_PROJECTILE) && SpriteProjectile[spriteNum].workslike & PROJECTILE_RADIUS_PICNUM)
|
|
#ifndef EDUKE32_STANDALONE
|
|
|| pSprite->picnum == RPG
|
|
#endif
|
|
))
|
|
#ifndef EDUKE32_STANDALONE
|
|
|| (pSprite->picnum == SHRINKSPARK)
|
|
#endif
|
|
)
|
|
dmgActor.picnum = pSprite->picnum;
|
|
else dmgActor.picnum = RADIUSEXPLOSION;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (pSprite->picnum != SHRINKSPARK)
|
|
#endif
|
|
{
|
|
// this is really weird
|
|
int const k = blastRadius/3;
|
|
int dmgBase, dmgFuzz;
|
|
|
|
if (spriteDist < k)
|
|
dmgBase = dmg3, dmgFuzz = dmg4;
|
|
else if (spriteDist < k*2)
|
|
dmgBase = dmg2, dmgFuzz = dmg3;
|
|
else if (spriteDist < blastRadius)
|
|
dmgBase = dmg1, dmgFuzz = dmg2;
|
|
|
|
if (dmgBase == dmgFuzz)
|
|
++dmgFuzz;
|
|
|
|
dmgActor.extra = dmgBase + (krand()%(dmgFuzz-dmgBase));
|
|
|
|
if (!A_CheckSpriteFlags(otherSprite, SFLAG_NODAMAGEPUSH))
|
|
{
|
|
if (pOther->xvel < 0) pOther->xvel = 0;
|
|
pOther->xvel += (pSprite->extra<<2);
|
|
}
|
|
|
|
if (A_CheckSpriteFlags(otherSprite, SFLAG_DAMAGEEVENT))
|
|
VM_OnEventWithReturn(EVENT_POSTDAMAGESPRITE, spriteNum, -1, otherSprite);
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
{
|
|
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 STATUE__STATIC:
|
|
case STATUEFLASH__STATIC:
|
|
case SPACEMARINE__STATIC:
|
|
case QUEBALL__STATIC:
|
|
case STRIPEBALL__STATIC:
|
|
A_DamageObject_Duke3D(otherSprite, spriteNum);
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
#ifndef EDUKE32_STANDALONE
|
|
else if (!FURY && pSprite->extra == 0) dmgActor.extra = 0;
|
|
#endif
|
|
|
|
if (pOther->picnum != RADIUSEXPLOSION &&
|
|
pSprite->owner >= 0 && sprite[pSprite->owner].statnum < MAXSTATUS)
|
|
{
|
|
if (pOther->picnum == APLAYER)
|
|
{
|
|
auto pPlayer = g_player[P_GetP(pOther)].ps;
|
|
|
|
if (pPlayer->newowner >= 0)
|
|
G_ClearCameraView(pPlayer);
|
|
}
|
|
|
|
dmgActor.owner = pSprite->owner;
|
|
}
|
|
}
|
|
}
|
|
|
|
#define MAXDAMAGESECTORS 256
|
|
|
|
void A_RadiusDamage(int const spriteNum, int const blastRadius, int const dmg1, int const dmg2, int const dmg3, int const dmg4)
|
|
{
|
|
// Allow checking for radius damage in EVENT_DAMAGE(SPRITE/WALL/FLOOR/CEILING) events.
|
|
decltype(ud.returnvar) const parms = { blastRadius, dmg1, dmg2, dmg3, dmg4 };
|
|
Bmemcpy(ud.returnvar, parms, sizeof(parms));
|
|
|
|
auto const pSprite = (uspriteptr_t)&sprite[spriteNum];
|
|
|
|
int16_t sectorList[MAXDAMAGESECTORS];
|
|
uint8_t sectorMap[(MAXSECTORS+7)>>3];
|
|
int16_t numSectors;
|
|
|
|
bfirst_search_init(sectorList, sectorMap, &numSectors, MAXSECTORS, pSprite->sectnum);
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY && (pSprite->picnum == RPG && pSprite->xrepeat < 11))
|
|
goto SKIPWALLCHECK;
|
|
#endif
|
|
|
|
for (int sectorCount=0; sectorCount < numSectors; ++sectorCount)
|
|
{
|
|
int const sectorNum = sectorList[sectorCount++];
|
|
auto const &listSector = sector[sectorNum];
|
|
vec2_t closest;
|
|
|
|
if (getsectordist(pSprite->pos.vec2, sectorNum, &closest) >= blastRadius)
|
|
continue;
|
|
|
|
int const startWall = listSector.wallptr;
|
|
int const endWall = listSector.wallnum + startWall;
|
|
|
|
int32_t floorZ, ceilZ;
|
|
getzsofslope(sectorNum, closest.x, closest.y, &ceilZ, &floorZ);
|
|
|
|
if (((ceilZ - pSprite->z) >> 8) < blastRadius)
|
|
Sect_DamageCeiling_Internal(spriteNum, sectorNum);
|
|
|
|
if (((pSprite->z - floorZ) >> 8) < blastRadius)
|
|
Sect_DamageFloor_Internal(spriteNum, sectorNum);
|
|
|
|
int w = startWall;
|
|
|
|
for (auto pWall = (uwallptr_t)&wall[startWall]; w < endWall; ++w, ++pWall)
|
|
{
|
|
if (getwalldist(pSprite->pos.vec2, w, &closest) >= blastRadius)
|
|
continue;
|
|
|
|
vec3_t const vect = { closest.x, closest.y, pSprite->z };
|
|
|
|
if (cansee(vect.x, vect.y, vect.z, sectorNum, pSprite->x, pSprite->y, pSprite->z, pSprite->sectnum))
|
|
{
|
|
A_DamageWall_Internal(spriteNum, w, vect, pSprite->picnum);
|
|
|
|
int const nextSector = pWall->nextsector;
|
|
|
|
if (nextSector >= 0)
|
|
bfirst_search_try(sectorList, sectorMap, &numSectors, nextSector);
|
|
|
|
if (numSectors == MAXDAMAGESECTORS)
|
|
goto SKIPWALLCHECK;
|
|
}
|
|
}
|
|
}
|
|
|
|
SKIPWALLCHECK:
|
|
int const randomZOffset = -ZOFFSET2 + (krand()&(ZOFFSET5-1));
|
|
|
|
for (int sectorCount=0; sectorCount < numSectors; ++sectorCount)
|
|
{
|
|
int damageSprite = headspritesect[sectorList[sectorCount]];
|
|
|
|
while (damageSprite >= 0)
|
|
{
|
|
int const nextSprite = nextspritesect[damageSprite];
|
|
auto pDamage = &sprite[damageSprite];
|
|
|
|
if (bitmap_test(g_radiusDmgStatnums, pDamage->statnum))
|
|
{
|
|
int const spriteDist = dist(pSprite, pDamage);
|
|
|
|
if (spriteDist < blastRadius)
|
|
A_RadiusDamageObject_Internal(spriteNum, damageSprite, blastRadius, spriteDist, randomZOffset, dmg1, dmg2, dmg3, dmg4);
|
|
}
|
|
|
|
damageSprite = nextSprite;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Maybe do a projectile transport via an SE7.
|
|
// <spritenum>: the projectile
|
|
// <i>: the SE7
|
|
// <fromunderp>: below->above change?
|
|
static int32_t Proj_MaybeDoTransport(int32_t spriteNum, uspriteptr_t const pSEffector, int32_t fromunderp, int32_t daz)
|
|
{
|
|
if (((int32_t) totalclock & UINT8_MAX) == actor[spriteNum].lasttransport)
|
|
return 0;
|
|
|
|
auto const pSprite = &sprite[spriteNum];
|
|
auto const otherse = (uspriteptr_t)&sprite[pSEffector->owner];
|
|
|
|
actor[spriteNum].lasttransport = ((int32_t) totalclock & UINT8_MAX);
|
|
|
|
pSprite->x += (otherse->x - pSEffector->x);
|
|
pSprite->y += (otherse->y - pSEffector->y);
|
|
|
|
// above->below
|
|
pSprite->z = (!fromunderp) ? sector[otherse->sectnum].ceilingz - daz + sector[pSEffector->sectnum].floorz
|
|
: sector[otherse->sectnum].floorz - daz + sector[pSEffector->sectnum].ceilingz;
|
|
// below->above
|
|
|
|
actor[spriteNum].bpos = sprite[spriteNum].pos;
|
|
changespritesect(spriteNum, otherse->sectnum);
|
|
|
|
return 1;
|
|
}
|
|
|
|
// 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(uspriteptr_t 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,
|
|
int32_t *ceilhit, int32_t *florhit)
|
|
{
|
|
if (zChange == 0)
|
|
return 0;
|
|
|
|
auto const pSprite = (uspriteptr_t)&sprite[spriteNum];
|
|
int const newZ = pSprite->z + (zChange >> 1);
|
|
|
|
*pZcoord = newZ;
|
|
|
|
int const clipDist = A_GetClipdist(spriteNum, -1);
|
|
|
|
VM_GetZRange(spriteNum, ceilhit, florhit, pSprite->statnum == STAT_PROJECTILE ? clipDist << 3 : clipDist);
|
|
|
|
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 2;
|
|
}
|
|
|
|
int A_GetClipdist(int spriteNum, int clipDist)
|
|
{
|
|
if (clipDist < 0)
|
|
{
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const isEnemy = A_CheckEnemySprite(pSprite);
|
|
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_REALCLIPDIST))
|
|
clipDist = pSprite->clipdist << 2;
|
|
else if ((pSprite->cstat & 48) == 16)
|
|
clipDist = 0;
|
|
else if (isEnemy)
|
|
{
|
|
if (pSprite->xrepeat > 60)
|
|
clipDist = 1024;
|
|
#ifndef EDUKE32_STANDALONE
|
|
else if (pSprite->picnum == LIZMAN)
|
|
clipDist = 292;
|
|
#endif
|
|
else if (A_CheckSpriteFlags(spriteNum, SFLAG_BADGUY))
|
|
clipDist = pSprite->clipdist << 2;
|
|
else
|
|
clipDist = 192;
|
|
}
|
|
else
|
|
{
|
|
if (pSprite->statnum == STAT_PROJECTILE && (SpriteProjectile[spriteNum].workslike & PROJECTILE_REALCLIPDIST) == 0)
|
|
clipDist = 16;
|
|
else
|
|
clipDist = pSprite->clipdist << 2;
|
|
}
|
|
}
|
|
|
|
return clipDist;
|
|
}
|
|
|
|
int32_t A_MoveSpriteClipdist(int32_t spriteNum, vec3_t const * const change, uint32_t clipType, int32_t clipDist)
|
|
{
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const isEnemy = A_CheckEnemySprite(pSprite);
|
|
vec2_t const oldPos = pSprite->pos.vec2;
|
|
|
|
// check to make sure the netcode didn't leave a deleted sprite in the sprite lists.
|
|
Bassert(pSprite->sectnum < MAXSECTORS);
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
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, &pSprite->pos);
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
|
|
clipDist = A_GetClipdist(spriteNum, clipDist);
|
|
|
|
int16_t newSectnum = pSprite->sectnum;
|
|
#ifndef EDUKE32_STANDALONE
|
|
int const oldSectnum = newSectnum;
|
|
#endif
|
|
|
|
// Handle horizontal movement first.
|
|
|
|
int returnValue;
|
|
int32_t diffZ;
|
|
spriteheightofs(spriteNum, &diffZ, 1);
|
|
|
|
if (pSprite->statnum == STAT_PROJECTILE)
|
|
returnValue = clipmovex(&pSprite->pos, &newSectnum, change->x << 13, change->y << 13, clipDist, diffZ >> 3, diffZ >> 3, clipType, 1);
|
|
else
|
|
{
|
|
pSprite->z -= diffZ >> 1;
|
|
returnValue = clipmove(&pSprite->pos, &newSectnum, change->x << 13, change->y << 13, clipDist, ZOFFSET6, ZOFFSET6, clipType);
|
|
pSprite->z += diffZ >> 1;
|
|
}
|
|
|
|
// Testing: For some reason the assert below this was tripping for clients
|
|
EDUKE32_UNUSED int16_t dbg_ClipMoveSectnum = newSectnum;
|
|
|
|
if (isEnemy)
|
|
{
|
|
// Handle potential stayput condition (map-provided or hard-coded).
|
|
if (newSectnum < 0 || ((actor[spriteNum].stayput >= 0 && actor[spriteNum].stayput != newSectnum)
|
|
|| ((g_tile[pSprite->picnum].flags & SFLAG_NOWATERSECTOR) && sector[newSectnum].lotag == ST_1_ABOVE_WATER)
|
|
#ifndef EDUKE32_STANDALONE
|
|
|| (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[oldSectnum].lotag != ST_1_ABOVE_WATER && sector[newSectnum].lotag == ST_1_ABOVE_WATER
|
|
&& (pSprite->picnum == LIZMAN || (pSprite->picnum == LIZTROOP && pSprite->zvel == 0)))
|
|
#endif
|
|
))
|
|
{
|
|
pSprite->pos.vec2 = oldPos;
|
|
|
|
// NOTE: in Duke3D, LIZMAN on water takes on random angle here.
|
|
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
|
|
if (newSectnum < 0)
|
|
newSectnum = 0;
|
|
|
|
return 16384+newSectnum;
|
|
}
|
|
|
|
if ((returnValue&49152) >= 32768 && actor[spriteNum].cgg==0)
|
|
pSprite->ang += 768;
|
|
}
|
|
|
|
EDUKE32_UNUSED int16_t dbg_newSectnum2 = newSectnum;
|
|
|
|
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 newZ = pSprite->z;
|
|
int32_t ceilhit, florhit;
|
|
int const doZUpdate = change->z ? A_CheckNeedZUpdate(spriteNum, change->z, &newZ, &ceilhit, &florhit) : 0;
|
|
|
|
// Update sprite's z positions and (for TROR) maybe the sector number.
|
|
if (doZUpdate == 2)
|
|
{
|
|
if (returnValue == 0)
|
|
returnValue = change->z < 0 ? ceilhit : florhit;
|
|
}
|
|
else 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, &pSprite->pos);
|
|
}
|
|
#endif
|
|
}
|
|
else if (change->z != 0 && returnValue == 0)
|
|
returnValue = 16384+newSectnum;
|
|
|
|
if (returnValue == 16384 + newSectnum)
|
|
{
|
|
if (pSprite->statnum == STAT_PROJECTILE)
|
|
{
|
|
// Projectile sector changes due to transport SEs (SE7_PROJECTILE).
|
|
// PROJECTILE_CHSECT
|
|
for (bssize_t SPRITES_OF(STAT_TRANSPORT, otherSpriteNum))
|
|
{
|
|
if (sprite[otherSpriteNum].sectnum == newSectnum)
|
|
{
|
|
int const sectLotag = sector[newSectnum].lotag;
|
|
|
|
if (sectLotag == ST_1_ABOVE_WATER && newZ >= actor[spriteNum].floorz)
|
|
if (Proj_MaybeDoTransport(spriteNum, (uspriteptr_t)&sprite[otherSpriteNum], 0, newZ))
|
|
return 0;
|
|
|
|
if (sectLotag == ST_2_UNDERWATER && newZ <= actor[spriteNum].ceilingz)
|
|
if (Proj_MaybeDoTransport(spriteNum, (uspriteptr_t)&sprite[otherSpriteNum], 1, newZ))
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
if (VM_HaveEvent(EVENT_KILLIT))
|
|
{
|
|
int32_t playerDist;
|
|
int playerNum = A_FindPlayer(&sprite[spriteNum], &playerDist);
|
|
|
|
if (VM_ExecuteEvent(EVENT_KILLIT, spriteNum, playerNum, playerDist))
|
|
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);
|
|
|
|
#ifdef NETCODE_DISABLE
|
|
deletesprite(spriteNum);
|
|
#else
|
|
Net_DeleteSprite(spriteNum);
|
|
#endif
|
|
}
|
|
|
|
void A_AddToDeleteQueue(int spriteNum)
|
|
{
|
|
if (g_netClient || (g_deleteQueueSize == 0)) // [75] Clients should not use SpriteDeletionQueue[] and just set the sprites invisible immediately in A_DeleteSprite
|
|
{
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
|
|
auto &deleteSpriteNum = SpriteDeletionQueue[g_spriteDeleteQueuePos];
|
|
|
|
if (deleteSpriteNum >= 0 && actor[deleteSpriteNum].flags & SFLAG_QUEUEDFORDELETE)
|
|
A_DeleteSprite(deleteSpriteNum);
|
|
|
|
deleteSpriteNum = spriteNum;
|
|
actor[spriteNum].flags |= SFLAG_QUEUEDFORDELETE;
|
|
g_spriteDeleteQueuePos = (g_spriteDeleteQueuePos+1)%g_deleteQueueSize;
|
|
}
|
|
|
|
void A_SpawnMultiple(int spriteNum, int tileNum, int spawnCnt)
|
|
{
|
|
auto const pSprite = &sprite[spriteNum];
|
|
|
|
for (; spawnCnt>0; spawnCnt--)
|
|
{
|
|
int const j = A_InsertSprite(pSprite->sectnum, pSprite->x, pSprite->y, pSprite->z - (krand() % (47 << 8)), tileNum, -32, 8,
|
|
8, krand() & 2047, 0, 0, spriteNum, 5);
|
|
A_Spawn(-1, j);
|
|
sprite[j].cstat = krand()&12;
|
|
}
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
void A_DoGuts(int spriteNum, int tileNum, int spawnCnt)
|
|
{
|
|
auto const pSprite = (uspriteptr_t)&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 (pSprite->picnum == COMMANDER)
|
|
gutZ -= (24<<8);
|
|
|
|
for (bssize_t j=spawnCnt; j>0; j--)
|
|
{
|
|
int const i = A_InsertSprite(pSprite->sectnum, pSprite->x + (krand() & 255) - 128,
|
|
pSprite->y + (krand() & 255) - 128, gutZ - (krand() & 8191), tileNum, -32, repeat.x,
|
|
repeat.y, krand() & 2047, 48 + (krand() & 31), -512 - (krand() & 2047), spriteNum, 5);
|
|
|
|
if (PN(i) == JIBS2)
|
|
{
|
|
sprite[i].xrepeat >>= 2;
|
|
sprite[i].yrepeat >>= 2;
|
|
}
|
|
|
|
sprite[i].pal = pSprite->pal;
|
|
}
|
|
}
|
|
|
|
void A_DoGutsDir(int spriteNum, int tileNum, int spawnCnt)
|
|
{
|
|
auto const s = (uspriteptr_t)&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 (s->picnum == COMMANDER)
|
|
gutZ -= (24<<8);
|
|
|
|
for (bssize_t j=spawnCnt; j>0; j--)
|
|
{
|
|
int const i = A_InsertSprite(s->sectnum, s->x, s->y, gutZ, tileNum, -32, repeat.x, repeat.y, krand() & 2047,
|
|
256 + (krand() & 127), -512 - (krand() & 2047), spriteNum, 5);
|
|
sprite[i].pal = s->pal;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
LUNATIC_EXTERN 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);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int32_t move_rotfixed_sprite(int32_t spriteNum, int32_t pivotSpriteNum, int32_t pivotAngle)
|
|
{
|
|
if ((ROTFIXSPR_STATNUMP(sprite[spriteNum].statnum) ||
|
|
((sprite[spriteNum].statnum == STAT_ACTOR || sprite[spriteNum].statnum == STAT_ZOMBIEACTOR) &&
|
|
A_CheckSpriteFlags(spriteNum, SFLAG_ROTFIXED))) &&
|
|
actor[spriteNum].t_data[7] == (ROTFIXSPR_MAGIC | pivotSpriteNum))
|
|
{
|
|
rotatepoint(zerovec, *(vec2_t *)&actor[spriteNum].t_data[8], pivotAngle & 2047, &sprite[spriteNum].pos.vec2);
|
|
sprite[spriteNum].x += sprite[pivotSpriteNum].x;
|
|
sprite[spriteNum].y += sprite[pivotSpriteNum].y;
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
void A_MoveSector(int spriteNum)
|
|
{
|
|
// T1,T2 and T3 are used for all the sector moving stuff!!!
|
|
|
|
int32_t playerDist;
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
int const rotateAngle = VM_OnEvent(EVENT_MOVESECTOR, spriteNum, playerNum, playerDist, 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++;
|
|
}
|
|
}
|
|
|
|
#if !defined LUNATIC
|
|
// 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)
|
|
#else
|
|
// startframe + viewtype*[cyclic counter]
|
|
# define LIGHTRAD_PICOFS(i) (actor[i].ac.startframe + actor[i].ac.viewtype * AC_CURFRAME(actor[i].t_data))
|
|
#endif
|
|
|
|
// 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
|
|
auto const 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
|
|
}
|
|
|
|
ACTOR_STATIC void A_MaybeAwakenBadGuys(int const spriteNum)
|
|
{
|
|
if (sprite[spriteNum].sectnum == MAXSECTORS)
|
|
return;
|
|
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_WAKEUPBADGUYS))
|
|
{
|
|
auto const pSprite = (uspriteptr_t)&sprite[spriteNum];
|
|
|
|
for (bssize_t nextSprite, SPRITES_OF_STAT_SAFE(STAT_ZOMBIEACTOR, spriteNum, nextSprite))
|
|
{
|
|
if (A_CheckEnemySprite(&sprite[spriteNum]))
|
|
{
|
|
if (sprite[spriteNum].sectnum == pSprite->sectnum
|
|
|| sprite[spriteNum].sectnum == nextsectorneighborz(pSprite->sectnum, sector[pSprite->sectnum].floorz, 1, 1)
|
|
|| cansee(pSprite->x, pSprite->y, pSprite->z - PHEIGHT, pSprite->sectnum, sprite[spriteNum].x, sprite[spriteNum].y,
|
|
sprite[spriteNum].z - PHEIGHT, sprite[spriteNum].sectnum))
|
|
{
|
|
actor[spriteNum].timetosleep = SLEEPTIME;
|
|
A_PlayAlertSound(spriteNum);
|
|
changespritestat(spriteNum, STAT_ACTOR);
|
|
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_WAKEUPBADGUYS))
|
|
A_MaybeAwakenBadGuys(spriteNum);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// sleeping monsters, etc
|
|
ACTOR_STATIC void G_MoveZombieActors(void)
|
|
{
|
|
int spriteNum = headspritestat[STAT_ZOMBIEACTOR], canSeePlayer;
|
|
|
|
while (spriteNum >= 0)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
int32_t playerDist;
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
auto 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 (pPlayer->newowner == -1 && A_CheckEnemySprite(pSprite))
|
|
{
|
|
vec3_t const p = { pPlayer->pos.x + 64 - (krand() & 127),
|
|
pPlayer->pos.y + 64 - (krand() & 127),
|
|
pPlayer->pos.z - (krand() % ZOFFSET5) };
|
|
|
|
int16_t pSectnum = pPlayer->cursectnum;
|
|
|
|
updatesector(p.x, p.y, &pSectnum);
|
|
|
|
if (pSectnum == -1)
|
|
{
|
|
spriteNum = nextSprite;
|
|
continue;
|
|
}
|
|
|
|
vec3_t const s = { pSprite->x + 64 - (krand() & 127),
|
|
pSprite->y + 64 - (krand() & 127),
|
|
pSprite->z - (krand() % (52 << 8)) };
|
|
|
|
int16_t sectNum = pSprite->sectnum;
|
|
|
|
updatesector(s.x, s.y, §Num);
|
|
|
|
if (sectNum == -1)
|
|
{
|
|
spriteNum = nextSprite;
|
|
continue;
|
|
}
|
|
|
|
canSeePlayer = cansee(s.x, s.y, s.z, sectNum, p.x, p.y, p.z, pSectnum);
|
|
}
|
|
else
|
|
canSeePlayer = cansee(pSprite->x, pSprite->y, pSprite->z - ((krand() & 31) << 8), pSprite->sectnum, pPlayer->opos.x,
|
|
pPlayer->opos.y, pPlayer->opos.z - ((krand() & 31) << 8), pPlayer->cursectnum);
|
|
|
|
if (canSeePlayer)
|
|
{
|
|
switch (DYNAMICTILEMAP(pSprite->picnum))
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
case RUBBERCAN__STATIC:
|
|
case EXPLODINGBARREL__STATIC:
|
|
case WOODENHORSE__STATIC:
|
|
case HORSEONSIDE__STATIC:
|
|
case CANWITHSOMETHING__STATIC:
|
|
case CANWITHSOMETHING2__STATIC:
|
|
case CANWITHSOMETHING3__STATIC:
|
|
case CANWITHSOMETHING4__STATIC:
|
|
case FIREBARREL__STATIC:
|
|
case FIREVASE__STATIC:
|
|
case NUKEBARREL__STATIC:
|
|
case NUKEBARRELDENTED__STATIC:
|
|
case NUKEBARRELLEAKED__STATIC:
|
|
case TRIPBOMB__STATIC:
|
|
pSprite->shade = ((sector[pSprite->sectnum].ceilingstat & 1) && A_CheckSpriteFlags(spriteNum, SFLAG_NOSHADE) == 0)
|
|
? sector[pSprite->sectnum].ceilingshade
|
|
: sector[pSprite->sectnum].floorshade;
|
|
actor[spriteNum].timetosleep = 0;
|
|
changespritestat(spriteNum, STAT_STANDABLE);
|
|
break;
|
|
|
|
case RECON__STATIC:
|
|
CS(spriteNum) |= 257;
|
|
fallthrough__;
|
|
#endif
|
|
default:
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_USEACTIVATOR) && sector[sprite[spriteNum].sectnum].lotag & 16384)
|
|
break;
|
|
|
|
actor[spriteNum].timetosleep = 0;
|
|
A_PlayAlertSound(spriteNum);
|
|
changespritestat(spriteNum, STAT_ACTOR);
|
|
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_WAKEUPBADGUYS))
|
|
A_MaybeAwakenBadGuys(spriteNum);
|
|
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
actor[spriteNum].timetosleep = 0;
|
|
}
|
|
}
|
|
|
|
if (A_CheckEnemySprite(pSprite) && A_CheckSpriteFlags(spriteNum,SFLAG_NOSHADE) == 0)
|
|
{
|
|
pSprite->shade = (sector[pSprite->sectnum].ceilingstat & 1)
|
|
? sector[pSprite->sectnum].ceilingshade
|
|
: sector[pSprite->sectnum].floorshade;
|
|
}
|
|
}
|
|
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
|
|
// stupid name, but it's what the function does.
|
|
static FORCE_INLINE int G_FindExplosionInSector(int const sectNum)
|
|
{
|
|
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)
|
|
{
|
|
auto const pSprite = &sprite[spriteNum];
|
|
auto 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 && pActor->picnum != SHRINKSPARK)
|
|
return -1;
|
|
|
|
int const playerNum = P_GetP(pSprite);
|
|
|
|
if (pActor->owner >= 0 && (sprite[pActor->owner].picnum == APLAYER))
|
|
{
|
|
if (
|
|
(ud.ffire == 0) &&
|
|
(spriteNum != pActor->owner) && // Not damaging self.
|
|
((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))
|
|
)
|
|
{
|
|
// Nullify damage and cancel.
|
|
pActor->owner = -1;
|
|
pActor->extra = -1;
|
|
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 RADIUSEXPLOSION__STATIC:
|
|
case SEENINE__STATIC:
|
|
#ifndef EDUKE32_STANDALONE
|
|
case RPG__STATIC:
|
|
case HYDRENT__STATIC:
|
|
case HEAVYHBOMB__STATIC:
|
|
case OOZFILTER__STATIC:
|
|
case EXPLODINGBARREL__STATIC:
|
|
#endif
|
|
P_Nudge(playerNum, spriteNum, 2);
|
|
break;
|
|
|
|
default:
|
|
P_Nudge(playerNum, spriteNum, (A_CheckSpriteFlags(pActor->owner, SFLAG_PROJECTILE) &&
|
|
(SpriteProjectile[pActor->owner].workslike & PROJECTILE_RPG))
|
|
? 2
|
|
: 1);
|
|
break;
|
|
}
|
|
|
|
pActor->extra = -1;
|
|
return pActor->picnum;
|
|
}
|
|
|
|
if (pActor->extra == 0 && 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));
|
|
auto const pPlayer = g_player[playerNum].ps;
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
int const playerSectnum = pPlayer->cursectnum;
|
|
|
|
if (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, &sprite[spriteNum].pos);
|
|
|
|
next_sprite:
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
|
|
|
|
static int P_Submerge(int, DukePlayer_t *, int, int);
|
|
static int P_Emerge(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];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
auto 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, &pSprite->pos);
|
|
}
|
|
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((uspriteptr_t)pSprite, playerSectnum, sectorLotag, &otherSector))
|
|
{
|
|
// NOTE: Compare with G_MoveTransports().
|
|
pPlayer->on_warping_sector = 1;
|
|
|
|
if ((sectorLotag == ST_1_ABOVE_WATER ?
|
|
P_Submerge(P_GetP(pSprite), pPlayer, playerSectnum, otherSector) :
|
|
P_Emerge(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)
|
|
{
|
|
// Don't stomp teammates.
|
|
if (
|
|
((g_gametypeFlags[ud.coop] & GAMETYPE_TDM) && pPlayer->team != g_player[otherp].ps->team) ||
|
|
(!(g_gametypeFlags[ud.coop] & GAMETYPE_PLAYERSFRIENDLY) && !(g_gametypeFlags[ud.coop] & GAMETYPE_TDM))
|
|
)
|
|
{
|
|
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 (!WW2GI)
|
|
pPlayer->inv_amount[GET_JETPACK] = 1599;
|
|
}
|
|
|
|
if (pSprite->extra > 0)
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
{
|
|
actor[spriteNum].owner = spriteNum;
|
|
|
|
if (ud.god == 0)
|
|
if (G_CheckForSpaceCeiling(pSprite->sectnum) || G_CheckForSpaceFloor(pSprite->sectnum))
|
|
{
|
|
OSD_Printf(OSD_ERROR "%s: player killed by space sector!\n", EDUKE32_FUNCTION);
|
|
P_QuickKill(pPlayer);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
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);
|
|
|
|
actor[spriteNum].bpos = pSprite->pos;
|
|
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,&pSprite->pos);
|
|
}
|
|
}
|
|
|
|
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)
|
|
{
|
|
auto 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));
|
|
// 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;
|
|
auto 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);
|
|
FX_SetReverbDelay(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)
|
|
{
|
|
if (g_sounds[pSprite->lotag].m & 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.
|
|
|
|
char om = g_sounds[pSprite->lotag].m;
|
|
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;
|
|
}
|
|
|
|
g_sounds[pSprite->lotag].m |= SF_LOOP;
|
|
A_PlaySound(pSprite->lotag,spriteNum);
|
|
g_sounds[pSprite->lotag].m = om;
|
|
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 ((g_sounds[pSprite->lotag].m & (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];
|
|
auto 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 (j == FIREEXT || j == RPG || 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 ((int16_t)pSprite->lotag <= 0)
|
|
{
|
|
pSprite->xvel = (32+(krand()&63));
|
|
pSprite->zvel = -(1024+(krand()&1023));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int32_t spriteGravity = g_spriteGravity;
|
|
|
|
if (pSprite->xvel > 0)
|
|
{
|
|
pSprite->xvel -= 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)
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
for (int x = 0, x_end = 1+(krand()&7); x < x_end; ++x)
|
|
RANDOMSCRAP(pSprite, spriteNum);
|
|
#endif
|
|
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)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
auto const pData = &actor[spriteNum].t_data[0];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const sectNum = pSprite->sectnum;
|
|
|
|
if (sectNum < 0)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
// Rotation-fixed sprites in rotating sectors already have bpos* updated.
|
|
if ((pData[7]&(0xffff0000))!=ROTFIXSPR_MAGIC)
|
|
actor[spriteNum].bpos = pSprite->pos;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
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(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(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,&pSprite->pos);
|
|
|
|
actor[pSprite->owner].bpos = pSprite->pos;
|
|
|
|
pSprite->zvel = 0;
|
|
}
|
|
else if (pSprite->owner == -2)
|
|
{
|
|
auto 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, &ps->pos);
|
|
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-(krand()&7);
|
|
if (j < 10)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
pSprite->xrepeat = j;
|
|
|
|
j = pSprite->yrepeat-(krand()&7);
|
|
if (j < 4)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
pSprite->yrepeat = j;
|
|
}
|
|
if (pSprite->picnum == BOX)
|
|
{
|
|
A_Fall(spriteNum);
|
|
actor[spriteNum].ceilingz = sector[pSprite->sectnum].ceilingz;
|
|
}
|
|
goto next_sprite;
|
|
}
|
|
else if (pSprite->picnum == TRIPBOMB)
|
|
{
|
|
// TIMER_CONTROL
|
|
if (actor[spriteNum].t_data[6] == 1)
|
|
{
|
|
|
|
if (actor[spriteNum].t_data[7] >= 1)
|
|
{
|
|
actor[spriteNum].t_data[7]--;
|
|
}
|
|
|
|
if (actor[spriteNum].t_data[7] <= 0)
|
|
{
|
|
T3(spriteNum)=16;
|
|
actor[spriteNum].t_data[6]=3;
|
|
A_PlaySound(LASERTRIP_ARMING,spriteNum);
|
|
}
|
|
// we're on a timer....
|
|
}
|
|
if (T3(spriteNum) > 0 && actor[spriteNum].t_data[6] == 3)
|
|
{
|
|
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)
|
|
{
|
|
actor[spriteNum].t_data[6] = 3;
|
|
T3(spriteNum) = 16;
|
|
}
|
|
pSprite->extra = oldExtra;
|
|
pSprite->ang = oldAng;
|
|
}
|
|
|
|
switch (T1(spriteNum))
|
|
{
|
|
default:
|
|
{
|
|
int32_t playerDist;
|
|
A_FindPlayer(pSprite, &playerDist);
|
|
if (playerDist > 768 || T1(spriteNum) > 16) T1(spriteNum)++;
|
|
break;
|
|
}
|
|
|
|
case 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,&pSprite->pos);
|
|
|
|
int hitDist = A_CheckHitSprite(spriteNum, &hitSprite);
|
|
|
|
actor[spriteNum].lastv.x = hitDist;
|
|
pSprite->ang = oldAng;
|
|
|
|
// if(lTripBombControl & TRIPBOMB_TRIPWIRE)
|
|
if (actor[spriteNum].t_data[6] != 1)
|
|
{
|
|
// we're on a trip wire
|
|
int16_t cursectnum;
|
|
|
|
while (hitDist > 0)
|
|
{
|
|
j = A_Spawn(spriteNum,LASERLINE);
|
|
setsprite(j,&sprite[j].pos);
|
|
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,&pSprite->pos);
|
|
T4(spriteNum) = T3(spriteNum) = 0;
|
|
|
|
if (hitSprite >= 0 && actor[spriteNum].t_data[6] != 1)
|
|
{
|
|
actor[spriteNum].t_data[6] = 3;
|
|
T3(spriteNum) = 13;
|
|
A_PlaySound(LASERTRIP_ARMING,spriteNum);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 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, &pSprite->pos);
|
|
|
|
int hitDist = A_CheckHitSprite(spriteNum, NULL);
|
|
|
|
pSprite->x = T4(spriteNum);
|
|
pSprite->y = T5(spriteNum);
|
|
pSprite->z += (3<<8);
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
|
|
// if( Actor[i].lastvx != x && lTripBombControl & TRIPBOMB_TRIPWIRE)
|
|
if (actor[spriteNum].lastv.x != hitDist && actor[spriteNum].t_data[6] != 1)
|
|
{
|
|
actor[spriteNum].t_data[6] = 3;
|
|
T3(spriteNum) = 13;
|
|
A_PlaySound(LASERTRIP_ARMING, spriteNum);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
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 FIREEXT__STATIC:
|
|
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 (pSprite->picnum == FIREEXT)
|
|
{
|
|
if (A_IncurDamage(spriteNum) < 0)
|
|
goto next_sprite;
|
|
|
|
for (int k=0; k<16; k++)
|
|
{
|
|
j = A_InsertSprite(SECT(spriteNum), SX(spriteNum), SY(spriteNum), SZ(spriteNum) - (krand() % (48 << 8)),
|
|
SCRAP3 + (krand() & 3), -8, 48, 48, krand() & 2047, (krand() & 63) + 64,
|
|
-(krand() & 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
|
|
#endif
|
|
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;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (pSprite->xrepeat)
|
|
for (int x=0; x<8; x++)
|
|
RANDOMSCRAP(pSprite, spriteNum);
|
|
#endif
|
|
|
|
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);
|
|
}
|
|
|
|
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;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (switchPic > SIDEBOLT1 && switchPic <= SIDEBOLT1 + 3)
|
|
switchPic = SIDEBOLT1;
|
|
else if (switchPic > BOLT1 && switchPic <= BOLT1 + 3)
|
|
switchPic = BOLT1;
|
|
#endif
|
|
switch (DYNAMICTILEMAP(switchPic))
|
|
{
|
|
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 VIEWSCREEN__STATIC:
|
|
case VIEWSCREEN2__STATIC:
|
|
|
|
if (pSprite->xrepeat == 0)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
{
|
|
int32_t playerDist;
|
|
int const p = A_FindPlayer(pSprite, &playerDist);
|
|
auto 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;
|
|
#ifndef EDUKE32_STANDALONE
|
|
case TRASH__STATIC:
|
|
|
|
if (pSprite->xvel == 0)
|
|
pSprite->xvel = 1;
|
|
if (A_SetSprite(spriteNum, CLIPMASK0))
|
|
{
|
|
A_Fall(spriteNum);
|
|
if (krand() & 1)
|
|
pSprite->zvel -= 256;
|
|
if ((pSprite->xvel) < 48)
|
|
pSprite->xvel += (krand() & 3);
|
|
}
|
|
else
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
break;
|
|
|
|
case SIDEBOLT1__STATIC:
|
|
// case SIDEBOLT1+1:
|
|
// case SIDEBOLT1+2:
|
|
// case SIDEBOLT1+3:
|
|
{
|
|
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 ((krand() & 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 ((krand() & 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 ((krand() & 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)
|
|
&& (krand() & 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 && 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 + (krand() & 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 CANWITHSOMETHING__STATIC:
|
|
case CANWITHSOMETHING2__STATIC:
|
|
case CANWITHSOMETHING3__STATIC:
|
|
case CANWITHSOMETHING4__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:
|
|
case FIREBARREL__STATIC:
|
|
case FIREVASE__STATIC:
|
|
case EXPLODINGBARREL__STATIC:
|
|
case WOODENHORSE__STATIC:
|
|
case HORSEONSIDE__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;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
next_sprite:
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
|
|
ACTOR_STATIC void A_DoProjectileBounce(int const spriteNum)
|
|
{
|
|
auto 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+(krand()&3);
|
|
ps->numloogs = j;
|
|
ps->loogcnt = 24*4;
|
|
for (bssize_t x=0; x < j; x++)
|
|
{
|
|
ps->loogiex[x] = krand()%xdim;
|
|
ps->loogiey[x] = krand()%ydim;
|
|
}
|
|
}
|
|
|
|
static void A_DoProjectileEffects(int spriteNum, const vec3_t *davect, int radiusDamage)
|
|
{
|
|
auto const pProj = &SpriteProjectile[spriteNum];
|
|
|
|
if (pProj->spawns >= 0)
|
|
{
|
|
int const newSpr = A_Spawn(spriteNum,pProj->spawns);
|
|
|
|
if (davect)
|
|
Bmemcpy(&sprite[newSpr],davect,sizeof(vec3_t));
|
|
|
|
if (pProj->sxrepeat > 4)
|
|
sprite[newSpr].xrepeat=pProj->sxrepeat;
|
|
if (pProj->syrepeat > 4)
|
|
sprite[newSpr].yrepeat=pProj->syrepeat;
|
|
}
|
|
|
|
if (pProj->isound >= 0)
|
|
A_PlaySound(pProj->isound,spriteNum);
|
|
|
|
if (!radiusDamage)
|
|
return;
|
|
|
|
auto const pSprite = &sprite[spriteNum];
|
|
pSprite->extra = Proj_GetDamage(pProj);
|
|
int const dmg = pSprite->extra;
|
|
A_RadiusDamage(spriteNum, pProj->hitradius, dmg >> 2, dmg >> 1, dmg - (dmg >> 2), dmg);
|
|
}
|
|
|
|
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 + 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)
|
|
{
|
|
auto const s = (uspriteptr_t)&sprite[spriteNum];
|
|
|
|
if (s->zvel < 0)
|
|
{
|
|
if ((sector[s->sectnum].ceilingstat&1) && sector[s->sectnum].ceilingpal == 0)
|
|
return 1;
|
|
|
|
Sect_DamageCeiling(spriteNum, s->sectnum);
|
|
}
|
|
else if (s->zvel > 0)
|
|
{
|
|
if ((sector[s->sectnum].floorstat&1) && sector[s->sectnum].floorpal == 0)
|
|
{
|
|
// Keep original Duke3D behavior: pass projectiles through
|
|
// parallaxed ceilings, but NOT through such floors.
|
|
return 0;
|
|
}
|
|
|
|
Sect_DamageFloor(spriteNum, s->sectnum);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
ACTOR_STATIC void Proj_MoveCustom(int const spriteNum)
|
|
{
|
|
int projectileMoved = SpriteProjectile[spriteNum].workslike & PROJECTILE_MOVED;
|
|
SpriteProjectile[spriteNum].workslike |= PROJECTILE_MOVED;
|
|
|
|
auto const pProj = &SpriteProjectile[spriteNum];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
vec3_t davect;
|
|
int otherSprite = 0;
|
|
|
|
switch (pProj->workslike & PROJECTILE_TYPE_MASK)
|
|
{
|
|
case PROJECTILE_HITSCAN:
|
|
{
|
|
if (!G_HaveActor(sprite[spriteNum].picnum))
|
|
return;
|
|
int32_t playerDist;
|
|
int const playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
A_Execute(spriteNum, playerNum, playerDist);
|
|
return;
|
|
}
|
|
|
|
case PROJECTILE_KNEE:
|
|
case PROJECTILE_BLOOD: A_DeleteSprite(spriteNum); return;
|
|
|
|
default:
|
|
case PROJECTILE_RPG:
|
|
{
|
|
davect = pSprite->pos;
|
|
|
|
VM_UpdateAnim(spriteNum, &actor[spriteNum].t_data[0]);
|
|
|
|
if (pProj->flashcolor)
|
|
G_AddGameLight(0, spriteNum, ((pSprite->yrepeat * tilesiz[pSprite->picnum].y) << 1), 2048, pProj->flashcolor,
|
|
PR_LIGHT_PRIO_LOW_GAME);
|
|
|
|
if ((pProj->workslike & (PROJECTILE_BOUNCESOFFWALLS | PROJECTILE_EXPLODEONTIMER)) == PROJECTILE_BOUNCESOFFWALLS
|
|
&& pSprite->yvel < 1)
|
|
{
|
|
A_DoProjectileEffects(spriteNum, &davect, 1);
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
|
|
if (pProj->workslike & PROJECTILE_COOLEXPLOSION1 && ++pSprite->shade >= 40)
|
|
{
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
|
|
pSprite->zvel -= pProj->drop;
|
|
|
|
if (pProj->workslike & PROJECTILE_SPIT && pSprite->zvel < 6144)
|
|
pSprite->zvel += g_spriteGravity - 112;
|
|
|
|
A_GetZLimits(spriteNum);
|
|
|
|
if (pProj->trail >= 0)
|
|
{
|
|
for (bssize_t cnt = 0; cnt <= pProj->tnum; cnt++)
|
|
{
|
|
otherSprite = A_Spawn(spriteNum, pProj->trail);
|
|
|
|
sprite[otherSprite].z += (pProj->toffset << 8);
|
|
|
|
if (pProj->txrepeat >= 0)
|
|
sprite[otherSprite].xrepeat = pProj->txrepeat;
|
|
|
|
if (pProj->tyrepeat >= 0)
|
|
sprite[otherSprite].yrepeat = pProj->tyrepeat;
|
|
}
|
|
}
|
|
|
|
int projMoveCnt = pProj->movecnt;
|
|
int projVel = pSprite->xvel;
|
|
int projZvel = pSprite->zvel;
|
|
|
|
if (sector[pSprite->sectnum].lotag == ST_2_UNDERWATER)
|
|
{
|
|
projVel >>= 1;
|
|
projZvel >>= 1;
|
|
}
|
|
|
|
do
|
|
{
|
|
vec3_t tmpvect = { (projVel * (sintable[(pSprite->ang + 512) & 2047])) >> 14 >> (int)!projectileMoved,
|
|
(projVel * (sintable[pSprite->ang & 2047])) >> 14 >> (int)!projectileMoved, projZvel >> (int)!projectileMoved };
|
|
Bmemcpy(&davect, pSprite, sizeof(vec3_t));
|
|
projectileMoved++;
|
|
otherSprite = A_MoveSprite(spriteNum, &tmpvect, (A_CheckSpriteFlags(spriteNum, SFLAG_NOCLIP) ? 0 : CLIPMASK1));
|
|
}
|
|
while (!otherSprite && --projMoveCnt > 0);
|
|
|
|
if (!(pProj->workslike & PROJECTILE_BOUNCESOFFWALLS) && // NOT_BOUNCESOFFWALLS_YVEL
|
|
(unsigned)pSprite->yvel < MAXSPRITES
|
|
&& sprite[pSprite->yvel].sectnum != MAXSECTORS)
|
|
if (FindDistance2D(pSprite->x - sprite[pSprite->yvel].x, pSprite->y - sprite[pSprite->yvel].y) < 256)
|
|
otherSprite = 49152 | pSprite->yvel;
|
|
|
|
actor[spriteNum].movflag = otherSprite;
|
|
|
|
if (pSprite->sectnum < 0)
|
|
{
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
|
|
if (pProj->workslike & PROJECTILE_TIMED && pProj->range > 0)
|
|
{
|
|
if (++actor[spriteNum].t_data[8] > pProj->range)
|
|
{
|
|
if (pProj->workslike & PROJECTILE_EXPLODEONTIMER)
|
|
A_DoProjectileEffects(spriteNum, &davect, 1);
|
|
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if ((otherSprite & 49152) != 49152 && !(pProj->workslike & PROJECTILE_BOUNCESOFFWALLS))
|
|
G_WeaponHitCeilingOrFloor(spriteNum, pSprite, &otherSprite);
|
|
|
|
if (pProj->workslike & PROJECTILE_WATERBUBBLES && sector[pSprite->sectnum].lotag == ST_2_UNDERWATER && rnd(140))
|
|
A_Spawn(spriteNum, WATERBUBBLE);
|
|
|
|
if (otherSprite != 0)
|
|
{
|
|
if (pProj->workslike & PROJECTILE_COOLEXPLOSION1)
|
|
{
|
|
pSprite->xvel = 0;
|
|
pSprite->zvel = 0;
|
|
}
|
|
|
|
switch (otherSprite & 49152)
|
|
{
|
|
case 49152:
|
|
otherSprite &= (MAXSPRITES - 1);
|
|
|
|
if (pProj->workslike & PROJECTILE_BOUNCESOFFSPRITES)
|
|
{
|
|
pSprite->yvel--;
|
|
|
|
int const projAngle = getangle(sprite[otherSprite].x - pSprite->x, sprite[otherSprite].y - pSprite->y)
|
|
+ ((sprite[otherSprite].cstat & 16) ? 0 : 512);
|
|
pSprite->ang = ((projAngle << 1) - pSprite->ang) & 2047;
|
|
|
|
if (pProj->bsound >= 0)
|
|
A_PlaySound(pProj->bsound, spriteNum);
|
|
|
|
if (pProj->workslike & PROJECTILE_LOSESVELOCITY)
|
|
PROJ_DECAYVELOCITY(pSprite);
|
|
|
|
if (!(pProj->workslike & PROJECTILE_FORCEIMPACT))
|
|
return;
|
|
}
|
|
|
|
A_DamageObject(otherSprite, spriteNum);
|
|
|
|
if (sprite[otherSprite].picnum == APLAYER)
|
|
{
|
|
int playerNum = P_Get(otherSprite);
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
A_PlaySound(PISTOL_BODYHIT, otherSprite);
|
|
#endif
|
|
if (pProj->workslike & PROJECTILE_SPIT)
|
|
P_HandleBeingSpitOn(g_player[playerNum].ps);
|
|
}
|
|
|
|
if (pProj->workslike & PROJECTILE_RPG_IMPACT)
|
|
{
|
|
actor[otherSprite].owner = pSprite->owner;
|
|
actor[otherSprite].picnum = pSprite->picnum;
|
|
|
|
if (pProj->workslike & PROJECTILE_RPG_IMPACT_DAMAGE)
|
|
actor[otherSprite].extra += pProj->extra;
|
|
|
|
A_DoProjectileEffects(spriteNum, &davect, 0);
|
|
|
|
if (!(pProj->workslike & PROJECTILE_FORCEIMPACT))
|
|
{
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (pProj->workslike & PROJECTILE_FORCEIMPACT)
|
|
return;
|
|
break;
|
|
|
|
case 32768:
|
|
otherSprite &= (MAXWALLS - 1);
|
|
|
|
if (pProj->workslike & PROJECTILE_BOUNCESOFFMIRRORS
|
|
&& (wall[otherSprite].overpicnum == MIRROR || wall[otherSprite].picnum == MIRROR))
|
|
{
|
|
Proj_BounceOffWall(pSprite, otherSprite);
|
|
pSprite->owner = spriteNum;
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
A_Spawn(spriteNum, TRANSPORTERSTAR);
|
|
#endif
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
setsprite(spriteNum, &davect);
|
|
A_DamageWall(spriteNum, otherSprite, pSprite->pos, pSprite->picnum);
|
|
|
|
if (pProj->workslike & PROJECTILE_BOUNCESOFFWALLS)
|
|
{
|
|
if (wall[otherSprite].overpicnum != MIRROR && wall[otherSprite].picnum != MIRROR)
|
|
pSprite->yvel--;
|
|
|
|
Proj_BounceOffWall(pSprite, otherSprite);
|
|
|
|
if (pProj->bsound >= 0)
|
|
A_PlaySound(pProj->bsound, spriteNum);
|
|
|
|
if (pProj->workslike & PROJECTILE_LOSESVELOCITY)
|
|
PROJ_DECAYVELOCITY(pSprite);
|
|
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 16384:
|
|
setsprite(spriteNum, &davect);
|
|
|
|
if (Proj_MaybeDamageCF(spriteNum))
|
|
{
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
|
|
if (pProj->workslike & PROJECTILE_BOUNCESOFFWALLS)
|
|
{
|
|
A_DoProjectileBounce(spriteNum);
|
|
A_SetSprite(spriteNum, CLIPMASK1);
|
|
|
|
pSprite->yvel--;
|
|
|
|
if (pProj->bsound >= 0)
|
|
A_PlaySound(pProj->bsound, spriteNum);
|
|
|
|
if (pProj->workslike & PROJECTILE_LOSESVELOCITY)
|
|
PROJ_DECAYVELOCITY(pSprite);
|
|
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
|
|
A_DoProjectileEffects(spriteNum, &davect, 1);
|
|
A_DeleteSprite(spriteNum);
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
ACTOR_STATIC void G_MoveWeapons(void)
|
|
{
|
|
int spriteNum = headspritestat[STAT_PROJECTILE];
|
|
|
|
while (spriteNum >= 0)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
|
|
if (pSprite->sectnum < 0)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
actor[spriteNum].bpos = pSprite->pos;
|
|
|
|
/* Custom projectiles */
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_PROJECTILE))
|
|
{
|
|
Proj_MoveCustom(spriteNum);
|
|
goto next_sprite;
|
|
}
|
|
|
|
// hard coded projectiles
|
|
switch (DYNAMICTILEMAP(pSprite->picnum))
|
|
{
|
|
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;
|
|
}
|
|
|
|
case RADIUSEXPLOSION__STATIC:
|
|
case KNEE__STATIC: DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
#ifndef EDUKE32_STANDALONE
|
|
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 SHRINKSPARK__STATIC:
|
|
case RPG__STATIC:
|
|
case FIRELASER__STATIC:
|
|
case SPIT__STATIC:
|
|
case COOLEXPLOSION1__STATIC:
|
|
{
|
|
int const projectileMoved = SpriteProjectile[spriteNum].workslike & PROJECTILE_MOVED;
|
|
SpriteProjectile[spriteNum].workslike |= PROJECTILE_MOVED;
|
|
|
|
if (pSprite->picnum == COOLEXPLOSION1)
|
|
if (!S_CheckSoundPlaying(WIERDSHOT_FLY))
|
|
A_PlaySound(WIERDSHOT_FLY, spriteNum);
|
|
|
|
int spriteXvel = pSprite->xvel;
|
|
int spriteZvel = pSprite->zvel;
|
|
|
|
if (pSprite->picnum == RPG && sector[pSprite->sectnum].lotag == ST_2_UNDERWATER)
|
|
{
|
|
spriteXvel >>= 1;
|
|
spriteZvel >>= 1;
|
|
}
|
|
|
|
vec3_t davect = pSprite->pos;
|
|
|
|
A_GetZLimits(spriteNum);
|
|
|
|
if (pSprite->picnum == RPG && actor[spriteNum].picnum != BOSS2 && pSprite->xrepeat >= 10
|
|
&& sector[pSprite->sectnum].lotag != ST_2_UNDERWATER
|
|
&& g_scriptVersion >= 13)
|
|
{
|
|
int const newSprite = A_Spawn(spriteNum, SMALLSMOKE);
|
|
sprite[newSprite].z += (1 << 8);
|
|
}
|
|
|
|
vec3_t const tmpvect = { (spriteXvel * (sintable[(pSprite->ang + 512) & 2047])) >> 14 >> (int)!projectileMoved,
|
|
(spriteXvel * (sintable[pSprite->ang & 2047])) >> 14 >> (int)!projectileMoved, spriteZvel >> (int)!projectileMoved };
|
|
|
|
int moveSprite = A_MoveSprite(spriteNum, &tmpvect, (A_CheckSpriteFlags(spriteNum, SFLAG_NOCLIP) ? 0 : CLIPMASK1));
|
|
|
|
if (pSprite->picnum == RPG && (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 ((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 (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 (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)
|
|
P_HandleBeingSpitOn(g_player[playerNum].ps);
|
|
}
|
|
break;
|
|
|
|
case 32768:
|
|
moveSprite &= (MAXWALLS - 1);
|
|
|
|
if (pSprite->picnum != RPG && pSprite->picnum != FREEZEBLAST && pSprite->picnum != SPIT
|
|
&& (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, pSprite->pos, pSprite->picnum);
|
|
|
|
if (pSprite->picnum == FREEZEBLAST)
|
|
{
|
|
if (wall[moveSprite].overpicnum != MIRROR && wall[moveSprite].picnum != MIRROR)
|
|
{
|
|
pSprite->extra >>= 1;
|
|
pSprite->yvel--;
|
|
}
|
|
|
|
Proj_BounceOffWall(pSprite, moveSprite);
|
|
goto next_sprite;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 16384:
|
|
setsprite(spriteNum, &davect);
|
|
|
|
if (Proj_MaybeDamageCF(spriteNum))
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
if (pSprite->picnum == FREEZEBLAST)
|
|
{
|
|
A_DoProjectileBounce(spriteNum);
|
|
A_SetSprite(spriteNum, CLIPMASK1);
|
|
|
|
pSprite->extra >>= 1;
|
|
pSprite->yvel--;
|
|
|
|
if (pSprite->xrepeat > 8)
|
|
{
|
|
pSprite->xrepeat -= 2;
|
|
|
|
if (pSprite->yrepeat > 8)
|
|
pSprite->yrepeat -= 2;
|
|
}
|
|
|
|
goto next_sprite;
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
|
|
switch (DYNAMICTILEMAP(pSprite->picnum))
|
|
{
|
|
case SPIT__STATIC:
|
|
case COOLEXPLOSION1__STATIC:
|
|
case FREEZEBLAST__STATIC:
|
|
case FIRELASER__STATIC: break;
|
|
|
|
case RPG__STATIC:
|
|
{
|
|
int const newSprite = A_Spawn(spriteNum, EXPLOSION2);
|
|
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 (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:
|
|
A_Spawn(spriteNum, SHRINKEREXPLOSION);
|
|
A_PlaySound(SHRINKER_HIT, spriteNum);
|
|
A_RadiusDamage(spriteNum, g_shrinkerRadius, 0, 0, 0, 0);
|
|
break;
|
|
|
|
default:
|
|
{
|
|
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 (pSprite->picnum != COOLEXPLOSION1)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
|
|
if (pSprite->picnum == COOLEXPLOSION1)
|
|
{
|
|
COOLEXPLOSION:
|
|
pSprite->shade++;
|
|
if (pSprite->shade >= 40)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
else if (pSprite->picnum == RPG && sector[pSprite->sectnum].lotag == ST_2_UNDERWATER && pSprite->xrepeat >= 10 && rnd(140))
|
|
A_Spawn(spriteNum, WATERBUBBLE);
|
|
|
|
goto next_sprite;
|
|
}
|
|
#endif
|
|
}
|
|
next_sprite:
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
|
|
|
|
static int P_Submerge(int const playerNum, DukePlayer_t * const pPlayer, int const sectNum, int const otherSect)
|
|
{
|
|
if (pPlayer->on_ground && pPlayer->pos.z >= sector[sectNum].floorz
|
|
&& (TEST_SYNC_KEY(g_player[playerNum].input->bits, SK_CROUCH) || pPlayer->vel.z > 2048))
|
|
// if( onfloorz && sectlotag == 1 && ps->pos.z > (sector[sect].floorz-(6<<8)) )
|
|
{
|
|
if (screenpeek == playerNum)
|
|
{
|
|
FX_StopAllSounds();
|
|
S_ClearSoundLocks();
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (sprite[pPlayer->i].extra > 0)
|
|
A_PlaySound(DUKE_UNDERWATER, pPlayer->i);
|
|
#endif
|
|
|
|
pPlayer->opos.z = pPlayer->pos.z = sector[otherSect].ceilingz;
|
|
|
|
if (TEST_SYNC_KEY(g_player[playerNum].input->bits, SK_CROUCH))
|
|
pPlayer->vel.z += 512;
|
|
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int P_Emerge(int const playerNum, DukePlayer_t * const pPlayer, int const sectNum, int const otherSect)
|
|
{
|
|
// r1449-:
|
|
if (pPlayer->pos.z < (sector[sectNum].ceilingz+1080) && pPlayer->vel.z <= 0)
|
|
// r1450+, breaks submergible slime in bobsp2:
|
|
// if (onfloorz && sectlotag == 2 && ps->pos.z <= sector[sect].ceilingz /*&& ps->vel.z == 0*/)
|
|
{
|
|
// if( sprite[j].extra <= 0) break;
|
|
if (screenpeek == playerNum)
|
|
{
|
|
FX_StopAllSounds();
|
|
S_ClearSoundLocks();
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
A_PlaySound(DUKE_GASP, pPlayer->i);
|
|
#endif
|
|
|
|
pPlayer->opos.z = pPlayer->pos.z = sector[otherSect].floorz;
|
|
pPlayer->vel.z = 0;
|
|
// ps->vel.z += 1024;
|
|
|
|
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);
|
|
|
|
vec3_t vect = pPlayer->pos;
|
|
vect.z += PHEIGHT;
|
|
setsprite(pPlayer->i, &vect);
|
|
|
|
P_UpdateScreenPal(pPlayer);
|
|
|
|
if ((krand()&255) < 32)
|
|
A_Spawn(playerNum, WATERSPLASH2);
|
|
|
|
if (sectLotag == ST_1_ABOVE_WATER)
|
|
{
|
|
for (bssize_t l = 0; l < 9; l++)
|
|
sprite[A_Spawn(pPlayer->i, WATERBUBBLE)].z += krand() & 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;
|
|
return !!(A_CheckSpriteFlags(spriteNum, SFLAG_NOTELEPORT) || 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);
|
|
auto 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)
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (sprite[spriteNum].pal == 0)
|
|
{
|
|
A_Spawn(spriteNum, TRANSPORTERBEAM);
|
|
A_PlaySound(TELEPORTER, spriteNum);
|
|
}
|
|
#endif
|
|
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 = sprite[OW(spriteNum)].pos;
|
|
pPlayer->pos.z -= PHEIGHT;
|
|
pPlayer->opos = pPlayer->pos;
|
|
pPlayer->bobpos = pPlayer->pos.vec2;
|
|
|
|
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
|
|
pPlayer->cursectnum = sprite[sectSprite].sectnum;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (sprite[spriteNum].pal == 0)
|
|
{
|
|
int const newSprite = A_Spawn(OW(spriteNum), TRANSPORTERBEAM);
|
|
A_PlaySound(TELEPORTER, newSprite);
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
if (onFloor == 0 && klabs(SZ(spriteNum) - pPlayer->pos.z) < 6144)
|
|
if (!pPlayer->jetpack_on || TEST_SYNC_KEY(g_player[playerNum].input->bits, SK_JUMP)
|
|
|| TEST_SYNC_KEY(g_player[playerNum].input->bits, SK_CROUCH))
|
|
{
|
|
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].input->bits, SK_JUMP)
|
|
|| pPlayer->jetpack_on < 11))
|
|
? sprite[OW(spriteNum)].z - 6144
|
|
: sprite[OW(spriteNum)].z + 6144;
|
|
|
|
actor[pPlayer->i].bpos = pPlayer->pos;
|
|
pPlayer->opos = pPlayer->pos;
|
|
pPlayer->bobpos = pPlayer->pos.vec2;
|
|
|
|
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
|
|
pPlayer->cursectnum = sprite[OW(spriteNum)].sectnum;
|
|
|
|
break;
|
|
}
|
|
|
|
int doWater = 0;
|
|
|
|
if (onFloor)
|
|
{
|
|
if (sectLotag == ST_1_ABOVE_WATER)
|
|
doWater = P_Submerge(playerNum, pPlayer, sectNum, sprite[OW(spriteNum)].sectnum);
|
|
else if (sectLotag == ST_2_UNDERWATER)
|
|
doWater = P_Emerge(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 (!(sectLotag == ST_1_ABOVE_WATER && pPlayer->on_ground == 1))
|
|
break;
|
|
}
|
|
break;
|
|
|
|
|
|
////////// Non-player teleportation //////////
|
|
|
|
case STAT_PROJECTILE:
|
|
// SE7_PROJECTILE, PROJECTILE_CHSECT.
|
|
// comment out to make RPGs pass through water: (r1450 breaks this)
|
|
// if (sectlotag != 0) goto JBOLT;
|
|
case STAT_ACTOR:
|
|
if (sprite[sectSprite].extra > 0 && A_CheckNonTeleporting(sectSprite))
|
|
goto JBOLT;
|
|
fallthrough__;
|
|
case STAT_MISC:
|
|
case STAT_FALLER:
|
|
case STAT_DUMMYPLAYER:
|
|
{
|
|
if (((int32_t) totalclock & UINT8_MAX) != actor[sectSprite].lasttransport)
|
|
{
|
|
int const zvel = sprite[sectSprite].zvel;
|
|
int const absZvel = klabs(zvel);
|
|
int doWarp = 0;
|
|
|
|
if (absZvel != 0)
|
|
{
|
|
if (sectLotag == ST_2_UNDERWATER && sprite[sectSprite].z < (sector[sectNum].ceilingz + absZvel) && zvel < 0)
|
|
doWarp = 1;
|
|
if (sectLotag == ST_1_ABOVE_WATER && sprite[sectSprite].z > (sector[sectNum].floorz - absZvel) && zvel > 0)
|
|
doWarp = 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)
|
|
{
|
|
if (A_CheckSpriteFlags(sectSprite, SFLAG_DECAL))
|
|
goto JBOLT;
|
|
|
|
switch (DYNAMICTILEMAP(sprite[sectSprite].picnum))
|
|
{
|
|
#ifndef EDUKE32_STANDALONE
|
|
case TRANSPORTERSTAR__STATIC:
|
|
case TRANSPORTERBEAM__STATIC:
|
|
case TRIPBOMB__STATIC:
|
|
case BULLETHOLE__STATIC:
|
|
case WATERSPLASH2__STATIC:
|
|
case BURNING__STATIC:
|
|
case BURNING2__STATIC:
|
|
case FIRE__STATIC:
|
|
case FIRE2__STATIC:
|
|
case TOILETWATER__STATIC:
|
|
case LASERLINE__STATIC: goto JBOLT;
|
|
#endif
|
|
case PLAYERONWATER__STATIC:
|
|
if (sectLotag == ST_2_UNDERWATER)
|
|
{
|
|
sprite[sectSprite].cstat &= 32768;
|
|
break;
|
|
}
|
|
fallthrough__;
|
|
default:
|
|
if (sprite[sectSprite].statnum == STAT_MISC && !(sectLotag == ST_1_ABOVE_WATER || sectLotag == ST_2_UNDERWATER))
|
|
break;
|
|
fallthrough__;
|
|
case WATERBUBBLE__STATIC:
|
|
// if( rnd(192) && sprite[j].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);
|
|
#ifndef EDUKE32_STANDALONE
|
|
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);
|
|
}
|
|
#endif
|
|
actor[sectSprite].lasttransport = ((int32_t) totalclock & UINT8_MAX);
|
|
|
|
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 : sector[osect].floorz;
|
|
|
|
actor[sectSprite].bpos = sprite[sectSprite].pos;
|
|
|
|
changespritesect(sectSprite, sprite[OW(spriteNum)].sectnum);
|
|
}
|
|
else if (Bassert(sectLotag == 0), 1)
|
|
{
|
|
// Non-water SE7 teleportation.
|
|
|
|
if (onFloor)
|
|
{
|
|
if (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 = sprite[sectSprite].pos;
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (sprite[spriteNum].pal == 0)
|
|
{
|
|
int newSprite = A_Spawn(spriteNum, TRANSPORTERBEAM);
|
|
A_PlaySound(TELEPORTER, newSprite);
|
|
|
|
newSprite = A_Spawn(OW(spriteNum), TRANSPORTERBEAM);
|
|
A_PlaySound(TELEPORTER, newSprite);
|
|
}
|
|
#endif
|
|
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 = sprite[sectSprite].pos;
|
|
|
|
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;
|
|
}
|
|
|
|
static int A_FindLocatorWithHiLoTags(int const hitag, int const tag, int const sectNum)
|
|
{
|
|
for (bssize_t SPRITES_OF(STAT_LOCATOR, spriteNum))
|
|
{
|
|
if ((sectNum == -1 || sectNum == SECT(spriteNum)) && tag == SLT(spriteNum) && hitag == SHT(spriteNum))
|
|
return spriteNum;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
ACTOR_STATIC void G_MoveActors(void)
|
|
{
|
|
int spriteNum = headspritestat[STAT_ACTOR];
|
|
|
|
while (spriteNum >= 0)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int const sectNum = pSprite->sectnum;
|
|
auto const pData = actor[spriteNum].t_data;
|
|
|
|
int switchPic;
|
|
|
|
if (pSprite->xrepeat == 0 || sectNum < 0 || sectNum >= MAXSECTORS)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
actor[spriteNum].bpos = pSprite->pos;
|
|
|
|
switchPic = pSprite->picnum;
|
|
|
|
if (pSprite->picnum > GREENSLIME && pSprite->picnum <= GREENSLIME+7)
|
|
switchPic = GREENSLIME;
|
|
|
|
switch (DYNAMICTILEMAP(switchPic))
|
|
{
|
|
case OOZ__STATIC:
|
|
case OOZ2__STATIC:
|
|
{
|
|
A_GetZLimits(spriteNum);
|
|
|
|
int const yrepeat = clamp((actor[spriteNum].floorz - actor[spriteNum].ceilingz) >> 9, 8, 255);
|
|
int const xrepeat = clamp(25 - (yrepeat >> 1), 8, 48);
|
|
|
|
pSprite->yrepeat = yrepeat;
|
|
pSprite->xrepeat = xrepeat;
|
|
pSprite->z = actor[spriteNum].floorz;
|
|
|
|
goto next_sprite;
|
|
}
|
|
#ifndef EDUKE32_STANDALONE
|
|
case DUCK__STATIC:
|
|
case TARGET__STATIC:
|
|
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:
|
|
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 ((krand()&255) < 3) A_PlaySound(RATTY,spriteNum);
|
|
pSprite->ang += (krand()&31)-15+(sintable[(pData[0]<<8)&2047]>>11);
|
|
}
|
|
else
|
|
{
|
|
T1(spriteNum)++;
|
|
if (T1(spriteNum) > 1)
|
|
{
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
else pSprite->ang = (krand()&2047);
|
|
}
|
|
if (pSprite->xvel < 128)
|
|
pSprite->xvel+=2;
|
|
pSprite->ang += (krand()&3)-6;
|
|
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(&pSprite->pos, &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);
|
|
auto 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].input->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 + (krand() & 63);
|
|
}
|
|
|
|
pData[3] = 64;
|
|
}
|
|
|
|
goto next_sprite;
|
|
|
|
case RECON__STATIC:
|
|
{
|
|
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;
|
|
}
|
|
|
|
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) || 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, newSprite);
|
|
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;
|
|
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) << 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)
|
|
{
|
|
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+(krand()&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 - 512)
|
|
pSprite->z += 512;
|
|
else if (pSprite->z > sprite[locatorSprite].z + 512)
|
|
pSprite->z -= 512;
|
|
else
|
|
pSprite->z = sprite[locatorSprite].z;
|
|
}
|
|
|
|
if (!A_CheckSoundPlaying(spriteNum,RECO_ROAM))
|
|
A_PlaySound(RECO_ROAM,spriteNum);
|
|
|
|
A_SetSprite(spriteNum,CLIPMASK0);
|
|
|
|
goto next_sprite;
|
|
}
|
|
|
|
case GREENSLIME__STATIC:
|
|
{
|
|
// #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);
|
|
auto 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 newSprite = A_InsertSprite(SECT(spriteNum), SX(spriteNum), SY(spriteNum), SZ(spriteNum),
|
|
GLASSPIECES + (j % 3), -32, 36, 36, krand() & 2047, 32 + (krand() & 63),
|
|
1024 - (krand() & 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,&pSprite->pos);
|
|
|
|
pSprite->ang = fix16_to_int(pPlayer->q16ang);
|
|
|
|
if ((TEST_SYNC_KEY(g_player[playerNum].input->bits, SK_FIRE) || (pPlayer->quick_kick > 0)) && sprite[pPlayer->i].extra > 0)
|
|
if (pPlayer->quick_kick > 0 ||
|
|
(PWEAPON(playerNum, pPlayer->curr_weapon, WorksLike) != HANDREMOTE_WEAPON && PWEAPON(playerNum, pPlayer->curr_weapon, WorksLike) != HANDBOMB_WEAPON &&
|
|
PWEAPON(playerNum, pPlayer->curr_weapon, WorksLike) != TRIPBOMB_WEAPON && pPlayer->ammo_amount[pPlayer->curr_weapon] >= 0))
|
|
{
|
|
for (bssize_t x = 0; x < 8; ++x)
|
|
{
|
|
int const j
|
|
= A_InsertSprite(sectNum, pSprite->x, pSprite->y, pSprite->z - ZOFFSET3, SCRAP3 + (krand() & 3), -8, 48, 48,
|
|
krand() & 2047, (krand() & 63) + 64, -(krand() & 4095) - (pSprite->zvel >> 2), spriteNum, 5);
|
|
sprite[j].pal = 6;
|
|
}
|
|
|
|
A_PlaySound(SLIM_DYING,spriteNum);
|
|
A_PlaySound(SQUISHED,spriteNum);
|
|
|
|
if ((krand()&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 + (krand() & 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);
|
|
|
|
if (pPlayer->somethingonplayer == spriteNum)
|
|
pPlayer->somethingonplayer = -1;
|
|
|
|
if (damageTile == FREEZEBLAST)
|
|
{
|
|
A_PlaySound(SOMETHINGFROZE, spriteNum);
|
|
pData[0] = -5;
|
|
pData[3] = 0;
|
|
goto next_sprite;
|
|
}
|
|
|
|
P_AddKills(pPlayer, 1);
|
|
|
|
if ((krand()&255) < 32)
|
|
{
|
|
int const j = A_Spawn(spriteNum,BLOODPOOL);
|
|
sprite[j].pal = 0;
|
|
}
|
|
|
|
for (bssize_t x=0; x<8; x++)
|
|
{
|
|
int const j = A_InsertSprite(sectNum, pSprite->x, pSprite->y, pSprite->z - ZOFFSET3, SCRAP3 + (krand() & 3), -8,
|
|
48, 48, krand() & 2047, (krand() & 63) + 64, -(krand() & 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 -= krand()&7;
|
|
if (pSprite->yrepeat > 16) pSprite->yrepeat -= krand()&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;
|
|
|
|
// JBF 20041129: a slimer eating another enemy really ought
|
|
// to decrease the maximum kill count by one.
|
|
if (sprite[pData[5]].extra > 0)
|
|
g_player[myconnectindex].ps->max_actors_killed--;
|
|
}
|
|
}
|
|
}
|
|
|
|
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 ((krand()&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
|
|
{
|
|
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 BOUNCEMINE__STATIC:
|
|
if (pSprite->xvel != 0)
|
|
case MORTER__STATIC:
|
|
{
|
|
int const j = A_Spawn(spriteNum, (PLUTOPAK ? FRAMEEFFECT1 : FRAMEEFFECT1_13));
|
|
actor[j].t_data[0] = 3;
|
|
}
|
|
fallthrough__;
|
|
case HEAVYHBOMB__STATIC:
|
|
{
|
|
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[2] = 0;
|
|
detonatePlayer = 0;
|
|
pSprite->xvel = 0;
|
|
goto DETONATEB;
|
|
}
|
|
}
|
|
|
|
if (pSprite->picnum != BOUNCEMINE)
|
|
{
|
|
A_Fall(spriteNum);
|
|
|
|
if ((sector[sectNum].lotag != ST_1_ABOVE_WATER || actor[spriteNum].floorz != sector[sectNum].floorz) && pSprite->z >= actor[spriteNum].floorz-(ZOFFSET) && pSprite->yvel < 3)
|
|
{
|
|
if (pSprite->yvel > 0 || (pSprite->yvel == 0 && actor[spriteNum].floorz == sector[sectNum].floorz))
|
|
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 (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 && actor[spriteNum].floorz == sector[sectNum].floorz)
|
|
{
|
|
pSprite->z += ZOFFSET5;
|
|
if (pData[5] == 0)
|
|
{
|
|
pData[5] = 1;
|
|
A_Spawn(spriteNum,WATERSPLASH2);
|
|
}
|
|
}
|
|
else pData[5] = 0;
|
|
|
|
if (pData[3] == 0 && (pSprite->picnum == BOUNCEMINE || pSprite->picnum == MORTER) && (moveSprite || playerDist < 844))
|
|
{
|
|
pData[3] = 1;
|
|
pData[2] = 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)
|
|
{
|
|
moveSprite &= (MAXWALLS - 1);
|
|
A_DamageWall(spriteNum, moveSprite, pSprite->pos, pSprite->picnum);
|
|
Proj_BounceOffWall(pSprite, moveSprite);
|
|
pSprite->xvel >>= 1;
|
|
}
|
|
|
|
DETONATEB:
|
|
// Pipebomb control set to timer? (see player.c)
|
|
// TIMER_CONTROL
|
|
if (pSprite->picnum == HEAVYHBOMB && pData[6] == 1)
|
|
{
|
|
if (pData[7] >= 1)
|
|
pData[7]--;
|
|
|
|
if (pData[7] <= 0)
|
|
pData[6] = 3;
|
|
}
|
|
|
|
if ((detonatePlayer >= 0 && g_player[detonatePlayer].ps->hbomb_on == 0 && pData[6] == 2) || pData[3] == 1)
|
|
pData[6] = 3;
|
|
|
|
if (pData[6] == 3)
|
|
{
|
|
pData[2]++;
|
|
|
|
if (pData[2] == 2)
|
|
{
|
|
int const x = pSprite->extra;
|
|
int radius = 0;
|
|
|
|
switch (DYNAMICTILEMAP(pSprite->picnum))
|
|
{
|
|
case HEAVYHBOMB__STATIC: radius = g_pipebombRadius; break;
|
|
case MORTER__STATIC: radius = g_morterRadius; break;
|
|
case BOUNCEMINE__STATIC: radius = g_bouncemineRadius; break;
|
|
}
|
|
|
|
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 (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[2] > 20)
|
|
{
|
|
if (pSprite->owner != spriteNum || ud.respawn_items == 0)
|
|
{
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
else
|
|
{
|
|
pData[2] = g_itemRespawnTime;
|
|
A_Spawn(spriteNum,RESPAWNMARKERRED);
|
|
pSprite->cstat = 32768;
|
|
pSprite->yrepeat = 9;
|
|
goto next_sprite;
|
|
}
|
|
}
|
|
}
|
|
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])
|
|
{
|
|
if ((g_gametypeFlags[ud.coop] & GAMETYPE_WEAPSTAY) && pSprite->owner == spriteNum)
|
|
{
|
|
for (bssize_t j = 0; j < pPlayer->weapreccnt; j++)
|
|
{
|
|
if (pPlayer->weaprecs[j] == pSprite->picnum)
|
|
goto next_sprite;
|
|
}
|
|
|
|
if (pPlayer->weapreccnt < MAX_WEAPONS)
|
|
pPlayer->weaprecs[pPlayer->weapreccnt++] = pSprite->picnum;
|
|
}
|
|
|
|
P_AddAmmo(pPlayer, HANDBOMB_WEAPON, 1);
|
|
A_PlaySound(DUKE_GET, pPlayer->i);
|
|
|
|
if ((pPlayer->gotweapon & (1<<HANDBOMB_WEAPON)) == 0 || pSprite->owner == pPlayer->i)
|
|
{
|
|
int doSwitch = ((pPlayer->weaponswitch & 1) ||
|
|
PWEAPON(playerNum, pPlayer->curr_weapon, WorksLike) == HANDREMOTE_WEAPON);
|
|
P_AddWeapon(pPlayer, HANDBOMB_WEAPON, doSwitch);
|
|
}
|
|
|
|
if (sprite[pSprite->owner].picnum != APLAYER)
|
|
P_PalFrom(pPlayer, 32, 0, 32, 0);
|
|
|
|
if (pSprite->owner != spriteNum || ud.respawn_items == 0)
|
|
{
|
|
if (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);
|
|
auto const pPlayer = g_player[playerNum].ps;
|
|
|
|
if (++pData[2] == 4)
|
|
pData[2] = 0;
|
|
|
|
if (playerDist < 4096)
|
|
{
|
|
if ((krand() & 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 - (krand() % (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;
|
|
}
|
|
#endif // EDUKE32_STANDALONE
|
|
|
|
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;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
for (bssize_t x = 0; x < 5; x++)
|
|
RANDOMSCRAP(pSprite, spriteNum);
|
|
#endif
|
|
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 = 0;
|
|
if (pSprite->extra)
|
|
pSprite->cstat = 257;
|
|
}
|
|
}
|
|
|
|
if (G_HaveActor(sprite[spriteNum].picnum))
|
|
{
|
|
int32_t playerDist;
|
|
int const playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
A_Execute(spriteNum, playerNum, playerDist);
|
|
}
|
|
next_sprite:
|
|
A_MaybeAwakenBadGuys(spriteNum);
|
|
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;
|
|
auto const pData = actor[spriteNum].t_data;
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int sectNum = pSprite->sectnum; // XXX: not const
|
|
int switchPic;
|
|
|
|
if (sectNum < 0 || pSprite->xrepeat == 0)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
actor[spriteNum].bpos = pSprite->pos;
|
|
|
|
switchPic = pSprite->picnum;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (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) || (pSprite->picnum == MAIL+1) || (pSprite->picnum == PAPER+1))
|
|
actor[spriteNum].floorz = pSprite->z = getflorzofslope(pSprite->sectnum,pSprite->x,pSprite->y);
|
|
else
|
|
#endif
|
|
switch (DYNAMICTILEMAP(switchPic))
|
|
{
|
|
case APLAYER__STATIC: pSprite->cstat = 32768; goto next_sprite;
|
|
case FRAMEEFFECT1_13__STATIC:
|
|
if (PLUTOPAK) goto next_sprite; // JBF: ideally this should never happen...
|
|
fallthrough__;
|
|
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 EXPLOSION2__STATIC:
|
|
case EXPLOSION2BOT__STATIC:
|
|
case FORCERIPPLE__STATIC:
|
|
case TRANSPORTERSTAR__STATIC:
|
|
case TRANSPORTERBEAM__STATIC:
|
|
case SMALLSMOKE__STATIC:
|
|
#ifndef EDUKE32_STANDALONE
|
|
case WATERBUBBLE__STATIC:
|
|
case BURNING__STATIC:
|
|
case BURNING2__STATIC:
|
|
case FECES__STATIC:
|
|
case SHRINKEREXPLOSION__STATIC:
|
|
case BLOOD__STATIC:
|
|
case LASERSITE__STATIC:
|
|
#endif
|
|
{
|
|
if (!G_HaveActor(sprite[spriteNum].picnum))
|
|
goto next_sprite;
|
|
int const playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
A_Execute(spriteNum, playerNum, playerDist);
|
|
goto next_sprite;
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
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] == 3 * GAMETICSPERSEC)
|
|
goto next_sprite;
|
|
|
|
actor[spriteNum].bpos.z -= pSprite->z;
|
|
|
|
if ((++pData[0] % 9) == 0)
|
|
{
|
|
pSprite->yrepeat++;
|
|
pSprite->z += (tilesiz[pSprite->picnum].y * pSprite->yrepeat) >> 2;
|
|
}
|
|
else
|
|
pSprite->z += 16 + (krand() & 15);
|
|
|
|
actor[spriteNum].bpos.z += pSprite->z;
|
|
goto next_sprite;
|
|
|
|
case NUKEBUTTON__STATIC:
|
|
// case NUKEBUTTON+1:
|
|
// case NUKEBUTTON+2:
|
|
// case NUKEBUTTON+3:
|
|
|
|
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;
|
|
}
|
|
|
|
pSprite->pos = sprite[pSprite->owner].pos;
|
|
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 (int j = pData[0]; j > 0; j--)
|
|
A_SetSprite(spriteNum, CLIPMASK0);
|
|
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(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 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 FIRELASER__STATIC:
|
|
if (pSprite->extra != 5)
|
|
pSprite->extra = 5;
|
|
else DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
break;
|
|
case TONGUE__STATIC:
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
case MONEY__STATIC:
|
|
case MAIL__STATIC:
|
|
case PAPER__STATIC:
|
|
{
|
|
pSprite->xvel = (krand()&7)+(sintable[T1(spriteNum)&2047]>>9);
|
|
T1(spriteNum) += (krand()&63);
|
|
if ((T1(spriteNum)&2047) > 512 && (T1(spriteNum)&2047) < 1536)
|
|
{
|
|
if (sector[sectNum].lotag == ST_2_UNDERWATER)
|
|
{
|
|
if (pSprite->zvel < 64)
|
|
pSprite->zvel += (g_spriteGravity>>5)+(krand()&7);
|
|
}
|
|
else if (pSprite->zvel < 144)
|
|
pSprite->zvel += (g_spriteGravity>>5)+(krand()&7);
|
|
}
|
|
|
|
A_SetSprite(spriteNum, CLIPMASK0);
|
|
|
|
if ((krand()&3) == 0)
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case JIBS1__STATIC:
|
|
case JIBS2__STATIC:
|
|
case JIBS3__STATIC:
|
|
case JIBS4__STATIC:
|
|
case JIBS5__STATIC:
|
|
case JIBS6__STATIC:
|
|
case HEADJIB1__STATIC:
|
|
case ARMJIB1__STATIC:
|
|
case LEGJIB1__STATIC:
|
|
case LIZMANHEAD1__STATIC:
|
|
case LIZMANARM1__STATIC:
|
|
case LIZMANLEG1__STATIC:
|
|
case DUKETORSO__STATIC:
|
|
case DUKEGUN__STATIC:
|
|
case DUKELEG__STATIC:
|
|
{
|
|
pSprite->xvel = (pSprite->xvel > 0) ? pSprite->xvel - 1 : 0;
|
|
|
|
if (++pData[5] == (30*10))
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
|
|
if (pSprite->zvel > 1024 && pSprite->zvel < 1280)
|
|
{
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
sectNum = pSprite->sectnum;
|
|
}
|
|
|
|
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;
|
|
}
|
|
else
|
|
{
|
|
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;
|
|
}
|
|
}
|
|
goto next_sprite;
|
|
}
|
|
|
|
case BLOODPOOL__STATIC:
|
|
case PUKE__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 - 1;
|
|
|
|
auto 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 += krand()&3;
|
|
pSprite->yrepeat += krand()&3;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (pSprite->xrepeat < 32 && pSprite->yrepeat < 32)
|
|
{
|
|
pSprite->xrepeat += krand()&3;
|
|
pSprite->yrepeat += krand()&3;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (playerDist < 844 && pSprite->xrepeat > 6 && pSprite->yrepeat > 6)
|
|
{
|
|
if (pSprite->pal == 0 && pSprite->picnum != PUKE && (krand()&255) < 16)
|
|
{
|
|
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;
|
|
goto next_sprite;
|
|
}
|
|
|
|
case SHELL__STATIC:
|
|
case SHOTGUNSHELL__STATIC:
|
|
|
|
A_SetSprite(spriteNum,CLIPMASK0);
|
|
|
|
if (sectNum < 0 || (sector[sectNum].floorz + 256) < 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 KILLIT(i);
|
|
}
|
|
|
|
goto next_sprite;
|
|
|
|
case GLASSPIECES__STATIC:
|
|
// 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)-(krand()&511);
|
|
if (sector[sectNum].lotag == ST_2_UNDERWATER)
|
|
pSprite->zvel >>= 1;
|
|
pSprite->xrepeat >>= 1;
|
|
pSprite->yrepeat >>= 1;
|
|
if (rnd(96))
|
|
setsprite(spriteNum,&pSprite->pos);
|
|
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;
|
|
#endif
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
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,&pSprite->pos);
|
|
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,&pSprite->pos);
|
|
A_GetZLimits(j);
|
|
sprite[j].hitag = sprite[j].lotag = 0;
|
|
}
|
|
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
goto next_sprite;
|
|
}
|
|
#endif
|
|
|
|
next_sprite:
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
|
|
|
|
// i: SE spritenum
|
|
static void HandleSE31(int spriteNum, int setFloorZ, int spriteZ, int SEdir, int zDifference)
|
|
{
|
|
auto const pSprite = &sprite[spriteNum];
|
|
auto const pSector = §or[sprite[spriteNum].sectnum];
|
|
auto const pData = actor[spriteNum].t_data;
|
|
|
|
if (klabs(pSector->floorz - spriteZ) < SP(spriteNum))
|
|
{
|
|
if (setFloorZ)
|
|
pSector->floorz = spriteZ;
|
|
|
|
pData[2] = SEdir;
|
|
pData[0] = 0;
|
|
pData[3] = pSprite->hitag;
|
|
|
|
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, j))
|
|
{
|
|
if (sprite[j].zvel == 0 && sprite[j].statnum != STAT_EFFECTOR && sprite[j].statnum != STAT_PROJECTILE)
|
|
{
|
|
actor[j].bpos.z = sprite[j].z;
|
|
actor[j].floorz = pSector->floorz;
|
|
}
|
|
}
|
|
|
|
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 && sprite[j].statnum != STAT_PROJECTILE)
|
|
{
|
|
actor[j].bpos.z = sprite[j].z;
|
|
sprite[j].z += zChange;
|
|
actor[j].floorz = pSector->floorz;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// s: SE sprite
|
|
static void MaybeTrainKillPlayer(const spritetype *pSprite, int const setOPos)
|
|
{
|
|
for (bssize_t TRAVERSE_CONNECT(playerNum))
|
|
{
|
|
auto 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 (pPlayer->cursectnum != pSprite->sectnum && (playerSectnum == -1 || playerSectnum == pSprite->sectnum))
|
|
{
|
|
pPlayer->pos.vec2 = pSprite->pos.vec2;
|
|
|
|
if (setOPos)
|
|
pPlayer->opos.vec2 = pPlayer->pos.vec2;
|
|
|
|
pPlayer->cursectnum = pSprite->sectnum;
|
|
|
|
setsprite(pPlayer->i, (vec3_t const *)pSprite);
|
|
P_QuickKill(pPlayer);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// i: SE spritenum
|
|
static void MaybeTrainKillEnemies(int const spriteNum)
|
|
{
|
|
int findSprite = headspritesect[sprite[OW(spriteNum)].sectnum];
|
|
|
|
do
|
|
{
|
|
int const nextSprite = nextspritesect[findSprite];
|
|
|
|
if (sprite[findSprite].extra >= 0 && sprite[findSprite].statnum == STAT_ACTOR && A_CheckEnemySprite(&sprite[findSprite]))
|
|
{
|
|
int16_t sectNum = sprite[findSprite].sectnum;
|
|
|
|
updatesector(sprite[findSprite].x,sprite[findSprite].y,§Num);
|
|
|
|
if (sectNum == sprite[spriteNum].sectnum || sectNum == -1)
|
|
{
|
|
actor[findSprite].picnum = RADIUSEXPLOSION;
|
|
actor[findSprite].extra = g_impactDamage << 10;
|
|
actor[findSprite].owner = spriteNum;
|
|
}
|
|
}
|
|
|
|
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];
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (!FURY)
|
|
{
|
|
for (native_t TRAVERSE_CONNECT(playerNum))
|
|
{
|
|
vec2_t & fric = g_player[playerNum].ps->fric;
|
|
fric.x = fric.y = 0;
|
|
}
|
|
}
|
|
#endif
|
|
while (spriteNum >= 0)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
auto const pSprite = &sprite[spriteNum];
|
|
int32_t playerDist;
|
|
int playerNum = A_FindPlayer(pSprite, &playerDist);
|
|
auto const pPlayer = g_player[playerNum].ps;
|
|
|
|
if (VM_OnEvent(EVENT_MOVEEFFECTORS, spriteNum, playerNum, playerDist, 0))
|
|
{
|
|
spriteNum = nextSprite;
|
|
continue;
|
|
}
|
|
|
|
sectortype *const pSector = §or[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))
|
|
{
|
|
auto 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(sprite[j].pos.vec2,pPlayer->pos.vec2,(q*l),&r);
|
|
|
|
pPlayer->bobpos.x += r.x-pPlayer->pos.x;
|
|
pPlayer->bobpos.y += r.y-pPlayer->pos.y;
|
|
|
|
pPlayer->pos.vec2 = r;
|
|
|
|
if (sprite[pPlayer->i].extra <= 0)
|
|
sprite[pPlayer->i].pos.vec2 = r;
|
|
}
|
|
}
|
|
|
|
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, p))
|
|
{
|
|
// KEEPINSYNC1
|
|
if (sprite[p].statnum != STAT_EFFECTOR && sprite[p].statnum != STAT_PROJECTILE)
|
|
if (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;
|
|
|
|
// interpolation fix
|
|
actor[p].bpos.vec2 = sprite[p].pos.vec2;
|
|
|
|
if (move_rotfixed_sprite(p, j, pData[2]))
|
|
rotatepoint(sprite[j].pos.vec2, sprite[p].pos.vec2, (q * l), &sprite[p].pos.vec2);
|
|
}
|
|
}
|
|
|
|
}
|
|
else if (l==0 && (pSector->floorstat&64))
|
|
{
|
|
// fix for jittering of sprites in halted rotating sectors
|
|
for (bssize_t SPRITES_OF_SECT(pSprite->sectnum, p))
|
|
{
|
|
// KEEPINSYNC1
|
|
if (sprite[p].statnum != STAT_EFFECTOR && sprite[p].statnum != STAT_PROJECTILE)
|
|
if (sprite[p].picnum != LASERLINE)
|
|
{
|
|
if (sprite[p].picnum == APLAYER && sprite[p].owner >= 0)
|
|
continue;
|
|
|
|
actor[p].bpos.vec2 = sprite[p].pos.vec2;
|
|
}
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
else pSprite->xvel = k;
|
|
|
|
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 ((pSector->floorstat&1) == 0 && (pSector->ceilingstat&1) == 0)
|
|
{
|
|
if (!S_CheckSoundPlaying(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 && (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))
|
|
{
|
|
auto 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 == pPlayer->cursectnum
|
|
#ifdef YAX_ENABLE
|
|
|| (pData[9]>=0 && pData[9] == pPlayer->cursectnum)
|
|
#endif
|
|
)
|
|
{
|
|
rotatepoint(pSprite->pos.vec2, pPlayer->pos.vec2, q, &pPlayer->pos.vec2);
|
|
|
|
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)
|
|
{
|
|
// fix interpolation
|
|
if (numplayers < 2 && !g_netServer)
|
|
actor[j].bpos.vec2 = sprite[j].pos.vec2;
|
|
|
|
if (move_rotfixed_sprite(j, pSprite-sprite, pData[2]))
|
|
rotatepoint(pSprite->pos.vec2, sprite[j].pos.vec2, q, &sprite[j].pos.vec2);
|
|
|
|
sprite[j].x+= m;
|
|
sprite[j].y+= x;
|
|
|
|
sprite[j].ang+=q;
|
|
|
|
if (g_netServer || numplayers > 1)
|
|
actor[j].bpos.vec2 = sprite[j].pos.vec2;
|
|
}
|
|
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,&pSprite->pos);
|
|
|
|
if ((pSector->floorz-pSector->ceilingz) < (108<<8))
|
|
{
|
|
if (ud.noclip == 0 && pSprite->xvel >= 192)
|
|
MaybeTrainKillPlayer(pSprite, 1);
|
|
|
|
MaybeTrainKillEnemies(spriteNum);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// fix for jittering of sprites in halted subways
|
|
for (SPRITES_OF_SECT(pSprite->sectnum, j))
|
|
{
|
|
// KEEPINSYNC2
|
|
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)
|
|
actor[j].bpos.vec2 = sprite[j].pos.vec2;
|
|
}
|
|
}
|
|
|
|
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.vec2 = sprite[j].pos.vec2;
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
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))
|
|
{
|
|
auto const pPlayer = g_player[playerNum].ps;
|
|
|
|
if (pPlayer->cursectnum == pSprite->sectnum)
|
|
{
|
|
pPlayer->pos.x += l;
|
|
pPlayer->pos.y += x;
|
|
|
|
if (g_netServer || numplayers > 1)
|
|
pPlayer->opos.vec2 = pPlayer->pos.vec2;
|
|
|
|
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.vec2 = sprite[j].pos.vec2;
|
|
|
|
sprite[j].x += l;
|
|
sprite[j].y += x;
|
|
|
|
if (g_netServer || numplayers > 1)
|
|
actor[j].bpos.vec2 = sprite[j].pos.vec2;
|
|
}
|
|
}
|
|
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum,&pSprite->pos);
|
|
|
|
if (pSector->floorz-pSector->ceilingz < (108<<8))
|
|
{
|
|
if (ud.noclip == 0)
|
|
MaybeTrainKillPlayer(pSprite, 1);
|
|
|
|
MaybeTrainKillEnemies(spriteNum);
|
|
}
|
|
}
|
|
|
|
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))
|
|
{
|
|
auto 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,&sprite[sectSprite].pos);
|
|
}
|
|
}
|
|
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum,&pSprite->pos);
|
|
}
|
|
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 && A_CheckSpriteFlags(sectSprite,SFLAG_NOSHADE) == 0)
|
|
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, &pSprite->pos);
|
|
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 (int 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];
|
|
}
|
|
}
|
|
}
|
|
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 == pPlayer->cursectnum)
|
|
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 (GetAnimationGoal(§or[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;
|
|
|
|
for (j=pSector->wallptr; j<endWall; j++)
|
|
{
|
|
for (SPRITES_OF(STAT_ACTOR, k))
|
|
{
|
|
if (sprite[k].extra > 0 &&
|
|
(pSprite->sectnum == sprite[k].sectnum ||
|
|
sectoradjacent(pSprite->sectnum, sprite[k].sectnum)) &&
|
|
A_CheckEnemySprite(&sprite[k]) &&
|
|
clipinsidebox(&sprite[k].pos.vec2, j, 256) == 1)
|
|
goto next_sprite;
|
|
}
|
|
}
|
|
|
|
l = (SP(spriteNum) >> 3) * pData[3];
|
|
pData[2] += l;
|
|
pData[4] += l;
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
|
|
for (j=pSector->wallptr; j<endWall; j++)
|
|
{
|
|
for (SPRITES_OF(STAT_PLAYER, k))
|
|
{
|
|
if (sprite[k].owner >= 0 && clipinsidebox(&sprite[k].pos.vec2, j, pPlayer->clipdist))
|
|
{
|
|
pData[5] = 8; // Delay
|
|
pData[2] -= l;
|
|
pData[4] -= l;
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
goto next_sprite;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pData[4] <= -511 || pData[4] >= 512)
|
|
{
|
|
pData[4] = 0;
|
|
pData[2] &= 0xffffff00;
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum, &pSprite->pos);
|
|
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) && (A_CheckSpriteFlags(j, SFLAG_NOSHADE) == 0))
|
|
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) && (A_CheckSpriteFlags(j, SFLAG_NOSHADE) == 0))
|
|
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->parallax_sectnum >= 0)
|
|
{
|
|
pSector->ceilingpicnum = sector[g_player[0].ps->parallax_sectnum].ceilingpicnum;
|
|
pSector->ceilingshade = sector[g_player[0].ps->parallax_sectnum].ceilingshade;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (++pData[2] > 256)
|
|
DELETE_SPRITE_AND_CONTINUE(spriteNum);
|
|
}
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
if (pData[2] == 4 && pSprite->ang != 512)
|
|
for (x=0; x<7; x++) RANDOMSCRAP(pSprite, spriteNum);
|
|
#endif
|
|
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,&pSprite->pos);
|
|
}
|
|
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,&pSprite->pos);
|
|
|
|
break;
|
|
|
|
case SE_17_WARP_ELEVATOR:
|
|
{
|
|
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);
|
|
auto 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;
|
|
|
|
int32_t nextk;
|
|
|
|
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);
|
|
auto const pPlayer = g_player[warpPlayer].ps;
|
|
|
|
pPlayer->pos.x += sprite[j].x - pSprite->x;
|
|
pPlayer->pos.y += sprite[j].y - pSprite->y;
|
|
pPlayer->opos.z -= pPlayer->pos.z;
|
|
pPlayer->pos.z = sector[sprite[j].sectnum].floorz - (pSector->floorz - pPlayer->pos.z);
|
|
pPlayer->opos.z += pPlayer->pos.z;
|
|
|
|
actor[k].floorz = sector[sprite[j].sectnum].floorz;
|
|
actor[k].ceilingz = sector[sprite[j].sectnum].ceilingz;
|
|
pPlayer->opos.vec2 = pPlayer->pos.vec2;
|
|
pPlayer->bobpos = pPlayer->pos.vec2;
|
|
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;
|
|
|
|
actor[k].bpos.vec2 = sprite[k].pos.vec2;
|
|
|
|
actor[k].bpos.z -= sprite[k].z;
|
|
sprite[k].z = sector[sprite[j].sectnum].floorz - (pSector->floorz - sprite[k].z);
|
|
actor[k].bpos.z += sprite[k].z;
|
|
|
|
changespritesect(k,sprite[j].sectnum);
|
|
setsprite(k,&sprite[k].pos);
|
|
|
|
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;
|
|
|
|
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;
|
|
|
|
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 = §or[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, &sprite[sectSprite].pos);
|
|
|
|
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))
|
|
{
|
|
auto 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, &pPlayer->pos);
|
|
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(§or[pData[0]].ceilingz) >= 0)
|
|
pSector->ceilingz += pSector->extra*9;
|
|
else pData[1] = 0;
|
|
}
|
|
break;
|
|
|
|
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 BLOODPOOL__STATIC:
|
|
case PUKE__STATIC:
|
|
case FOOTPRINTS__STATIC:
|
|
case FOOTPRINTS2__STATIC:
|
|
case FOOTPRINTS3__STATIC:
|
|
case FOOTPRINTS4__STATIC:
|
|
case BULLETHOLE__STATIC:
|
|
case BLOODSPLAT1__STATIC:
|
|
case BLOODSPLAT2__STATIC:
|
|
case BLOODSPLAT3__STATIC:
|
|
case BLOODSPLAT4__STATIC: sprite[sectSprite].xrepeat = sprite[sectSprite].yrepeat = 0; continue;
|
|
|
|
case LASERLINE__STATIC: continue;
|
|
}
|
|
fallthrough__;
|
|
case STAT_STANDABLE:
|
|
if (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
|
|
|| 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.vec2 = sprite[sectSprite].pos.vec2;
|
|
|
|
sprite[sectSprite].x += vect.x >> 2;
|
|
sprite[sectSprite].y += vect.y >> 2;
|
|
|
|
setsprite(sectSprite, &sprite[sectSprite].pos);
|
|
|
|
if (sector[sprite[sectSprite].sectnum].floorstat & 2)
|
|
if (sprite[sectSprite].statnum == STAT_ZOMBIEACTOR)
|
|
A_Fall(sectSprite);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (bssize_t TRAVERSE_CONNECT(playerNum))
|
|
{
|
|
auto 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;
|
|
}
|
|
}
|
|
}
|
|
pSector->floorxpanning += SP(spriteNum)>>7;
|
|
|
|
break;
|
|
}
|
|
|
|
case SE_35:
|
|
if (pSector->ceilingz > pSprite->z)
|
|
{
|
|
for (j = 0; j < 8; j++)
|
|
{
|
|
pSprite->ang += krand()&511;
|
|
k = A_Spawn(spriteNum, SMALLSMOKE);
|
|
sprite[k].xvel = 96+(krand()&127);
|
|
A_SetSprite(k, CLIPMASK0);
|
|
setsprite(k, &sprite[k].pos);
|
|
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[3])
|
|
pSprite->shade = 1;
|
|
|
|
if (pSprite->shade)
|
|
{
|
|
pSector->ceilingz += SP(spriteNum)<<4;
|
|
if (pSector->ceilingz > pSector->floorz)
|
|
pSector->ceilingz = pSector->floorz;
|
|
}
|
|
else
|
|
{
|
|
pSector->ceilingz -= SP(spriteNum)<<4;
|
|
if (pSector->ceilingz < pData[3])
|
|
pSector->ceilingz = pData[3];
|
|
}
|
|
|
|
break;
|
|
|
|
case SE_26:
|
|
{
|
|
int32_t p, nextj;
|
|
|
|
pSprite->xvel = pSector->extra != 0 ? pSector->extra : 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 && sprite[j].statnum != STAT_PROJECTILE)
|
|
{
|
|
actor[j].bpos.vec2 = sprite[j].pos.vec2;
|
|
|
|
sprite[j].x += l;
|
|
sprite[j].y += x;
|
|
sprite[j].z += pSprite->zvel;
|
|
|
|
setsprite(j, &sprite[j].pos);
|
|
}
|
|
}
|
|
|
|
for (TRAVERSE_CONNECT(p))
|
|
{
|
|
auto const pPlayer = g_player[p].ps;
|
|
|
|
if (pSprite->sectnum == pPlayer->cursectnum && pPlayer->on_ground)
|
|
{
|
|
pPlayer->pos.x += l;
|
|
pPlayer->pos.y += x;
|
|
pPlayer->pos.z += pSprite->zvel;
|
|
|
|
updatesector(pPlayer->pos.x, pPlayer->pos.y, &pPlayer->cursectnum);
|
|
changespritesect(pPlayer->i, pPlayer->cursectnum);
|
|
|
|
pPlayer->bobpos.x += l;
|
|
pPlayer->bobpos.y += x;
|
|
|
|
if (g_netServer || numplayers > 1)
|
|
pPlayer->opos.vec2 = pPlayer->pos.vec2;
|
|
|
|
if (sprite[pPlayer->i].extra <= 0)
|
|
sprite[pPlayer->i].pos.vec2 = pPlayer->pos.vec2;
|
|
}
|
|
}
|
|
|
|
A_MoveSector(spriteNum);
|
|
setsprite(spriteNum,&pSprite->pos);
|
|
|
|
break;
|
|
}
|
|
|
|
case SE_27_DEMO_CAM:
|
|
{
|
|
if (pSprite->extra < 1 && (ud.recstat == 0 || !cl_democams)) break;
|
|
|
|
if (klabs(pSprite->extra) == 2)
|
|
{
|
|
actor[spriteNum].tempang = pSprite->ang;
|
|
if (ud.camerasprite != spriteNum)
|
|
{
|
|
//level the camera out by default (yvel stores the up/down angle)
|
|
pSprite->yvel = 100;
|
|
ud.camerasprite = spriteNum;
|
|
}
|
|
|
|
findCameraDestination:
|
|
if (pSprite->owner == spriteNum)
|
|
{
|
|
pSprite->owner = A_FindLocatorWithHiLoTags(pSprite->hitag, pData[0], -1);
|
|
|
|
//reset our elapsed time since reaching a locator
|
|
pData[1] = 0;
|
|
//store our starting point
|
|
pData[2] = pSprite->x;
|
|
pData[3] = pSprite->y;
|
|
pData[4] = pSprite->z;
|
|
pData[5] = pSprite->ang;
|
|
pData[6] = pSprite->yvel;
|
|
if (pSprite->owner != -1)
|
|
{
|
|
spritetype* const destLocator = &sprite[pSprite->owner];
|
|
int32_t subjectLocatorIndex = A_FindLocatorWithHiLoTags(pSprite->hitag, destLocator->owner, -1);
|
|
pData[7] = G_GetAngleDelta(pData[5], destLocator->ang);
|
|
//level the camera out by default (pData[8] stores our destination up/down angle)
|
|
pData[8] = 100;
|
|
if (subjectLocatorIndex != -1)
|
|
{
|
|
spritetype* const subjectLocator = &sprite[subjectLocatorIndex];
|
|
const vec3_t cameraDirection = {subjectLocator->x - destLocator->x,
|
|
subjectLocator->y - destLocator->y,
|
|
subjectLocator->z - destLocator->z};
|
|
pData[7] = G_GetAngleDelta(pData[5], getangle(cameraDirection.x,
|
|
cameraDirection.y));
|
|
pData[8] = (((int32_t) getangle(-ksqrt(cameraDirection.x*cameraDirection.x+cameraDirection.y*cameraDirection.y), cameraDirection.z)*(400.f/1024.f)))-300;
|
|
}
|
|
}
|
|
}
|
|
if (pSprite->owner == -1)
|
|
{
|
|
break;
|
|
}
|
|
|
|
spritetype* const destLocator = &sprite[pSprite->owner];
|
|
if (pData[1] == destLocator->extra)
|
|
{
|
|
pSprite->owner = spriteNum;
|
|
++pData[0];
|
|
goto findCameraDestination;
|
|
}
|
|
|
|
//smoothstep to the new location and camera direction over the duration (in ticks) stored in the destLocator's extra value
|
|
const vec3_t heading = {destLocator->x-pData[2],
|
|
destLocator->y-pData[3],
|
|
destLocator->z-pData[4]};
|
|
float interpolation = (pData[1]/(float) destLocator->extra);
|
|
interpolation = interpolation*interpolation*(3-2*interpolation);
|
|
pSprite->x = pData[2]+interpolation*heading.x;
|
|
pSprite->y = pData[3]+interpolation*heading.y;
|
|
pSprite->z = pData[4]+interpolation*heading.z;
|
|
pSprite->ang = pData[5]+interpolation*pData[7];
|
|
pSprite->yvel = (pData[6]+((int32_t) interpolation*pData[8])+100)%400-100;
|
|
|
|
//increment elapsed time
|
|
++pData[1];
|
|
}
|
|
else
|
|
{
|
|
actor[spriteNum].tempang = pSprite->ang;
|
|
|
|
int const p = A_FindPlayer(pSprite, &x);
|
|
auto 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 (pData[5] > 0)
|
|
{
|
|
pData[5]--;
|
|
break;
|
|
}
|
|
|
|
if (T1(spriteNum) == 0)
|
|
{
|
|
A_FindPlayer(pSprite,&x);
|
|
if (x > 15500)
|
|
break;
|
|
T1(spriteNum) = 1;
|
|
T2(spriteNum) = 64 + (krand()&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+(krand()&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 (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;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
case SE_33_QUAKE_DEBRIS:
|
|
if (g_earthquakeTime > 0 && (krand()&7) == 0)
|
|
RANDOMSCRAP(pSprite, spriteNum);
|
|
break;
|
|
#endif
|
|
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+(krand()&7);
|
|
sprite[k].z = pSector->floorz-(krand()%x);
|
|
sprite[k].ang += 256-(krand()%511);
|
|
sprite[k].xvel = krand()&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+(krand()&3);
|
|
sprite[k].z = pSector->floorz-(krand()%x);
|
|
sprite[k].ang += 256-(krand()%511);
|
|
sprite[k].xvel = krand()&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))
|
|
{
|
|
auto const s = &sprite[spriteNum];
|
|
|
|
if (s->lotag == SE_29_WAVES)
|
|
{
|
|
auto const sc = (usectorptr_t)§or[s->sectnum];
|
|
|
|
if (sc->wallnum == 4)
|
|
{
|
|
auto 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)
|
|
{
|
|
auto 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;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
for (int 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, §num);
|
|
|
|
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
|
|
}
|
|
}
|
|
#endif // POLYMER
|
|
|
|
void A_PlayAlertSound(int spriteNum)
|
|
{
|
|
if (sprite[spriteNum].extra > 0)
|
|
{
|
|
if ((VM_OnEventWithReturn(EVENT_RECOGSOUND, spriteNum, screenpeek, 0)) != 0)
|
|
return;
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
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:
|
|
case BOSS1STAYPUT__STATIC: S_PlaySound(BOS1_RECOG); break;
|
|
case BOSS2__STATIC: S_PlaySound((sprite[spriteNum].pal != 0) ? BOS2_RECOG : WHIPYOURASS); break;
|
|
case BOSS3__STATIC: S_PlaySound((sprite[spriteNum].pal != 0) ? BOS3_RECOG : RIPHEADNECK); break;
|
|
case BOSS4__STATIC:
|
|
case BOSS4STAYPUT__STATIC: S_PlaySound((sprite[spriteNum].pal != 0) ? BOS4_RECOG : BOSS4_FIRSTSEE); break;
|
|
case GREENSLIME__STATIC: A_PlaySound(SLIM_RECOG, spriteNum); break;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
int A_CheckSwitchTile(int spriteNum)
|
|
{
|
|
UNREFERENCED_PARAMETER(spriteNum);
|
|
|
|
#ifndef EDUKE32_STANDALONE
|
|
// 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;
|
|
|
|
// Loop to catch both states of switches.
|
|
for (bssize_t j=1; j>=0; j--)
|
|
{
|
|
switch (DYNAMICTILEMAP(PN(spriteNum)-j))
|
|
{
|
|
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;
|
|
}
|
|
}
|
|
#endif
|
|
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
|
|
}
|
|
|
|
static void G_DoEventGame(int const nEventID)
|
|
{
|
|
if (VM_HaveEvent(nEventID))
|
|
{
|
|
int statNum = 0;
|
|
|
|
do
|
|
{
|
|
int spriteNum = headspritestat[statNum++];
|
|
|
|
while (spriteNum >= 0)
|
|
{
|
|
int const nextSprite = nextspritestat[spriteNum];
|
|
|
|
if (A_CheckSpriteFlags(spriteNum, SFLAG_NOEVENTCODE))
|
|
{
|
|
spriteNum = nextSprite;
|
|
continue;
|
|
}
|
|
|
|
int32_t playerDist;
|
|
int const playerNum = A_FindPlayer(&sprite[spriteNum], &playerDist);
|
|
VM_ExecuteEvent(nEventID, spriteNum, playerNum, playerDist);
|
|
|
|
spriteNum = nextSprite;
|
|
}
|
|
}
|
|
while (statNum < MAXSTATUS);
|
|
}
|
|
}
|
|
|
|
void G_MoveWorld(void)
|
|
{
|
|
extern double g_moveActorsTime, g_moveWorldTime;
|
|
const double worldTime = timerGetHiTicks();
|
|
|
|
VM_OnEvent(EVENT_PREWORLD);
|
|
|
|
G_DoEventGame(EVENT_PREGAME);
|
|
|
|
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
|
|
|
|
|
|
VM_OnEvent(EVENT_WORLD);
|
|
|
|
G_DoEventGame(EVENT_GAME);
|
|
|
|
G_RefreshLights();
|
|
G_DoSectorAnimations();
|
|
G_MoveFX(); //ST 11
|
|
|
|
g_moveWorldTime = (1-0.033)*g_moveWorldTime + 0.033*(timerGetHiTicks()-worldTime);
|
|
}
|
|
|
|
END_DUKE_NS
|