Refactor blending functions

This commit is contained in:
Elias Batek 2024-05-26 16:26:09 +02:00
parent 1aae3674dc
commit 0b2481bab6
1 changed files with 269 additions and 263 deletions

View File

@ -846,7 +846,7 @@ in (opacity <= 1.0) {
pixmap.opacity = opacity255;
}
// ==== Alpha-blending functions ====
// ==== Blending functions ====
/++
Alpha-blending accuracy level
@ -882,66 +882,6 @@ enum BlendAccuracy {
rgba = true,
}
///
public void alphaBlend(
BlendAccuracy accuracy,
ubyte function(const ubyte, const ubyte) blend = null,
)(
scope Pixel[] target,
scope const Pixel[] source,
) @trusted
in (source.length == target.length) {
foreach (immutable idx, ref pxTarget; target) {
alphaBlend(pxTarget, source.ptr[idx]);
}
}
/// ditto
public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @safe {
return alphaBlend!(BlendAccuracy.rgba, null)(target, source);
}
///
public void alphaBlend(
BlendAccuracy accuracy,
ubyte function(const ubyte, const ubyte) blend = null,
)(
ref Pixel pxTarget,
const Pixel pxSource,
) @trusted {
pragma(inline, true);
static if (accuracy) {
immutable alphaResult = clamp255(pxSource.a + n255thsOf(pxTarget.a, (0xFF - pxSource.a)));
//immutable alphaResult = clamp255(pxTarget.a + n255thsOf(pxSource.a, (0xFF - pxTarget.a)));
}
immutable alphaSource = (pxSource.a | (pxSource.a << 8));
immutable alphaTarget = (0xFFFF - alphaSource);
foreach (immutable ib, ref px; pxTarget.components) {
static if (blend !is null) {
immutable bx = blend(px, pxSource.components.ptr[ib]);
} else {
immutable bx = pxSource.components.ptr[ib];
}
immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16);
immutable s = cast(ubyte)(((bx * alphaSource) + 0x8080) >> 16);
px = cast(ubyte)(d + s);
}
static if (accuracy) {
pxTarget.a = alphaResult;
}
}
/// ditto
public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @safe {
return alphaBlend!(BlendAccuracy.rgba, null)(pxTarget, pxSource);
}
// ==== Blending functions ====
/++
Blend modes
@ -1002,74 +942,154 @@ alias Blend = BlendMode;
// undocumented
enum blendNormal = BlendMode.normal;
///
alias BlendFn = ubyte function(const ubyte background, const ubyte foreground) pure nothrow @nogc;
/++
Blends pixel `source` into pixel `target`.
Blends `source` into `target`
with respect to the opacity of the source image (as stored in the alpha channel).
See_Also:
[alphaBlendRGBA] and [alphaBlendRGB] are shorthand functions
in cases where no special blending algorithm is needed.
+/
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == BlendMode.replace) {
template alphaBlend(BlendFn blend = null, BlendAccuracy accuracy = BlendAccuracy.rgba) {
/// ditto
public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted
in (source.length == target.length) {
foreach (immutable idx, ref pxTarget; target) {
alphaBlend(pxTarget, source.ptr[idx]);
}
}
/// ditto
public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted {
pragma(inline, true);
static if (accuracy == BlendAccuracy.rgba) {
immutable alphaResult = clamp255(pxSource.a + n255thsOf(pxTarget.a, (0xFF - pxSource.a)));
//immutable alphaResult = clamp255(pxTarget.a + n255thsOf(pxSource.a, (0xFF - pxTarget.a)));
}
immutable alphaSource = (pxSource.a | (pxSource.a << 8));
immutable alphaTarget = (0xFFFF - alphaSource);
foreach (immutable ib, ref px; pxTarget.components) {
static if (blend !is null) {
immutable bx = blend(px, pxSource.components.ptr[ib]);
} else {
immutable bx = pxSource.components.ptr[ib];
}
immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16);
immutable s = cast(ubyte)(((bx * alphaSource) + 0x8080) >> 16);
px = cast(ubyte)(d + s);
}
static if (accuracy == BlendAccuracy.rgba) {
pxTarget.a = alphaResult;
}
}
}
/// ditto
template alphaBlend(BlendAccuracy accuracy, BlendFn blend = null) {
alias alphaBlend = alphaBlend!(blend, accuracy);
}
/++
Blends `source` into `target`
with respect to the opacity of the source image (as stored in the alpha channel).
This variant is $(slower than) [alphaBlendRGB],
but calculates the correct alpha-channel value of the target.
See [BlendAccuracy] for further explanation.
+/
public void alphaBlendRGBA(scope Pixel[] target, scope const Pixel[] source) @safe {
return alphaBlend!(null, BlendAccuracy.rgba)(target, source);
}
/// ditto
public void alphaBlendRGBA(ref Pixel pxTarget, const Pixel pxSource) @safe {
return alphaBlend!(null, BlendAccuracy.rgba)(pxTarget, pxSource);
}
/++
Blends `source` into `target`
with respect to the opacity of the source image (as stored in the alpha channel).
This variant is $(B faster than) [alphaBlendRGBA],
but leads to a wrong alpha-channel value in the target.
Useful because of the performance advantage in cases where the resulting
alpha does not matter.
See [BlendAccuracy] for further explanation.
+/
public void alphaBlendRGB(scope Pixel[] target, scope const Pixel[] source) @safe {
return alphaBlend!(null, BlendAccuracy.rgb)(target, source);
}
/// ditto
public void alphaBlendRGB(ref Pixel pxTarget, const Pixel pxSource) @safe {
return alphaBlend!(null, BlendAccuracy.rgb)(pxTarget, pxSource);
}
/++
Blends pixel `source` into pixel `target`
using the requested $(B blending mode).
+/
template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) {
static if (mode == BlendMode.replace) {
/// ditto
void blendPixel(ref Pixel target, const Pixel source) {
target = source;
}
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.alpha) {
return alphaBlend!accuracy(target, source);
}
static if (mode == BlendMode.alpha) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.multiply) {
void blendPixel(ref Pixel target, const Pixel source) {
return alphaBlend!accuracy(target, source);
}
}
static if (mode == BlendMode.multiply) {
/// ditto
void blendPixel(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(a, b) => n255thsOf(a, b)
)(target, source);
}
}
static if (mode == BlendMode.screen) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.screen) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(a, b) => castTo!ubyte(0xFF - n255thsOf((0xFF - a), (0xFF - b)))
)(target, source);
}
}
static if (mode == BlendMode.darken) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.darken) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(a, b) => min(a, b)
)(target, source);
}
}
static if (mode == BlendMode.lighten) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.lighten) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(a, b) => max(a, b)
)(target, source);
}
}
static if (mode == BlendMode.overlay) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.overlay) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) {
if (b < 0x80) {
return n255thsOf((2 * b).castTo!ubyte, f);
@ -1079,13 +1099,11 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
);
})(target, source);
}
}
static if (mode == BlendMode.hardLight) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.hardLight) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) {
if (f < 0x80) {
return n255thsOf(castTo!ubyte(2 * f), b);
@ -1095,13 +1113,11 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
);
})(target, source);
}
}
static if (mode == BlendMode.softLight) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.softLight) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) {
if (f < 0x80) {
// dfmt off
@ -1125,13 +1141,11 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
);
})(target, source);
}
}
static if (mode == BlendMode.colorDodge) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.colorDodge) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) {
if (b == 0x00) {
return ubyte(0x00);
@ -1145,13 +1159,11 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
);
})(target, source);
}
}
static if (mode == BlendMode.colorBurn) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.colorBurn) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) {
if (b == 0xFF) {
return ubyte(0xFF);
@ -1167,57 +1179,55 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
return castTo!ubyte(0xFF - m);
})(target, source);
}
}
static if (mode == BlendMode.difference) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.difference) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(b, f) => (b > f) ? castTo!ubyte(b - f) : castTo!ubyte(f - b)
)(target, source);
}
}
static if (mode == BlendMode.exclusion) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.exclusion) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(b, f) => castTo!ubyte(b + f - (2 * n255thsOf(f, b)))
)(target, source);
}
}
static if (mode == BlendMode.subtract) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.subtract) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(b, f) => (b > f) ? castTo!ubyte(b - f) : ubyte(0)
)(target, source);
}
}
static if (mode == BlendMode.divide) {
/// ditto
void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)(
ref Pixel target,
const Pixel source,
) if (mode == Blend.divide) {
void blendPixel()(ref Pixel target, const Pixel source) {
return alphaBlend!(accuracy,
(b, f) => (f == 0) ? ubyte(0xFF) : clamp255(0xFF * b / f)
)(target, source);
}
}
}
/++
Blends the pixel data of `source` into `target`.
Blends the pixel data of `source` into `target`
using the requested $(B blending mode).
`source` and `target` MUST have the same length.
+/
void blendPixels(BlendMode mode, BlendAccuracy accuracy)(scope Pixel[] target, scope const Pixel[] source) @trusted
void blendPixels(
BlendMode mode,
BlendAccuracy accuracy,
)(scope Pixel[] target, scope const Pixel[] source) @trusted
in (source.length == target.length) {
static if (mode == BlendMode.replace) {
// explicit optimization
@ -1236,11 +1246,7 @@ in (source.length == target.length) {
}
/// ditto
void blendPixels(BlendAccuracy accuracy = BlendAccuracy.rgba)(
scope Pixel[] target,
scope const Pixel[] source,
BlendMode mode,
) {
void blendPixels(BlendAccuracy accuracy)(scope Pixel[] target, scope const Pixel[] source, BlendMode mode) {
import std.meta : NoDuplicates;
import std.traits : EnumMembers;