aeon-199x/shaders/composite.fsh

129 lines
3.9 KiB
GLSL

#version 120
#include "/module/common.glsl"
float renderRes = pixelSize;
#define dithering // whether or not to apply dithering
uniform sampler2D gcolor;
uniform float viewWidth, viewHeight;
varying vec2 texcoord;
#include "/module/color.frag"
#include "/module/dof.frag"
#include "/module/interlace.frag"
const float steps = 4.0;
uniform vec3 palette[8];
uniform int paletteSize;
const int indexMatrix4x4[16] = int[](0, 8, 2, 10,
12, 4, 14, 6,
3, 11, 1, 9,
15, 7, 13, 5);
const int indexMatrix2x2[4] = int[](0, 2,
3, 1);
const int indexMatrix8x8[64] = int[](0, 32, 8, 40, 2, 34, 10, 42,
48, 16, 56, 24, 50, 18, 58, 26,
12, 44, 4, 36, 14, 46, 6, 38,
60, 28, 52, 20, 62, 30, 54, 22,
3, 35, 11, 43, 1, 33, 9, 41,
51, 19, 59, 27, 49, 17, 57, 25,
15, 47, 7, 39, 13, 45, 5, 37,
63, 31, 55, 23, 61, 29, 53, 21);
float indexValue() {
#if defined(dof) && defined(dofRes)
vec2 coord = gl_FragCoord.xy / renderRes;
#elif pixelSize > 1
vec2 coord = gl_FragCoord.xy / pixelSize;
#else
vec2 coord = gl_FragCoord.xy;
#endif
int x = int(mod(coord.x, 4));
int y = int(mod(coord.y, 4));
return indexMatrix4x4[(x + y * 4)] / 16.0;
}
float hueDistance(float h1, float h2) {
float diff = abs((h1 - h2));
return min(abs((1.0 - diff)), diff);
}
vec3[2] closestColors(float hue) {
vec3 ret[2];
vec3 closest = vec3(-2, 0, 0);
vec3 secondClosest = vec3(-2, 0, 0);
vec3 temp;
for (int i = 0; i < paletteSize; ++i) {
temp = palette[i];
float tempDistance = hueDistance(temp.x, hue);
if (tempDistance < hueDistance(closest.x, hue)) {
secondClosest = closest;
closest = temp;
} else {
if (tempDistance < hueDistance(secondClosest.x, hue)) {
secondClosest = temp;
}
}
}
ret[0] = closest;
ret[1] = secondClosest;
return ret;
}
float lightnessStep(float l, float lightnessSteps) {
/* Quantize the lightness to one of `lightnessSteps` values */;
return floor((0.5 + l * (lightnessSteps - 1.0))) / (lightnessSteps - 1.0);
}
float dither(float color, float dithersteps) {
float d = indexValue();
float l1 = lightnessStep(max((color - 0.125), 0.0), dithersteps);
float l2 = lightnessStep(min((color + 0.124), 1.0), dithersteps);
float lightnessDiff = (color - l1) / (l2 - l1);
float resultColor = (lightnessDiff < d) ? l1 : l2;
return resultColor;
}
void main() {
// adjust texture coordinate based on pixel size if needed
vec2 newcoord = texcoord;
#ifdef dof
newcoord = depthOfField();
#elif pixelSize > 1
vec2 view = vec2(viewWidth, viewHeight) / float(pixelSize);
float offset = (ceil(pixelSize * 0.5) - 0.5) / float(pixelSize);
newcoord = (floor(newcoord * view) + offset) / view;
#endif
vec3 color = texture2D(gcolor, newcoord).rgb;
color = to(color).xyz;
#ifdef dithering
vec3 final = vec3(dither(color.x, firstMax), dither(color.y, secondMax), dither(color.z, thirdMax)).rgb;
#else
vec3 final = vec3(lightnessStep(color.x, firstMax), lightnessStep(color.y, secondMax), lightnessStep(color.z, thirdMax)).rgb;
#endif
final = from(final);
#ifdef interlacing
/* DRAWBUFFERS:01 */
// pull previous buffer
vec3 prev = texture2D(colortex1, texcoord).rgb;
// interlace alternates between odd and even lines on world time
if(mod(int(gl_FragCoord.y / pixelSize), 2) == mod(frameCounter, 2))
gl_FragData[0] = vec4(prev, 1);
else
gl_FragData[0] = vec4(final, 1);
gl_FragData[1] = vec4(final, 1);
#else
gl_FragData[0] = vec4(final, 1.0);
#endif
}