raze/source/games/exhumed/src/lighting.cpp

831 lines
19 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 2010-2019 EDuke32 developers and contributors
Copyright (C) 2019 sirlemonhead, Nuke.YKT
This file is part of PCExhumed.
PCExhumed 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"
#include "aistuff.h"
#include "player.h"
#include "engine.h"
#include "exhumed.h"
#include "sound.h"
#include "interpolate.h"
#include <string.h>
#include <assert.h>
BEGIN_PS_NS
enum
{
kMaxFlashes = 2000,
kMaxFlickerMask = 25,
kMaxGlows = 50,
kMaxFlickers = 100,
kMaxFlows = 375,
};
struct Flash
{
2021-11-22 17:51:22 +00:00
union
{
walltype* pWall;
sectortype* pSector;
2021-12-07 17:53:02 +00:00
TObjPtr<DExhumedActor*> pActor;
2021-11-22 17:51:22 +00:00
};
int next;
2021-10-27 17:35:08 +00:00
int8_t nType;
int8_t shade;
};
struct Glow
{
2021-11-22 18:08:49 +00:00
sectortype* pSector;
int16_t nShade;
int16_t nCounter;
int16_t nThreshold;
};
struct Flicker
{
2021-11-22 18:14:26 +00:00
sectortype* pSector;
int16_t nShade;
unsigned int nMask;
};
struct Flow
{
union
{
walltype* pWall;
sectortype* pSector;
};
2021-11-09 16:32:50 +00:00
int type;
int xdelta;
int ydelta;
int angcos;
int angsin;
int xacc;
int yacc;
};
2020-11-30 00:05:17 +00:00
FreeListArray<Flash, kMaxFlashes> sFlash;
Glow sGlow[kMaxGlows];
Flicker sFlicker[kMaxFlickers];
Flow sFlowInfo[kMaxFlows];
int flickermask[kMaxFlickerMask];
int bTorch = 0;
int nFirstFlash = -1;
int nLastFlash = -1;
int nFlashDepth = 2;
int nFlowCount;
int nFlickerCount;
int nGlowCount;
int bDoFlicks = 0;
int bDoGlows = 0;
2021-12-07 17:53:02 +00:00
size_t MarkLighting()
{
for (int i = 0; i < kMaxFlashes; i++)
{
auto& f = sFlash[i];
if (f.nType & 4) GC::Mark(f.pActor);
}
return kMaxFlashes;
}
2020-11-30 00:05:17 +00:00
FSerializer& Serialize(FSerializer& arc, const char* keyname, Flash& w, Flash* def)
{
if (arc.BeginObject(keyname))
{
2021-11-22 17:51:22 +00:00
arc("type", w.nType)
2020-11-30 00:05:17 +00:00
("shade", w.shade)
2021-11-22 17:51:22 +00:00
("next", w.next);
if (w.nType & 4) arc("index", w.pActor);
else if (w.nType & 1) arc("index", w.pSector);
else arc("index", w.pWall);
arc.EndObject();
2020-11-30 00:05:17 +00:00
}
return arc;
}
FSerializer& Serialize(FSerializer& arc, const char* keyname, Glow& w, Glow* def)
{
if (arc.BeginObject(keyname))
{
2021-11-22 18:14:26 +00:00
arc("shade", w.nShade)
("counter", w.nCounter)
2021-11-22 18:08:49 +00:00
("sector", w.pSector)
2021-11-22 18:14:26 +00:00
("threshold", w.nThreshold)
2020-11-30 00:05:17 +00:00
.EndObject();
}
return arc;
}
FSerializer& Serialize(FSerializer& arc, const char* keyname, Flicker& w, Flicker* def)
{
if (arc.BeginObject(keyname))
{
2021-11-22 18:14:26 +00:00
arc("shade", w.nShade)
("sector", w.pSector)
("mask", w.nMask)
2020-11-30 00:05:17 +00:00
.EndObject();
}
return arc;
}
FSerializer& Serialize(FSerializer& arc, const char* keyname, Flow& w, Flow* def)
{
if (arc.BeginObject(keyname))
{
arc("type", w.type)
2020-11-30 00:05:17 +00:00
("xdelta", w.xdelta)
("ydelta", w.ydelta)
2021-11-22 18:14:26 +00:00
("angcos", w.angcos)
("angsin", w.angsin)
2020-11-30 00:05:17 +00:00
("xacc", w.xacc)
("yacc", w.yacc);
if (w.type < 2) arc("index", w.pSector);
else arc("index", w.pWall);
arc.EndObject();
2020-11-30 00:05:17 +00:00
}
return arc;
}
2020-11-30 00:05:17 +00:00
void SerializeLighting(FSerializer& arc)
{
if (arc.BeginObject("lighting"))
{
arc("flash", sFlash)
("glowcount", nGlowCount)
.Array("glow", sGlow, nGlowCount)
("flickercount", nFlickerCount)
.Array("flicker", sFlicker, nFlickerCount)
("flowcount", nFlowCount)
.Array("flow", sFlowInfo, nFlowCount)
.Array("flickermask", flickermask, countof(flickermask))
("torch", bTorch)
("firstflash", nFirstFlash)
("lastflash", nLastFlash)
("flashdepth", nFlashDepth)
("doflicks", bDoFlicks)
("doglows", bDoGlows)
.EndObject();
}
}
// done
int GrabFlash()
{
2020-11-30 00:05:17 +00:00
int nFlash = sFlash.Get();
if (nFlash < 0) {
return -1;
}
2020-11-30 00:05:17 +00:00
sFlash[nFlash].next = -1;
if (nLastFlash <= -1)
{
nFirstFlash = nFlash;
}
else
{
2020-11-30 00:05:17 +00:00
sFlash[nLastFlash].next = nFlash;
}
nLastFlash = nFlash;
return nLastFlash;
}
void InitLights()
{
int i;
nFlickerCount = 0;
for (i = 0; i < kMaxFlickerMask; i++) {
flickermask[i] = RandomSize(0x1F) * 2;
}
nGlowCount = 0;
nFlowCount = 0;
bDoFlicks = false;
bDoGlows = false;
2020-11-30 00:05:17 +00:00
sFlash.Clear();
nFirstFlash = -1;
nLastFlash = -1;
}
2021-11-22 17:51:22 +00:00
void AddFlash(sectortype* pSector, int x, int y, int z, int val)
{
int var_28 = 0;
2021-05-12 15:09:07 +00:00
int var_1C = val >> 8;
if (var_1C >= nFlashDepth) {
return;
}
unsigned int var_20 = val & 0x80;
unsigned int var_18 = val & 0x40;
2021-05-12 15:09:07 +00:00
val = ((var_1C + 1) << 8) | (val & 0xff);
int var_14 = 0;
2021-11-22 17:51:22 +00:00
for (auto& wal : wallsofsector(pSector))
{
2021-11-17 23:52:32 +00:00
auto average = wal.center();
sectortype *pNextSector = NULL;
2021-11-22 16:54:32 +00:00
if (wal.twoSided())
2021-11-17 23:52:32 +00:00
pNextSector = wal.nextSector();
int ebx = -255;
if (!var_18)
{
2021-12-22 09:26:51 +00:00
int x2 = x - average.X;
if (x2 < 0) {
x2 = -x2;
}
ebx = x2;
2021-12-22 09:28:51 +00:00
int y2 = y - average.Y;
if (y2 < 0) {
y2 = -y2;
}
ebx = ((y2 + ebx) >> 4) - 255;
}
if (ebx < 0)
{
var_14++;
var_28 += ebx;
2021-11-17 23:52:32 +00:00
if (wal.pal < 5)
{
2021-11-22 17:51:22 +00:00
if (!pNextSector || pNextSector->floorz < pSector->floorz)
{
int nFlash = GrabFlash();
if (nFlash < 0) {
return;
}
2021-10-27 17:35:08 +00:00
sFlash[nFlash].nType = var_20 | 2;
2021-11-17 23:52:32 +00:00
sFlash[nFlash].shade = wal.shade;
2021-11-22 17:51:22 +00:00
sFlash[nFlash].pWall = &wal;
2021-11-17 23:52:32 +00:00
wal.pal += 7;
2021-11-17 23:52:32 +00:00
ebx += wal.shade;
int eax = ebx;
if (ebx < -127) {
eax = -127;
}
2021-11-17 23:52:32 +00:00
wal.shade = eax;
2021-11-17 23:52:32 +00:00
if (!var_1C && !wal.overpicnum && pNextSector)
{
2021-11-22 17:51:22 +00:00
AddFlash(pNextSector, x, y, z, val);
}
}
}
}
}
2021-11-22 17:51:22 +00:00
if (var_14 && pSector->floorpal < 4)
{
int nFlash = GrabFlash();
if (nFlash < 0) {
return;
}
2021-10-27 17:35:08 +00:00
sFlash[nFlash].nType = var_20 | 1;
2021-11-22 17:51:22 +00:00
sFlash[nFlash].pSector = pSector;;
sFlash[nFlash].shade = pSector->floorshade;
2021-11-22 17:51:22 +00:00
pSector->floorpal += 7;
2021-11-22 17:51:22 +00:00
int edx = pSector->floorshade + var_28;
int eax = edx;
if (edx < -127) {
eax = -127;
}
2021-11-22 17:51:22 +00:00
pSector->floorshade = eax;
if (!(pSector->ceilingstat & CSTAT_SECTOR_SKY))
{
2021-11-22 17:51:22 +00:00
if (pSector->ceilingpal < 4)
{
int nFlash2 = GrabFlash();
if (nFlash2 >= 0)
{
2021-10-27 17:35:08 +00:00
sFlash[nFlash2].nType = var_20 | 3;
2021-11-22 17:51:22 +00:00
sFlash[nFlash2].pSector = pSector;
sFlash[nFlash2].shade = pSector->ceilingshade;
2021-11-22 17:51:22 +00:00
pSector->ceilingpal += 7;
2021-11-22 17:51:22 +00:00
int edx = pSector->ceilingshade + var_28;
int eax = edx;
if (edx < -127) {
eax = -127;
}
2021-11-22 17:51:22 +00:00
pSector->ceilingshade = eax;
}
}
}
2021-11-22 17:51:22 +00:00
ExhumedSectIterator it(pSector);
while (auto pActor = it.Next())
{
auto pSprite = &pActor->s();
if (pSprite->pal < 4)
{
int nFlash3 = GrabFlash();
if (nFlash3 >= 0)
{
2021-10-27 17:35:08 +00:00
sFlash[nFlash3].nType = var_20 | 4;
sFlash[nFlash3].shade = pSprite->shade;
2021-10-21 08:55:16 +00:00
sFlash[nFlash3].pActor = pActor;
pSprite->pal += 7;
int eax = -255;
if (!var_18)
{
int xDiff = x - pSprite->x;
if (xDiff < 0) {
xDiff = -xDiff;
}
int yDiff = y - pSprite->y;
if (yDiff < 0) {
yDiff = -yDiff;
}
eax = ((xDiff + yDiff) >> 4) - 255;
}
if (eax < 0)
{
int shade = pSprite->shade + eax;
if (shade < -127) {
shade = -127;
}
pSprite->shade = (int8_t)shade;
}
}
}
}
}
}
void UndoFlashes()
{
int var_24 = 0; // CHECKME - Watcom error "initializer for variable var_24 may not execute
int edi = -1;
2021-11-09 16:32:50 +00:00
for (int nFlash = nFirstFlash; nFlash >= 0; nFlash = sFlash[nFlash].next)
{
assert(nFlash < 2000 && nFlash >= 0);
2021-11-22 17:51:22 +00:00
auto pFlash = &sFlash[nFlash];
2021-11-22 17:51:22 +00:00
uint8_t type = pFlash->nType & 0x3F;
2021-11-22 17:51:22 +00:00
if (pFlash->nType & 0x80)
{
2021-10-27 17:35:08 +00:00
int flashtype = type - 1;
assert(flashtype >= 0);
int8_t *pShade = NULL;
2021-10-27 17:35:08 +00:00
switch (flashtype)
{
case 0:
{
2021-11-22 17:51:22 +00:00
pShade = &pFlash->pSector->floorshade;
break;
}
case 1:
{
2021-11-22 17:51:22 +00:00
pShade = &pFlash->pWall->shade;
break;
}
case 2:
{
2021-11-22 17:51:22 +00:00
pShade = &pFlash->pSector->ceilingshade;
break;
}
case 3:
{
2021-12-07 17:53:02 +00:00
DExhumedActor* ac = pFlash->pActor;
if (ac && ac->spr.pal >= 7)
{
2021-12-07 17:53:02 +00:00
pShade = &ac->spr.shade;
}
else {
goto loc_1868A;
}
break;
}
default:
break;
}
assert(pShade != NULL);
2021-05-12 15:09:07 +00:00
int thisshade = (*pShade) + 6;
2021-11-22 17:51:22 +00:00
int maxshade = pFlash->shade;
2021-05-12 15:09:07 +00:00
if (thisshade < maxshade)
{
2021-05-12 15:09:07 +00:00
*pShade = (int8_t)thisshade;
edi = nFlash;
continue;
}
}
// loc_185FE
2021-10-27 17:35:08 +00:00
var_24 = type - 1; // CHECKME - Watcom error "initializer for variable var_24 may not execute
assert(var_24 >= 0);
switch (var_24)
{
default:
break;
case 0:
{
2021-11-22 17:51:22 +00:00
pFlash->pSector->floorpal -= 7;
pFlash->pSector->floorshade = pFlash->shade;
break;
}
case 1:
{
2021-11-22 17:51:22 +00:00
pFlash->pWall->pal -= 7;
pFlash->pWall->shade = pFlash->shade;
break;
}
case 2:
{
2021-11-22 17:51:22 +00:00
pFlash->pSector->ceilingpal -= 7;
pFlash->pSector->ceilingshade = pFlash->shade;
break;
}
case 3:
{
2021-12-07 17:53:02 +00:00
DExhumedActor* ac = pFlash->pActor;
if (ac && ac->spr.pal >= 7)
{
2021-12-07 17:53:02 +00:00
ac->spr.pal -= 7;
ac->spr.shade = pFlash->shade;
}
break;
}
}
loc_1868A:
if (edi != -1)
{
2021-11-22 17:51:22 +00:00
sFlash[edi].next = pFlash->next;
}
if (nFlash == nFirstFlash)
{
2020-11-30 00:05:17 +00:00
nFirstFlash = sFlash[nFirstFlash].next;
}
if (nFlash == nLastFlash)
{
nLastFlash = edi;
}
2021-12-07 18:05:40 +00:00
pFlash->pActor = nullptr;
2020-11-30 00:05:17 +00:00
sFlash.Release(nFlash);
}
}
2021-11-22 18:08:49 +00:00
void AddGlow(sectortype* pSector, int nVal)
{
if (nGlowCount >= kMaxGlows) {
return;
}
sGlow[nGlowCount].nThreshold = nVal;
2021-11-22 18:08:49 +00:00
sGlow[nGlowCount].pSector = pSector;
sGlow[nGlowCount].nShade = -1;
sGlow[nGlowCount].nCounter = 0;
nGlowCount++;
}
// ok
2021-11-22 18:14:26 +00:00
void AddFlicker(sectortype* pSector, int nVal)
{
if (nFlickerCount >= kMaxFlickers) {
return;
}
sFlicker[nFlickerCount].nShade = nVal;
2021-11-22 18:14:26 +00:00
sFlicker[nFlickerCount].pSector = pSector;
if (nVal >= 25) {
nVal = 24;
}
sFlicker[nFlickerCount].nMask = flickermask[nVal];
nFlickerCount++;
}
void DoGlows()
{
bDoGlows++;
if (bDoGlows < 3) {
return;
}
bDoGlows = 0;
for (int i = 0; i < nGlowCount; i++)
{
sGlow[i].nCounter++;
2021-11-22 18:08:49 +00:00
auto pSector = sGlow[i].pSector;
int nShade = sGlow[i].nShade;
if (sGlow[i].nCounter >= sGlow[i].nThreshold)
{
sGlow[i].nCounter = 0;
sGlow[i].nShade = -sGlow[i].nShade;
}
2021-11-22 18:08:49 +00:00
pSector->ceilingshade += nShade;
pSector->floorshade += nShade;
2021-11-22 18:08:49 +00:00
for(auto& wal : wallsofsector(pSector))
{
2021-11-21 08:31:40 +00:00
wal.shade += nShade;
}
}
}
void DoFlickers()
{
bDoFlicks ^= 1;
if (!bDoFlicks) {
return;
}
for (int i = 0; i < nFlickerCount; i++)
{
2021-11-22 18:14:26 +00:00
auto pSector = sFlicker[i].pSector;
unsigned int eax = (sFlicker[i].nMask & 1);
unsigned int edx = (sFlicker[i].nMask & 1) << 31;
unsigned int ebp = sFlicker[i].nMask >> 1;
ebp |= edx;
edx = ebp & 1;
sFlicker[i].nMask = ebp;
if (edx ^ eax)
{
int shade;
if (eax)
{
shade = sFlicker[i].nShade;
}
else
{
shade = -sFlicker[i].nShade;
}
2021-11-22 18:14:26 +00:00
pSector->ceilingshade += shade;
pSector->floorshade += shade;
2021-11-22 18:14:26 +00:00
for(auto& wal : wallsofsector(pSector))
{
2021-11-21 08:31:40 +00:00
wal.shade += shade;
}
}
}
}
void AddFlow(sectortype* pSector, int nSpeed, int b, int nAngle)
{
if (nFlowCount >= kMaxFlows || b >= 2)
return;
int nFlow = nFlowCount;
nFlowCount++;
int nPic = pSector->floorpicnum;
sFlowInfo[nFlow].xacc = (tileWidth(nPic) << 14) - 1;
sFlowInfo[nFlow].yacc = (tileHeight(nPic) << 14) - 1;
sFlowInfo[nFlow].angcos = -bcos(nAngle) * nSpeed;
sFlowInfo[nFlow].angsin = bsin(nAngle) * nSpeed;
sFlowInfo[nFlow].pSector = pSector;
StartInterpolation(pSector, b ? Interp_Sect_CeilingPanX : Interp_Sect_FloorPanX);
StartInterpolation(pSector, b ? Interp_Sect_CeilingPanY : Interp_Sect_FloorPanY);
sFlowInfo[nFlow].ydelta = 0;
sFlowInfo[nFlow].xdelta = 0;
sFlowInfo[nFlow].type = b;
}
void AddFlow(walltype* pWall, int nSpeed, int b, int nAngle)
{
if (nFlowCount >= kMaxFlows || b < 2)
return;
int nFlow = nFlowCount;
nFlowCount++;
StartInterpolation(pWall, Interp_Wall_PanX);
StartInterpolation(pWall, Interp_Wall_PanY);
if (b == 2) {
nAngle = 512;
}
else {
nAngle = 1536;
}
int nPic = pWall->picnum;
sFlowInfo[nFlow].xacc = (tileWidth(nPic) * pWall->xrepeat) << 8;
sFlowInfo[nFlow].yacc = (tileHeight(nPic) * pWall->yrepeat) << 8;
sFlowInfo[nFlow].angcos = -bcos(nAngle) * nSpeed;
sFlowInfo[nFlow].angsin = bsin(nAngle) * nSpeed;
sFlowInfo[nFlow].pWall = pWall;
sFlowInfo[nFlow].ydelta = 0;
sFlowInfo[nFlow].xdelta = 0;
sFlowInfo[nFlow].type = b;
}
void DoFlows()
{
for (int i = 0; i < nFlowCount; i++)
{
sFlowInfo[i].xdelta += sFlowInfo[i].angcos;
sFlowInfo[i].ydelta += sFlowInfo[i].angsin;
switch (sFlowInfo[i].type)
{
case 0:
{
sFlowInfo[i].xdelta &= sFlowInfo[i].xacc;
sFlowInfo[i].ydelta &= sFlowInfo[i].yacc;
auto pSector =sFlowInfo[i].pSector;
pSector->setfloorxpan(sFlowInfo[i].xdelta / 16384.f);
pSector->setfloorypan(sFlowInfo[i].ydelta / 16384.f);
break;
}
case 1:
{
auto pSector = sFlowInfo[i].pSector;
pSector->setceilingxpan(sFlowInfo[i].xdelta / 16384.f);
pSector->setceilingypan(sFlowInfo[i].ydelta / 16384.f);
sFlowInfo[i].xdelta &= sFlowInfo[i].xacc;
sFlowInfo[i].ydelta &= sFlowInfo[i].yacc;
break;
}
case 2:
{
auto pWall = sFlowInfo[i].pWall;
pWall->setxpan(sFlowInfo[i].xdelta / 16384.f);
pWall->setypan(sFlowInfo[i].ydelta / 16384.f);
if (sFlowInfo[i].xdelta < 0)
{
sFlowInfo[i].xdelta += sFlowInfo[i].xacc;
}
if (sFlowInfo[i].ydelta < 0)
{
sFlowInfo[i].ydelta += sFlowInfo[i].yacc;
}
break;
}
case 3:
{
auto pWall = sFlowInfo[i].pWall;
pWall->setxpan(sFlowInfo[i].xdelta / 16384.f);
pWall->setypan(sFlowInfo[i].ydelta / 16384.f);
if (sFlowInfo[i].xdelta >= sFlowInfo[i].xacc)
{
sFlowInfo[i].xdelta -= sFlowInfo[i].xacc;
}
if (sFlowInfo[i].ydelta >= sFlowInfo[i].yacc)
{
sFlowInfo[i].ydelta -= sFlowInfo[i].yacc;
}
break;
}
}
}
}
void DoLights()
{
DoFlickers();
DoGlows();
DoFlows();
}
void SetTorch(int nPlayer, int bTorchOnOff)
{
if (bTorchOnOff == bTorch) {
return;
}
if (nPlayer != nLocalPlayer) {
return;
}
if (bTorchOnOff == 2) {
bTorch = !bTorch;
}
else {
bTorch = bTorchOnOff;
}
if (bTorch) {
PlayLocalSound(StaticSound[kSoundTorchOn], 0);
}
const char* buf = bTorch ? "TXT_EX_TORCHLIT" : "TXT_EX_TORCHOUT";
StatusMessage(150, GStrings(buf));
}
void BuildFlash(int nPlayer, int nVal)
{
if (nPlayer == nLocalPlayer)
{
flash = nVal;
flash = -nVal; // ???
}
}
END_PS_NS