slang-shaders/crt/shaders/dotmask.slang

114 lines
2.9 KiB
Plaintext
Raw Normal View History

#version 450
2016-10-07 01:21:19 +11:00
/*
Dot Mask
Authors: cgwg, Timothy Lottes
License: GPL
Note: This shader is just the dotmask functions from cgwg's CRT shader and crt-lottes.
*/
layout(std140, set = 0, binding = 0) uniform UBO
{
mat4 MVP;
vec4 OutputSize;
vec4 OriginalSize;
vec4 SourceSize;
2016-10-07 01:21:19 +11:00
float DOTMASK_STRENGTH;
float maskDark;
float maskLight;
float shadowMask;
} global;
2016-10-07 01:21:19 +11:00
#pragma parameter DOTMASK_STRENGTH "CGWG Dot Mask Strength" 0.3 0.0 1.0 0.01
#pragma parameter maskDark "Lottes maskDark" 0.5 0.0 2.0 0.1
#pragma parameter maskLight "Lottes maskLight" 1.5 0.0 2.0 0.1
#pragma parameter shadowMask "Mask Style" 3.0 0.0 4.0 1.0
#define mod_factor vTexCoord.x * global.SourceSize.x * global.OutputSize.x / global.SourceSize.x
#pragma stage vertex
layout(location = 0) in vec4 Position;
layout(location = 1) in vec2 TexCoord;
layout(location = 0) out vec2 vTexCoord;
void main()
{
gl_Position = global.MVP * Position;
vTexCoord = TexCoord;
}
#pragma stage fragment
layout(location = 0) in vec2 vTexCoord;
layout(location = 0) out vec4 FragColor;
layout(set = 0, binding = 2) uniform sampler2D Source;
// Shadow mask.
vec3 Mask(vec2 pos){
2016-10-07 01:21:19 +11:00
vec3 mask=vec3(global.maskDark,global.maskDark,global.maskDark);
// Very compressed TV style shadow mask.
2016-10-07 01:21:19 +11:00
if (global.shadowMask == 1.0) {
float line=global.maskLight;
float odd=0.0;
if(fract(pos.x/6.0)<0.5)odd=1.0;
2016-10-07 01:21:19 +11:00
if(fract((pos.y+odd)/2.0)<0.5)line=global.maskDark;
pos.x=fract(pos.x/3.0);
2016-10-07 01:21:19 +11:00
if(pos.x<0.333)mask.r=global.maskLight;
else if(pos.x<0.666)mask.g=global.maskLight;
else mask.b=global.maskLight;
mask*=line;
}
// Aperture-grille.
2016-10-07 01:21:19 +11:00
else if (global.shadowMask == 2.0) {
pos.x=fract(pos.x/3.0);
2016-10-07 01:21:19 +11:00
if(pos.x<0.333)mask.r=global.maskLight;
else if(pos.x<0.666)mask.g=global.maskLight;
else mask.b=global.maskLight;
}
// Stretched VGA style shadow mask (same as prior shaders).
2016-10-07 01:21:19 +11:00
else if (global.shadowMask == 3.0) {
pos.x+=pos.y*3.0;
pos.x=fract(pos.x/6.0);
2016-10-07 01:21:19 +11:00
if(pos.x<0.333)mask.r=global.maskLight;
else if(pos.x<0.666)mask.g=global.maskLight;
else mask.b=global.maskLight;
}
// VGA style shadow mask.
2016-10-07 01:21:19 +11:00
else if (global.shadowMask == 4.0) {
pos.xy=floor(pos.xy*vec2(1.0,0.5));
pos.x+=pos.y*3.0;
pos.x=fract(pos.x/6.0);
2016-10-07 01:21:19 +11:00
if(pos.x<0.333)mask.r=global.maskLight;
else if(pos.x<0.666)mask.g=global.maskLight;
else mask.b=global.maskLight;
}
return mask;}
void main()
{
vec3 res = pow(texture(Source, vTexCoord).rgb, vec3(2.2));
2016-10-07 01:21:19 +11:00
float mask = 1.0 - global.DOTMASK_STRENGTH;
//cgwg's dotmask emulation:
//Output pixels are alternately tinted green and magenta
vec3 dotMaskWeights = mix(vec3(1.0, mask, 1.0),
vec3(mask, 1.0, mask),
floor(mod(mod_factor, 2.0)));
2016-10-07 01:21:19 +11:00
if (global.shadowMask == 0) {
res *= dotMaskWeights;
}
else {
res *= Mask(floor(1.000001 * vTexCoord.xy * global.OutputSize.xy + vec2(0.5)));
}
FragColor = vec4(pow(res, vec3(1.0/2.2)), 1.0);
}