2016-12-26 16:11:13 +00:00
|
|
|
#version 420 core
|
|
|
|
|
2017-01-17 04:20:07 +00:00
|
|
|
#include "constants.glsl"
|
|
|
|
|
2016-12-26 16:11:13 +00:00
|
|
|
struct ColorStop {
|
|
|
|
float offset;
|
|
|
|
vec4 color;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RoundedRect {
|
|
|
|
vec4 bounds;
|
|
|
|
vec4 corners;
|
|
|
|
};
|
|
|
|
|
|
|
|
layout(location = 0) in vec2 inPos;
|
|
|
|
layout(location = 1) in float inGradientPos;
|
|
|
|
layout(location = 2) in flat int inRepeating;
|
|
|
|
layout(location = 3) in flat int inStopCount;
|
2017-01-17 04:20:07 +00:00
|
|
|
layout(location = 4) in flat ColorStop inStops[8];
|
2016-12-26 16:11:13 +00:00
|
|
|
|
|
|
|
layout(location = 0) out vec4 outColor;
|
|
|
|
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
float clip(vec2 pos, RoundedRect r) {
|
|
|
|
vec2 ref_tl = r.bounds.xy + vec2( r.corners.x, r.corners.x);
|
|
|
|
vec2 ref_tr = r.bounds.zy + vec2(-r.corners.y, r.corners.y);
|
|
|
|
vec2 ref_br = r.bounds.zw + vec2(-r.corners.z, -r.corners.z);
|
|
|
|
vec2 ref_bl = r.bounds.xw + vec2( r.corners.w, -r.corners.w);
|
|
|
|
|
|
|
|
float d_tl = distance(pos, ref_tl);
|
|
|
|
float d_tr = distance(pos, ref_tr);
|
|
|
|
float d_br = distance(pos, ref_br);
|
|
|
|
float d_bl = distance(pos, ref_bl);
|
|
|
|
|
|
|
|
float pixels_per_fragment = length(fwidth(pos.xy));
|
|
|
|
float nudge = 0.5 * pixels_per_fragment;
|
|
|
|
vec4 distances = vec4(d_tl, d_tr, d_br, d_bl) - r.corners + nudge;
|
|
|
|
|
|
|
|
bvec4 is_out = bvec4(pos.x < ref_tl.x && pos.y < ref_tl.y,
|
|
|
|
pos.x > ref_tr.x && pos.y < ref_tr.y,
|
|
|
|
pos.x > ref_br.x && pos.y > ref_br.y,
|
|
|
|
pos.x < ref_bl.x && pos.y > ref_bl.y);
|
|
|
|
|
|
|
|
float distance_from_border = dot(vec4(is_out),
|
|
|
|
max(vec4(0.0, 0.0, 0.0, 0.0), distances));
|
|
|
|
|
|
|
|
// Move the distance back into pixels.
|
|
|
|
distance_from_border /= pixels_per_fragment;
|
|
|
|
// Apply a more gradual fade out to transparent.
|
|
|
|
//distance_from_border -= 0.5;
|
|
|
|
|
|
|
|
return 1.0 - smoothstep(0.0, 1.0, distance_from_border);
|
|
|
|
}
|
|
|
|
|
|
|
|
void main()
|
|
|
|
{
|
2017-01-17 04:20:07 +00:00
|
|
|
RoundedRect r = RoundedRect(vec4(push.clip_bounds.xy, push.clip_bounds.xy + push.clip_bounds.zw), push.clip_widths);
|
2016-12-26 16:11:13 +00:00
|
|
|
|
|
|
|
float pos;
|
|
|
|
if (inRepeating != 0)
|
|
|
|
pos = fract (inGradientPos);
|
|
|
|
else
|
|
|
|
pos = clamp (inGradientPos, 0, 1);
|
|
|
|
|
|
|
|
vec4 color = inStops[0].color;
|
|
|
|
int n = clamp (inStopCount, 2, 8);
|
|
|
|
for (int i = 1; i < n; i++)
|
|
|
|
{
|
|
|
|
if (inStops[i].offset > inStops[i-1].offset)
|
|
|
|
color = mix (color, inStops[i].color, clamp((pos - inStops[i-1].offset) / (inStops[i].offset - inStops[i-1].offset), 0, 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
//outColor = vec4(pos, pos, pos, 1.0);
|
|
|
|
outColor = color * clip (inPos, r);
|
|
|
|
}
|