mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-22 00:41:11 +00:00
9db80259d1
The static variable meant that Fog_GetColor was not thread-safe (though multiple calls in the one thread look to be ok for now). However, this change takes it one step closer to being more generally usable. Patch found in an old stash.
354 lines
7.3 KiB
C
354 lines
7.3 KiB
C
/*
|
|
Copyright (C) 1996-2001 Id Software, Inc.
|
|
Copyright (C) 2002-2009 John Fitzgibbons and others
|
|
|
|
This program 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
//gl_fog.c -- global and volumetric fog
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#define NH_DEFINE
|
|
#include "namehack.h"
|
|
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
#ifdef HAVE_STRINGS_H
|
|
# include <strings.h>
|
|
#endif
|
|
|
|
#include "QF/cmd.h"
|
|
#include "QF/qfplist.h"
|
|
#include "QF/render.h"
|
|
#include "QF/sys.h"
|
|
|
|
#include "QF/GL/defines.h"
|
|
#include "QF/GL/funcs.h"
|
|
|
|
#include "compat.h"
|
|
#include "r_internal.h"
|
|
|
|
//==============================================================================
|
|
//
|
|
// GLOBAL FOG
|
|
//
|
|
//==============================================================================
|
|
|
|
static float fog_density;
|
|
static float fog_red;
|
|
static float fog_green;
|
|
static float fog_blue;
|
|
|
|
static float old_density;
|
|
static float old_red;
|
|
static float old_green;
|
|
static float old_blue;
|
|
|
|
static float fade_time; //duration of fade
|
|
static float fade_done; //time when fade will be done
|
|
|
|
/*
|
|
Fog_Update
|
|
|
|
update internal variables
|
|
*/
|
|
void
|
|
gl_Fog_Update (float density, float red, float green, float blue, float time)
|
|
{
|
|
//save previous settings for fade
|
|
if (time > 0) {
|
|
//check for a fade in progress
|
|
if (fade_done > vr_data.realtime) {
|
|
float f;
|
|
|
|
f = (fade_done - vr_data.realtime) / fade_time;
|
|
old_density = f * old_density + (1.0 - f) * fog_density;
|
|
old_red = f * old_red + (1.0 - f) * fog_red;
|
|
old_green = f * old_green + (1.0 - f) * fog_green;
|
|
old_blue = f * old_blue + (1.0 - f) * fog_blue;
|
|
} else {
|
|
old_density = fog_density;
|
|
old_red = fog_red;
|
|
old_green = fog_green;
|
|
old_blue = fog_blue;
|
|
}
|
|
}
|
|
|
|
fog_density = density;
|
|
fog_red = red;
|
|
fog_green = green;
|
|
fog_blue = blue;
|
|
fade_time = time;
|
|
fade_done = vr_data.realtime + time;
|
|
}
|
|
|
|
/*
|
|
Fog_FogCommand_f
|
|
|
|
handle the 'fog' console command
|
|
*/
|
|
static void
|
|
Fog_FogCommand_f (void)
|
|
{
|
|
float density = fog_density;
|
|
float red = fog_red;
|
|
float green = fog_green;
|
|
float blue = fog_blue;
|
|
float time = 0.0;
|
|
|
|
switch (Cmd_Argc ()) {
|
|
default:
|
|
case 1:
|
|
Sys_Printf ("usage:\n");
|
|
Sys_Printf (" fog <density>\n");
|
|
Sys_Printf (" fog <red> <green> <blue>\n");
|
|
Sys_Printf (" fog <density> <red> <green> <blue>\n");
|
|
Sys_Printf ("current values:\n");
|
|
Sys_Printf (" \"density\" is \"%f\"\n", fog_density);
|
|
Sys_Printf (" \"red\" is \"%f\"\n", fog_red);
|
|
Sys_Printf (" \"green\" is \"%f\"\n", fog_green);
|
|
Sys_Printf (" \"blue\" is \"%f\"\n", fog_blue);
|
|
return;
|
|
case 2:
|
|
density = atof (Cmd_Argv(1));
|
|
break;
|
|
case 3: //TEST
|
|
density = atof (Cmd_Argv(1));
|
|
time = atof (Cmd_Argv(2));
|
|
break;
|
|
case 4:
|
|
red = atof (Cmd_Argv(1));
|
|
green = atof (Cmd_Argv(2));
|
|
blue = atof (Cmd_Argv(3));
|
|
break;
|
|
case 5:
|
|
density = atof (Cmd_Argv(1));
|
|
red = atof (Cmd_Argv(2));
|
|
green = atof (Cmd_Argv(3));
|
|
blue = atof (Cmd_Argv(4));
|
|
break;
|
|
case 6: //TEST
|
|
density = atof (Cmd_Argv(1));
|
|
red = atof (Cmd_Argv(2));
|
|
green = atof (Cmd_Argv(3));
|
|
blue = atof (Cmd_Argv(4));
|
|
time = atof (Cmd_Argv(5));
|
|
break;
|
|
}
|
|
density = max (0.0, density);
|
|
red = bound (0.0, red, 1.0);
|
|
green = bound (0.0, green, 1.0);
|
|
blue = bound (0.0, blue, 1.0);
|
|
gl_Fog_Update (density, red, green, blue, time);
|
|
}
|
|
|
|
/*
|
|
Fog_ParseWorldspawn
|
|
|
|
called at map load
|
|
*/
|
|
void
|
|
gl_Fog_ParseWorldspawn (plitem_t *worldspawn)
|
|
{
|
|
plitem_t *fog;
|
|
const char *value;
|
|
|
|
//initially no fog
|
|
fog_density = 0.0;
|
|
old_density = 0.0;
|
|
fade_time = 0.0;
|
|
fade_done = 0.0;
|
|
|
|
if (!worldspawn)
|
|
return; // error
|
|
if ((fog = PL_ObjectForKey (worldspawn, "fog"))
|
|
&& (value = PL_String (fog))) {
|
|
sscanf (value, "%f %f %f %f", &fog_density,
|
|
&fog_red, &fog_green, &fog_blue);
|
|
}
|
|
}
|
|
|
|
/*
|
|
Fog_GetColor
|
|
|
|
calculates fog color for this frame, taking into account fade times
|
|
*/
|
|
void
|
|
gl_Fog_GetColor (quat_t fogcolor)
|
|
{
|
|
float f;
|
|
int i;
|
|
|
|
if (fade_done > vr_data.realtime) {
|
|
f = (fade_done - vr_data.realtime) / fade_time;
|
|
fogcolor[0] = f * old_red + (1.0 - f) * fog_red;
|
|
fogcolor[1] = f * old_green + (1.0 - f) * fog_green;
|
|
fogcolor[2] = f * old_blue + (1.0 - f) * fog_blue;
|
|
fogcolor[3] = 1.0;
|
|
} else {
|
|
fogcolor[0] = fog_red;
|
|
fogcolor[1] = fog_green;
|
|
fogcolor[2] = fog_blue;
|
|
fogcolor[3] = 1.0;
|
|
}
|
|
|
|
//find closest 24-bit RGB value, so solid-colored sky can match the fog
|
|
//perfectly
|
|
for (i = 0; i < 3; i++)
|
|
fogcolor[i] = (float) (rint (fogcolor[i] * 255)) / 255.0f;
|
|
}
|
|
|
|
/*
|
|
Fog_GetDensity
|
|
|
|
returns current density of fog
|
|
*/
|
|
float
|
|
gl_Fog_GetDensity (void)
|
|
{
|
|
float f;
|
|
|
|
if (fade_done > vr_data.realtime) {
|
|
f = (fade_done - vr_data.realtime) / fade_time;
|
|
return f * old_density + (1.0 - f) * fog_density;
|
|
} else {
|
|
return fog_density;
|
|
}
|
|
}
|
|
|
|
/*
|
|
Fog_SetupFrame
|
|
|
|
called at the beginning of each frame
|
|
*/
|
|
void
|
|
gl_Fog_SetupFrame (void)
|
|
{
|
|
quat_t fogcolor;
|
|
|
|
gl_Fog_GetColor (fogcolor);
|
|
qfglFogfv (GL_FOG_COLOR, fogcolor);
|
|
qfglFogf (GL_FOG_DENSITY, gl_Fog_GetDensity () / 64.0);
|
|
}
|
|
|
|
/*
|
|
Fog_EnableGFog
|
|
|
|
called before drawing stuff that should be fogged
|
|
*/
|
|
void
|
|
gl_Fog_EnableGFog (void)
|
|
{
|
|
if (gl_Fog_GetDensity () > 0)
|
|
qfglEnable (GL_FOG);
|
|
}
|
|
|
|
/*
|
|
Fog_DisableGFog
|
|
|
|
called after drawing stuff that should be fogged
|
|
*/
|
|
void
|
|
gl_Fog_DisableGFog (void)
|
|
{
|
|
if (gl_Fog_GetDensity () > 0)
|
|
qfglDisable (GL_FOG);
|
|
}
|
|
|
|
/*
|
|
Fog_StartAdditive
|
|
|
|
called before drawing stuff that is additive blended -- sets fog color
|
|
to black
|
|
*/
|
|
void
|
|
gl_Fog_StartAdditive (void)
|
|
{
|
|
vec3_t color = {0, 0, 0};
|
|
|
|
if (gl_Fog_GetDensity () > 0)
|
|
qfglFogfv (GL_FOG_COLOR, color);
|
|
}
|
|
|
|
/*
|
|
Fog_StopAdditive
|
|
|
|
called after drawing stuff that is additive blended -- restores fog color
|
|
*/
|
|
void
|
|
gl_Fog_StopAdditive (void)
|
|
{
|
|
if (gl_Fog_GetDensity () > 0) {
|
|
quat_t fogcolor;
|
|
gl_Fog_GetColor (fogcolor);
|
|
qfglFogfv (GL_FOG_COLOR, fogcolor);
|
|
}
|
|
}
|
|
|
|
//==============================================================================
|
|
//
|
|
// VOLUMETRIC FOG
|
|
//
|
|
//==============================================================================
|
|
|
|
//cvar_t r_vfog = {"r_vfog", "1"};
|
|
|
|
//void Fog_DrawVFog (void) {}
|
|
//void Fog_MarkModels (void) {}
|
|
|
|
//==============================================================================
|
|
//
|
|
// INIT
|
|
//
|
|
//==============================================================================
|
|
|
|
/*
|
|
Fog_NewMap
|
|
|
|
called whenever a map is loaded
|
|
*/
|
|
#if 0
|
|
void
|
|
gl_Fog_NewMap (void)
|
|
{
|
|
Fog_ParseWorldspawn (); //for global fog
|
|
Fog_MarkModels (); //for volumetric fog
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
Fog_Init
|
|
|
|
called when quake initializes
|
|
*/
|
|
void
|
|
gl_Fog_Init (void)
|
|
{
|
|
Cmd_AddCommand ("fog", Fog_FogCommand_f, "");
|
|
|
|
//Cvar_RegisterVariable (&r_vfog, NULL);
|
|
|
|
//set up global fog
|
|
fog_density = 0.0;
|
|
fog_red = 0.3;
|
|
fog_green = 0.3;
|
|
fog_blue = 0.3;
|
|
|
|
qfglFogi (GL_FOG_MODE, GL_EXP2);
|
|
}
|