2019-10-06 10:42:35 +00:00
#version 330
2019-07-12 08:44:36 +00:00
2019-10-19 20:46:37 +00:00
const int RF_ColorOnly = 1;
const int RF_UsePalette = 2;
const int RF_DetailMapping = 4;
const int RF_GlowMapping = 8;
const int RF_Brightmapping = 16;
const int RF_NPOTEmulation = 32;
const int RF_ShadeInterpolate = 64;
const int RF_FogDisabled = 128;
2020-01-11 21:18:06 +00:00
const int RF_HICTINT_Grayscale = 0x1;
const int RF_HICTINT_Invert = 0x2;
const int RF_HICTINT_Colorize = 0x4;
const int RF_HICTINT_BLEND_Screen = 64;
const int RF_HICTINT_BLEND_Overlay = 128;
const int RF_HICTINT_BLEND_Hardlight = 192;
2019-10-19 20:46:37 +00:00
const int RF_HICTINT_BLENDMASK = RF_HICTINT_BLEND_Screen | RF_HICTINT_BLEND_Overlay | RF_HICTINT_BLEND_Hardlight;
2019-07-12 08:44:36 +00:00
//s_texture points to an indexed color texture
uniform sampler2D s_texture;
//s_palswap is the palette swap texture where u is the color index and v is the shade
uniform sampler2D s_palswap;
//s_palette is the base palette texture where u is the color index
uniform sampler2D s_palette;
uniform sampler2D s_detail;
uniform sampler2D s_glow;
2019-11-10 18:42:26 +00:00
uniform sampler2D s_brightmap;
2019-07-12 08:44:36 +00:00
uniform float u_shade;
uniform float u_numShades;
2019-10-19 23:14:48 +00:00
uniform float u_shadeDiv;
2019-07-12 08:44:36 +00:00
uniform float u_visFactor;
2019-10-19 20:46:37 +00:00
uniform int u_flags;
2019-11-10 09:01:31 +00:00
uniform float u_alphaThreshold;
2019-07-12 08:44:36 +00:00
2020-01-11 21:18:06 +00:00
uniform vec4 u_tintOverlay, u_tintModulate;
uniform int u_tintFlags;
2020-02-11 19:55:47 +00:00
uniform vec4 u_fullscreenTint;
2020-01-11 21:18:06 +00:00
2019-07-12 08:44:36 +00:00
uniform float u_npotEmulationFactor;
uniform float u_npotEmulationXOffset;
2019-09-20 13:02:49 +00:00
uniform float u_brightness;
2019-10-06 10:42:35 +00:00
uniform vec4 u_fogColor;
2019-10-06 08:07:09 +00:00
uniform vec3 u_tintcolor;
2019-10-19 20:46:37 +00:00
uniform vec3 u_tintmodulate;
2019-10-06 08:07:09 +00:00
2019-10-06 10:42:35 +00:00
in vec4 v_color;
in float v_distance;
in vec4 v_texCoord;
in vec4 v_detailCoord;
in float v_fogCoord;
2020-01-03 17:34:43 +00:00
in vec4 v_eyeCoordPosition;
2019-07-12 08:44:36 +00:00
const float c_basepalScale = 255.0/256.0;
const float c_basepalOffset = 0.5/256.0;
const float c_zero = 0.0;
const float c_one = 1.0;
const float c_two = 2.0;
const vec4 c_vec4_one = vec4(c_one);
const float c_wrapThreshold = 0.9;
2019-10-06 10:42:35 +00:00
layout(location=0) out vec4 fragColor;
2020-01-03 17:34:43 +00:00
layout(location=1) out vec4 fragFog;
layout(location=2) out vec4 fragNormal;
2019-10-06 10:42:35 +00:00
2019-10-05 19:59:03 +00:00
//===========================================================================
//
// Color to grayscale
//
//===========================================================================
float grayscale(vec4 color)
{
return dot(color.rgb, vec3(0.3, 0.56, 0.14));
}
//===========================================================================
//
// Hightile tinting code. (hictinting[dapalnum]) This can be done inside the shader
// to avoid costly texture duplication (but needs a more modern GLSL than 1.10.)
//
//===========================================================================
2020-01-11 21:18:06 +00:00
vec4 convertColor(vec4 color)
2019-10-05 19:59:03 +00:00
{
2020-01-11 21:18:06 +00:00
int effect = u_tintFlags;
if ((effect & RF_HICTINT_Grayscale) != 0)
2019-10-05 19:59:03 +00:00
{
float g = grayscale(color);
color = vec4(g, g, g, color.a);
}
2020-01-11 21:18:06 +00:00
if ((effect & RF_HICTINT_Invert) != 0)
2019-10-05 19:59:03 +00:00
{
2020-01-11 21:18:06 +00:00
color = vec4(1.0 - color.r, 1.0 - color.g, 1.0 - color.b, color.a);
2019-10-05 19:59:03 +00:00
}
vec3 tcol = color.rgb * 255.0; // * 255.0 to make it easier to reuse the integer math.
2020-01-11 21:18:06 +00:00
// Much of this looks quite broken by design. Why is this effectively multplied by 4 if the flag is set...? :(
if ((effect & RF_HICTINT_Colorize) != 0)
{
2020-02-05 19:02:50 +00:00
tcol.r = min(((tcol.b) * u_tintModulate.r)* 4, 255.0);
2020-01-11 21:18:06 +00:00
tcol.g = min(((tcol.g) * u_tintModulate.g)* 4, 255.0);
2020-02-05 19:02:50 +00:00
tcol.b = min(((tcol.r) * u_tintModulate.b)* 4, 255.0);
2020-01-11 21:18:06 +00:00
}
else
2019-10-05 19:59:03 +00:00
{
2020-02-05 19:02:50 +00:00
tcol.r = min(((tcol.b) * u_tintModulate.r), 255.0);
2020-01-11 21:18:06 +00:00
tcol.g = min(((tcol.g) * u_tintModulate.g), 255.0);
2020-02-05 19:02:50 +00:00
tcol.b = min(((tcol.r) * u_tintModulate.b), 255.0);
2019-10-05 19:59:03 +00:00
}
2020-01-11 21:18:06 +00:00
vec4 ov = u_tintOverlay * 255.0;
2019-10-19 20:46:37 +00:00
switch (effect & RF_HICTINT_BLENDMASK)
2019-10-05 19:59:03 +00:00
{
2019-10-19 20:46:37 +00:00
case RF_HICTINT_BLEND_Screen:
2020-02-05 19:02:50 +00:00
tcol.r = 255.0 - (((255.0 - tcol.r) * (255.0 - ov.r)) / 256.0);
2020-01-11 21:18:06 +00:00
tcol.g = 255.0 - (((255.0 - tcol.g) * (255.0 - ov.g)) / 256.0);
2020-02-05 19:02:50 +00:00
tcol.b = 255.0 - (((255.0 - tcol.b) * (255.0 - ov.b)) / 256.0);
2019-10-05 19:59:03 +00:00
break;
2019-10-19 20:46:37 +00:00
case RF_HICTINT_BLEND_Overlay:
2020-02-05 19:02:50 +00:00
tcol.r = tcol.b < 128.0? (tcol.r * ov.r) / 128.0 : 255.0 - (((255.0 - tcol.r) * (255.0 - ov.r)) / 128.0);
2020-01-11 21:18:06 +00:00
tcol.g = tcol.g < 128.0? (tcol.g * ov.g) / 128.0 : 255.0 - (((255.0 - tcol.g) * (255.0 - ov.g)) / 128.0);
2020-02-05 19:02:50 +00:00
tcol.b = tcol.r < 128.0? (tcol.b * ov.b) / 128.0 : 255.0 - (((255.0 - tcol.b) * (255.0 - ov.b)) / 128.0);
2019-10-05 19:59:03 +00:00
break;
2019-10-19 20:46:37 +00:00
case RF_HICTINT_BLEND_Hardlight:
2020-02-05 19:02:50 +00:00
tcol.r = ov.r < 128.0 ? (tcol.r * ov.r) / 128.0 : 255.0 - (((255.0 - tcol.r) * (255.0 - ov.r)) / 128.0);
2020-01-11 21:18:06 +00:00
tcol.g = ov.g < 128.0 ? (tcol.g * ov.g) / 128.0 : 255.0 - (((255.0 - tcol.g) * (255.0 - ov.g)) / 128.0);
2020-02-05 19:02:50 +00:00
tcol.b = ov.b < 128.0 ? (tcol.b * ov.b) / 128.0 : 255.0 - (((255.0 - tcol.b) * (255.0 - ov.b)) / 128.0);
2019-10-05 19:59:03 +00:00
break;
}
color.rgb = tcol / 255.0;
return color;
}
//===========================================================================
//
2019-10-07 22:02:37 +00:00
//
2019-10-05 19:59:03 +00:00
//
//===========================================================================
2019-07-12 08:44:36 +00:00
void main()
{
2019-10-06 19:36:48 +00:00
float fullbright = 0.0;
vec4 color;
2019-10-19 20:46:37 +00:00
if ((u_flags & RF_ColorOnly) == 0)
2019-10-06 19:23:51 +00:00
{
2019-10-06 19:36:48 +00:00
float coordX = v_texCoord.x;
float coordY = v_texCoord.y;
vec2 newCoord;
// Coordinate adjustment for NPOT textures (something must have gone very wrong to make this necessary...)
2019-10-19 20:46:37 +00:00
if ((u_flags & RF_NPOTEmulation) != 0)
2019-10-06 19:36:48 +00:00
{
float period = floor(coordY / u_npotEmulationFactor);
coordX += u_npotEmulationXOffset * floor(mod(coordY, u_npotEmulationFactor));
coordY = period + mod(coordY, u_npotEmulationFactor);
}
newCoord = vec2(coordX, coordY);
2019-10-07 21:11:59 +00:00
// Paletted textures are stored in column major order rather than row major so coordinates need to be swapped here.
2020-01-26 14:44:08 +00:00
color = texture(s_texture, newCoord);
2019-10-06 19:36:48 +00:00
// This was further down but it really should be done before applying any kind of depth fading, not afterward.
vec4 detailColor = vec4(1.0);
2019-10-19 20:46:37 +00:00
if ((u_flags & RF_DetailMapping) != 0)
2019-10-06 19:36:48 +00:00
{
2020-01-26 14:44:08 +00:00
detailColor = texture(s_detail, v_detailCoord.xy);
2019-10-06 19:36:48 +00:00
detailColor = mix(vec4(1.0), 2.0 * detailColor, detailColor.a);
2019-10-07 21:11:59 +00:00
// Application of this differs based on render mode because for paletted rendering with palettized shade tables it can only be done after processing the shade table. We only have a palette index before.
2019-10-06 19:36:48 +00:00
}
2019-10-07 21:11:59 +00:00
2019-10-19 20:46:37 +00:00
float visibility = max(u_visFactor * v_distance - ((u_flags & RF_ShadeInterpolate) != 0.0? 0.5 : 0.0), 0.0);
2019-10-06 19:36:48 +00:00
float shade = clamp((u_shade + visibility), 0.0, u_numShades - 1.0);
2019-10-07 21:11:59 +00:00
2019-10-19 20:46:37 +00:00
if ((u_flags & RF_UsePalette) != 0)
2019-10-06 19:36:48 +00:00
{
2019-10-07 23:08:08 +00:00
int palindex = int(color.r * 255.0 + 0.1); // The 0.1 is for roundoff error compensation.
int shadeindex = int(floor(shade));
float colorIndexF = texelFetch(s_palswap, ivec2(palindex, shadeindex), 0).r;
int colorIndex = int(colorIndexF * 255.0 + 0.1); // The 0.1 is for roundoff error compensation.
vec4 palettedColor = texelFetch(s_palette, ivec2(colorIndex, 0), 0);
2019-10-06 19:36:48 +00:00
2019-10-19 20:46:37 +00:00
if ((u_flags & RF_ShadeInterpolate) != 0)
2019-10-06 19:36:48 +00:00
{
// Get the next shaded palette index for interpolation
2019-10-07 23:08:08 +00:00
colorIndexF = texelFetch(s_palswap, ivec2(palindex, shadeindex+1), 0).r;
colorIndex = int(colorIndexF * 255.0 + 0.1); // The 0.1 is for roundoff error compensation.
vec4 palettedColorNext = texelFetch(s_palette, ivec2(colorIndex, 0), 0);
2019-10-06 19:36:48 +00:00
float shadeFrac = mod(shade, 1.0);
2019-10-07 21:32:58 +00:00
palettedColor.rgb = mix(palettedColor.rgb, palettedColorNext.rgb, shadeFrac);
2019-10-06 19:36:48 +00:00
}
2019-10-07 21:11:59 +00:00
2019-10-06 19:36:48 +00:00
fullbright = palettedColor.a; // This only gets set for paletted rendering.
2019-10-07 21:11:59 +00:00
palettedColor.a = c_one-floor(color.r);
2019-10-19 08:21:07 +00:00
color = palettedColor;
2019-10-06 19:36:48 +00:00
color.rgb *= detailColor.rgb; // with all this palettizing, this can only be applied afterward, even though it is wrong to do it this way.
2019-11-10 18:42:26 +00:00
if (fullbright == 0.0) color.rgb *= v_color.rgb; // Well, this is dead wrong but unavoidable. For colored fog it applies the light to the fog as well...
2019-10-06 19:36:48 +00:00
}
else
{
color.rgb *= detailColor.rgb;
2020-02-18 19:43:16 +00:00
if (u_tintFlags != -1) color = convertColor(color);
if ((u_flags & RF_FogDisabled) == 0)
2019-11-10 18:42:26 +00:00
{
2020-02-18 19:43:16 +00:00
shade = clamp(shade * u_shadeDiv, 0.0, 1.0); // u_shadeDiv is really 1/shadeDiv.
vec3 lightcolor = v_color.rgb * (1.0 - shade);
if ((u_flags & RF_Brightmapping) != 0)
2019-11-11 22:54:09 +00:00
{
2020-02-18 19:43:16 +00:00
lightcolor = texture(s_brightmap, v_texCoord.xy).rgb;
2019-11-11 22:54:09 +00:00
}
2019-11-10 18:42:26 +00:00
color.rgb *= lightcolor;
2020-02-18 19:43:16 +00:00
color.rgb += u_fogColor.rgb * shade;
2019-10-19 16:14:13 +00:00
}
2019-10-06 19:36:48 +00:00
}
2019-11-10 09:01:31 +00:00
if (color.a < u_alphaThreshold) discard; // it's only here that we have the alpha value available to be able to perform the alpha test.
2019-11-10 18:42:26 +00:00
2019-10-06 19:36:48 +00:00
color.a *= v_color.a;
2019-10-04 21:29:00 +00:00
}
2019-10-06 19:36:48 +00:00
else
2019-10-06 10:42:35 +00:00
{
2019-10-06 19:36:48 +00:00
// untextured rendering
color = v_color;
2019-10-06 10:42:35 +00:00
}
2019-07-12 08:44:36 +00:00
2019-10-19 20:46:37 +00:00
if ((u_flags & (RF_ColorOnly|RF_GlowMapping)) == RF_GlowMapping)
2019-10-04 21:29:00 +00:00
{
2020-01-26 14:44:08 +00:00
vec4 glowColor = texture(s_glow, v_texCoord.xy);
2019-10-06 10:42:35 +00:00
color.rgb = mix(color.rgb, glowColor.rgb, glowColor.a);
2019-10-04 21:29:00 +00:00
}
2019-10-06 19:23:51 +00:00
color.rgb = pow(color.rgb, vec3(u_brightness));
2020-02-11 19:55:47 +00:00
color.rgb *= u_fullscreenTint.rgb; // must be the last thing to be done.
2019-10-06 19:23:51 +00:00
fragColor = color;
2020-01-03 17:34:43 +00:00
fragFog = vec4(0.0, 0.0, 0.0, 1.0); // Does build have colored fog?
vec3 normal = normalize(cross(dFdx(v_eyeCoordPosition.xyz), dFdy(v_eyeCoordPosition.xyz)));
normal.x = -normal.x;
normal.y = -normal.y;
fragNormal = vec4(normal * 0.5 + 0.5, 1.0);
2019-07-12 08:44:36 +00:00
}