blob: ab6dcb8922ddfbf8f9d4e2f708116e1962076b1e [file] [log] [blame]
R"SKSL(
// Definitions of functions implementing all of the SkBlendMode blends.
half4 blend_clear(half4 src, half4 dst) { return half4(0); }
half4 blend_src(half4 src, half4 dst) { return src; }
half4 blend_dst(half4 src, half4 dst) { return dst; }
half4 blend_src_over(half4 src, half4 dst) { return src + (1 - src.a)*dst; }
half4 blend_dst_over(half4 src, half4 dst) { return (1 - dst.a)*src + dst; }
half4 blend_src_in(half4 src, half4 dst) {
@if (sk_Caps.inBlendModesFailRandomlyForAllZeroVec) {
if (src == half4(0)) {
return half4(0);
}
}
return src*dst.a;
}
half4 blend_dst_in(half4 src, half4 dst) { return blend_src_in(dst, src); }
half4 blend_src_out(half4 src, half4 dst) { return (1 - dst.a)*src; }
half4 blend_dst_out(half4 src, half4 dst) { return (1 - src.a)*dst; }
half4 blend_src_atop(half4 src, half4 dst) { return dst.a*src + (1 - src.a)*dst; }
half4 blend_dst_atop(half4 src, half4 dst) { return (1 - dst.a) * src + src.a*dst; }
half4 blend_xor(half4 src, half4 dst) { return (1 - dst.a)*src + (1 - src.a)*dst; }
half4 blend_plus(half4 src, half4 dst) { return min(src + dst, 1); }
half4 blend_modulate(half4 src, half4 dst) { return src*dst; }
half4 blend_screen(half4 src, half4 dst) { return src + (1 - src)*dst; }
half _blend_overlay_component(half sc, half sa, half dc, half da) {
if (2*dc <= da) {
return 2*sc*dc;
}
return sa*da - 2*(da - dc)*(sa - sc);
}
half4 blend_overlay(half4 src, half4 dst) {
half4 result = half4(_blend_overlay_component(src.r, src.a, dst.r, dst.a),
_blend_overlay_component(src.g, src.a, dst.g, dst.a),
_blend_overlay_component(src.b, src.a, dst.b, dst.a),
src.a + (1 - src.a)*dst.a);
result.rgb += dst.rgb*(1 - src.a) + src.rgb*(1 - dst.a);
return result;
}
half4 blend_darken(half4 src, half4 dst) {
half4 result = blend_src_over(src, dst);
result.rgb = min(result.rgb, (1 - dst.a)*src.rgb + dst.rgb);
return result;
}
half4 blend_lighten(half4 src, half4 dst) {
half4 result = blend_src_over(src, dst);
result.rgb = max(result.rgb, (1 - dst.a)*src.rgb + dst.rgb);
return result;
}
half _guarded_divide(half n, half d) {
@if (sk_Caps.mustGuardDivisionEvenAfterExplicitZeroCheck) {
return n/(d + 0.00000001);
} else {
return n/d;
}
}
half _color_dodge_component(half sc, half sa, half dc, half da) {
if (dc == 0) {
return sc*(1 - da);
} else {
half d = sa - sc;
if (d == 0) {
return sa*da + sc*(1 - da) + dc*(1 - sa);
}
d = min(da, _guarded_divide(dc*sa, d));
return d*sa + sc*(1 - da) + dc*(1 - sa);
}
}
half4 blend_color_dodge(half4 src, half4 dst) {
return half4(_color_dodge_component(src.r, src.a, dst.r, dst.a),
_color_dodge_component(src.g, src.a, dst.g, dst.a),
_color_dodge_component(src.b, src.a, dst.b, dst.a),
src.a + (1 - src.a)*dst.a);
}
half _color_burn_component(half sc, half sa, half dc, half da) {
if (da == dc) {
return sa*da + sc*(1 - da) + dc*(1 - sa);
} else if (sc == 0) {
return dc*(1 - sa);
}
half d = max(0, da - _guarded_divide((da - dc)*sa, sc));
return d*sa + sc*(1 - da) + dc*(1 - sa);
}
half4 blend_color_burn(half4 src, half4 dst) {
return half4(_color_burn_component(src.r, src.a, dst.r, dst.a),
_color_burn_component(src.g, src.a, dst.g, dst.a),
_color_burn_component(src.b, src.a, dst.b, dst.a),
src.a + (1 - src.a)*dst.a);
}
half4 blend_hard_light(half4 src, half4 dst) { return blend_overlay(dst, src); }
half _soft_light_component(half sc, half sa, half dc, half da) {
if (2*sc <= sa) {
return _guarded_divide(dc*dc*(sa - 2*sc), da) + (1 - da)*sc + dc*(-sa + 2*sc + 1);
} else if (4.0 * dc <= da) {
half DSqd = dc*dc;
half DCub = DSqd*dc;
half DaSqd = da*da;
half DaCub = DaSqd*da;
return _guarded_divide(DaSqd*(sc - dc*(3*sa - 6*sc - 1)) + 12*da*DSqd*(sa - 2*sc)
- 16*DCub * (sa - 2*sc) - DaCub*sc, DaSqd);
}
return dc*(sa - 2*sc + 1) + sc - sqrt(da*dc)*(sa - 2*sc) - da*sc;
}
half4 blend_soft_light(half4 src, half4 dst) {
if (dst.a == 0) {
return src;
}
return half4(_soft_light_component(src.r, src.a, dst.r, dst.a),
_soft_light_component(src.g, src.a, dst.g, dst.a),
_soft_light_component(src.b, src.a, dst.b, dst.a),
src.a + (1 - src.a)*dst.a);
}
half4 blend_difference(half4 src, half4 dst) {
return half4(src.rgb + dst.rgb - 2*min(src.rgb*dst.a, dst.rgb*src.a),
src.a + (1 - src.a)*dst.a);
}
half4 blend_exclusion(half4 src, half4 dst) {
return half4(dst.rgb + src.rgb - 2*dst.rgb*src.rgb, src.a + (1 - src.a)*dst.a);
}
half4 blend_multiply(half4 src, half4 dst) {
return half4((1 - src.a)*dst.rgb + (1 - dst.a)*src.rgb + src.rgb*dst.rgb,
src.a + (1 - src.a)*dst.a);
}
half _blend_color_luminance(half3 color) { return dot(half3(0.3, 0.59, 0.11), color); }
half3 _blend_set_color_luminance(half3 hueSatColor, half alpha, half3 lumColor) {
half lum = _blend_color_luminance(lumColor);
half3 result = lum - _blend_color_luminance(hueSatColor) + hueSatColor;
half minComp = min(min(result.r, result.g), result.b);
half maxComp = max(max(result.r, result.g), result.b);
if (minComp < 0 && lum != minComp) {
result = lum + (result - lum) * lum/(lum - minComp);
}
if (maxComp > alpha && maxComp != lum) {
return lum + ((result - lum) * (alpha - lum))/(maxComp - lum);
}
return result;
}
half _blend_color_saturation(half3 color) {
return max(max(color.r, color.g), color.b) - min(min(color.r, color.g), color.b);
}
half3 _blend_set_color_saturation_helper(half3 minMidMax, half sat) {
if (minMidMax.r < minMidMax.b) {
return half3(0, sat*(minMidMax.g - minMidMax.r)/(minMidMax.b - minMidMax.r), sat);
}
return half3(0);
}
half3 _blend_set_color_saturation(half3 hueLumColor, half3 satColor) {
half sat = _blend_color_saturation(satColor);
if (hueLumColor.r <= hueLumColor.g) {
if (hueLumColor.g <= hueLumColor.b) {
hueLumColor.rgb = _blend_set_color_saturation_helper(hueLumColor.rgb, sat);
} else if (hueLumColor.r <= hueLumColor.b) {
hueLumColor.rbg = _blend_set_color_saturation_helper(hueLumColor.rbg, sat);
} else {
hueLumColor.brg = _blend_set_color_saturation_helper(hueLumColor.brg, sat);
}
} else if (hueLumColor.r <= hueLumColor.b) {
hueLumColor.grb = _blend_set_color_saturation_helper(hueLumColor.grb, sat);
} else if (hueLumColor.g <= hueLumColor.b) {
hueLumColor.gbr = _blend_set_color_saturation_helper(hueLumColor.gbr, sat);
} else {
hueLumColor.bgr = _blend_set_color_saturation_helper(hueLumColor.bgr, sat);
}
return hueLumColor;
}
half4 blend_hue(half4 src, half4 dst) {
half alpha = dst.a*src.a;
half3 sda = src.rgb*dst.a;
half3 dsa = dst.rgb*src.a;
return half4(_blend_set_color_luminance(_blend_set_color_saturation(sda, dsa), alpha, dsa) +
dst.rgb - dsa + src.rgb - sda,
src.a + dst.a - alpha);
}
half4 blend_saturation(half4 src, half4 dst) {
half alpha = dst.a*src.a;
half3 sda = src.rgb*dst.a;
half3 dsa = dst.rgb*src.a;
return half4(_blend_set_color_luminance(_blend_set_color_saturation(dsa, sda), alpha, dsa) +
dst.rgb - dsa + src.rgb - sda,
src.a + dst.a - alpha);
}
half4 blend_color(half4 src, half4 dst) {
half alpha = dst.a*src.a;
half3 sda = src.rgb*dst.a;
half3 dsa = dst.rgb*src.a;
return half4(_blend_set_color_luminance(sda, alpha, dsa) + dst.rgb - dsa + src.rgb - sda,
src.a + dst.a - alpha);
}
half4 blend_luminosity(half4 src, half4 dst) {
half alpha = dst.a*src.a;
half3 sda = src.rgb*dst.a;
half3 dsa = dst.rgb*src.a;
return half4(_blend_set_color_luminance(dsa, alpha, sda) + dst.rgb - dsa + src.rgb - sda,
src.a + dst.a - alpha);
}
enum class SkBlendMode {
kClear = 0,
kSrc = 1,
kDst = 2,
kSrcOver = 3,
kDstOver = 4,
kSrcIn = 5,
kDstIn = 6,
kSrcOut = 7,
kDstOut = 8,
kSrcATop = 9,
kDstATop = 10,
kXor = 11,
kPlus = 12,
kModulate = 13,
kScreen = 14,
kOverlay = 15,
kDarken = 16,
kLighten = 17,
kColorDodge = 18,
kColorBurn = 19,
kHardLight = 20,
kSoftLight = 21,
kDifference = 22,
kExclusion = 23,
kMultiply = 24,
kHue = 25,
kSaturation = 26,
kColor = 27,
kLuminosity = 28
};
half4 blend(SkBlendMode mode, half4 src, half4 dst) {
switch (mode) {
case SkBlendMode::kClear: return blend_clear(src, dst);
case SkBlendMode::kSrc: return blend_src(src, dst);
case SkBlendMode::kDst: return blend_dst(src, dst);
case SkBlendMode::kSrcOver: return blend_src_over(src, dst);
case SkBlendMode::kDstOver: return blend_dst_over(src, dst);
case SkBlendMode::kSrcIn: return blend_src_in(src, dst);
case SkBlendMode::kDstIn: return blend_dst_in(src, dst);
case SkBlendMode::kSrcOut: return blend_src_out(src, dst);
case SkBlendMode::kDstOut: return blend_dst_out(src, dst);
case SkBlendMode::kSrcATop: return blend_src_atop(src, dst);
case SkBlendMode::kDstATop: return blend_dst_atop(src, dst);
case SkBlendMode::kXor: return blend_xor(src, dst);
case SkBlendMode::kPlus: return blend_plus(src, dst);
case SkBlendMode::kModulate: return blend_modulate(src, dst);
case SkBlendMode::kScreen: return blend_screen(src, dst);
case SkBlendMode::kOverlay: return blend_overlay(src, dst);
case SkBlendMode::kDarken: return blend_darken(src, dst);
case SkBlendMode::kLighten: return blend_lighten(src, dst);
case SkBlendMode::kColorDodge: return blend_color_dodge(src, dst);
case SkBlendMode::kColorBurn: return blend_color_burn(src, dst);
case SkBlendMode::kHardLight: return blend_hard_light(src, dst);
case SkBlendMode::kSoftLight: return blend_soft_light(src, dst);
case SkBlendMode::kDifference: return blend_difference(src, dst);
case SkBlendMode::kExclusion: return blend_exclusion(src, dst);
case SkBlendMode::kMultiply: return blend_multiply(src, dst);
case SkBlendMode::kHue: return blend_hue(src, dst);
case SkBlendMode::kSaturation: return blend_saturation(src, dst);
case SkBlendMode::kColor: return blend_color(src, dst);
case SkBlendMode::kLuminosity: return blend_luminosity(src, dst);
}
return half4(0); // Avoids "'blend' can exit without returning a value."
}
// The max() guards against division by zero when the incoming color is transparent black
half4 unpremul(half4 color) { return half4(color.rgb / max(color.a, 0.0001), color.a); }
float4 unpremul_float(float4 color) { return float4(color.rgb / max(color.a, 0.0001), color.a); }
)SKSL"