raze/source/games/sw/src/light.cpp

435 lines
12 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
Copyright (C) 1997, 2005 - 3D Realms Entertainment
This file is part of Shadow Warrior version 1.2
Shadow Warrior is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
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.
Original Source: 1997 - Frank Maddin and Jim Norwood
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
*/
//-------------------------------------------------------------------------
#include "ns.h"
#include "build.h"
#include "names2.h"
#include "game.h"
#include "light.h"
BEGIN_SW_NS
/* #define LIGHT_Match(sp) (SP_TAG2((sp))) */
/* #define LIGHT_Type(sp) (SP_TAG3((sp))) */
/* #define LIGHT_MaxTics(sp) (SP_TAG4((sp))) */
/* #define LIGHT_MaxBright(sp) (SP_TAG5((sp))) */
/* #define LIGHT_MaxDark(sp) (SP_TAG6((sp))) */
/* #define LIGHT_ShadeInc(sp) (SP_TAG7((sp))) */
/* */
/* #define LIGHT_Dir(sp) (!!(TEST((sp)->extra, SPRX_BOOL10))) */
/* #define LIGHT_DirChange(sp) (FLIP((sp)->extra, SPRX_BOOL10)) */
/* */
/* #define LIGHT_Shade(sp) ((sp)->shade) */
/* #define LIGHT_FloorShade(sp) ((sp)->xoffset) */
/* #define LIGHT_CeilingShade(sp) ((sp)->yoffset) */
/* #define LIGHT_Tics(sp) ((sp)->z) */
/* */
/* #define LIGHT_DiffuseNum(sp) (SP_TAG3((sp))) */
/* #define LIGHT_DiffuseMult(sp) (SP_TAG4((sp))) */
2021-11-01 13:44:27 +00:00
void SectorLightShade(DSWActor* actor, short intensity)
{
2021-11-02 17:58:02 +00:00
auto u = actor->hasU()? actor->u() : nullptr;
2021-11-01 13:44:27 +00:00
auto sp = &actor->s();
int8_t* wall_shade;
short base_shade;
short wallcount;
if (TEST_BOOL8(sp))
intensity = -intensity;
if (!TEST_BOOL2(sp))
{
if (!TEST_BOOL6(sp))
sp->sector()->floorpal = sp->pal;
sp->sector()->floorshade = LIGHT_FloorShade(sp) + intensity; // floor change
}
if (!TEST_BOOL3(sp))
{
if (!TEST_BOOL6(sp))
sp->sector()->ceilingpal = sp->pal;
sp->sector()->ceilingshade = LIGHT_CeilingShade(sp) + intensity; // ceiling change
}
// change wall
if (!TEST_BOOL4(sp))
{
2021-11-02 17:58:02 +00:00
ASSERT(u && u->WallShade.Data());
wall_shade = u->WallShade.Data();
2021-11-24 19:11:50 +00:00
int wallcount = 0;
2021-11-24 19:11:50 +00:00
for(auto &wal : wallsofsector(sp->sector()))
{
base_shade = wall_shade[wallcount];
2021-11-24 19:11:50 +00:00
wal.shade = base_shade + intensity;
if (!TEST_BOOL6(sp))
2021-11-24 19:11:50 +00:00
wal.pal = sp->pal;
wallcount++;
if (TEST(sp->extra, SPRX_BOOL5))
{
2021-11-24 19:11:50 +00:00
if (wal.twoSided())
{
2021-11-24 19:11:50 +00:00
auto nextWall = wal.nextWall();
base_shade = wall_shade[wallcount];
2021-11-24 19:11:50 +00:00
nextWall->shade = base_shade + intensity;
if (!TEST_BOOL6(sp))
2021-11-24 19:11:50 +00:00
nextWall->pal = sp->pal;
wallcount++;
}
}
}
}
}
2021-11-01 13:44:27 +00:00
void DiffuseLighting(DSWActor* actor)
{
2021-11-01 13:44:27 +00:00
auto sp = &actor->s();
2020-10-15 15:45:07 +00:00
int i;
short count;
short shade;
SPRITEp dsp;
// diffused lighting
count = 0;
2021-11-01 13:44:27 +00:00
SWStatIterator it(STAT_LIGHTING_DIFFUSE);
while (auto itActor = it.Next())
{
2021-11-01 13:44:27 +00:00
dsp = &itActor->s();
// make sure matchs match
if (LIGHT_Match(dsp) != LIGHT_Match(sp))
continue;
shade = sp->shade + ((LIGHT_DiffuseNum(dsp) + 1) * LIGHT_DiffuseMult(dsp));
if (shade > LIGHT_MaxDark(sp))
shade = LIGHT_MaxDark(sp);
if (!TEST_BOOL6(dsp))
dsp->pal = sp->pal;
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, shade);
count++;
}
}
void DoLightingMatch(short match, short state)
{
SPRITEp sp;
2021-11-01 13:44:27 +00:00
SWStatIterator it(STAT_LIGHTING);
while (auto itActor = it.Next())
{
2021-11-01 13:44:27 +00:00
auto u = itActor->u();
sp = &itActor->s();
if (LIGHT_Match(sp) != match)
continue;
switch (LIGHT_Type(sp))
{
case LIGHT_CONSTANT:
// initialized
SET_BOOL9(sp);
// toggle
if (state == -1)
state = !TEST_BOOL1(sp);
if (state == ON)
{
SET_BOOL1(sp);
sp->shade = -LIGHT_MaxBright(sp);
2021-10-31 11:20:03 +00:00
sp->pal = u->spal; // on
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
else
{
RESET_BOOL1(sp);
sp->shade = LIGHT_MaxDark(sp);
sp->pal = 0; // off
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
break;
case LIGHT_FLICKER:
case LIGHT_FADE:
// initialized
SET_BOOL9(sp);
// toggle
if (state == -1)
state = !TEST_BOOL1(sp);
if (state == ON)
{
// allow fade or flicker
SET_BOOL1(sp);
}
else
{
RESET_BOOL1(sp);
sp->shade = LIGHT_MaxDark(sp);
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
break;
case LIGHT_FADE_TO_ON_OFF:
// initialized
SET_BOOL9(sp);
// toggle
//if (state == -1)
// state = !TEST_BOOL1(sp);
if (state == ON)
{
if (LIGHT_Dir(sp) == 1)
{
LIGHT_DirChange(sp);
}
}
else if (state == OFF)
{
if (LIGHT_Dir(sp) == 0)
{
LIGHT_DirChange(sp);
}
}
// allow fade or flicker
SET_BOOL1(sp);
break;
case LIGHT_FLICKER_ON:
// initialized
SET_BOOL9(sp);
// toggle
if (state == -1)
state = !TEST_BOOL1(sp);
if (state == ON)
{
// allow fade or flicker
SET_BOOL1(sp);
}
else
{
// turn it off till next switch
2021-05-12 00:02:26 +00:00
auto spal = sp->pal;
RESET_BOOL1(sp);
sp->pal = 0;
sp->shade = LIGHT_MaxDark(sp);
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
sp->pal = spal;
}
break;
}
}
}
void InitLighting(void)
{
SPRITEp sp;
// processed on level startup
// puts lights in correct state
2021-11-01 13:44:27 +00:00
SWStatIterator it(STAT_LIGHTING);
while (auto actor = it.Next())
{
2021-11-01 13:44:27 +00:00
sp = &actor->s();
if (!TEST_BOOL9(sp))
{
DoLightingMatch(LIGHT_Match(sp), !!TEST_BOOL1(sp));
}
}
}
void DoLighting(void)
{
SPRITEp sp;
2021-11-01 13:44:27 +00:00
SWStatIterator it(STAT_LIGHTING);
while (auto itActor = it.Next())
{
2021-11-01 13:44:27 +00:00
auto u = itActor->u();
sp = &itActor->s();
// on/off test
if (TEST_BOOL1(sp) == OFF)
continue;
switch (LIGHT_Type(sp))
{
case LIGHT_CONSTANT:
break;
case LIGHT_FLICKER:
LIGHT_Tics(sp) += synctics;
while (LIGHT_Tics(sp) >= LIGHT_MaxTics(sp))
{
LIGHT_Tics(sp) -= LIGHT_MaxTics(sp);
if ((RANDOM_P2(128 << 8) >> 8) > 64)
{
sp->shade = -LIGHT_MaxBright(sp) + RandomRange(LIGHT_MaxBright(sp) + LIGHT_MaxDark(sp));
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
else
{
// turn off lighting - even colored lighting
2021-05-12 00:02:26 +00:00
auto spal = sp->pal;
sp->pal = 0;
sp->shade = LIGHT_MaxDark(sp);
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
sp->pal = spal;
}
}
break;
case LIGHT_FADE:
LIGHT_Tics(sp) += synctics;
while (LIGHT_Tics(sp) >= LIGHT_MaxTics(sp))
{
LIGHT_Tics(sp) -= LIGHT_MaxTics(sp);
if (LIGHT_Dir(sp) == 1)
{
sp->shade += LIGHT_ShadeInc(sp);
if (sp->shade >= LIGHT_MaxDark(sp))
LIGHT_DirChange(sp);
}
else
{
sp->shade -= LIGHT_ShadeInc(sp);
if (sp->shade <= -LIGHT_MaxBright(sp))
LIGHT_DirChange(sp);
}
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
break;
case LIGHT_FADE_TO_ON_OFF:
LIGHT_Tics(sp) += synctics;
while (LIGHT_Tics(sp) >= LIGHT_MaxTics(sp))
{
LIGHT_Tics(sp) -= LIGHT_MaxTics(sp);
if (LIGHT_Dir(sp) == 1)
{
sp->shade += LIGHT_ShadeInc(sp);
if (sp->shade >= LIGHT_MaxDark(sp))
{
sp->pal = 0; // off
LIGHT_DirChange(sp);
// stop it until switch is hit
RESET_BOOL1(sp);
}
}
else
{
sp->shade -= LIGHT_ShadeInc(sp);
2021-10-31 11:20:03 +00:00
sp->pal = u->spal; // on
if (sp->shade <= -LIGHT_MaxBright(sp))
{
LIGHT_DirChange(sp);
// stop it until switch is hit
RESET_BOOL1(sp);
}
}
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
break;
case LIGHT_FLICKER_ON:
LIGHT_Tics(sp) += synctics;
while (LIGHT_Tics(sp) >= LIGHT_MaxTics(sp))
{
LIGHT_Tics(sp) -= LIGHT_MaxTics(sp);
if ((RANDOM_P2(128 << 8) >> 8) > 64)
{
sp->shade = -LIGHT_MaxBright(sp) + RandomRange(LIGHT_MaxBright(sp) + LIGHT_MaxDark(sp));
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
}
else
{
// turn off lighting - even colored lighting
2021-05-12 00:02:26 +00:00
auto spal = sp->pal;
sp->pal = 0;
sp->shade = LIGHT_MaxDark(sp);
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
sp->pal = spal;
}
if ((RANDOM_P2(128 << 8) >> 8) < 8)
{
// set to full brightness
sp->shade = -LIGHT_MaxBright(sp);
2021-11-01 13:44:27 +00:00
SectorLightShade(itActor, sp->shade);
DiffuseLighting(itActor);
// turn it off until a swith happens
RESET_BOOL1(sp);
}
}
break;
}
}
}
END_SW_NS