raze/source/games/blood/src/gib.cpp
2023-10-13 23:08:06 +02:00

531 lines
12 KiB
C++

//-------------------------------------------------------------------------
/*
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 "build.h"
#include "blood.h"
#include "bloodactor.h"
BEGIN_BLD_NS
struct GIBFX
{
FX_ID fxId;
int palcopy;
int chance;
int counts;
int randxy;
int randz;
};
struct GIBTHING
{
int type;
int picno;
int chance;
int randxy;
int randz;
FTextureID textureID() const { return tileGetTextureID(picno); }
};
struct GIBLIST
{
GIBFX* gibFX;
int FXCount;
GIBTHING* things;
int thingcount;
int sound;
};
GIBFX gibFxGlassT[] = {
{ FX_18, 0, 65536, 3, 200, 400 },
{ FX_31, 0, 32768, 5, 200, 400 }
};
GIBFX gibFxGlassS[] = {
{ FX_18, 0, 65536, 8, 200, 400 }
};
GIBFX gibFxBurnShard[] = {
{ FX_16, 0, 65536, 12, 500, 1000 }
};
GIBFX gibFxWoodShard[] = {
{ FX_17, 0, 65536, 12, 500, 1000 }
};
GIBFX gibFxMetalShard[] = {
{ FX_30, 0, 65536, 12, 500, 1000 }
};
GIBFX gibFxFireSpark[] = {
{ FX_14, 0, 65536, 8, 500, 1000 }
};
GIBFX gibFxShockSpark[] = {
{ FX_15, 0, 65536, 8, 500, 1000 }
};
GIBFX gibFxBloodChunks[] = {
{ FX_13, 0, 65536, 8, 90, 600 }
};
GIBFX gibFxBubblesS[] = {
{ FX_25, 0, 65536, 8, 200, 400 }
};
GIBFX gibFxBubblesM[] = {
{ FX_24, 0, 65536, 8, 200, 400 }
};
GIBFX gibFxBubblesL[] = {
{ FX_23, 0, 65536, 8, 200, 400 }
};
GIBFX gibFxIcicles[] = {
{ FX_31, 0, 65536, 15, 200, 400 }
};
GIBFX gibFxGlassCombo1[] = {
{ FX_18, 0, 65536, 15, 200, 400 },
{ FX_31, 0, 65536, 10, 200, 400 }
};
GIBFX gibFxGlassCombo2[] = {
{ FX_18, 0, 65536, 5, 200, 400 },
{ FX_20, 0, 53248, 5, 200, 400 },
{ FX_21, 0, 53248, 5, 200, 400 },
{ FX_19, 0, 53248, 5, 200, 400 },
{ FX_22, 0, 53248, 5, 200, 400 }
};
GIBFX gibFxWoodCombo[] = {
{ FX_16, 0, 65536, 8, 500, 1000 },
{ FX_17, 0, 65536, 8, 500, 1000 },
{ FX_14, 0, 65536, 8, 500, 1000 }
};
GIBFX gibFxMedicCombo[] = {
{ FX_18, 0, 32768, 7, 200, 400 },
{ FX_30, 0, 65536, 7, 500, 1000 },
{ FX_13, 0, 65536, 10, 90, 600 },
{ FX_14, 0, 32768, 7, 500, 1000 }
};
GIBFX gibFxFlareSpark[] = {
{ FX_28, 0, 32768, 15, 128, -128 }
};
GIBFX gibFxBloodBits[] = {
{ FX_13, 0, 45056, 8, 90, 600 }
};
GIBFX gibFxRockShards[] = {
{ FX_46, 0, 65536, 10, 300, 800 },
{ FX_31, 0, 32768, 10, 200, 1000 }
};
GIBFX gibFxPaperCombo1[] = {
{ FX_47, 0, 65536, 12, 300, 600 },
{ FX_14, 0, 65536, 8, 500, 1000 }
};
GIBFX gibFxPlantCombo1[] = {
{ FX_44, 0, 45056, 8, 400, 800 },
{ FX_45, 0, 45056, 8, 300, 800 },
{ FX_14, 0, 45056, 6, 500, 1000 }
};
GIBFX gibFx13BBA8[] = {
{ FX_49, 0, 65536, 4, 80, 300 }
};
GIBFX gibFx13BBC0[] = {
{ FX_50, 0, 65536, 4, 80, 0 }
};
GIBFX gibFx13BBD8[] = {
{ FX_50, 0, 65536, 20, 800, -40 },
{ FX_15, 0, 65536, 15, 400, 10 }
};
GIBFX gibFx13BC04[] = {
{ FX_32, 0, 65536, 8, 100, 0 }
};
GIBFX gibFx13BC1C[] = {
{ FX_56, 0, 65536, 8, 100, 0 }
};
GIBTHING gibHuman[] = {
{ 425, 1454, 917504, 300, 900 },
{ 425, 1454, 917504, 300, 900 },
{ 425, 1267, 917504, 300, 900 },
{ 425, 1267, 917504, 300, 900 },
{ 425, 1268, 917504, 300, 900 },
{ 425, 1269, 917504, 300, 900 },
{ 425, 1456, 917504, 300, 900 }
};
GIBTHING gibMime[] = {
{ 425, 2405, 917504, 300, 900 },
{ 425, 2405, 917504, 300, 900 },
{ 425, 2404, 917504, 300, 900 },
{ 425, 1268, 32768, 300, 900 },
{ 425, 1269, 32768, 300, 900 },
{ 425, 1456, 32768, 300, 900 },
};
GIBTHING gibHound[] = {
{ 425, 1326, 917504, 300, 900 },
{ 425, 1268, 32768, 300, 900 },
{ 425, 1269, 32768, 300, 900 },
{ 425, 1456, 32768, 300, 900 }
};
GIBTHING gibFleshGargoyle[] = {
{ 425, 1369, 917504, 300, 900 },
{ 425, 1361, 917504, 300, 900 },
{ 425, 1268, 32768, 300, 900 },
{ 425, 1269, 32768, 300, 900 },
{ 425, 1456, 32768, 300, 900 }
};
GIBTHING gibAxeZombieHead[] = {
{ 427, 3405, 917504, 0, 0 }
};
GIBLIST gibList[] = {
{ gibFxGlassT, 2, NULL, 0, 300 },
{ gibFxGlassS, 1, NULL, 0, 300 },
{ gibFxBurnShard, 1, NULL, 0, 0 },
{ gibFxWoodShard, 1, NULL, 0, 0 },
{ gibFxMetalShard, 1, NULL, 0, 0 },
{ gibFxFireSpark, 1, NULL, 0, 0 },
{ gibFxShockSpark, 1, NULL, 0, 0 },
{ gibFxBloodChunks, 1, NULL, 0, 0 },
{ gibFxBubblesS, 1, NULL, 0, 0 },
{ gibFxBubblesM, 1, NULL, 0, 0 },
{ gibFxBubblesL, 1, NULL, 0, 0 },
{ gibFxIcicles, 1, NULL, 0, 0 },
{ gibFxGlassCombo1, 2, NULL, 0, 300 },
{ gibFxGlassCombo2, 5, NULL, 0, 300 },
{ gibFxWoodCombo, 3, NULL, 0, 0 },
{ NULL, 0, gibHuman, 7, 0 },
{ gibFxMedicCombo, 4, NULL, 0, 0 },
{ gibFxFlareSpark, 1, NULL, 0, 0 },
{ gibFxBloodBits, 1, NULL, 0, 0 },
{ gibFxRockShards, 2, NULL, 0, 0 },
{ gibFxPaperCombo1, 2, NULL, 0, 0 },
{ gibFxPlantCombo1, 3, NULL, 0, 0 },
{ gibFx13BBA8, 1, NULL, 0, 0 },
{ gibFx13BBC0, 1, NULL, 0, 0 },
{ gibFx13BBD8, 2, NULL, 0, 0 },
{ gibFx13BC04, 1, NULL, 0, 0 },
{ gibFx13BC1C, 1, NULL, 0, 0 },
{ NULL, 0, gibAxeZombieHead, 1, 0 },
{ NULL, 0, gibMime, 6, 0 },
{ NULL, 0, gibHound, 4, 0 },
{ NULL, 0, gibFleshGargoyle, 5, 0 },
};
int ChanceToCount(int a1, int a2)
{
int vb = a2;
if (a1 < 0x10000)
{
for (int i = 0; i < a2; i++)
if (!Chance(a1))
vb--;
}
return vb;
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GibFX(DBloodActor* actor, GIBFX* pGFX, DVector3* pPos, DVector3* pVel)
{
auto pSector = actor->sector();
if (adult_lockout && gGameOptions.nGameType == 0 && pGFX->fxId == FX_13)
return;
auto gPos = pPos? *pPos : actor->spr.pos;
double ceilZ, floorZ;
calcSlope(pSector, gPos.XY(), &ceilZ, &floorZ);
int nCount = ChanceToCount(pGFX->chance, pGFX->counts);
double dz1 = floorZ - gPos.Z;
double dz2 = gPos.Z - ceilZ;
double top, bottom;
GetActorExtents(actor, &top, &bottom);
for (int i = 0; i < nCount; i++)
{
if (!pPos && (actor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_MASK) == 0)
{
DAngle nAngle = RandomAngle();
gPos = DVector3(actor->spr.pos.XY() + actor->clipdist * nAngle.ToVector(), bottom - RandomD(bottom - top, 8));
}
auto pFX = gFX.fxSpawnActor(pGFX->fxId, pSector, gPos);
if (pFX)
{
if (pGFX->palcopy < 0)
pFX->spr.pal = actor->spr.pal;
if (pVel)
{
pFX->vel = *pVel + DVector3(Random2F(pGFX->randxy, 4), Random2F(pGFX->randxy, 4), -RandomF(pGFX->randz, 8));
}
else
{
pFX->vel.X = Random2F((pGFX->randxy << 18) / 120);
pFX->vel.Y = Random2F((pGFX->randxy << 18) / 120);
switch (actor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_MASK)
{
case 16:
pFX->vel.Z = Random2F((pGFX->randz << 18) / 120);
break;
default:
if (dz2 < dz1 && dz2 < 0x400)
{
pFX->vel.Z = 0;
}
else if (dz2 > dz1 && dz1 < 0x400)
{
pFX->vel.Z = -Random2F((abs(pGFX->randz) << 18) / 120);
}
else
{
if ((pGFX->randz << 18) / 120 < 0)
pFX->vel.Z = -Random2F((abs(pGFX->randz) << 18) / 120);
else
pFX->vel.Z = Random2F((pGFX->randz << 18) / 120);
}
break;
}
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GibThing(DBloodActor* actor, GIBTHING* pGThing, DVector3* pPos, DVector3* pVel)
{
if (adult_lockout && gGameOptions.nGameType == kGameTypeSinglePlayer)
switch (pGThing->type) {
case kThingBloodBits:
case kThingZombieHead:
return;
}
if (pGThing->chance == 65536 || Chance(pGThing->chance))
{
auto pSector = actor->sector();
double top, bottom;
GetActorExtents(actor, &top, &bottom);
DVector3 gPos;
if (!pPos)
{
DAngle nAngle = RandomAngle();
gPos = DVector3(actor->spr.pos.XY() + actor->clipdist * nAngle.ToVector(), bottom - RandomD(bottom - top, 8));
}
else
{
gPos = *pPos;
}
double ceilZ, floorZ;
calcSlope(pSector, gPos, &ceilZ, &floorZ);
double dz1 = floorZ - gPos.Z;
double dz2 = gPos.Z - ceilZ;
auto gibactor = actSpawnThing(pSector, gPos, pGThing->type);
if (!gibactor) return;
if (pGThing->textureID().isValid())
gibactor->spr.setspritetexture(pGThing->textureID());
if (pVel)
{
gibactor->vel = *pVel + DVector3(Random2F(pGThing->randxy, 4), Random2F(pGThing->randxy, 4), -RandomF(pGThing->randz, 8));
}
else
{
gibactor->vel.X = Random2F((pGThing->randxy << 18) / 120);
gibactor->vel.Y = Random2F((pGThing->randxy << 18) / 120);
switch (actor->spr.cstat & CSTAT_SPRITE_ALIGNMENT_MASK)
{
case 16:
gibactor->vel.Z = Random2F((pGThing->randz << 18) / 120);
break;
default:
if (dz2 < dz1 && dz2 < 0x400)
{
gibactor->vel.Z = 0;
}
else if (dz2 > dz1 && dz1 < 0x400)
{
gibactor->vel.Z = -Random2F((pGThing->randz << 18) / 120);
}
else
{
gibactor->vel.Z = Random2F((pGThing->randz << 18) / 120);
}
break;
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GibSprite(DBloodActor* actor, GIBTYPE nGibType, DVector3* pPos, DVector3* pVel)
{
assert(actor != NULL);
assert(nGibType >= 0 && nGibType < kGibMax);
if (!actor->insector())
return;
GIBLIST* pGib = &gibList[nGibType];
for (int i = 0; i < pGib->FXCount; i++)
{
GIBFX* pGibFX = &pGib->gibFX[i];
assert(pGibFX->chance > 0);
GibFX(actor, pGibFX, pPos, pVel);
}
for (int i = 0; i < pGib->thingcount; i++)
{
GIBTHING* pGibThing = &pGib->things[i];
assert(pGibThing->chance > 0);
GibThing(actor, pGibThing, pPos, pVel);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GibFX(walltype* pWall, GIBFX* pGFX, double ceilZ, const DVector3& spread, DVector3* pVel)
{
assert(pWall);
int nCount = ChanceToCount(pGFX->chance, pGFX->counts);
auto pSector = pWall->sectorp();
for (int i = 0; i < nCount; i++)
{
DVector3 r;
r.Z = RandomD(spread.Z, 8) + ceilZ;
r.Y = RandomD(spread.Y, 4) + pWall->pos.Y;
r.X = RandomD(spread.X, 4) + pWall->pos.X;
auto pGib = gFX.fxSpawnActor(pGFX->fxId, pSector, r);
if (pGib)
{
if (pGFX->palcopy < 0)
pGib->spr.pal = pWall->pal;
if (!pVel)
{
pGib->vel.X = Random2F((pGFX->randxy << 18) / 120);
pGib->vel.Y = Random2F((pGFX->randxy << 18) / 120);
pGib->vel.Z = -Random2F((pGFX->randz << 18) / 120);
}
else
{
pGib->vel.X = Random2F(int((pVel->X * 17179869184.) / 120));
pGib->vel.Y = Random2F(int((pVel->Y * 17179869184.) / 120));
pGib->vel.Z = -Random2F(int((pVel->Z * 17179869184.) / 120));
}
}
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void GibWall(walltype* pWall, GIBTYPE nGibType, DVector3* pVel)
{
assert(pWall);
assert(nGibType >= 0 && nGibType < kGibMax);
DVector3 center;
center.XY() = pWall->center();
auto pSector = pWall->sectorp();
double ceilZ, floorZ;
calcSlope(pSector, center, &ceilZ, &floorZ);
double ceilZ2, floorZ2;
calcSlope(pWall->nextSector(), center, &ceilZ2, &floorZ2);
ceilZ = max(ceilZ, ceilZ2);
floorZ = min(floorZ, floorZ2);
DVector3 w;
w.Z = floorZ - ceilZ;
w.XY() = pWall->delta();
center.Z = (ceilZ + floorZ) * 0.5;
GIBLIST* pGib = &gibList[nGibType];
sfxPlay3DSectorSound(center, pGib->sound, pSector);
for (int i = 0; i < pGib->FXCount; i++)
{
GIBFX* pGibFX = &pGib->gibFX[i];
assert(pGibFX->chance > 0);
GibFX(pWall, pGibFX, ceilZ, w, pVel);
}
}
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
void gibPrecache()
{
for (int i = 0; i < kGibMax; i++)
{
auto const pThing = gibList[i].things;
if (pThing)
{
for (int j = 0; j < gibList[i].thingcount; j++)
{
if (pThing[j].textureID().isValid())
tilePrecacheTile(pThing[j].textureID(), -1, 0);
}
}
}
}
END_BLD_NS