2019-10-19 20:46:37 +00:00
|
|
|
const int RF_ColorOnly = 1;
|
|
|
|
const int RF_UsePalette = 2;
|
|
|
|
const int RF_ShadeInterpolate = 64;
|
|
|
|
|
2020-06-07 08:15:31 +00:00
|
|
|
const int TEXF_Brightmap = 0x10000;
|
|
|
|
const int TEXF_Detailmap = 0x20000;
|
|
|
|
const int TEXF_Glowmap = 0x40000;
|
|
|
|
|
2020-06-07 06:24:12 +00:00
|
|
|
|
|
|
|
struct Material
|
|
|
|
{
|
|
|
|
vec4 Base;
|
|
|
|
vec4 Bright;
|
|
|
|
vec4 Glow;
|
|
|
|
vec3 Normal;
|
|
|
|
vec3 Specular;
|
|
|
|
float Glossiness;
|
|
|
|
float SpecularLevel;
|
|
|
|
float Metallic;
|
|
|
|
float Roughness;
|
|
|
|
float AO;
|
|
|
|
};
|
|
|
|
|
|
|
|
Material material;
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
2019-10-19 20:46:37 +00:00
|
|
|
uniform int u_flags;
|
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 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;
|
2020-06-05 15:02:21 +00:00
|
|
|
in vec4 v_worldPosition;
|
2019-07-12 08:44:36 +00:00
|
|
|
|
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));
|
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
|
2019-10-05 19:59:03 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
2020-06-05 15:02:21 +00:00
|
|
|
// Desaturate a color
|
2019-10-05 19:59:03 +00:00
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
vec4 dodesaturate(vec4 texel, float factor)
|
2019-10-05 19:59:03 +00:00
|
|
|
{
|
2020-06-05 15:02:21 +00:00
|
|
|
if (factor != 0.0)
|
2019-10-05 19:59:03 +00:00
|
|
|
{
|
2020-06-05 15:02:21 +00:00
|
|
|
float gray = grayscale(texel);
|
|
|
|
return mix (texel, vec4(gray,gray,gray,texel.a), factor);
|
2019-10-05 19:59:03 +00:00
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
else
|
2019-10-05 19:59:03 +00:00
|
|
|
{
|
2020-06-05 15:02:21 +00:00
|
|
|
return texel;
|
2019-10-05 19:59:03 +00:00
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Texture tinting code originally from JFDuke but with a few more options
|
|
|
|
//
|
|
|
|
//===========================================================================
|
2019-10-05 19:59:03 +00:00
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
const int Tex_Blend_Alpha = 1;
|
|
|
|
const int Tex_Blend_Screen = 2;
|
|
|
|
const int Tex_Blend_Overlay = 3;
|
|
|
|
const int Tex_Blend_Hardlight = 4;
|
|
|
|
|
|
|
|
vec4 ApplyTextureManipulation(vec4 texel, int blendflags)
|
|
|
|
{
|
|
|
|
// Step 1: desaturate according to the material's desaturation factor.
|
|
|
|
texel = dodesaturate(texel, uTextureModulateColor.a);
|
|
|
|
|
|
|
|
// Step 2: Invert if requested
|
|
|
|
if ((blendflags & 8) != 0)
|
2020-01-11 21:18:06 +00:00
|
|
|
{
|
2020-06-05 15:02:21 +00:00
|
|
|
texel.rgb = vec3(1.0 - texel.r, 1.0 - texel.g, 1.0 - texel.b);
|
2020-01-11 21:18:06 +00:00
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
|
|
|
|
// Step 3: Apply additive color
|
|
|
|
texel.rgb += uTextureAddColor.rgb;
|
|
|
|
|
|
|
|
// Step 4: Colorization, including gradient if set.
|
|
|
|
texel.rgb *= uTextureModulateColor.rgb;
|
|
|
|
|
|
|
|
// Before applying the blend the value needs to be clamped to [0..1] range.
|
|
|
|
texel.rgb = clamp(texel.rgb, 0.0, 1.0);
|
|
|
|
|
|
|
|
// Step 5: Apply a blend. This may just be a translucent overlay or one of the blend modes present in current Build engines.
|
|
|
|
if ((blendflags & 7) != 0)
|
2019-10-05 19:59:03 +00:00
|
|
|
{
|
2020-06-05 15:02:21 +00:00
|
|
|
vec3 tcol = texel.rgb * 255.0; // * 255.0 to make it easier to reuse the integer math.
|
|
|
|
vec4 tint = uTextureBlendColor * 255.0;
|
2019-10-05 19:59:03 +00:00
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
switch (blendflags & 7)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
tcol.b = tcol.b * (1.0 - uTextureBlendColor.a) + tint.b * uTextureBlendColor.a;
|
|
|
|
tcol.g = tcol.g * (1.0 - uTextureBlendColor.a) + tint.g * uTextureBlendColor.a;
|
|
|
|
tcol.r = tcol.r * (1.0 - uTextureBlendColor.a) + tint.r * uTextureBlendColor.a;
|
|
|
|
break;
|
|
|
|
// The following 3 are taken 1:1 from the Build engine
|
|
|
|
case Tex_Blend_Screen:
|
|
|
|
tcol.b = 255.0 - (((255.0 - tcol.b) * (255.0 - tint.r)) / 256.0);
|
|
|
|
tcol.g = 255.0 - (((255.0 - tcol.g) * (255.0 - tint.g)) / 256.0);
|
|
|
|
tcol.r = 255.0 - (((255.0 - tcol.r) * (255.0 - tint.b)) / 256.0);
|
|
|
|
break;
|
|
|
|
case Tex_Blend_Overlay:
|
|
|
|
tcol.b = tcol.b < 128.0? (tcol.b * tint.b) / 128.0 : 255.0 - (((255.0 - tcol.b) * (255.0 - tint.b)) / 128.0);
|
|
|
|
tcol.g = tcol.g < 128.0? (tcol.g * tint.g) / 128.0 : 255.0 - (((255.0 - tcol.g) * (255.0 - tint.g)) / 128.0);
|
|
|
|
tcol.r = tcol.r < 128.0? (tcol.r * tint.r) / 128.0 : 255.0 - (((255.0 - tcol.r) * (255.0 - tint.r)) / 128.0);
|
|
|
|
break;
|
|
|
|
case Tex_Blend_Hardlight:
|
|
|
|
tcol.b = tint.b < 128.0 ? (tcol.b * tint.b) / 128.0 : 255.0 - (((255.0 - tcol.b) * (255.0 - tint.b)) / 128.0);
|
|
|
|
tcol.g = tint.g < 128.0 ? (tcol.g * tint.g) / 128.0 : 255.0 - (((255.0 - tcol.g) * (255.0 - tint.g)) / 128.0);
|
|
|
|
tcol.r = tint.r < 128.0 ? (tcol.r * tint.r) / 128.0 : 255.0 - (((255.0 - tcol.r) * (255.0 - tint.r)) / 128.0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
texel.rgb = tcol / 255.0;
|
2019-10-05 19:59:03 +00:00
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
return texel;
|
2019-10-05 19:59:03 +00:00
|
|
|
}
|
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
|
2019-10-05 19:59:03 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
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...)
|
2020-06-07 07:30:55 +00:00
|
|
|
if (u_npotEmulationFactor != 0.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
|
|
|
|
2020-01-26 14:44:08 +00:00
|
|
|
color = texture(s_texture, newCoord);
|
2019-10-06 19:36:48 +00:00
|
|
|
|
2020-06-05 22:44:57 +00:00
|
|
|
float visibility = max(uGlobVis * uLightFactor * v_distance - ((u_flags & RF_ShadeInterpolate) != 0.0? 0.5 : 0.0), 0.0);
|
2020-06-04 18:14:48 +00:00
|
|
|
float numShades = float(uPalLightLevels & 255);
|
2020-06-05 22:44:57 +00:00
|
|
|
float shade = clamp((uLightLevel + visibility), 0.0, numShades - 1.0);
|
2019-10-06 19:36:48 +00:00
|
|
|
|
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
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
palettedColor.a = color.r == 0.0? 0.0 : 1.0;// 1.0-floor(color.r);
|
2020-06-07 12:50:12 +00:00
|
|
|
color.rgb = palettedColor.rgb * v_color.rgb;
|
|
|
|
color.a = palettedColor.a;
|
2019-10-06 19:36:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-06-07 08:15:31 +00:00
|
|
|
// This was further down but it really should be done before applying any kind of depth fading, not afterward.
|
|
|
|
if ((uTextureMode & TEXF_Detailmap) != 0)
|
|
|
|
{
|
|
|
|
vec4 detailColor = texture(detailtexture, newCoord * uDetailParms.xy) * uDetailParms.z;
|
|
|
|
detailColor = mix(vec4(1.0), 2.0 * detailColor, detailColor.a);
|
|
|
|
color.rgb *= detailColor.rgb;
|
|
|
|
}
|
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
// Apply the texture modification colors.
|
|
|
|
int blendflags = int(uTextureAddColor.a); // this alpha is unused otherwise
|
|
|
|
if (blendflags != 0)
|
|
|
|
{
|
|
|
|
// only apply the texture manipulation if it contains something.
|
|
|
|
color = ApplyTextureManipulation(color, blendflags);
|
|
|
|
}
|
|
|
|
|
2020-06-05 22:44:57 +00:00
|
|
|
if (uFogEnabled != 0) // Right now this code doesn't care about the fog modes yet.
|
2019-11-10 18:42:26 +00:00
|
|
|
{
|
2020-06-07 08:15:31 +00:00
|
|
|
shade = clamp(shade * uLightDist, 0.0, 1.0);
|
2020-02-18 19:43:16 +00:00
|
|
|
vec3 lightcolor = v_color.rgb * (1.0 - shade);
|
|
|
|
|
2020-06-07 08:15:31 +00:00
|
|
|
if ((uTextureMode & TEXF_Brightmap) != 0)
|
2019-11-11 22:54:09 +00:00
|
|
|
{
|
2020-06-07 06:16:04 +00:00
|
|
|
lightcolor = clamp(lightcolor + texture(brighttexture, v_texCoord.xy).rgb, 0.0, 1.0);
|
2019-11-11 22:54:09 +00:00
|
|
|
}
|
2019-11-10 18:42:26 +00:00
|
|
|
color.rgb *= lightcolor;
|
2020-06-07 06:16:04 +00:00
|
|
|
if (uFogDensity == 0.0) color.rgb += uFogColor.rgb * shade;
|
2019-10-19 16:14:13 +00:00
|
|
|
}
|
2020-04-12 07:44:28 +00:00
|
|
|
else color.rgb *= v_color.rgb;
|
2020-06-07 08:15:31 +00:00
|
|
|
|
|
|
|
if ((uTextureMode & TEXF_Glowmap) != 0)
|
|
|
|
{
|
|
|
|
vec4 glowColor = texture(glowtexture, v_texCoord.xy);
|
|
|
|
color.rgb = mix(color.rgb, glowColor.rgb, glowColor.a);
|
|
|
|
}
|
|
|
|
|
2019-10-06 19:36:48 +00:00
|
|
|
}
|
2020-06-05 22:44:57 +00:00
|
|
|
if (uFogDensity != 0.0) // fog hack for RRRA E2L1. Needs to be done better, this is gross, but still preferable to the broken original implementation.
|
2020-04-11 22:21:35 +00:00
|
|
|
{
|
2020-06-07 06:24:12 +00:00
|
|
|
float fogfactor = 0.55 + 0.3 * exp2 (uFogDensity * v_fogCoord / 1024.0);
|
|
|
|
color.rgb = uFogColor.rgb * (1.0-fogfactor) + color.rgb * fogfactor;// mix(vec3(0.6), color.rgb, fogfactor);
|
2020-04-11 22:21:35 +00:00
|
|
|
}
|
2020-06-05 15:02:21 +00:00
|
|
|
if (color.a < uAlphaThreshold) discard; // it's only here that we have the alpha value available to be able to perform the alpha test.
|
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-04 21:29:00 +00:00
|
|
|
|
2020-06-05 15:02:21 +00:00
|
|
|
/*
|
|
|
|
int ix = int (v_worldPosition.x);
|
|
|
|
int iy = int (v_worldPosition.z);
|
|
|
|
int iz = int (v_worldPosition.y);
|
|
|
|
if ((ix & 64) == 1) color.r = 0;
|
|
|
|
if ((iy & 64) == 1) color.g = 0;
|
|
|
|
if ((iz & 64) == 1) color.b = 0;
|
|
|
|
*/
|
|
|
|
|
2019-10-06 19:23:51 +00:00
|
|
|
color.rgb = pow(color.rgb, vec3(u_brightness));
|
|
|
|
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
|
|
|
}
|