raze/source/games/blood/src/animatesprite.cpp

1000 lines
36 KiB
C++
Raw Normal View History

2020-08-02 17:59:14 +00:00
//-------------------------------------------------------------------------
/*
Copyright (C) 2010-2019 EDuke32 developers and contributors
Copyright (C) 2019 Nuke.YKT
This file is part of NBlood.
NBlood 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!
#include <stdlib.h>
#include <string.h>
#include "build.h"
#include "v_font.h"
#include "blood.h"
2020-08-02 17:59:14 +00:00
#include "zstring.h"
#include "razemenu.h"
2020-08-02 17:59:14 +00:00
#include "gstrings.h"
#include "v_2ddrawer.h"
#include "v_video.h"
#include "v_font.h"
#include "hw_voxels.h"
2021-04-13 18:33:21 +00:00
#include "gamefuncs.h"
2020-08-02 17:59:14 +00:00
#include "glbackend/glbackend.h"
BEGIN_BLD_NS
static fixed_t gCameraAng;
2020-08-02 17:59:14 +00:00
int dword_172CE0[16][3];
static void RotateYZ(int *, int *pY, int *pZ, int ang)
{
int oY, oZ, angSin, angCos;
oY = *pY;
oZ = *pZ;
angSin = Sin(ang);
angCos = Cos(ang);
*pY = dmulscale30r(oY,angCos,oZ,-angSin);
*pZ = dmulscale30r(oY,angSin,oZ,angCos);
}
static void RotateXZ(int *pX, int *, int *pZ, int ang)
{
int oX, oZ, angSin, angCos;
oX = *pX;
oZ = *pZ;
angSin = Sin(ang);
angCos = Cos(ang);
*pX = dmulscale30r(oX,angCos,oZ,-angSin);
*pZ = dmulscale30r(oX,angSin,oZ,angCos);
}
tspritetype* viewInsertTSprite(tspritetype* tsprite, int& spritesortcnt, sectortype* pSector, int nStatnum, tspritetype const * const parentTSprite)
2020-08-02 17:59:14 +00:00
{
if (spritesortcnt >= MAXSPRITESONSCREEN)
return nullptr;
2020-08-02 17:59:14 +00:00
int nTSprite = spritesortcnt;
tspritetype *pTSprite = &tsprite[nTSprite];
memset(pTSprite, 0, sizeof(tspritetype));
pTSprite->cstat = CSTAT_SPRITE_YCENTER;
2020-08-02 17:59:14 +00:00
pTSprite->xrepeat = 64;
pTSprite->yrepeat = 64;
pTSprite->ownerActor = nullptr;
2020-08-02 17:59:14 +00:00
pTSprite->type = -spritesortcnt;
pTSprite->statnum = nStatnum;
2021-11-23 23:28:50 +00:00
pTSprite->setsector(pSector);
2020-08-02 17:59:14 +00:00
spritesortcnt++;
2021-11-19 20:11:41 +00:00
if (parentTSprite)
2020-08-02 17:59:14 +00:00
{
pTSprite->pos = parentTSprite->pos;
pTSprite->ownerActor = parentTSprite->ownerActor;
2021-11-19 20:11:41 +00:00
pTSprite->ang = parentTSprite->ang;
2020-08-02 17:59:14 +00:00
}
pTSprite->pos.X += Cos(gCameraAng)>>25;
pTSprite->pos.Y += Sin(gCameraAng)>>25;
2020-08-02 17:59:14 +00:00
return pTSprite;
}
static const int effectDetail[kViewEffectMax] = {
2020-08-02 17:59:14 +00:00
4, 4, 4, 4, 0, 0, 0, 0, 0, 1, 4, 4, 0, 0, 0, 1, 0, 0, 0
};
struct WEAPONICON {
int16_t nTile;
uint8_t zOffset;
2020-08-02 17:59:14 +00:00
};
static const WEAPONICON gWeaponIcon[] = {
{ -1, 0 },
{ -1, 0 }, // 1: pitchfork
{ 524, 6 }, // 2: flare gun
{ 559, 6 }, // 3: shotgun
{ 558, 8 }, // 4: tommy gun
{ 526, 6 }, // 5: napalm launcher
{ 589, 11 }, // 6: dynamite
{ 618, 11 }, // 7: spray can
{ 539, 6 }, // 8: tesla gun
{ 800, 0 }, // 9: life leech
{ 525, 11 }, // 10: voodoo doll
{ 811, 11 }, // 11: proxy bomb
{ 810, 11 }, // 12: remote bomb
{ -1, 0 },
};
static tspritetype *viewAddEffect(tspritetype* tsprite, int& spritesortcnt, int nTSprite, VIEW_EFFECT nViewEffect)
2020-08-02 17:59:14 +00:00
{
assert(nViewEffect >= 0 && nViewEffect < kViewEffectMax);
2020-08-02 17:59:14 +00:00
auto pTSprite = &tsprite[nTSprite];
auto owneractor = static_cast<DBloodActor*>(pTSprite->ownerActor);
if (gDetail < effectDetail[nViewEffect] || nTSprite >= MAXSPRITESONSCREEN) return NULL;
2020-08-02 17:59:14 +00:00
switch (nViewEffect)
{
case kViewEffectSpotProgress: {
2021-09-05 07:30:46 +00:00
XSPRITE* pXSprite = &owneractor->x();
int perc = (100 * pXSprite->data3) / kMaxPatrolSpotValue;
int width = (94 * pXSprite->data3) / kMaxPatrolSpotValue;
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
2021-11-23 23:28:50 +00:00
auto pNSprite2 = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite2)
break;
pNSprite2->picnum = 2203;
pNSprite2->xrepeat = width;
pNSprite2->yrepeat = 20;
pNSprite2->pal = 10;
if (perc >= 75) pNSprite2->pal = 0;
else if (perc >= 50) pNSprite2->pal = 6;
pNSprite2->pos.Z = top - 2048;
pNSprite2->shade = -128;
break;
}
case kViewEffectAtom:
2020-08-02 17:59:14 +00:00
for (int i = 0; i < 16; i++)
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
int ang = (PlayClock*2048)/120;
2020-08-02 17:59:14 +00:00
int nRand1 = dword_172CE0[i][0];
int nRand2 = dword_172CE0[i][1];
int nRand3 = dword_172CE0[i][2];
ang += nRand3;
int x = MulScale(512, Cos(ang), 30);
int y = MulScale(512, Sin(ang), 30);
2020-08-02 17:59:14 +00:00
int z = 0;
RotateYZ(&x, &y, &z, nRand1);
RotateXZ(&x, &y, &z, nRand2);
pNSprite->pos.X = pTSprite->pos.X + x;
pNSprite->pos.Y = pTSprite->pos.Y + y;
pNSprite->pos.Z = pTSprite->pos.Z + (z<<4);
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 1720;
pNSprite->shade = -128;
}
break;
case kViewEffectFlag:
case kViewEffectBigFlag:
2020-08-02 17:59:14 +00:00
{
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
pNSprite->shade = -128;
pNSprite->pal = 0;
pNSprite->pos.Z = top;
if (nViewEffect == kViewEffectFlag)
2020-08-02 17:59:14 +00:00
pNSprite->xrepeat = pNSprite->yrepeat = 24;
else
pNSprite->xrepeat = pNSprite->yrepeat = 64;
pNSprite->picnum = 3558;
return pNSprite;
}
case kViewEffectTesla:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
pNSprite->pos.Z = pTSprite->pos.Z;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->shade = -128;
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat;
pNSprite->picnum = 2135;
break;
}
case kViewEffectShoot:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
pNSprite->shade = -128;
pNSprite->pal = 0;
pNSprite->xrepeat = pNSprite->yrepeat = 64;
pNSprite->picnum = 2605;
return pNSprite;
}
case kViewEffectReflectiveBall:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
pNSprite->shade = 26;
pNSprite->pal = 0;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->xrepeat = pNSprite->yrepeat = 64;
pNSprite->picnum = 2089;
break;
}
case kViewEffectPhase:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
pNSprite->shade = 26;
pNSprite->pal = 0;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->xrepeat = pNSprite->yrepeat = 24;
pNSprite->picnum = 626;
pNSprite->pos.Z = top;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectTrail:
2020-08-02 17:59:14 +00:00
{
int nAng = pTSprite->ang;
2021-12-18 15:45:05 +00:00
if (pTSprite->cstat & CSTAT_SPRITE_ALIGNMENT_WALL)
2020-08-02 17:59:14 +00:00
{
nAng = (nAng+512)&2047;
}
else
{
nAng = (nAng+1024)&2047;
}
for (int i = 0; i < 5 && spritesortcnt < MAXSPRITESONSCREEN; i++)
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pSector = pTSprite->sector();
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pSector, 32767, NULL);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int nLen = 128+(i<<7);
int x = MulScale(nLen, Cos(nAng), 30);
pNSprite->pos.X = pTSprite->pos.X + x;
int y = MulScale(nLen, Sin(nAng), 30);
pNSprite->pos.Y = pTSprite->pos.Y + y;
pNSprite->pos.Z = pTSprite->pos.Z;
2021-11-23 23:28:50 +00:00
assert(pSector);
FindSector(pNSprite->pos.X, pNSprite->pos.Y, pNSprite->pos.Z, &pSector);
2021-11-23 23:28:50 +00:00
pNSprite->setsector(pSector);
pNSprite->ownerActor = pTSprite->ownerActor;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = pTSprite->picnum;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
if (i < 2)
2021-12-18 18:34:18 +00:00
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_TRANS_FLIP;
2020-08-02 17:59:14 +00:00
pNSprite->shade = ClipLow(pTSprite->shade-16, -128);
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat;
pNSprite->picnum = pTSprite->picnum;
}
break;
}
case kViewEffectFlame:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
pNSprite->shade = -128;
pNSprite->pos.Z = pTSprite->pos.Z;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 908;
pNSprite->statnum = kStatDecoration;
pNSprite->xrepeat = pNSprite->yrepeat = (tileWidth(pTSprite->picnum)*pTSprite->xrepeat)/64;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectSmokeHigh:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
pNSprite->pos.Z = top;
2020-08-02 17:59:14 +00:00
if (IsDudeSprite(pTSprite))
pNSprite->picnum = 672;
else
pNSprite->picnum = 754;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->shade = 8;
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat;
break;
}
case kViewEffectSmokeLow:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
pNSprite->pos.Z = bottom;
2020-08-02 17:59:14 +00:00
if (pTSprite->type >= kDudeBase && pTSprite->type < kDudeMax)
pNSprite->picnum = 672;
else
pNSprite->picnum = 754;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->shade = 8;
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat;
break;
}
case kViewEffectTorchHigh:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
pNSprite->pos.Z = top;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 2101;
pNSprite->shade = -128;
pNSprite->xrepeat = pNSprite->yrepeat = (tileWidth(pTSprite->picnum)*pTSprite->xrepeat)/32;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectTorchLow:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
pNSprite->pos.Z = bottom;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 2101;
pNSprite->shade = -128;
pNSprite->xrepeat = pNSprite->yrepeat = (tileWidth(pTSprite->picnum)*pTSprite->xrepeat)/32;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectShadow:
2020-08-02 17:59:14 +00:00
{
if (r_shadows)
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
pNSprite->pos.Z = getflorzofslopeptr(pTSprite->sector(), pNSprite->pos.X, pNSprite->pos.Y);
2020-08-02 17:59:14 +00:00
pNSprite->shade = 127;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat >> 2;
pNSprite->picnum = pTSprite->picnum;
pNSprite->pal = 5;
int height = tileHeight(pNSprite->picnum);
2020-08-02 17:59:14 +00:00
int center = height / 2 + tileTopOffset(pNSprite->picnum);
pNSprite->pos.Z -= (pNSprite->yrepeat << 2) * (height - center);
2020-08-02 17:59:14 +00:00
}
break;
}
case kViewEffectFlareHalo:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2020-08-02 17:59:14 +00:00
pNSprite->shade = -128;
pNSprite->pal = 2;
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
pNSprite->pos.Z = pTSprite->pos.Z;
2020-08-02 17:59:14 +00:00
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = pTSprite->yrepeat;
pNSprite->picnum = 2427;
break;
}
case kViewEffectCeilGlow:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2021-11-23 23:28:50 +00:00
sectortype *pSector = pTSprite->sector();
pNSprite->pos.X = pTSprite->pos.X;
pNSprite->pos.Y = pTSprite->pos.Y;
pNSprite->pos.Z = pSector->ceilingz;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 624;
pNSprite->shade = ((pTSprite->pos.Z-pSector->ceilingz)>>8)-64;
2020-08-02 17:59:14 +00:00
pNSprite->pal = 2;
pNSprite->xrepeat = pNSprite->yrepeat = 64;
2021-12-18 18:34:18 +00:00
pNSprite->cstat |= CSTAT_SPRITE_ONE_SIDE | CSTAT_SPRITE_ALIGNMENT_FLOOR | CSTAT_SPRITE_YFLIP | CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->ang = pTSprite->ang;
pNSprite->ownerActor = pTSprite->ownerActor;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectFloorGlow:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
2021-11-23 23:28:50 +00:00
sectortype *pSector = pTSprite->sector();
pNSprite->pos.X = pTSprite->pos.X;
pNSprite->pos.Y = pTSprite->pos.Y;
pNSprite->pos.Z = pSector->floorz;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = 624;
uint8_t nShade = (pSector->floorz-pTSprite->pos.Z)>>8;
2020-08-02 17:59:14 +00:00
pNSprite->shade = nShade-32;
pNSprite->pal = 2;
pNSprite->xrepeat = pNSprite->yrepeat = nShade;
2021-12-18 18:34:18 +00:00
pNSprite->cstat |= CSTAT_SPRITE_ONE_SIDE | CSTAT_SPRITE_ALIGNMENT_FLOOR | CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pNSprite->ang = pTSprite->ang;
pNSprite->ownerActor = pTSprite->ownerActor;
2020-08-02 17:59:14 +00:00
break;
}
case kViewEffectSpear:
2020-08-02 17:59:14 +00:00
{
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
pNSprite->pos.Z = pTSprite->pos.Z;
2020-08-02 17:59:14 +00:00
if (gDetail > 1)
2021-12-18 18:34:18 +00:00
pNSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_TRANS_FLIP;
2020-08-02 17:59:14 +00:00
pNSprite->shade = ClipLow(pTSprite->shade-32, -128);
pNSprite->xrepeat = pTSprite->xrepeat;
pNSprite->yrepeat = 64;
pNSprite->picnum = 775;
break;
}
case kViewEffectShowWeapon:
2020-08-02 17:59:14 +00:00
{
assert(pTSprite->type >= kDudePlayer1 && pTSprite->type <= kDudePlayer8);
2020-08-02 17:59:14 +00:00
PLAYER *pPlayer = &gPlayer[pTSprite->type-kDudePlayer1];
WEAPONICON weaponIcon = gWeaponIcon[pPlayer->curWeapon];
auto& nTile = weaponIcon.nTile;
2020-08-02 17:59:14 +00:00
if (nTile < 0) break;
2021-11-23 23:28:50 +00:00
auto pNSprite = viewInsertTSprite(tsprite, spritesortcnt, pTSprite->sector(), 32767, pTSprite);
if (!pNSprite)
break;
pNSprite->pos.X = pTSprite->pos.X;
pNSprite->pos.Y = pTSprite->pos.Y;
pNSprite->pos.Z = pTSprite->pos.Z-(32<<8);
pNSprite->pos.Z -= weaponIcon.zOffset<<8; // offset up
2020-08-02 17:59:14 +00:00
pNSprite->picnum = nTile;
pNSprite->shade = pTSprite->shade;
pNSprite->xrepeat = 32;
pNSprite->yrepeat = 32;
auto& nVoxel = voxelIndex[nTile];
2020-11-11 12:03:47 +00:00
if (cl_showweapon == 2 && r_voxels && nVoxel != -1)
2020-08-02 17:59:14 +00:00
{
pNSprite->ang = (gView->pSprite->ang + 512) & 2047; // always face viewer
2021-12-18 18:34:18 +00:00
pNSprite->cstat |= CSTAT_SPRITE_ALIGNMENT_SLAB;
2021-12-18 15:45:05 +00:00
pNSprite->cstat &= ~CSTAT_SPRITE_YFLIP;
2020-08-02 17:59:14 +00:00
pNSprite->picnum = nVoxel;
if (pPlayer->curWeapon == kWeapLifeLeech) // position lifeleech behind player
2020-08-02 17:59:14 +00:00
{
pNSprite->pos.X += MulScale(128, Cos(gView->pSprite->ang), 30);
pNSprite->pos.Y += MulScale(128, Sin(gView->pSprite->ang), 30);
2020-08-02 17:59:14 +00:00
}
if ((pPlayer->curWeapon == kWeapLifeLeech) || (pPlayer->curWeapon == kWeapVoodooDoll)) // make lifeleech/voodoo doll always face viewer like sprite
pNSprite->ang = (pNSprite->ang + 512) & 2047; // offset angle 90 degrees
2020-08-02 17:59:14 +00:00
}
break;
}
}
return NULL;
}
static void viewApplyDefaultPal(tspritetype *pTSprite, sectortype const *pSector)
{
XSECTOR const *pXSector = pSector->hasX()? &pSector->xs() : nullptr;
2020-08-02 17:59:14 +00:00
if (pXSector && pXSector->color && (VanillaMode() || pSector->floorpal != 0))
{
2021-04-13 18:33:21 +00:00
copyfloorpal(pTSprite, pSector);
2020-08-02 17:59:14 +00:00
}
}
void viewProcessSprites(tspritetype* tsprite, int& spritesortcnt, int32_t cX, int32_t cY, int32_t cZ, int32_t cA, int32_t smoothratio)
2020-08-02 17:59:14 +00:00
{
// shift before interpolating to increase precision.
int myclock = (PlayClock<<3) + MulScale(4<<3, smoothratio, 16);
assert(spritesortcnt <= MAXSPRITESONSCREEN);
2020-08-02 17:59:14 +00:00
gCameraAng = cA;
int nViewSprites = spritesortcnt;
for (int nTSprite = spritesortcnt-1; nTSprite >= 0; nTSprite--)
{
tspritetype *pTSprite = &tsprite[nTSprite];
auto owneractor = static_cast<DBloodActor*>(pTSprite->ownerActor);
2020-08-02 17:59:14 +00:00
XSPRITE *pTXSprite = NULL;
if (owneractor->spr.detail > gDetail)
2020-08-02 17:59:14 +00:00
{
pTSprite->xrepeat = 0;
continue;
}
2021-09-05 07:30:46 +00:00
if (owneractor->hasX())
2020-08-02 17:59:14 +00:00
{
2021-09-05 07:30:46 +00:00
pTXSprite = &owneractor->x();
2020-08-02 17:59:14 +00:00
}
int nTile = pTSprite->picnum;
if (nTile < 0 || nTile >= kMaxTiles)
{
pTSprite->xrepeat = 0;
continue;
}
// skip picnum 0 on face sprites. picnum 0 is a simple wall texture in Blood,
// but there are maps that use 0 on some operator sprites that may show up in potals as a result.
// Since the wall texture is perfectly fine for wall and floor sprites, these will be allowed to pass.
if (nTile == 0 && (pTSprite->cstat & CSTAT_SPRITE_ALIGNMENT_MASK) == CSTAT_SPRITE_ALIGNMENT_FACING)
2020-08-02 17:59:14 +00:00
{
pTSprite->xrepeat = 0;
2020-08-02 17:59:14 +00:00
continue;
}
if (cl_interpolate && owneractor->interpolated && !(pTSprite->flags&512))
2020-08-02 17:59:14 +00:00
{
pTSprite->pos = pTSprite->interpolatedvec3(gInterpolate);
pTSprite->ang = pTSprite->interpolatedang(gInterpolate);
2020-08-02 17:59:14 +00:00
}
int nAnim = 0;
switch (picanm[nTile].extra & 7) {
case 0:
2021-09-05 07:30:46 +00:00
if (pTXSprite == nullptr) break;
2020-08-02 17:59:14 +00:00
switch (pTSprite->type) {
case kSwitchToggle:
case kSwitchOneWay:
2021-09-05 07:30:46 +00:00
if (pTXSprite->state) nAnim = 1;
2020-08-02 17:59:14 +00:00
break;
case kSwitchCombo:
2021-09-05 07:30:46 +00:00
nAnim = pTXSprite->data1;
2020-08-02 17:59:14 +00:00
break;
}
break;
case 1:
{
2021-12-22 20:06:31 +00:00
if (tilehasmodelorvoxel(pTSprite->picnum, pTSprite->pal) && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
{
pTSprite->cstat &= ~CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
break;
}
int dX = cX - pTSprite->pos.X;
int dY = cY - pTSprite->pos.Y;
2020-08-02 17:59:14 +00:00
RotateVector(&dX, &dY, 128-pTSprite->ang);
nAnim = GetOctant(dX, dY);
if (nAnim <= 4)
{
pTSprite->cstat &= ~CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
else
{
nAnim = 8 - nAnim;
pTSprite->cstat |= CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
break;
}
case 2:
{
2021-12-22 20:06:31 +00:00
if (tilehasmodelorvoxel(pTSprite->picnum, pTSprite->pal) && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
{
pTSprite->cstat &= ~CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
break;
}
int dX = cX - pTSprite->pos.X;
int dY = cY - pTSprite->pos.Y;
2020-08-02 17:59:14 +00:00
RotateVector(&dX, &dY, 128-pTSprite->ang);
nAnim = GetOctant(dX, dY);
break;
}
case 3:
{
2021-09-05 07:30:46 +00:00
if (pTXSprite)
2020-08-02 17:59:14 +00:00
{
2021-08-29 17:09:29 +00:00
if (owneractor->hit.florhit.type == kHitNone)
2020-08-02 17:59:14 +00:00
nAnim = 1;
}
else
{
int top, bottom;
GetSpriteExtents(pTSprite, &top, &bottom);
if (getflorzofslopeptr(pTSprite->sector(), pTSprite->pos.X, pTSprite->pos.Y) > bottom)
2020-08-02 17:59:14 +00:00
nAnim = 1;
}
break;
}
case 6:
case 7:
{
2021-12-22 20:06:31 +00:00
if (hw_models && md_tilehasmodel(pTSprite->picnum, pTSprite->pal) >= 0 && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
break;
// Can be overridden by def script
2021-12-22 20:06:31 +00:00
if (r_voxels && tiletovox[pTSprite->picnum] == -1 && voxelIndex[pTSprite->picnum] != -1 && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
{
if ((pTSprite->flags&kHitagRespawn) == 0)
{
2021-12-18 18:34:18 +00:00
pTSprite->cstat |= CSTAT_SPRITE_ALIGNMENT_SLAB;
pTSprite->cstat &= ~(CSTAT_SPRITE_XFLIP | CSTAT_SPRITE_YFLIP);
2020-08-02 17:59:14 +00:00
pTSprite->yoffset += tileTopOffset(pTSprite->picnum);
pTSprite->picnum = voxelIndex[pTSprite->picnum];
if ((picanm[nTile].extra&7) == 7)
{
pTSprite->ang = myclock & 2047;
2020-08-02 17:59:14 +00:00
}
}
}
break;
}
}
while (nAnim > 0)
{
pTSprite->picnum += picanm[pTSprite->picnum].num+1;
nAnim--;
}
2021-12-22 20:06:31 +00:00
if ((pTSprite->cstat & CSTAT_SPRITE_ALIGNMENT_MASK) != CSTAT_SPRITE_ALIGNMENT_SLAB && r_voxels && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
{
int const nRootTile = pTSprite->picnum;
int nAnimTile = pTSprite->picnum + qanimateoffs(pTSprite->picnum, 32768 + (pTSprite->ownerActor->GetIndex() & 16383));
2020-08-02 17:59:14 +00:00
#if 0
if (tiletovox[nAnimTile] != -1)
{
pTSprite->yoffset += tileTopOffset(nAnimTile);
pTSprite->xoffset += tileLeftOffset(nAnimTile);
}
#endif
int const nVoxel = tiletovox[pTSprite->picnum];
if (nVoxel != -1 && (picanm[nRootTile].extra & 7) == 7)
pTSprite->cstat2 |= CSTAT2_SPRITE_MDLROTATE; // per-sprite rotation setting.
2020-08-02 17:59:14 +00:00
}
2021-12-22 20:06:31 +00:00
if ((pTSprite->cstat & CSTAT_SPRITE_ALIGNMENT_MASK) != CSTAT_SPRITE_ALIGNMENT_SLAB && hw_models && !(owneractor->sprext.flags&SPREXT_NOTMD))
2020-08-02 17:59:14 +00:00
{
int const nRootTile = pTSprite->picnum;
int nAnimTile = pTSprite->picnum + qanimateoffs(pTSprite->picnum, 32768 + (pTSprite->ownerActor->GetIndex() & 16383));
2020-08-02 17:59:14 +00:00
if (tile2model[Ptile2tile(nAnimTile, pTSprite->pal)].modelid >= 0 &&
tile2model[Ptile2tile(nAnimTile, pTSprite->pal)].framenum >= 0)
{
pTSprite->yoffset += tileTopOffset(nAnimTile);
pTSprite->xoffset += tileLeftOffset(nAnimTile);
if ((picanm[nRootTile].extra&7) == 7)
pTSprite->cstat2 |= CSTAT2_SPRITE_MDLROTATE; // per-sprite rotation setting.
2020-08-02 17:59:14 +00:00
}
}
2021-11-23 23:28:50 +00:00
sectortype *pSector = pTSprite->sector();
XSECTOR const* pXSector = pSector->hasX() ? &pSector->xs() : nullptr;
2020-08-02 17:59:14 +00:00
int nShade = pTSprite->shade;
if ((pSector->ceilingstat & CSTAT_SECTOR_SKY) && (pSector->floorstat & CSTAT_SECTOR_NO_CEILINGSHADE) == 0)
2020-08-02 17:59:14 +00:00
{
nShade += tileShade[pSector->ceilingpicnum]+pSector->ceilingshade;
}
else
{
nShade += tileShade[pSector->floorpicnum]+pSector->floorshade;
}
nShade += tileShade[pTSprite->picnum];
pTSprite->shade = ClipRange(nShade, -128, 127);
if ((pTSprite->flags&kHitagRespawn) && pTSprite->ownerActor->spr.owner == 3) // Where does this 3 come from? Nothing sets it.
2020-08-02 17:59:14 +00:00
{
assert(pTXSprite != NULL);
2020-08-02 17:59:14 +00:00
pTSprite->xrepeat = 48;
pTSprite->yrepeat = 48;
pTSprite->shade = -128;
pTSprite->picnum = 2272 + 2*pTXSprite->respawnPending;
2021-12-18 18:34:18 +00:00
pTSprite->cstat &= ~(CSTAT_SPRITE_TRANSLUCENT | CSTAT_SPRITE_TRANS_FLIP);
2020-08-02 17:59:14 +00:00
if (((IsItemSprite(pTSprite) || IsAmmoSprite(pTSprite)) && gGameOptions.nItemSettings == 2)
|| (IsWeaponSprite(pTSprite) && gGameOptions.nWeaponSettings == 3))
{
pTSprite->xrepeat = pTSprite->yrepeat = 48;
}
else
{
pTSprite->xrepeat = pTSprite->yrepeat = 0;
}
}
if (spritesortcnt >= MAXSPRITESONSCREEN) continue;
2020-08-02 17:59:14 +00:00
if (pTXSprite && pTXSprite->burnTime > 0)
{
pTSprite->shade = ClipRange(pTSprite->shade-16-QRandom(8), -128, 127);
}
if (pTSprite->flags&256)
{
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectSmokeHigh);
2020-08-02 17:59:14 +00:00
}
if (pTSprite->flags&1024)
{
pTSprite->cstat |= CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
if (pTSprite->flags&2048)
{
pTSprite->cstat |= CSTAT_SPRITE_YFLIP;
2020-08-02 17:59:14 +00:00
}
switch (pTSprite->statnum) {
case kStatDecoration: {
switch (pTSprite->type) {
case kDecorationCandle:
if (!pTXSprite || pTXSprite->state == 1) {
pTSprite->shade = -128;
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectPhase);
2020-08-02 17:59:14 +00:00
} else {
pTSprite->shade = -8;
}
break;
case kDecorationTorch:
if (!pTXSprite || pTXSprite->state == 1) {
pTSprite->picnum++;
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectTorchHigh);
2020-08-02 17:59:14 +00:00
} else {
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectSmokeHigh);
2020-08-02 17:59:14 +00:00
}
break;
default:
viewApplyDefaultPal(pTSprite, pSector);
break;
}
}
break;
case kStatItem: {
switch (pTSprite->type) {
case kItemFlagABase:
if (pTXSprite && pTXSprite->state > 0 && gGameOptions.nGameType == 3) {
auto pNTSprite = viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectBigFlag);
2020-08-02 17:59:14 +00:00
if (pNTSprite) pNTSprite->pal = 10;
}
break;
case kItemFlagBBase:
if (pTXSprite && pTXSprite->state > 0 && gGameOptions.nGameType == 3) {
auto pNTSprite = viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectBigFlag);
2020-08-02 17:59:14 +00:00
if (pNTSprite) pNTSprite->pal = 7;
}
break;
case kItemFlagA:
pTSprite->pal = 10;
2021-12-18 18:34:18 +00:00
pTSprite->cstat |= CSTAT_SPRITE_BLOOD_BIT2;
2020-08-02 17:59:14 +00:00
break;
case kItemFlagB:
pTSprite->pal = 7;
2021-12-18 18:34:18 +00:00
pTSprite->cstat |= CSTAT_SPRITE_BLOOD_BIT2;
2020-08-02 17:59:14 +00:00
break;
default:
if (pTSprite->type >= kItemKeySkull && pTSprite->type < kItemKeyMax)
pTSprite->shade = -128;
viewApplyDefaultPal(pTSprite, pSector);
break;
}
}
break;
case kStatProjectile: {
switch (pTSprite->type) {
case kMissileTeslaAlt:
pTSprite->yrepeat = 128;
2021-12-18 18:34:18 +00:00
pTSprite->cstat |= CSTAT_SPRITE_ALIGNMENT_FLOOR;
2020-08-02 17:59:14 +00:00
break;
case kMissileTeslaRegular:
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectTesla);
2020-08-02 17:59:14 +00:00
break;
case kMissileButcherKnife:
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectTrail);
2020-08-02 17:59:14 +00:00
break;
case kMissileFlareRegular:
case kMissileFlareAlt:
if (pTSprite->statnum == kStatFlare) {
if (owneractor->GetTarget() == gView->actor)
{
2020-08-02 17:59:14 +00:00
pTSprite->xrepeat = 0;
break;
}
}
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectFlareHalo);
2020-08-02 17:59:14 +00:00
if (pTSprite->type != kMissileFlareRegular) break;
2021-11-23 23:28:50 +00:00
sectortype *pSector = pTSprite->sector();
2020-08-02 17:59:14 +00:00
int zDiff = (pTSprite->pos.Z - pSector->ceilingz) >> 8;
if ((pSector->ceilingstat & CSTAT_SECTOR_SKY) == 0 && zDiff < 64) {
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectCeilGlow);
2020-08-02 17:59:14 +00:00
}
zDiff = (pSector->floorz - pTSprite->pos.Z) >> 8;
if ((pSector->floorstat & CSTAT_SECTOR_SKY) == 0 && zDiff < 64) {
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectFloorGlow);
2020-08-02 17:59:14 +00:00
}
break;
}
break;
}
case kStatDude:
{
if (pTSprite->type == kDudeHand && pTXSprite->aiState == &hand13A3B4)
{
auto target = owneractor->GetTarget();
if (target && target->IsPlayerActor())
2020-08-02 17:59:14 +00:00
{
pTSprite->xrepeat = 0;
break;
}
}
2021-04-13 18:33:21 +00:00
if (pXSector && pXSector->color) copyfloorpal(pTSprite, pSector);
2020-08-02 17:59:14 +00:00
if (powerupCheck(gView, kPwUpBeastVision) > 0) pTSprite->shade = -128;
if (IsPlayerSprite(pTSprite)) {
PLAYER *pPlayer = &gPlayer[pTSprite->type-kDudePlayer1];
if (powerupCheck(pPlayer, kPwUpShadowCloak) && !powerupCheck(gView, kPwUpBeastVision)) {
pTSprite->cstat |= CSTAT_SPRITE_TRANSLUCENT;
2020-08-02 17:59:14 +00:00
pTSprite->pal = 5;
} else if (powerupCheck(pPlayer, kPwUpDeathMask)) {
pTSprite->shade = -128;
pTSprite->pal = 5;
} else if (powerupCheck(pPlayer, kPwUpDoppleganger)) {
pTSprite->pal = 11+(gView->teamId&3);
}
if (powerupCheck(pPlayer, kPwUpReflectShots)) {
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectReflectiveBall);
2020-08-02 17:59:14 +00:00
}
if (cl_showweapon && gGameOptions.nGameType > 0 && gView) {
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectShowWeapon);
2020-08-02 17:59:14 +00:00
}
if (pPlayer->flashEffect && (gView != pPlayer || gViewPos != VIEWPOS_0)) {
auto pNTSprite = viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectShoot);
2020-08-02 17:59:14 +00:00
if (pNTSprite) {
POSTURE *pPosture = &pPlayer->pPosture[pPlayer->lifeMode][pPlayer->posture];
pNTSprite->pos.X += MulScale(pPosture->zOffset, Cos(pTSprite->ang), 28);
pNTSprite->pos.Y += MulScale(pPosture->zOffset, Sin(pTSprite->ang), 28);
pNTSprite->pos.Z = pPlayer->pSprite->pos.Z-pPosture->xOffset;
2020-08-02 17:59:14 +00:00
}
}
if (pPlayer->hasFlag > 0 && gGameOptions.nGameType == 3) {
if (pPlayer->hasFlag&1) {
auto pNTSprite = viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectFlag);
2020-08-02 17:59:14 +00:00
if (pNTSprite)
{
pNTSprite->pal = 10;
pNTSprite->cstat |= CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
}
if (pPlayer->hasFlag&2) {
auto pNTSprite = viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectFlag);
2020-08-02 17:59:14 +00:00
if (pNTSprite)
{
pNTSprite->pal = 7;
pNTSprite->cstat |= CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
}
}
}
if (pTSprite->ownerActor != gView->actor || gViewPos != VIEWPOS_0) {
if (getflorzofslopeptr(pTSprite->sector(), pTSprite->pos.X, pTSprite->pos.Y) >= cZ)
2020-08-02 17:59:14 +00:00
{
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectShadow);
2020-08-02 17:59:14 +00:00
}
}
if (gModernMap) { // add target spot indicator for patrol dudes
if (pTXSprite->dudeFlag4 && aiInPatrolState(pTXSprite->aiState) && pTXSprite->data3 > 0 && pTXSprite->data3 <= kMaxPatrolSpotValue)
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectSpotProgress);
}
2020-08-02 17:59:14 +00:00
break;
}
case kStatTraps: {
if (pTSprite->type == kTrapSawCircular) {
if (pTXSprite->state) {
if (pTXSprite->data1) {
pTSprite->picnum = 772;
if (pTXSprite->data2)
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectSpear);
2020-08-02 17:59:14 +00:00
}
}
else if (pTXSprite->data1) pTSprite->picnum = 773;
else pTSprite->picnum = 656;
}
break;
}
case kStatThing: {
viewApplyDefaultPal(pTSprite, pSector);
2021-08-29 17:09:29 +00:00
if (pTSprite->type < kThingBase || pTSprite->type >= kThingMax || owneractor->hit.florhit.type == kHitNone)
{
if ((pTSprite->flags & kPhysMove) && getflorzofslopeptr(pTSprite->sector(), pTSprite->pos.X, pTSprite->pos.Y) >= cZ)
viewAddEffect(tsprite, spritesortcnt, nTSprite, kViewEffectShadow);
2020-08-02 17:59:14 +00:00
}
}
break;
}
}
for (int nTSprite = spritesortcnt-1; nTSprite >= nViewSprites; nTSprite--)
{
tspritetype *pTSprite = &tsprite[nTSprite];
int nAnim = 0;
switch (picanm[pTSprite->picnum].extra&7)
{
case 1:
{
int dX = cX - pTSprite->pos.X;
int dY = cY - pTSprite->pos.Y;
2020-08-02 17:59:14 +00:00
RotateVector(&dX, &dY, 128-pTSprite->ang);
nAnim = GetOctant(dX, dY);
if (nAnim <= 4)
{
pTSprite->cstat &= ~CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
else
{
nAnim = 8 - nAnim;
pTSprite->cstat |= CSTAT_SPRITE_XFLIP;
2020-08-02 17:59:14 +00:00
}
break;
}
case 2:
{
int dX = cX - pTSprite->pos.X;
int dY = cY - pTSprite->pos.Y;
2020-08-02 17:59:14 +00:00
RotateVector(&dX, &dY, 128-pTSprite->ang);
nAnim = GetOctant(dX, dY);
break;
}
}
while (nAnim > 0)
{
pTSprite->picnum += picanm[pTSprite->picnum].num+1;
nAnim--;
}
}
}
void GameInterface::processSprites(tspritetype* tsprite, int& spritesortcnt, int viewx, int viewy, int viewz, binangle viewang, double smoothRatio)
2021-03-20 22:01:16 +00:00
{
viewProcessSprites(tsprite, spritesortcnt, viewx, viewy, viewz, viewang.asbuild(), int(smoothRatio));
2021-03-20 22:01:16 +00:00
}
2021-03-29 19:48:23 +00:00
int display_mirror;
void GameInterface::EnterPortal(spritetype* viewer, int type)
{
if (type == PORTAL_WALL_MIRROR)
{
display_mirror++;
if (viewer) viewer->cstat &= ~CSTAT_SPRITE_INVISIBLE;
}
}
void GameInterface::LeavePortal(spritetype* viewer, int type)
{
if (type == PORTAL_WALL_MIRROR)
{
display_mirror--;
if (viewer && display_mirror == 0 && !(viewer->cstat & CSTAT_SPRITE_TRANSLUCENT)) viewer->cstat |= CSTAT_SPRITE_INVISIBLE;
}
}
2020-08-02 17:59:14 +00:00
END_BLD_NS