2024-04-24 11:48:45 +02:00
|
|
|
// 2024-04-24 10:09:27
|
2024-04-24 05:09:18 +02:00
|
|
|
varying vec2 v_vTexcoord;
|
|
|
|
varying vec4 v_vColour;
|
|
|
|
|
|
|
|
#define TAU 6.283185307179586
|
|
|
|
|
|
|
|
uniform vec2 dimension;
|
|
|
|
|
|
|
|
uniform vec2 borderStart;
|
|
|
|
uniform int borderStartUseSurf;
|
|
|
|
uniform sampler2D borderStartSurf;
|
|
|
|
|
|
|
|
uniform vec2 borderSize;
|
|
|
|
uniform int borderSizeUseSurf;
|
|
|
|
uniform sampler2D borderSizeSurf;
|
|
|
|
|
|
|
|
uniform vec4 borderColor;
|
|
|
|
uniform int side;
|
|
|
|
uniform int crop_border;
|
|
|
|
uniform int is_aa;
|
|
|
|
uniform int is_blend;
|
|
|
|
|
|
|
|
uniform vec2 blend_alpha;
|
|
|
|
uniform int blend_alphaUseSurf;
|
|
|
|
uniform sampler2D blend_alphaSurf;
|
|
|
|
|
|
|
|
uniform int sampleMode;
|
|
|
|
uniform int outline_only;
|
|
|
|
|
|
|
|
vec2 round(in vec2 v) { #region
|
|
|
|
v.x = fract(v.x) > 0.5? ceil(v.x) : floor(v.x);
|
|
|
|
v.y = fract(v.y) > 0.5? ceil(v.y) : floor(v.y);
|
|
|
|
return v;
|
|
|
|
} #endregion
|
|
|
|
|
|
|
|
vec4 sampleTexture(vec2 pos) { #region
|
|
|
|
if(pos.x >= 0. && pos.y >= 0. && pos.x <= 1. && pos.y <= 1.)
|
|
|
|
return texture2D(gm_BaseTexture, pos);
|
|
|
|
|
|
|
|
if(sampleMode == 0)
|
|
|
|
return vec4(0.);
|
|
|
|
|
|
|
|
else if(sampleMode == 1)
|
|
|
|
return texture2D(gm_BaseTexture, clamp(pos, 0., 1.));
|
|
|
|
|
|
|
|
else if(sampleMode == 2)
|
|
|
|
return texture2D(gm_BaseTexture, fract(pos));
|
|
|
|
|
|
|
|
else if(sampleMode == 3)
|
|
|
|
return vec4(vec3(0.), 1.);
|
|
|
|
|
|
|
|
return vec4(0.);
|
|
|
|
} #endregion
|
|
|
|
|
|
|
|
vec4 blendColor(vec4 base, vec4 colr, float alpha) {
|
|
|
|
|
|
|
|
float blend = base.a + colr.a * alpha * (1. - base.a);
|
|
|
|
|
|
|
|
vec4 col = (colr * alpha + base * base.a * ( 1. - alpha )) / blend;
|
|
|
|
col.a = base.a + colr.a * alpha;
|
|
|
|
|
|
|
|
return col;
|
|
|
|
}
|
|
|
|
|
|
|
|
void main() { #region
|
|
|
|
#region params
|
|
|
|
float bStr = borderStart.x;
|
|
|
|
if(borderStartUseSurf == 1) {
|
|
|
|
vec4 _vMap = texture2D( borderStartSurf, v_vTexcoord );
|
|
|
|
bStr = mix(borderStart.x, borderStart.y, (_vMap.r + _vMap.g + _vMap.b) / 3.);
|
|
|
|
}
|
|
|
|
|
|
|
|
float bSiz = borderSize.x;
|
|
|
|
if(borderSizeUseSurf == 1) {
|
|
|
|
vec4 _vMap = texture2D( borderSizeSurf, v_vTexcoord );
|
|
|
|
bSiz = mix(borderSize.x, borderSize.y, (_vMap.r + _vMap.g + _vMap.b) / 3.);
|
|
|
|
}
|
|
|
|
|
|
|
|
float bld = blend_alpha.x;
|
|
|
|
if(blend_alphaUseSurf == 1) {
|
|
|
|
vec4 _vMap = texture2D( blend_alphaSurf, v_vTexcoord );
|
|
|
|
bld = mix(blend_alpha.x, blend_alpha.y, (_vMap.r + _vMap.g + _vMap.b) / 3.);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
vec2 pixelPosition = v_vTexcoord * dimension;
|
|
|
|
vec4 baseColor = texture2D( gm_BaseTexture, v_vTexcoord );
|
|
|
|
vec4 col = outline_only == 0? baseColor : vec4(0.);
|
|
|
|
|
|
|
|
bool isOutline = false;
|
|
|
|
bool closetCollected = false;
|
|
|
|
vec4 closetColor;
|
|
|
|
float closetDistance = 9999.;
|
|
|
|
|
|
|
|
#region filter out filled ot empty pixel
|
|
|
|
bool isBorder = false;
|
|
|
|
if(side == 0) isBorder = baseColor.a > 0.;
|
|
|
|
else if(side == 1) isBorder = baseColor.a < 1.;
|
|
|
|
|
|
|
|
if(!isBorder) {
|
|
|
|
gl_FragColor = col;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
if(bSiz + bStr > 0.) {
|
|
|
|
for(float i = 1.; i <= 32.; i++) {
|
|
|
|
if(i > bStr + bSiz + float(is_aa)) break;
|
|
|
|
|
|
|
|
float base = 1.;
|
|
|
|
float top = 0.;
|
|
|
|
for(float j = 0.; j <= 64.; j++) {
|
|
|
|
float ang = top / base * TAU;
|
|
|
|
top += 2.;
|
|
|
|
if(top >= base) {
|
|
|
|
top = 1.;
|
|
|
|
base *= 2.;
|
|
|
|
}
|
|
|
|
|
|
|
|
vec2 pxs = pixelPosition + vec2( cos(ang), sin(ang)) * i;
|
|
|
|
vec2 txs = pxs / dimension;
|
|
|
|
pxs = floor(pxs) + 0.5;
|
|
|
|
|
|
|
|
if(side == 0 && crop_border == 1 && (txs.x < 0. || txs.x > 1. || txs.y < 0. || txs.y > 1.)) continue;
|
|
|
|
|
|
|
|
vec4 sam = sampleTexture( txs );
|
|
|
|
if(side == 0 && sam.a > 0.) continue; //inside border, skip if current pixel is filled
|
|
|
|
if(side == 1 && sam.a < 1.) continue; //outside border, skip if current pixel is empty
|
|
|
|
|
|
|
|
isOutline = true;
|
|
|
|
|
|
|
|
float dist = distance(pixelPosition, pxs);
|
|
|
|
if(dist < closetDistance) {
|
|
|
|
closetDistance = dist;
|
|
|
|
closetColor = sam;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
closetDistance = 0.;
|
|
|
|
|
|
|
|
float tauDiv = TAU / 4.;
|
|
|
|
for(float j = 0.; j < 4.; j++) {
|
|
|
|
float ang = j * tauDiv;
|
|
|
|
vec2 pxs = (pixelPosition + vec2( cos(ang), sin(ang)) ) / dimension;
|
|
|
|
if(side == 0 && crop_border == 1 && (pxs.x < 0. || pxs.x > 1. || pxs.y < 0. || pxs.y > 1.)) continue;
|
|
|
|
|
|
|
|
vec4 sam = sampleTexture( pxs );
|
|
|
|
|
|
|
|
if((side == 0 && sam.a == 0.) || (side == 1 && sam.a > 0.)) {
|
|
|
|
isOutline = true;
|
|
|
|
if(!closetCollected) {
|
|
|
|
closetCollected = true;
|
|
|
|
closetColor = sam;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!isOutline) {
|
|
|
|
gl_FragColor = col;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
float _aa = 1.;
|
|
|
|
|
|
|
|
if(is_aa == 1) _aa = min(smoothstep(bSiz + bStr + 1., bSiz + bStr, closetDistance), smoothstep(bStr - 1., bStr, closetDistance));
|
|
|
|
else _aa = min(step(-(bSiz + bStr + 0.5), -closetDistance), step(bStr - 0.5, closetDistance));
|
|
|
|
|
|
|
|
if(is_blend == 0) col = blendColor(baseColor, borderColor, _aa);
|
|
|
|
else col = blendColor(side == 0? baseColor : closetColor, borderColor, _aa * bld);
|
|
|
|
|
|
|
|
gl_FragColor = col;
|
|
|
|
} #endregion
|