2016-09-14 18:01:13 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// Copyright(C) 2002-2016 Christoph Oelckers
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 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 Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with this program. If not, see http://www.gnu.org/licenses/
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------------
|
|
|
|
//
|
2013-06-23 07:49:34 +00:00
|
|
|
/*
|
|
|
|
** gl_light.cpp
|
|
|
|
** Light level / fog management / dynamic lights
|
|
|
|
**
|
2016-09-14 18:01:13 +00:00
|
|
|
**/
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
#include "gl/system/gl_system.h"
|
2013-09-03 16:29:39 +00:00
|
|
|
#include "gl/system/gl_interface.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
#include "gl/system/gl_cvars.h"
|
|
|
|
#include "gl/data/gl_data.h"
|
|
|
|
#include "gl/renderer/gl_colormap.h"
|
|
|
|
#include "gl/renderer/gl_lightdata.h"
|
|
|
|
#include "gl/renderer/gl_renderstate.h"
|
|
|
|
#include "gl/shaders/gl_shader.h"
|
|
|
|
#include "gl/scene/gl_portal.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "g_level.h"
|
|
|
|
#include "r_sky.h"
|
2017-01-08 17:45:30 +00:00
|
|
|
#include "g_levellocals.h"
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
// externally settable lighting properties
|
|
|
|
static float distfogtable[2][256]; // light to fog conversion table for black fog
|
|
|
|
|
|
|
|
CVAR(Int, gl_weaponlight, 8, CVAR_ARCHIVE);
|
2014-10-05 07:40:36 +00:00
|
|
|
CUSTOM_CVAR(Bool, gl_enhanced_nightvision, true, CVAR_ARCHIVE|CVAR_NOINITCALL)
|
|
|
|
{
|
|
|
|
// The fixed colormap state needs to be reset because if this happens when
|
|
|
|
// a shader is set to CM_LITE or CM_TORCH it won't register the change in behavior caused by this CVAR.
|
2016-09-01 15:14:51 +00:00
|
|
|
if (GLRenderer != nullptr && GLRenderer->mShaderManager != nullptr)
|
2015-11-09 12:49:17 +00:00
|
|
|
{
|
|
|
|
GLRenderer->mShaderManager->ResetFixedColormap();
|
|
|
|
}
|
2014-10-05 07:40:36 +00:00
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
CVAR(Bool, gl_brightfog, false, CVAR_ARCHIVE);
|
2016-12-07 22:17:18 +00:00
|
|
|
CVAR(Bool, gl_lightadditivesurfaces, false, CVAR_ARCHIVE);
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Sets up the fog tables
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
CUSTOM_CVAR (Int, gl_distfog, 70, CVAR_ARCHIVE | CVAR_GLOBALCONFIG)
|
|
|
|
{
|
|
|
|
for (int i=0;i<256;i++)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (i<164)
|
|
|
|
{
|
|
|
|
distfogtable[0][i]= (gl_distfog>>1) + (gl_distfog)*(164-i)/164;
|
|
|
|
}
|
|
|
|
else if (i<230)
|
|
|
|
{
|
|
|
|
distfogtable[0][i]= (gl_distfog>>1) - (gl_distfog>>1)*(i-164)/(230-164);
|
|
|
|
}
|
|
|
|
else distfogtable[0][i]=0;
|
|
|
|
|
|
|
|
if (i<128)
|
|
|
|
{
|
|
|
|
distfogtable[1][i]= 6.f + (gl_distfog>>1) + (gl_distfog)*(128-i)/48;
|
|
|
|
}
|
|
|
|
else if (i<216)
|
|
|
|
{
|
|
|
|
distfogtable[1][i]= (216.f-i) / ((216.f-128.f)) * gl_distfog / 10;
|
|
|
|
}
|
|
|
|
else distfogtable[1][i]=0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CUSTOM_CVAR(Int,gl_fogmode,1,CVAR_ARCHIVE|CVAR_NOINITCALL)
|
|
|
|
{
|
|
|
|
if (self>2) self=2;
|
|
|
|
if (self<0) self=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
CUSTOM_CVAR(Int, gl_lightmode, 3 ,CVAR_ARCHIVE|CVAR_NOINITCALL)
|
|
|
|
{
|
|
|
|
int newself = self;
|
|
|
|
if (newself > 4) newself=8; // use 8 for software lighting to avoid conflicts with the bit mask
|
|
|
|
if (newself < 0) newself=0;
|
|
|
|
if (self != newself) self = newself;
|
|
|
|
glset.lightmode = newself;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Sets render state to draw the given render style
|
|
|
|
// includes: Texture mode, blend equation and blend mode
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void gl_GetRenderStyle(FRenderStyle style, bool drawopaque, bool allowcolorblending,
|
|
|
|
int *tm, int *sb, int *db, int *be)
|
|
|
|
{
|
|
|
|
static int blendstyles[] = { GL_ZERO, GL_ONE, GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA };
|
|
|
|
static int renderops[] = { 0, GL_FUNC_ADD, GL_FUNC_SUBTRACT, GL_FUNC_REVERSE_SUBTRACT, -1, -1, -1, -1,
|
|
|
|
-1, -1, -1, -1, -1, -1, -1, -1 };
|
|
|
|
|
|
|
|
int srcblend = blendstyles[style.SrcAlpha&3];
|
|
|
|
int dstblend = blendstyles[style.DestAlpha&3];
|
|
|
|
int blendequation = renderops[style.BlendOp&15];
|
|
|
|
int texturemode = drawopaque? TM_OPAQUE : TM_MODULATE;
|
|
|
|
|
2014-09-09 11:21:36 +00:00
|
|
|
if (style.Flags & STYLEF_RedIsAlpha)
|
|
|
|
{
|
|
|
|
texturemode = TM_REDTOALPHA;
|
|
|
|
}
|
|
|
|
else if (style.Flags & STYLEF_ColorIsFixed)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
texturemode = TM_MASK;
|
|
|
|
}
|
|
|
|
else if (style.Flags & STYLEF_InvertSource)
|
|
|
|
{
|
2014-05-11 17:44:19 +00:00
|
|
|
// The only place where InvertSource is used is for inverted sprites with the infrared powerup.
|
2014-05-11 15:56:38 +00:00
|
|
|
texturemode = TM_INVERSE;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (blendequation == -1)
|
|
|
|
{
|
|
|
|
srcblend = GL_DST_COLOR;
|
|
|
|
dstblend = GL_ONE_MINUS_SRC_ALPHA;
|
|
|
|
blendequation = GL_FUNC_ADD;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (allowcolorblending && srcblend == GL_SRC_ALPHA && dstblend == GL_ONE && blendequation == GL_FUNC_ADD)
|
|
|
|
{
|
|
|
|
srcblend = GL_SRC_COLOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
*tm = texturemode;
|
|
|
|
*be = blendequation;
|
|
|
|
*sb = srcblend;
|
|
|
|
*db = dstblend;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Get current light level
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
int gl_CalcLightLevel(int lightlevel, int rellight, bool weapon)
|
|
|
|
{
|
|
|
|
int light;
|
|
|
|
|
|
|
|
if (lightlevel == 0) return 0;
|
|
|
|
|
|
|
|
if ((glset.lightmode & 2) && lightlevel < 192 && !weapon)
|
|
|
|
{
|
2016-12-25 11:58:29 +00:00
|
|
|
if (lightlevel > 100)
|
|
|
|
{
|
|
|
|
light = xs_CRoundToInt(192.f - (192 - lightlevel)* 1.87f);
|
|
|
|
if (light + rellight < 20)
|
|
|
|
{
|
|
|
|
light = 20 + (light + rellight - 20) / 5;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
light += rellight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
light = (lightlevel + rellight) / 5;
|
|
|
|
}
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-25 11:58:29 +00:00
|
|
|
light=lightlevel+rellight;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2016-12-25 11:58:29 +00:00
|
|
|
return clamp(light, 0, 255);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Get current light color
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2014-05-11 22:13:19 +00:00
|
|
|
static PalEntry gl_CalcLightColor(int light, PalEntry pe, int blendfactor)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
int r,g,b;
|
|
|
|
|
2014-05-11 22:13:19 +00:00
|
|
|
if (glset.lightmode == 8)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
return pe;
|
|
|
|
}
|
|
|
|
else if (blendfactor == 0)
|
|
|
|
{
|
|
|
|
r = pe.r * light / 255;
|
|
|
|
g = pe.g * light / 255;
|
|
|
|
b = pe.b * light / 255;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-05-11 14:49:17 +00:00
|
|
|
// This is what Legacy does with colored light in 3D volumes. No, it doesn't really make sense...
|
|
|
|
// It also doesn't translate well to software style lighting.
|
2013-06-23 07:49:34 +00:00
|
|
|
int mixlight = light * (255 - blendfactor);
|
|
|
|
|
|
|
|
r = (mixlight + pe.r * blendfactor) / 255;
|
|
|
|
g = (mixlight + pe.g * blendfactor) / 255;
|
|
|
|
b = (mixlight + pe.b * blendfactor) / 255;
|
|
|
|
}
|
2017-03-09 18:54:41 +00:00
|
|
|
return PalEntry(255, uint8_t(r), uint8_t(g), uint8_t(b));
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// set current light color
|
|
|
|
//
|
|
|
|
//==========================================================================
|
2017-03-12 20:57:39 +00:00
|
|
|
void gl_SetColor(int sectorlightlevel, int rellight, bool fullbright, const FColormap &cm, float alpha, bool weapon)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2017-03-12 20:57:39 +00:00
|
|
|
if (fullbright)
|
2014-05-11 22:13:19 +00:00
|
|
|
{
|
|
|
|
gl_RenderState.SetColorAlpha(0xffffff, alpha, 0);
|
|
|
|
gl_RenderState.SetSoftLightLevel(255);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int hwlightlevel = gl_CalcLightLevel(sectorlightlevel, rellight, weapon);
|
2017-03-15 15:47:42 +00:00
|
|
|
PalEntry pe = gl_CalcLightColor(hwlightlevel, cm.LightColor, cm.BlendFactor);
|
|
|
|
gl_RenderState.SetColorAlpha(pe, alpha, cm.Desaturation);
|
2014-05-11 22:13:19 +00:00
|
|
|
gl_RenderState.SetSoftLightLevel(gl_ClampLight(sectorlightlevel + rellight));
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// calculates the current fog density
|
|
|
|
//
|
|
|
|
// Rules for fog:
|
|
|
|
//
|
|
|
|
// 1. If bit 4 of gl_lightmode is set always use the level's fog density.
|
|
|
|
// This is what Legacy's GL render does.
|
|
|
|
// 2. black fog means no fog and always uses the distfogtable based on the level's fog density setting
|
|
|
|
// 3. If outside fog is defined and the current fog color is the same as the outside fog
|
|
|
|
// the engine always uses the outside fog density to make the fog uniform across the level.
|
|
|
|
// If the outside fog's density is undefined it uses the level's fog density and if that is
|
|
|
|
// not defined it uses a default of 70.
|
|
|
|
// 4. If a global fog density is specified it is being used for all fog on the level
|
|
|
|
// 5. If none of the above apply fog density is based on the light level as for the software renderer.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-12-29 00:12:17 +00:00
|
|
|
float gl_GetFogDensity(int lightlevel, PalEntry fogcolor, int sectorfogdensity)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
float density;
|
|
|
|
|
2016-12-29 00:12:17 +00:00
|
|
|
if (glset.lightmode & 4)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
// uses approximations of Legacy's default settings.
|
2017-03-14 12:54:24 +00:00
|
|
|
density = level.fogdensity ? level.fogdensity : 18;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2017-01-28 20:49:52 +00:00
|
|
|
else if (sectorfogdensity != 0)
|
|
|
|
{
|
2017-01-29 10:45:28 +00:00
|
|
|
// case 1: Sector has an explicit fog density set.
|
2017-01-28 20:49:52 +00:00
|
|
|
density = sectorfogdensity;
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
else if ((fogcolor.d & 0xffffff) == 0)
|
|
|
|
{
|
2017-01-29 10:45:28 +00:00
|
|
|
// case 2: black fog
|
2017-03-14 17:31:11 +00:00
|
|
|
if (glset.lightmode != 8 && !(level.flags3 & LEVEL3_NOLIGHTFADE))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-12-29 00:12:17 +00:00
|
|
|
density = distfogtable[glset.lightmode != 0][gl_ClampLight(lightlevel)];
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
density = 0;
|
|
|
|
}
|
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (level.outsidefogdensity != 0 && APART(level.info->outsidefog) != 0xff && (fogcolor.d & 0xffffff) == (level.info->outsidefog & 0xffffff))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2016-12-29 00:12:17 +00:00
|
|
|
// case 3. outsidefogdensity has already been set as needed
|
2017-03-14 12:54:24 +00:00
|
|
|
density = level.outsidefogdensity;
|
2016-12-29 00:12:17 +00:00
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (level.fogdensity != 0)
|
2016-12-29 00:12:17 +00:00
|
|
|
{
|
|
|
|
// case 4: level has fog density set
|
2017-03-14 12:54:24 +00:00
|
|
|
density = level.fogdensity;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
else if (lightlevel < 248)
|
|
|
|
{
|
2016-12-29 00:12:17 +00:00
|
|
|
// case 5: use light level
|
|
|
|
density = clamp<int>(255 - lightlevel, 30, 255);
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2016-12-29 00:12:17 +00:00
|
|
|
else
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
density = 0.f;
|
|
|
|
}
|
|
|
|
return density;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Check if the current linedef is a candidate for a fog boundary
|
|
|
|
//
|
2014-05-11 14:49:17 +00:00
|
|
|
// Requirements for a fog boundary:
|
|
|
|
// - front sector has no fog
|
|
|
|
// - back sector has fog
|
|
|
|
// - at least one of both does not have a sky ceiling.
|
|
|
|
//
|
2013-06-23 07:49:34 +00:00
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool gl_CheckFog(sector_t *frontsector, sector_t *backsector)
|
|
|
|
{
|
2014-05-11 14:49:17 +00:00
|
|
|
if (frontsector == backsector) return false; // there can't be a boundary if both sides are in the same sector.
|
|
|
|
|
2013-06-23 07:49:34 +00:00
|
|
|
// Check for fog boundaries. This needs a few more checks for the sectors
|
|
|
|
|
2017-03-15 15:47:42 +00:00
|
|
|
PalEntry fogcolor = frontsector->Colormap.FadeColor;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
if ((fogcolor.d & 0xffffff) == 0)
|
|
|
|
{
|
2014-05-11 14:49:17 +00:00
|
|
|
return false;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (fogcolor.a != 0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else if (level.outsidefogdensity != 0 && APART(level.info->outsidefog) != 0xff && (fogcolor.d & 0xffffff) == (level.info->outsidefog & 0xffffff))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (level.fogdensity!=0 || (glset.lightmode & 4))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
// case 3: level has fog density set
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// case 4: use light level
|
2014-05-11 14:49:17 +00:00
|
|
|
if (frontsector->lightlevel >= 248) return false;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
2017-03-15 15:47:42 +00:00
|
|
|
fogcolor = backsector->Colormap.FadeColor;
|
2013-06-23 07:49:34 +00:00
|
|
|
|
|
|
|
if ((fogcolor.d & 0xffffff) == 0)
|
|
|
|
{
|
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (level.outsidefogdensity != 0 && APART(level.info->outsidefog) != 0xff && (fogcolor.d & 0xffffff) == (level.info->outsidefog & 0xffffff))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
2014-05-11 14:49:17 +00:00
|
|
|
return false;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
2017-03-14 12:54:24 +00:00
|
|
|
else if (level.fogdensity!=0 || (glset.lightmode & 4))
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
// case 3: level has fog density set
|
2014-05-11 14:49:17 +00:00
|
|
|
return false;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// case 4: use light level
|
2014-05-11 14:49:17 +00:00
|
|
|
if (backsector->lightlevel < 248) return false;
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// in all other cases this might create more problems than it solves.
|
2014-05-11 14:49:17 +00:00
|
|
|
return ((frontsector->GetTexture(sector_t::ceiling)!=skyflatnum ||
|
2013-06-23 07:49:34 +00:00
|
|
|
backsector->GetTexture(sector_t::ceiling)!=skyflatnum));
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Lighting stuff
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void gl_SetShaderLight(float level, float olight)
|
|
|
|
{
|
|
|
|
const float MAXDIST = 256.f;
|
|
|
|
const float THRESHOLD = 96.f;
|
|
|
|
const float FACTOR = 0.75f;
|
|
|
|
|
|
|
|
if (level > 0)
|
|
|
|
{
|
|
|
|
float lightdist, lightfactor;
|
|
|
|
|
|
|
|
if (olight < THRESHOLD)
|
|
|
|
{
|
|
|
|
lightdist = (MAXDIST/2) + (olight * MAXDIST / THRESHOLD / 2);
|
|
|
|
olight = THRESHOLD;
|
|
|
|
}
|
|
|
|
else lightdist = MAXDIST;
|
|
|
|
|
|
|
|
lightfactor = 1.f + ((olight/level) - 1.f) * FACTOR;
|
|
|
|
if (lightfactor == 1.f) lightdist = 0.f; // save some code in the shader
|
|
|
|
gl_RenderState.SetLightParms(lightfactor, lightdist);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gl_RenderState.SetLightParms(1.f, 0.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Sets the fog for the current polygon
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-03-12 20:57:39 +00:00
|
|
|
void gl_SetFog(int lightlevel, int rellight, bool fullbright, const FColormap *cmap, bool isadditive)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
PalEntry fogcolor;
|
|
|
|
float fogdensity;
|
|
|
|
|
|
|
|
if (level.flags&LEVEL_HASFADETABLE)
|
|
|
|
{
|
|
|
|
fogdensity=70;
|
|
|
|
fogcolor=0x808080;
|
|
|
|
}
|
2017-03-12 20:57:39 +00:00
|
|
|
else if (cmap != NULL && !fullbright)
|
2013-06-23 07:49:34 +00:00
|
|
|
{
|
|
|
|
fogcolor = cmap->FadeColor;
|
2017-03-15 15:47:42 +00:00
|
|
|
fogdensity = gl_GetFogDensity(lightlevel, fogcolor, cmap->FogDensity);
|
2013-06-23 07:49:34 +00:00
|
|
|
fogcolor.a=0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fogcolor = 0;
|
|
|
|
fogdensity = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make fog a little denser when inside a skybox
|
|
|
|
if (GLPortal::inskybox) fogdensity+=fogdensity/2;
|
|
|
|
|
|
|
|
|
|
|
|
// no fog in enhanced vision modes!
|
|
|
|
if (fogdensity==0 || gl_fogmode == 0)
|
|
|
|
{
|
|
|
|
gl_RenderState.EnableFog(false);
|
|
|
|
gl_RenderState.SetFog(0,0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (glset.lightmode == 2 && fogcolor == 0)
|
|
|
|
{
|
|
|
|
float light = gl_CalcLightLevel(lightlevel, rellight, false);
|
|
|
|
gl_SetShaderLight(light, lightlevel);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gl_RenderState.SetLightParms(1.f, 0.f);
|
|
|
|
}
|
|
|
|
|
|
|
|
// For additive rendering using the regular fog color here would mean applying it twice
|
|
|
|
// so always use black
|
|
|
|
if (isadditive)
|
|
|
|
{
|
|
|
|
fogcolor=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
gl_RenderState.EnableFog(true);
|
|
|
|
gl_RenderState.SetFog(fogcolor, fogdensity);
|
|
|
|
|
|
|
|
// Korshun: fullbright fog like in software renderer.
|
2013-12-23 08:56:16 +00:00
|
|
|
if (glset.lightmode == 8 && glset.brightfog && fogdensity != 0 && fogcolor != 0)
|
2014-05-11 22:13:19 +00:00
|
|
|
{
|
|
|
|
gl_RenderState.SetSoftLightLevel(255);
|
|
|
|
}
|
2013-06-23 07:49:34 +00:00
|
|
|
}
|
|
|
|
}
|