From 1ddb1e1e0e6b91b437bb176bf3a238aec5805a8c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 10 Jan 2025 04:14:37 +0100 Subject: [PATCH 01/38] Add UInt32p64 type --- pixmappaint.d | 172 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 172 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 15acaa0..a4c0ef9 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -374,6 +374,178 @@ static assert(Pixel.sizeof == uint.sizeof); } } +/++ + Unsigned 32-bit integer type with 64-bit precision for intermediate calculations + +/ +struct UInt32p64 { + private { + ulong _value = 0; + } + +@safe pure nothrow @nogc: + + /// + public this(uint initialValue) { + _value = (long(initialValue) << 32); + } + + private static UInt32p64 make(ulong internal) { + auto result = UInt32p64(); + result._value = internal; + return result; + } + + /// + T opCast(T : uint)() const { + return (_value >> 32).castTo!uint; + } + + /// + public UInt32p64 round() const { + const truncated = (_value & 0xFFFF_FFFF_0000_0000); + const delta = _value - truncated; + + // dfmt off + const rounded = (delta >= 0x8000_0000) + ? truncated + 0x1_0000_0000 + : truncated; + // dfmt on + + return UInt32p64.make(rounded); + } + + public { + /// + UInt32p64 opBinary(string op : "+")(const uint rhs) const { + return UInt32p64.make(_value + (ulong(rhs) << 32)); + } + + /// ditto + UInt32p64 opBinary(string op : "-")(const uint rhs) const { + return UInt32p64.make(_value - (ulong(rhs) << 32)); + } + + /// ditto + UInt32p64 opBinary(string op : "*")(const uint rhs) const { + return UInt32p64.make(_value * rhs); + } + + /// ditto + UInt32p64 opBinary(string op : "/")(const uint rhs) const { + return UInt32p64.make(_value / rhs); + } + } + + public { + /// + UInt32p64 opBinaryRight(string op : "+")(const uint lhs) const { + return UInt32p64.make((ulong(lhs) << 32) + _value); + } + + /// ditto + UInt32p64 opBinaryRight(string op : "-")(const uint lhs) const { + return UInt32p64.make((ulong(lhs) << 32) - _value); + } + + /// ditto + UInt32p64 opBinaryRight(string op : "*")(const uint lhs) const { + return UInt32p64.make(lhs * _value); + } + + /// ditto + UInt32p64 opBinaryRight(string op : "/")(const uint) const { + static assert(false, "Use `int() / cast(int)(UInt32p64())` instead."); + } + } + + public { + /// + auto opOpAssign(string op : "+")(const uint rhs) { + _value += (ulong(rhs) << 32); + return this; + } + + /// ditto + auto opOpAssign(string op : "-")(const uint rhs) { + _value -= (ulong(rhs) << 32); + return this; + } + + /// ditto + auto opOpAssign(string op : "*")(const uint rhs) { + _value *= rhs; + return this; + } + + /// ditto + auto opOpAssign(string op : "/")(const uint rhs) { + _value /= rhs; + return this; + } + } +} + +@safe unittest { + assert(UInt32p64(uint.max).castTo!uint == uint.max); + assert(UInt32p64(uint.min).castTo!uint == uint.min); + assert(UInt32p64(1).castTo!uint == 1); + assert(UInt32p64(2).castTo!uint == 2); + assert(UInt32p64(1_991_007).castTo!uint == 1_991_007); + + assert((UInt32p64(uint.max) / 2).castTo!uint == 2_147_483_647); + assert((UInt32p64(uint.max) / 2).round().castTo!uint == 2_147_483_648); + +} + +@safe unittest { + UInt32p64 val; + + val = UInt32p64(10); + val += 12; + assert(val.castTo!uint == 22); + + val = UInt32p64(1024); + val -= 24; + assert(val.castTo!uint == 1000); + val -= 100; + assert(val.castTo!uint == 900); + val += 5; + assert(val.castTo!uint == 905); + + val = UInt32p64(256); + val *= 4; + assert(val.castTo!uint == (256 * 4)); + + val = UInt32p64(2048); + val /= 10; + val *= 10; + assert(val.castTo!uint == 2047); +} + +@safe unittest { + UInt32p64 val; + + val = UInt32p64(9_000_000); + val /= 13; + val *= 4; + + // ≈ 2,769,230.8 + assert(val.castTo!uint == 2_769_230); + assert(val.round.castTo!uint == 2_769_231); + // assert(uint(9_000_000) / uint(13) * uint(4) == 2_769_228); + + val = UInt32p64(64); + val /= 31; + val *= 30; + val /= 29; + val *= 28; + + // ≈ 59.8 + assert(val.castTo!uint == 59); + assert(val.round().castTo!uint == 60); + // assert(((((64 / 31) * 30) / 29) * 28) == 56); +} + /++ $(I Advanced functionality.) From 9f280bfbb68eb2857786b7d1fbd080f3bead6358 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 10 Jan 2025 04:19:50 +0100 Subject: [PATCH 02/38] Increase unittest coverage of `UInt32p64` --- pixmappaint.d | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index a4c0ef9..5d5ebbd 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -492,6 +492,15 @@ struct UInt32p64 { assert(UInt32p64(2).castTo!uint == 2); assert(UInt32p64(1_991_007).castTo!uint == 1_991_007); + assert((UInt32p64(10) + 9).castTo!uint == 19); + assert((UInt32p64(10) - 9).castTo!uint == 1); + assert((UInt32p64(10) * 9).castTo!uint == 90); + assert((UInt32p64(99) / 9).castTo!uint == 11); + + assert((4 + UInt32p64(4)).castTo!uint == 8); + assert((4 - UInt32p64(4)).castTo!uint == 0); + assert((4 * UInt32p64(4)).castTo!uint == 16); + assert((UInt32p64(uint.max) / 2).castTo!uint == 2_147_483_647); assert((UInt32p64(uint.max) / 2).round().castTo!uint == 2_147_483_648); @@ -531,7 +540,7 @@ struct UInt32p64 { // ≈ 2,769,230.8 assert(val.castTo!uint == 2_769_230); - assert(val.round.castTo!uint == 2_769_231); + assert(val.round().castTo!uint == 2_769_231); // assert(uint(9_000_000) / uint(13) * uint(4) == 2_769_228); val = UInt32p64(64); From 3c40abb151f4a320797b63ce2f2d8ae8cf9e3c97 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 01:13:42 +0100 Subject: [PATCH 03/38] Implement "nearest neighbour" scaling in PixmapPaint --- pixmappaint.d | 124 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 124 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 5d5ebbd..90c2238 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2772,6 +2772,130 @@ PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } +/// +enum ScalingMethod { + /// + nearest, + + /// + linear, // TODO: Decide whether to replace this + // by moving over `ScalingFilter` from `arsd.pixmappresenter`. +} + +private alias Scale = ScalingMethod; + +private enum ScalingDirection { + none, + up, + down, +} + +private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc { + if (delta == 0) { + return ScalingDirection.none; + } else if (delta > 0) { + return ScalingDirection.up; + } else { + return ScalingDirection.down; + } +} + +private void scaleToImpl(Scale method)(const Pixmap source, Pixmap target) @nogc { + + enum none = ScalingDirection.none; + enum up = ScalingDirection.up; + enum down = ScalingDirection.down; + + const sourceMaxX = (source.width - 1); + const sourceMaxY = (source.height - 1); + + const Size delta = (target.size - source.size); + + const ScalingDirection directionX = scalingDirectionFromDelta(delta.width); + const ScalingDirection directionY = scalingDirectionFromDelta(delta.height); + + const ratioX = (UInt32p64(source.width) / target.width); + const ratioY = (UInt32p64(source.height) / target.height); + + Point translate(const Point dstPos) { + pragma(inline, true); + const x = min( + (dstPos.x * ratioX).round().castTo!int, + sourceMaxX + ); + const y = min( + (dstPos.y * ratioY).round().castTo!int, + sourceMaxY + ); + return Point(x, y); + } + + // Nearest Neighbour + static if (method == Scale.nearest) { + auto dst = PixmapScannerRW(target); + + size_t y = 0; + foreach (dstLine; dst) { + foreach (x, ref pxDst; dstLine) { + const posDst = Point(x.castTo!int, y.castTo!int); + const posSrc = translate(posDst); + const pxSrc = source.getPixel(posSrc); + pxDst = pxSrc; + } + ++y; + } + } else static if (method == Scale.linear) { + static assert(false, "Not implemented."); + } else { + static assert(false, "Scaling method not implemented yet."); + } +} + +void scaleTo(const Pixmap source, Pixmap target, ScalingMethod method) @nogc { + import std.meta : NoDuplicates; + import std.traits : EnumMembers; + + // dfmt off + final switch (method) { + static foreach (scalingMethod; NoDuplicates!(EnumMembers!ScalingMethod)) + case scalingMethod: { + scaleToImpl!scalingMethod(source, target); + return; + } + } + // dfmt on +} + +// consistency +private alias scaleInto = scaleTo; + +/++ + Scales an image to a new size. + + ``` + ╔═══╗ ╔═╗ + ║———║ → ║—║ + ╚═══╝ ╚═╝ + ``` + +/ +Pixmap scale(const Pixmap source, Pixmap target, Size scaleToSize, ScalingMethod method) @nogc { + target.adjustTo(scaleCalcDims(scaleToSize)); + source.scaleInto(target, method); + return target; +} + +/// ditto +Pixmap scaleNew(const Pixmap source, Size scaleToSize, ScalingMethod method) { + auto target = Pixmap.makeNew(scaleToSize); + source.scaleInto(target, method); + return target; +} + +/// ditto +PixmapBlueprint scaleCalcDims(Size scaleToSize) @nogc { + return PixmapBlueprint.fromSize(scaleToSize); +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From 9c5a341bce2da48453e49b9452ca098bb8dd472d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 01:57:24 +0100 Subject: [PATCH 04/38] Move `ScalingFilter` to PixmapPaint --- pixmappaint.d | 44 ++++++++++++++++++++++++++++---------------- pixmappresenter.d | 6 ------ 2 files changed, 28 insertions(+), 22 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 90c2238..ff52370 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2773,16 +2773,28 @@ PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { } /// -enum ScalingMethod { - /// +enum ScalingFilter { + /++ + Nearest neighbour interpolation + + Also known $(B proximal interpolation) + and $(B point sampling). + + $(TIP + Visual impression: “blocky”, “pixel’ish” + ) + +/ nearest, - /// - linear, // TODO: Decide whether to replace this - // by moving over `ScalingFilter` from `arsd.pixmappresenter`. -} + /++ + (Bi-)linear interpolation -private alias Scale = ScalingMethod; + $(TIP + Visual impression: “smooth”, “blurry” + ) + +/ + linear, +} private enum ScalingDirection { none, @@ -2800,7 +2812,7 @@ private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc } } -private void scaleToImpl(Scale method)(const Pixmap source, Pixmap target) @nogc { +private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap target) @nogc { enum none = ScalingDirection.none; enum up = ScalingDirection.up; @@ -2831,7 +2843,7 @@ private void scaleToImpl(Scale method)(const Pixmap source, Pixmap target) @nogc } // Nearest Neighbour - static if (method == Scale.nearest) { + static if (method == ScalingFilter.nearest) { auto dst = PixmapScannerRW(target); size_t y = 0; @@ -2844,22 +2856,22 @@ private void scaleToImpl(Scale method)(const Pixmap source, Pixmap target) @nogc } ++y; } - } else static if (method == Scale.linear) { + } else static if (method == ScalingFilter.linear) { static assert(false, "Not implemented."); } else { static assert(false, "Scaling method not implemented yet."); } } -void scaleTo(const Pixmap source, Pixmap target, ScalingMethod method) @nogc { +void scaleTo(const Pixmap source, Pixmap target, ScalingFilter method) @nogc { import std.meta : NoDuplicates; import std.traits : EnumMembers; // dfmt off final switch (method) { - static foreach (scalingMethod; NoDuplicates!(EnumMembers!ScalingMethod)) - case scalingMethod: { - scaleToImpl!scalingMethod(source, target); + static foreach (scalingFilter; NoDuplicates!(EnumMembers!ScalingFilter)) + case scalingFilter: { + scaleToImpl!scalingFilter(source, target); return; } } @@ -2878,14 +2890,14 @@ private alias scaleInto = scaleTo; ╚═══╝ ╚═╝ ``` +/ -Pixmap scale(const Pixmap source, Pixmap target, Size scaleToSize, ScalingMethod method) @nogc { +Pixmap scale(const Pixmap source, Pixmap target, Size scaleToSize, ScalingFilter method) @nogc { target.adjustTo(scaleCalcDims(scaleToSize)); source.scaleInto(target, method); return target; } /// ditto -Pixmap scaleNew(const Pixmap source, Size scaleToSize, ScalingMethod method) { +Pixmap scaleNew(const Pixmap source, Size scaleToSize, ScalingFilter method) { auto target = Pixmap.makeNew(scaleToSize); source.scaleInto(target, method); return target; diff --git a/pixmappresenter.d b/pixmappresenter.d index 858d65b..1b12910 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -372,12 +372,6 @@ enum Scaling { cssCover = cover, /// equivalent CSS: `object-fit: cover;` } -/// -enum ScalingFilter { - nearest, /// nearest neighbor → blocky/pixel’ish - linear, /// (bi-)linear interpolation → smooth/blurry -} - /// struct PresenterConfig { Window window; /// From a024404330402ddbf0ce32fe5a04aec25d905407 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 02:06:36 +0100 Subject: [PATCH 05/38] Add floor and ceiling rounding functions to `UInt32p64` --- pixmappaint.d | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index ff52370..b7188bc 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -414,6 +414,25 @@ struct UInt32p64 { return UInt32p64.make(rounded); } + /// + public UInt32p64 floor() const { + const truncated = (_value & 0xFFFF_FFFF_0000_0000); + return UInt32p64.make(truncated); + } + + /// + public UInt32p64 ceil() const { + const truncated = (_value & 0xFFFF_FFFF_0000_0000); + + // dfmt off + const ceiling = (truncated != _value) + ? truncated + 0x1_0000_0000 + : truncated; + // dfmt on + + return UInt32p64.make(ceiling); + } + public { /// UInt32p64 opBinary(string op : "+")(const uint rhs) const { @@ -504,6 +523,17 @@ struct UInt32p64 { assert((UInt32p64(uint.max) / 2).castTo!uint == 2_147_483_647); assert((UInt32p64(uint.max) / 2).round().castTo!uint == 2_147_483_648); + assert((UInt32p64(10) / 8).round().castTo!uint == 1); + assert((UInt32p64(10) / 8).floor().castTo!uint == 1); + assert((UInt32p64(10) / 8).ceil().castTo!uint == 2); + + assert((UInt32p64(10) / 4).round().castTo!uint == 3); + assert((UInt32p64(10) / 4).floor().castTo!uint == 2); + assert((UInt32p64(10) / 4).ceil().castTo!uint == 3); + + assert((UInt32p64(10) / 5).round().castTo!uint == 2); + assert((UInt32p64(10) / 5).floor().castTo!uint == 2); + assert((UInt32p64(10) / 5).ceil().castTo!uint == 2); } @safe unittest { From 78ed1bb287a1c8a12e89737a8eb8dca66c52df9c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 02:27:53 +0100 Subject: [PATCH 06/38] Rename `UInt32p64` to `UDecimal` --- pixmappaint.d | 121 ++++++++++++++++++++++++++------------------------ 1 file changed, 62 insertions(+), 59 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index b7188bc..c520613 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -375,9 +375,12 @@ static assert(Pixel.sizeof == uint.sizeof); } /++ - Unsigned 32-bit integer type with 64-bit precision for intermediate calculations + Unsigned 64-bit fixed-point decimal type + + Assigns 32 bits to the digits of the pre-decimal point portion + and the other 32 bits to the digits of the fractional part. +/ -struct UInt32p64 { +struct UDecimal { private { ulong _value = 0; } @@ -389,8 +392,8 @@ struct UInt32p64 { _value = (long(initialValue) << 32); } - private static UInt32p64 make(ulong internal) { - auto result = UInt32p64(); + private static UDecimal make(ulong internal) { + auto result = UDecimal(); result._value = internal; return result; } @@ -401,7 +404,7 @@ struct UInt32p64 { } /// - public UInt32p64 round() const { + public UDecimal round() const { const truncated = (_value & 0xFFFF_FFFF_0000_0000); const delta = _value - truncated; @@ -411,17 +414,17 @@ struct UInt32p64 { : truncated; // dfmt on - return UInt32p64.make(rounded); + return UDecimal.make(rounded); } /// - public UInt32p64 floor() const { + public UDecimal floor() const { const truncated = (_value & 0xFFFF_FFFF_0000_0000); - return UInt32p64.make(truncated); + return UDecimal.make(truncated); } /// - public UInt32p64 ceil() const { + public UDecimal ceil() const { const truncated = (_value & 0xFFFF_FFFF_0000_0000); // dfmt off @@ -430,50 +433,50 @@ struct UInt32p64 { : truncated; // dfmt on - return UInt32p64.make(ceiling); + return UDecimal.make(ceiling); } public { /// - UInt32p64 opBinary(string op : "+")(const uint rhs) const { - return UInt32p64.make(_value + (ulong(rhs) << 32)); + UDecimal opBinary(string op : "+")(const uint rhs) const { + return UDecimal.make(_value + (ulong(rhs) << 32)); } /// ditto - UInt32p64 opBinary(string op : "-")(const uint rhs) const { - return UInt32p64.make(_value - (ulong(rhs) << 32)); + UDecimal opBinary(string op : "-")(const uint rhs) const { + return UDecimal.make(_value - (ulong(rhs) << 32)); } /// ditto - UInt32p64 opBinary(string op : "*")(const uint rhs) const { - return UInt32p64.make(_value * rhs); + UDecimal opBinary(string op : "*")(const uint rhs) const { + return UDecimal.make(_value * rhs); } /// ditto - UInt32p64 opBinary(string op : "/")(const uint rhs) const { - return UInt32p64.make(_value / rhs); + UDecimal opBinary(string op : "/")(const uint rhs) const { + return UDecimal.make(_value / rhs); } } public { /// - UInt32p64 opBinaryRight(string op : "+")(const uint lhs) const { - return UInt32p64.make((ulong(lhs) << 32) + _value); + UDecimal opBinaryRight(string op : "+")(const uint lhs) const { + return UDecimal.make((ulong(lhs) << 32) + _value); } /// ditto - UInt32p64 opBinaryRight(string op : "-")(const uint lhs) const { - return UInt32p64.make((ulong(lhs) << 32) - _value); + UDecimal opBinaryRight(string op : "-")(const uint lhs) const { + return UDecimal.make((ulong(lhs) << 32) - _value); } /// ditto - UInt32p64 opBinaryRight(string op : "*")(const uint lhs) const { - return UInt32p64.make(lhs * _value); + UDecimal opBinaryRight(string op : "*")(const uint lhs) const { + return UDecimal.make(lhs * _value); } /// ditto - UInt32p64 opBinaryRight(string op : "/")(const uint) const { - static assert(false, "Use `int() / cast(int)(UInt32p64())` instead."); + UDecimal opBinaryRight(string op : "/")(const uint) const { + static assert(false, "Use `int() / cast(int)(UDecimal())` instead."); } } @@ -505,45 +508,45 @@ struct UInt32p64 { } @safe unittest { - assert(UInt32p64(uint.max).castTo!uint == uint.max); - assert(UInt32p64(uint.min).castTo!uint == uint.min); - assert(UInt32p64(1).castTo!uint == 1); - assert(UInt32p64(2).castTo!uint == 2); - assert(UInt32p64(1_991_007).castTo!uint == 1_991_007); + assert(UDecimal(uint.max).castTo!uint == uint.max); + assert(UDecimal(uint.min).castTo!uint == uint.min); + assert(UDecimal(1).castTo!uint == 1); + assert(UDecimal(2).castTo!uint == 2); + assert(UDecimal(1_991_007).castTo!uint == 1_991_007); - assert((UInt32p64(10) + 9).castTo!uint == 19); - assert((UInt32p64(10) - 9).castTo!uint == 1); - assert((UInt32p64(10) * 9).castTo!uint == 90); - assert((UInt32p64(99) / 9).castTo!uint == 11); + assert((UDecimal(10) + 9).castTo!uint == 19); + assert((UDecimal(10) - 9).castTo!uint == 1); + assert((UDecimal(10) * 9).castTo!uint == 90); + assert((UDecimal(99) / 9).castTo!uint == 11); - assert((4 + UInt32p64(4)).castTo!uint == 8); - assert((4 - UInt32p64(4)).castTo!uint == 0); - assert((4 * UInt32p64(4)).castTo!uint == 16); + assert((4 + UDecimal(4)).castTo!uint == 8); + assert((4 - UDecimal(4)).castTo!uint == 0); + assert((4 * UDecimal(4)).castTo!uint == 16); - assert((UInt32p64(uint.max) / 2).castTo!uint == 2_147_483_647); - assert((UInt32p64(uint.max) / 2).round().castTo!uint == 2_147_483_648); + assert((UDecimal(uint.max) / 2).castTo!uint == 2_147_483_647); + assert((UDecimal(uint.max) / 2).round().castTo!uint == 2_147_483_648); - assert((UInt32p64(10) / 8).round().castTo!uint == 1); - assert((UInt32p64(10) / 8).floor().castTo!uint == 1); - assert((UInt32p64(10) / 8).ceil().castTo!uint == 2); + assert((UDecimal(10) / 8).round().castTo!uint == 1); + assert((UDecimal(10) / 8).floor().castTo!uint == 1); + assert((UDecimal(10) / 8).ceil().castTo!uint == 2); - assert((UInt32p64(10) / 4).round().castTo!uint == 3); - assert((UInt32p64(10) / 4).floor().castTo!uint == 2); - assert((UInt32p64(10) / 4).ceil().castTo!uint == 3); + assert((UDecimal(10) / 4).round().castTo!uint == 3); + assert((UDecimal(10) / 4).floor().castTo!uint == 2); + assert((UDecimal(10) / 4).ceil().castTo!uint == 3); - assert((UInt32p64(10) / 5).round().castTo!uint == 2); - assert((UInt32p64(10) / 5).floor().castTo!uint == 2); - assert((UInt32p64(10) / 5).ceil().castTo!uint == 2); + assert((UDecimal(10) / 5).round().castTo!uint == 2); + assert((UDecimal(10) / 5).floor().castTo!uint == 2); + assert((UDecimal(10) / 5).ceil().castTo!uint == 2); } @safe unittest { - UInt32p64 val; + UDecimal val; - val = UInt32p64(10); + val = UDecimal(10); val += 12; assert(val.castTo!uint == 22); - val = UInt32p64(1024); + val = UDecimal(1024); val -= 24; assert(val.castTo!uint == 1000); val -= 100; @@ -551,20 +554,20 @@ struct UInt32p64 { val += 5; assert(val.castTo!uint == 905); - val = UInt32p64(256); + val = UDecimal(256); val *= 4; assert(val.castTo!uint == (256 * 4)); - val = UInt32p64(2048); + val = UDecimal(2048); val /= 10; val *= 10; assert(val.castTo!uint == 2047); } @safe unittest { - UInt32p64 val; + UDecimal val; - val = UInt32p64(9_000_000); + val = UDecimal(9_000_000); val /= 13; val *= 4; @@ -573,7 +576,7 @@ struct UInt32p64 { assert(val.round().castTo!uint == 2_769_231); // assert(uint(9_000_000) / uint(13) * uint(4) == 2_769_228); - val = UInt32p64(64); + val = UDecimal(64); val /= 31; val *= 30; val /= 29; @@ -2856,8 +2859,8 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe const ScalingDirection directionX = scalingDirectionFromDelta(delta.width); const ScalingDirection directionY = scalingDirectionFromDelta(delta.height); - const ratioX = (UInt32p64(source.width) / target.width); - const ratioY = (UInt32p64(source.height) / target.height); + const ratioX = (UDecimal(source.width) / target.width); + const ratioY = (UDecimal(source.height) / target.height); Point translate(const Point dstPos) { pragma(inline, true); From 266ae6f7dc7e3109b9d0691051b21f01c16af38f Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 02:32:25 +0100 Subject: [PATCH 07/38] Implement `opBinaryRight(string op : "/")` of `UDecimal` --- pixmappaint.d | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index c520613..d24be5d 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -475,8 +475,8 @@ struct UDecimal { } /// ditto - UDecimal opBinaryRight(string op : "/")(const uint) const { - static assert(false, "Use `int() / cast(int)(UDecimal())` instead."); + UDecimal opBinaryRight(string op : "/")(const uint lhs) const { + return UDecimal.make(lhs / _value); } } From bc196985b5ecaaecd58ebb22c12f027f1035e011 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 06:45:46 +0100 Subject: [PATCH 08/38] Implement bilinear image upscaling in PixmapPaint --- pixmappaint.d | 147 ++++++++++++++++++++++++++++++++++++++++++---- pixmappresenter.d | 2 + 2 files changed, 136 insertions(+), 13 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index d24be5d..910784e 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -378,7 +378,7 @@ static assert(Pixel.sizeof == uint.sizeof); Unsigned 64-bit fixed-point decimal type Assigns 32 bits to the digits of the pre-decimal point portion - and the other 32 bits to the digits of the fractional part. + and the other 32 bits to fractional digits. +/ struct UDecimal { private { @@ -436,6 +436,11 @@ struct UDecimal { return UDecimal.make(ceiling); } + /// + public uint fractionalDigits() const { + return (_value & 0x0000_0000_FFFF_FFFF); + } + public { /// UDecimal opBinary(string op : "+")(const uint rhs) const { @@ -2805,28 +2810,56 @@ PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } -/// +/++ + Interpolation methods to apply when scaling images + + Each filter has its own distinctive properties. + + $(TIP + Bilinear filtering (`linear`) is general-purpose. + Works well with photos. + + For pixel graphics the retro look of `nearest` (as + in $(I nearest neighbour)) is usually the option of choice. + ) + + $(NOTE + When used as a parameter, it shall be understood as a hint. + + Implementations are not required to support all enumerated options + and may pick a different filter as a substitute at their own discretion. + ) + +/ enum ScalingFilter { /++ Nearest neighbour interpolation - Also known $(B proximal interpolation) + Also known as $(B proximal interpolation) and $(B point sampling). $(TIP - Visual impression: “blocky”, “pixel’ish” + Visual impression: “blocky”, “pixelated”, “slightly displaced” ) +/ nearest, /++ - (Bi-)linear interpolation + Bilinear interpolation $(TIP - Visual impression: “smooth”, “blurry” + Visual impression: “smooth”, “blurred” ) +/ linear, + + /++ + Unweighted linear interpolation + + $(TIP + Visual impression: “blocky”, “pixelated” + ) + +/ + fauxLinear, } private enum ScalingDirection { @@ -2845,6 +2878,7 @@ private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc } } +// TODO: Rename `method` to `filter` private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap target) @nogc { enum none = ScalingDirection.none; @@ -2875,7 +2909,7 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe return Point(x, y); } - // Nearest Neighbour + // ==== Nearest Neighbour ==== static if (method == ScalingFilter.nearest) { auto dst = PixmapScannerRW(target); @@ -2884,15 +2918,102 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe foreach (x, ref pxDst; dstLine) { const posDst = Point(x.castTo!int, y.castTo!int); const posSrc = translate(posDst); - const pxSrc = source.getPixel(posSrc); - pxDst = pxSrc; + const pxInt = source.getPixel(posSrc); + pxDst = pxInt; } ++y; } - } else static if (method == ScalingFilter.linear) { - static assert(false, "Not implemented."); - } else { - static assert(false, "Scaling method not implemented yet."); + } + + // ==== Bilinear ==== + static if ((method == ScalingFilter.linear) || (method == ScalingFilter.fauxLinear)) { + auto dst = PixmapScannerRW(target); + + size_t y = 0; + foreach (dstLine; dst) { + foreach (x, ref pxDst; dstLine) { + const posDst = Point(x.castTo!int, y.castTo!int); + + const UDecimal[2] posSrc = [ + (posDst.x * ratioX), + (posDst.y * ratioY), + ]; + + const posSrcXF = (() @trusted => min(sourceMaxX, posSrc.ptr[0].floor().castTo!int))(); + const posSrcXC = (() @trusted => min(sourceMaxX, posSrc.ptr[0].ceil().castTo!int))(); + + const posSrcYF = (() @trusted => min(sourceMaxY, posSrc.ptr[1].floor().castTo!int))(); + const posSrcYC = (() @trusted => min(sourceMaxY, posSrc.ptr[1].ceil().castTo!int))(); + + const Point[4] posNeighs = [ + Point(posSrcXF, posSrcYF), + Point(posSrcXC, posSrcYF), + Point(posSrcXF, posSrcYC), + Point(posSrcXC, posSrcYC), + ]; + + const Color[4] pxNeighs = [ + source.getPixel((() @trusted => posNeighs.ptr[0])()), + source.getPixel((() @trusted => posNeighs.ptr[1])()), + source.getPixel((() @trusted => posNeighs.ptr[2])()), + source.getPixel((() @trusted => posNeighs.ptr[3])()), + ]; + + // TODO: Downscaling (currently equivalent to nearest neighbour but with extra steps!) + + // ====== Faux bilinear ====== + static if (method == ScalingFilter.fauxLinear) { + auto pxInt = Pixel(0, 0, 0, 0); + + foreach (immutable ib, ref c; pxInt.components) { + uint sum = 0; + foreach (const pxNeigh; pxNeighs) { + sum += (() @trusted => pxNeigh.components.ptr[ib])(); + } + c = (sum >> 2).castTo!ubyte; + } + } + + // ====== Proper bilinear ====== + static if (method == ScalingFilter.linear) { + const ulong[2] fract = [ + (() @trusted => posSrc.ptr[0].fractionalDigits)(), + (() @trusted => posSrc.ptr[1].fractionalDigits)(), + ]; + + const ulong[2] fractComplementary = ((ulong(uint.max) + 1) - fract[]); + + alias fractC = fract; + alias fractF = fractComplementary; + + auto pxInt = Pixel(0, 0, 0, 0); + foreach (immutable ib, ref c; pxInt.components) { + ulong[2] xSums = [0, 0]; + xSums[0] += (() @trusted => (pxNeighs.ptr[0].components.ptr[ib] * fractF.ptr[0]))(); + xSums[0] += (() @trusted => (pxNeighs.ptr[1].components.ptr[ib] * fractC.ptr[0]))(); + + xSums[1] += (() @trusted => (pxNeighs.ptr[2].components.ptr[ib] * fractF.ptr[0]))(); + xSums[1] += (() @trusted => (pxNeighs.ptr[3].components.ptr[ib] * fractC.ptr[0]))(); + + foreach (ref sum; xSums) { + sum >>= 32; + } + + ulong ySum = 0; + ySum += (() @trusted => (xSums.ptr[0] * fractF.ptr[1]))(); + ySum += (() @trusted => (xSums.ptr[1] * fractC.ptr[1]))(); + + const xySum = (ySum >> 32); + + c = clamp255(xySum); + } + } + + pxDst = pxInt; + } + + ++y; + } } } diff --git a/pixmappresenter.d b/pixmappresenter.d index 1b12910..0e46d3d 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -630,6 +630,7 @@ final class OpenGl3PixmapRenderer : PixmapRenderer { final switch (_poc.config.renderer.filter) with (ScalingFilter) { case nearest: + case fauxLinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; @@ -739,6 +740,7 @@ final class OpenGl1PixmapRenderer : PixmapRenderer { final switch (_poc.config.renderer.filter) with (ScalingFilter) { case nearest: + case fauxLinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; From f1d41403d17e332009e0653b5970b9ad8c9dfada Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 06:53:04 +0100 Subject: [PATCH 09/38] Rename scaling filter `linear` to `bilinear` --- pixmappaint.d | 9 ++++++--- pixmappresenter.d | 6 +++--- 2 files changed, 9 insertions(+), 6 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 910784e..f5e9c15 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2850,7 +2850,7 @@ enum ScalingFilter { Visual impression: “smooth”, “blurred” ) +/ - linear, + bilinear, /++ Unweighted linear interpolation @@ -2860,6 +2860,9 @@ enum ScalingFilter { ) +/ fauxLinear, + + /// + //linear = bilinear, } private enum ScalingDirection { @@ -2926,7 +2929,7 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe } // ==== Bilinear ==== - static if ((method == ScalingFilter.linear) || (method == ScalingFilter.fauxLinear)) { + static if ((method == ScalingFilter.bilinear) || (method == ScalingFilter.fauxLinear)) { auto dst = PixmapScannerRW(target); size_t y = 0; @@ -2975,7 +2978,7 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe } // ====== Proper bilinear ====== - static if (method == ScalingFilter.linear) { + static if (method == ScalingFilter.bilinear) { const ulong[2] fract = [ (() @trusted => posSrc.ptr[0].fractionalDigits)(), (() @trusted => posSrc.ptr[1].fractionalDigits)(), diff --git a/pixmappresenter.d b/pixmappresenter.d index 0e46d3d..c8bab90 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -391,7 +391,7 @@ struct PresenterConfig { Scaling scaling = Scaling.keepAspectRatio; /++ - Filter + Scaling filter +/ ScalingFilter filter = ScalingFilter.nearest; @@ -634,7 +634,7 @@ final class OpenGl3PixmapRenderer : PixmapRenderer { glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; - case linear: + case bilinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); break; @@ -744,7 +744,7 @@ final class OpenGl1PixmapRenderer : PixmapRenderer { glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; - case linear: + case bilinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); break; From ebd1c62d69cce212f2f12011a6956155421aa2cf Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 06:56:04 +0100 Subject: [PATCH 10/38] Rename param `method` of type `ScalingFilter` to `filter` --- pixmappaint.d | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index f5e9c15..b6bed8e 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2881,8 +2881,7 @@ private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc } } -// TODO: Rename `method` to `filter` -private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap target) @nogc { +private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap target) @nogc { enum none = ScalingDirection.none; enum up = ScalingDirection.up; @@ -2913,7 +2912,7 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe } // ==== Nearest Neighbour ==== - static if (method == ScalingFilter.nearest) { + static if (filter == ScalingFilter.nearest) { auto dst = PixmapScannerRW(target); size_t y = 0; @@ -2929,7 +2928,7 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe } // ==== Bilinear ==== - static if ((method == ScalingFilter.bilinear) || (method == ScalingFilter.fauxLinear)) { + static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { auto dst = PixmapScannerRW(target); size_t y = 0; @@ -3020,12 +3019,13 @@ private void scaleToImpl(ScalingFilter method)(const Pixmap source, Pixmap targe } } -void scaleTo(const Pixmap source, Pixmap target, ScalingFilter method) @nogc { +// TODO: Document this function +void scaleTo(const Pixmap source, Pixmap target, ScalingFilter filter) @nogc { import std.meta : NoDuplicates; import std.traits : EnumMembers; // dfmt off - final switch (method) { + final switch (filter) { static foreach (scalingFilter; NoDuplicates!(EnumMembers!ScalingFilter)) case scalingFilter: { scaleToImpl!scalingFilter(source, target); @@ -3047,16 +3047,16 @@ private alias scaleInto = scaleTo; ╚═══╝ ╚═╝ ``` +/ -Pixmap scale(const Pixmap source, Pixmap target, Size scaleToSize, ScalingFilter method) @nogc { +Pixmap scale(const Pixmap source, Pixmap target, Size scaleToSize, ScalingFilter filter) @nogc { target.adjustTo(scaleCalcDims(scaleToSize)); - source.scaleInto(target, method); + source.scaleInto(target, filter); return target; } /// ditto -Pixmap scaleNew(const Pixmap source, Size scaleToSize, ScalingFilter method) { +Pixmap scaleNew(const Pixmap source, Size scaleToSize, ScalingFilter filter) { auto target = Pixmap.makeNew(scaleToSize); - source.scaleInto(target, method); + source.scaleInto(target, filter); return target; } From 4f3dca5a324a36a0492ecc85fd584281e910a48f Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 22:07:27 +0100 Subject: [PATCH 11/38] Refactor, update and fix PixmapPaint --- pixmappaint.d | 43 ++++++++++++++++++++++--------------------- 1 file changed, 22 insertions(+), 21 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index b6bed8e..b14abfa 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -248,12 +248,12 @@ Depending on the operation, implementing in-place transformations can be either straightforward or a major undertaking (and topic of research). - This library focuses and the former and leaves out cases where the latter - applies. + This library focuses and the former case and leaves out those where the + latter applies. In particular, algorithms that require allocating further buffers to store temporary results or auxiliary data will probably not get implemented. - Furthermore, operations where to result is longer than the source cannot + Furthermore, operations where to result is larger than the source cannot be performed in-place. Certain in-place manipulation functions return a shallow-copy of the @@ -2770,7 +2770,7 @@ private void flipVerticallyInto(const Pixmap source, Pixmap target) @nogc { +/ Pixmap flipVertically(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.flipVerticallyCalcDims()); - flipVerticallyInto(source, target); + source.flipVerticallyInto(target); return target; } @@ -2862,7 +2862,7 @@ enum ScalingFilter { fauxLinear, /// - //linear = bilinear, + linear = bilinear, } private enum ScalingDirection { @@ -2892,22 +2892,23 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const Size delta = (target.size - source.size); - const ScalingDirection directionX = scalingDirectionFromDelta(delta.width); - const ScalingDirection directionY = scalingDirectionFromDelta(delta.height); + const ScalingDirection[2] directions = [ + scalingDirectionFromDelta(delta.width), + scalingDirectionFromDelta(delta.height), + ]; - const ratioX = (UDecimal(source.width) / target.width); - const ratioY = (UDecimal(source.height) / target.height); + const UDecimal[2] ratios = [ + (UDecimal(source.width) / target.width), + (UDecimal(source.height) / target.height), + ]; + // TODO: move Point translate(const Point dstPos) { pragma(inline, true); - const x = min( - (dstPos.x * ratioX).round().castTo!int, - sourceMaxX - ); - const y = min( - (dstPos.y * ratioY).round().castTo!int, - sourceMaxY - ); + const xCandidate = (() @trusted => (dstPos.x * ratios.ptr[0]).round().castTo!int)(); + const yCandidate = (() @trusted => (dstPos.y * ratios.ptr[1]).round().castTo!int)(); + const x = min(xCandidate, sourceMaxX); + const y = min(yCandidate, sourceMaxY); return Point(x, y); } @@ -2937,8 +2938,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const posDst = Point(x.castTo!int, y.castTo!int); const UDecimal[2] posSrc = [ - (posDst.x * ratioX), - (posDst.y * ratioY), + (() @trusted => posDst.x * ratios.ptr[0])(), + (() @trusted => posDst.y * ratios.ptr[1])(), ]; const posSrcXF = (() @trusted => min(sourceMaxX, posSrc.ptr[0].floor().castTo!int))(); @@ -2964,7 +2965,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // TODO: Downscaling (currently equivalent to nearest neighbour but with extra steps!) // ====== Faux bilinear ====== - static if (method == ScalingFilter.fauxLinear) { + static if (filter == ScalingFilter.fauxLinear) { auto pxInt = Pixel(0, 0, 0, 0); foreach (immutable ib, ref c; pxInt.components) { @@ -2977,7 +2978,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } // ====== Proper bilinear ====== - static if (method == ScalingFilter.bilinear) { + static if (filter == ScalingFilter.bilinear) { const ulong[2] fract = [ (() @trusted => posSrc.ptr[0].fractionalDigits)(), (() @trusted => posSrc.ptr[1].fractionalDigits)(), From 6eb6e8859472b30f9f6468348035242f3500b9c0 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 12 Jan 2025 22:59:37 +0100 Subject: [PATCH 12/38] Revert "Implement `opBinaryRight(string op : "/")` of `UDecimal`" This reverts commit 266ae6f7dc7e3109b9d0691051b21f01c16af38f. Implementation was impromper and an attempt to fix it outlined why it was't implemented in the first place. --- pixmappaint.d | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index b14abfa..1fa7b77 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -480,8 +480,8 @@ struct UDecimal { } /// ditto - UDecimal opBinaryRight(string op : "/")(const uint lhs) const { - return UDecimal.make(lhs / _value); + UDecimal opBinaryRight(string op : "/")(const uint) const { + static assert(false, "Use `uint(…) / cast(uint)(UDecimal(…))` instead."); } } From cce1a924ae627ca2fe628fcb5e2efd396133ffe1 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 13 Jan 2025 03:15:47 +0100 Subject: [PATCH 13/38] Add additional operator overloads to `UDecimal` --- pixmappaint.d | 44 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 44 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 1fa7b77..9590267 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -447,11 +447,21 @@ struct UDecimal { return UDecimal.make(_value + (ulong(rhs) << 32)); } + /// ditto + UDecimal opBinary(string op : "+")(const UDecimal rhs) const { + return UDecimal.make(_value + rhs._value); + } + /// ditto UDecimal opBinary(string op : "-")(const uint rhs) const { return UDecimal.make(_value - (ulong(rhs) << 32)); } + /// ditto + UDecimal opBinary(string op : "-")(const UDecimal rhs) const { + return UDecimal.make(_value - rhs._value); + } + /// ditto UDecimal opBinary(string op : "*")(const uint rhs) const { return UDecimal.make(_value * rhs); @@ -461,6 +471,16 @@ struct UDecimal { UDecimal opBinary(string op : "/")(const uint rhs) const { return UDecimal.make(_value / rhs); } + + /// ditto + UDecimal opBinary(string op : "<<")(const uint rhs) const { + return UDecimal.make(_value << rhs); + } + + /// ditto + UDecimal opBinary(string op : ">>")(const uint rhs) const { + return UDecimal.make(_value >> rhs); + } } public { @@ -492,12 +512,24 @@ struct UDecimal { return this; } + /// ditto + auto opOpAssign(string op : "+")(const UDecimal rhs) { + _value += rhs._value; + return this; + } + /// ditto auto opOpAssign(string op : "-")(const uint rhs) { _value -= (ulong(rhs) << 32); return this; } + /// ditto + auto opOpAssign(string op : "-")(const UDecimal rhs) { + _value -= rhs._value; + return this; + } + /// ditto auto opOpAssign(string op : "*")(const uint rhs) { _value *= rhs; @@ -509,6 +541,18 @@ struct UDecimal { _value /= rhs; return this; } + + /// ditto + auto opOpAssign(string op : "<<")(const uint rhs) const { + _value <<= rhs; + return this; + } + + /// ditto + auto opOpAssign(string op : ">>")(const uint rhs) const { + _value >>= rhs; + return this; + } } } From bb6ad459eb344f553efab781e37e8d069f8315b6 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 14 Jan 2025 00:40:22 +0100 Subject: [PATCH 14/38] Add improper downscaler implementation (WIP) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit To be replaced… --- pixmappaint.d | 205 +++++++++++++++++++++++++++++++++++++------------- 1 file changed, 152 insertions(+), 53 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 9590267..7eeeb26 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2934,30 +2934,23 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const sourceMaxX = (source.width - 1); const sourceMaxY = (source.height - 1); - const Size delta = (target.size - source.size); - - const ScalingDirection[2] directions = [ - scalingDirectionFromDelta(delta.width), - scalingDirectionFromDelta(delta.height), - ]; - const UDecimal[2] ratios = [ (UDecimal(source.width) / target.width), (UDecimal(source.height) / target.height), ]; - // TODO: move - Point translate(const Point dstPos) { - pragma(inline, true); - const xCandidate = (() @trusted => (dstPos.x * ratios.ptr[0]).round().castTo!int)(); - const yCandidate = (() @trusted => (dstPos.y * ratios.ptr[1]).round().castTo!int)(); - const x = min(xCandidate, sourceMaxX); - const y = min(yCandidate, sourceMaxY); - return Point(x, y); - } - // ==== Nearest Neighbour ==== static if (filter == ScalingFilter.nearest) { + + Point translate(const Point dstPos) { + pragma(inline, true); + const xCandidate = (() @trusted => (dstPos.x * ratios.ptr[0]).round().castTo!int)(); + const yCandidate = (() @trusted => (dstPos.y * ratios.ptr[1]).round().castTo!int)(); + const x = min(xCandidate, sourceMaxX); + const y = min(yCandidate, sourceMaxY); + return Point(x, y); + } + auto dst = PixmapScannerRW(target); size_t y = 0; @@ -2974,6 +2967,13 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ==== Bilinear ==== static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { + const Size delta = (target.size - source.size); + + const ScalingDirection[2] directions = [ + scalingDirectionFromDelta(delta.width), + scalingDirectionFromDelta(delta.height), + ]; + auto dst = PixmapScannerRW(target); size_t y = 0; @@ -2986,28 +2986,64 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe (() @trusted => posDst.y * ratios.ptr[1])(), ]; - const posSrcXF = (() @trusted => min(sourceMaxX, posSrc.ptr[0].floor().castTo!int))(); - const posSrcXC = (() @trusted => min(sourceMaxX, posSrc.ptr[0].ceil().castTo!int))(); + const int[2] posSrcX = () { + int[2] result; + if (directions[0] == none) { + result = [ + posSrc[0].castTo!int, + posSrc[0].castTo!int, + ]; + } else if (directions[0] == up) { + result = [ + min(sourceMaxX, posSrc[0].floor().castTo!int), + min(sourceMaxX, posSrc[0].ceil().castTo!int), + ]; + } else { + const ratioXHalf = (ratios[0] >> 1); + result = [ + max((posSrc[0] - ratioXHalf).round().castTo!int, 0), + min((posSrc[0] + ratioXHalf).round().castTo!int, sourceMaxX), + ]; + } + return result; + }(); - const posSrcYF = (() @trusted => min(sourceMaxY, posSrc.ptr[1].floor().castTo!int))(); - const posSrcYC = (() @trusted => min(sourceMaxY, posSrc.ptr[1].ceil().castTo!int))(); + const int[2] posSrcY = () { + int[2] result; + if (directions[1] == none) { + result = [ + posSrc[1].castTo!int, + posSrc[1].castTo!int, + ]; + } else if (directions[1] == up) { + result = [ + min(sourceMaxY, posSrc[1].floor().castTo!int), + min(sourceMaxY, posSrc[1].ceil().castTo!int), + ]; + } else { + const ratioHalf = (ratios[1] >> 1); + result = [ + max((posSrc[1] - ratioHalf).round().castTo!int, 0), + min((posSrc[1] + ratioHalf).round().castTo!int, sourceMaxY), + ]; + } + return result; + }(); const Point[4] posNeighs = [ - Point(posSrcXF, posSrcYF), - Point(posSrcXC, posSrcYF), - Point(posSrcXF, posSrcYC), - Point(posSrcXC, posSrcYC), + Point(posSrcX[0], posSrcY[0]), + Point(posSrcX[1], posSrcY[0]), + Point(posSrcX[0], posSrcY[1]), + Point(posSrcX[1], posSrcY[1]), ]; const Color[4] pxNeighs = [ - source.getPixel((() @trusted => posNeighs.ptr[0])()), - source.getPixel((() @trusted => posNeighs.ptr[1])()), - source.getPixel((() @trusted => posNeighs.ptr[2])()), - source.getPixel((() @trusted => posNeighs.ptr[3])()), + source.getPixel(posNeighs[0]), + source.getPixel(posNeighs[1]), + source.getPixel(posNeighs[2]), + source.getPixel(posNeighs[3]), ]; - // TODO: Downscaling (currently equivalent to nearest neighbour but with extra steps!) - // ====== Faux bilinear ====== static if (filter == ScalingFilter.fauxLinear) { auto pxInt = Pixel(0, 0, 0, 0); @@ -3023,36 +3059,99 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ====== Proper bilinear ====== static if (filter == ScalingFilter.bilinear) { - const ulong[2] fract = [ - (() @trusted => posSrc.ptr[0].fractionalDigits)(), - (() @trusted => posSrc.ptr[1].fractionalDigits)(), - ]; - - const ulong[2] fractComplementary = ((ulong(uint.max) + 1) - fract[]); - - alias fractC = fract; - alias fractF = fractComplementary; - + // TODO: Downscaling looks bad as-is. auto pxInt = Pixel(0, 0, 0, 0); foreach (immutable ib, ref c; pxInt.components) { - ulong[2] xSums = [0, 0]; - xSums[0] += (() @trusted => (pxNeighs.ptr[0].components.ptr[ib] * fractF.ptr[0]))(); - xSums[0] += (() @trusted => (pxNeighs.ptr[1].components.ptr[ib] * fractC.ptr[0]))(); + ulong[2] xSums; - xSums[1] += (() @trusted => (pxNeighs.ptr[2].components.ptr[ib] * fractF.ptr[0]))(); - xSums[1] += (() @trusted => (pxNeighs.ptr[3].components.ptr[ib] * fractC.ptr[0]))(); + // ======== X ======== + if (directions[0] == none) { + xSums = () @trusted { + ulong[2] result = [ + pxNeighs[0].components.ptr[ib], + pxNeighs[2].components.ptr[ib], + ]; + return result; + }(); + } else if (directions[1] == down) { + xSums = [0, 0]; - foreach (ref sum; xSums) { - sum >>= 32; + const UDecimal[2] deltasX = [ + posSrc[0] - posSrcX[0], + posSrcX[1] - posSrc[0], + ]; + + const deltasXSum = (deltasX[0] + deltasX[1]).round().castTo!uint; + const UDecimal[2] weightsX = [ + deltasX[0] / deltasXSum, + deltasX[1] / deltasXSum, + ]; + + () @trusted { + xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]).round().castTo!uint; + xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]).round().castTo!uint; + + xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]).round().castTo!uint; + xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]).round().castTo!uint; + }(); + } else { + xSums = [0, 0]; + + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + () @trusted { + xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]); + xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]); + + xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]); + xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]); + }(); + foreach (ref sum; xSums) { + sum >>= 32; + } } - ulong ySum = 0; - ySum += (() @trusted => (xSums.ptr[0] * fractF.ptr[1]))(); - ySum += (() @trusted => (xSums.ptr[1] * fractC.ptr[1]))(); + // ======== Y ======== + if (directions[1] == none) { + c = clamp255(xSums[0]); + } else if (directions[1] == down) { + const UDecimal[2] deltasY = [ + posSrc[1] - posSrcY[0], + posSrcY[1] - posSrc[1], + ]; - const xySum = (ySum >> 32); + const deltasYSum = (deltasY[0] + deltasY[1]).round().castTo!uint; + const UDecimal[2] weightsY = [ + deltasY[0] / deltasYSum, + deltasY[1] / deltasYSum, + ]; - c = clamp255(xySum); + auto ySum = UDecimal(0); + ySum += ((xSums[0] & 0xFFFF_FFFF) * weightsY[0]); + ySum += ((xSums[1] & 0xFFFF_FFFF) * weightsY[1]); + + c = clamp255(ySum.round().castTo!uint); + } else { + const ulong[2] weightsY = () { + ulong[2] result; + result[1] = posSrc[1].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + ulong ySum = 0; + ySum += (xSums[0] * weightsY[0]); + ySum += (xSums[1] * weightsY[1]); + + const xySum = (ySum >> 32); + + c = clamp255(xySum); + } } } From 247cee88d0879789e1ac5b085c0bcb0b1d48fa93 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 Jan 2025 04:09:03 +0100 Subject: [PATCH 15/38] Make pixel mapping examples render in visually distinguishable blocks --- pixmappaint.d | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 7eeeb26..5c101cd 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -21,7 +21,12 @@ In the case of this library, a “width” field is used to map a specified number of pixels to a row of an image. - ``` + + + + ### Pixel mapping + + ```text pixels := [ 0, 1, 2, 3 ] width := 2 @@ -32,7 +37,7 @@ ] ``` - ``` + ```text pixels := [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 ] width := 3 @@ -45,7 +50,7 @@ ] ``` - ``` + ```text pixels := [ 0, 1, 2, 3, 4, 5, 6, 7 ] width := 4 @@ -132,8 +137,8 @@ prone to such errors. (Slicing of the 1D array data can actually be utilized to cut off the - bottom part of an image. Any other naiv cropping operations will run into - the aforementioned issues.) + top or bottom part of an image. Any other naiv cropping operations will run + into the aforementioned issues.) From b031783e845abf89f9fe9c39afeb63cfa1a037d2 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 Jan 2025 04:10:09 +0100 Subject: [PATCH 16/38] Spell "neighbor" without a 'u' --- pixmappaint.d | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 5c101cd..a105be3 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2869,7 +2869,7 @@ PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { Works well with photos. For pixel graphics the retro look of `nearest` (as - in $(I nearest neighbour)) is usually the option of choice. + in $(I nearest neighbor)) is usually the option of choice. ) $(NOTE @@ -2881,7 +2881,7 @@ PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { +/ enum ScalingFilter { /++ - Nearest neighbour interpolation + Nearest neighbor interpolation Also known as $(B proximal interpolation) and $(B point sampling). @@ -2944,7 +2944,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe (UDecimal(source.height) / target.height), ]; - // ==== Nearest Neighbour ==== + // ==== Nearest Neighbor ==== static if (filter == ScalingFilter.nearest) { Point translate(const Point dstPos) { From 68a94f03c3bcb460158978323dda9ab308a5cca2 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:01:31 +0100 Subject: [PATCH 17/38] Finish downscaler implementation --- pixmappaint.d | 333 +++++++++++++++++++++++++++++++++++--------------- 1 file changed, 236 insertions(+), 97 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index a105be3..1dd5b0f 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -334,8 +334,7 @@ private float round(float f) pure @nogc nothrow @trusted { ## TODO: - Refactoring the template-mess of blendPixel() & co. - - Scaling - - Rotating + - Rotating (by arbitrary angles) - Skewing - HSL - Advanced blend modes (maybe) @@ -422,6 +421,29 @@ struct UDecimal { return UDecimal.make(rounded); } + /// + public UDecimal roundEven() const { + const truncated = (_value & 0xFFFF_FFFF_0000_0000); + const delta = _value - truncated; + + ulong rounded; + + if (delta == 0x8000_0000) { + const bool floorIsOdd = ((truncated & 0x1_0000_0000) != 0); + // dfmt off + rounded = (floorIsOdd) + ? truncated + 0x1_0000_0000 // ceil + : truncated; // floor + // dfmt on + } else if (delta > 0x8000_0000) { + rounded = truncated + 0x1_0000_0000; + } else { + rounded = truncated; + } + + return UDecimal.make(rounded); + } + /// public UDecimal floor() const { const truncated = (_value & 0xFFFF_FFFF_0000_0000); @@ -2895,6 +2917,8 @@ enum ScalingFilter { /++ Bilinear interpolation + (Uses arithmetic mean for downscaling.) + $(TIP Visual impression: “smooth”, “blurred” ) @@ -2931,7 +2955,6 @@ private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc } private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap target) @nogc { - enum none = ScalingDirection.none; enum up = ScalingDirection.up; enum down = ScalingDirection.down; @@ -2943,6 +2966,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe (UDecimal(source.width) / target.width), (UDecimal(source.height) / target.height), ]; + enum idxX = 0, idxY = 1; // ==== Nearest Neighbor ==== static if (filter == ScalingFilter.nearest) { @@ -2987,27 +3011,27 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const posDst = Point(x.castTo!int, y.castTo!int); const UDecimal[2] posSrc = [ - (() @trusted => posDst.x * ratios.ptr[0])(), - (() @trusted => posDst.y * ratios.ptr[1])(), + posDst.x * ratios[idxX], + posDst.y * ratios[idxY], ]; const int[2] posSrcX = () { int[2] result; - if (directions[0] == none) { + if (directions[idxX] == none) { result = [ - posSrc[0].castTo!int, - posSrc[0].castTo!int, + posSrc[idxX].castTo!int, + posSrc[idxX].castTo!int, ]; - } else if (directions[0] == up) { + } else if (directions[idxX] == up) { result = [ - min(sourceMaxX, posSrc[0].floor().castTo!int), - min(sourceMaxX, posSrc[0].ceil().castTo!int), + min(sourceMaxX, posSrc[idxX].floor().castTo!int), + min(sourceMaxX, posSrc[idxX].ceil().castTo!int), ]; - } else { - const ratioXHalf = (ratios[0] >> 1); + } else /* if (directions[0] == down) */ { + const ratioXHalf = (ratios[idxX] >> 1); result = [ - max((posSrc[0] - ratioXHalf).round().castTo!int, 0), - min((posSrc[0] + ratioXHalf).round().castTo!int, sourceMaxX), + max((posSrc[idxX] - ratioXHalf).roundEven().castTo!int, 0), + min((posSrc[idxX] + ratioXHalf).roundEven().castTo!int, sourceMaxX), ]; } return result; @@ -3015,31 +3039,34 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const int[2] posSrcY = () { int[2] result; - if (directions[1] == none) { + if (directions[idxY] == none) { result = [ - posSrc[1].castTo!int, - posSrc[1].castTo!int, + posSrc[idxY].castTo!int, + posSrc[idxY].castTo!int, ]; - } else if (directions[1] == up) { + } else if (directions[idxY] == up) { result = [ - min(sourceMaxY, posSrc[1].floor().castTo!int), - min(sourceMaxY, posSrc[1].ceil().castTo!int), + min(sourceMaxY, posSrc[idxY].floor().castTo!int), + min(sourceMaxY, posSrc[idxY].ceil().castTo!int), ]; - } else { - const ratioHalf = (ratios[1] >> 1); + } else /* if (directions[idxY] == down) */ { + const ratioHalf = (ratios[idxY] >> 1); result = [ - max((posSrc[1] - ratioHalf).round().castTo!int, 0), - min((posSrc[1] + ratioHalf).round().castTo!int, sourceMaxY), + max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0), + min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMaxY), ]; } return result; }(); + enum idxL = 0, idxR = 1; + enum idxT = 0, idxB = 1; + const Point[4] posNeighs = [ - Point(posSrcX[0], posSrcY[0]), - Point(posSrcX[1], posSrcY[0]), - Point(posSrcX[0], posSrcY[1]), - Point(posSrcX[1], posSrcY[1]), + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxR], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + Point(posSrcX[idxR], posSrcY[idxB]), ]; const Color[4] pxNeighs = [ @@ -3049,6 +3076,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe source.getPixel(posNeighs[3]), ]; + enum idxTL = 0, idxTR = 1, idxBL = 2, idxBR = 3; + // ====== Faux bilinear ====== static if (filter == ScalingFilter.fauxLinear) { auto pxInt = Pixel(0, 0, 0, 0); @@ -3062,96 +3091,206 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } - // ====== Proper bilinear ====== + // ====== Proper bilinear (up) + Avg (down) ====== static if (filter == ScalingFilter.bilinear) { - // TODO: Downscaling looks bad as-is. auto pxInt = Pixel(0, 0, 0, 0); foreach (immutable ib, ref c; pxInt.components) { - ulong[2] xSums; + ulong sampleX() { + pragma(inline, true); - // ======== X ======== - if (directions[0] == none) { - xSums = () @trusted { - ulong[2] result = [ - pxNeighs[0].components.ptr[ib], - pxNeighs[2].components.ptr[ib], - ]; - return result; - }(); - } else if (directions[1] == down) { - xSums = [0, 0]; + if (directions[0] == none) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } else if (directions[0] == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); + const samplingOffset = source.scanTo(posSampling); + const srcSamples = () @trusted { + return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; + }(); - const UDecimal[2] deltasX = [ - posSrc[0] - posSrcX[0], - posSrcX[1] - posSrc[0], - ]; + ulong xSum = 0; - const deltasXSum = (deltasX[0] + deltasX[1]).round().castTo!uint; - const UDecimal[2] weightsX = [ - deltasX[0] / deltasXSum, - deltasX[1] / deltasXSum, - ]; + foreach (srcSample; srcSamples) { + xSum += (() @trusted => srcSample.components.ptr[ib])(); + } - () @trusted { - xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]).round().castTo!uint; - xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]).round().castTo!uint; + return (xSum / nSamples); + } else /* if (directions[0] == up) */ { + ulong xSum = 0; - xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]).round().castTo!uint; - xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]).round().castTo!uint; - }(); - } else { - xSums = [0, 0]; + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); + () @trusted { + xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + }(); - () @trusted { - xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]); - xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]); - - xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]); - xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]); - }(); - foreach (ref sum; xSums) { - sum >>= 32; + return (xSum >> 32); } } - // ======== Y ======== - if (directions[1] == none) { - c = clamp255(xSums[0]); - } else if (directions[1] == down) { - const UDecimal[2] deltasY = [ - posSrc[1] - posSrcY[0], - posSrcY[1] - posSrc[1], - ]; + ulong[2] sampleXDual() { + pragma(inline, true); - const deltasYSum = (deltasY[0] + deltasY[1]).round().castTo!uint; - const UDecimal[2] weightsY = [ - deltasY[0] / deltasYSum, - deltasY[1] / deltasYSum, - ]; + if (directions[0] == none) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } else if (directions[0] == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const Point[2] posSampling = [ + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + ]; - auto ySum = UDecimal(0); - ySum += ((xSums[0] & 0xFFFF_FFFF) * weightsY[0]); - ySum += ((xSums[1] & 0xFFFF_FFFF) * weightsY[1]); + const int[2] samplingOffsets = [ + source.scanTo(posSampling[0]), + source.scanTo(posSampling[1]), + ]; - c = clamp255(ySum.round().castTo!uint); - } else { + const srcSamples2 = () @trusted { + const(const(Pixel)[])[2] result = [ + source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], + source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], + ]; + return result; + }(); + + ulong[2] xSums = [0, 0]; + + foreach (idx, srcSamples; srcSamples2) { + foreach (srcSample; srcSamples) { + () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); + } + } + + xSums[] /= nSamples; + return xSums; + } else /* if (directions[0] == up) */ { + ulong[2] xSums = [0, 0]; + + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + () @trusted { + xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + + xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); + xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); + }(); + + foreach (ref sum; xSums) { + sum >>= 32; + } + + return xSums; + } + } + + ulong sampleXMulti() { + pragma(inline, true); + + const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + ulong ySum = 0; + + alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; + ulong foreachLine(scope ForeachLineCallback apply) { + ulong linesSum = 0; + foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { + const posLine = Point(posSrcX[idxL], lineY); + linesSum += apply(posLine); + } + return linesSum; + } + + if (directions[0] == none) { + ySum = foreachLine(delegate(const Point posLine) { + const pxSrc = source.getPixel(posLine); + return ulong((() @trusted => pxSrc.components.ptr[ib])()); + }); + } else if (directions[0] == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + + ySum = foreachLine(delegate(const Point posLine) { + const samplingOffset = source.scanTo(posLine); + const srcSamples = () @trusted { + return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; + }(); + + ulong xSum = 0; + + foreach (srcSample; srcSamples) { + xSum += (() @trusted => srcSample.components.ptr[ib])(); + } + + return xSum; + }); + + ySum /= nSamples; + } else /* if (directions[0] == up) */ { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + + ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; + + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + const samplingOffset = source.scanTo(posLine); + ubyte[2] pxcLR = () @trusted { + ubyte[2] result = [ + source.data.ptr[samplingOffset].components.ptr[ib], + source.data.ptr[samplingOffset + nSamples].components.ptr[ib], + ]; + return result; + }(); + + xSum += (pxcLR[idxL] * weightsX[idxL]); + xSum += (pxcLR[idxR] * weightsX[idxR]); + + return (xSum >> 32); + }); + } + + return (ySum / nLines); + } + + if (directions[idxY] == none) { + c = clamp255(sampleX()); + } else if (directions[idxY] == down) { + c = clamp255(sampleXMulti()); + } else /* if (directions[idxY] == up) */ { + // looks ass const ulong[2] weightsY = () { ulong[2] result; - result[1] = posSrc[1].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; + result[idxB] = posSrc[1].fractionalDigits; + result[idxT] = ulong(uint.max) + 1 - result[idxB]; return result; }(); + const xSums = sampleXDual(); + ulong ySum = 0; - ySum += (xSums[0] * weightsY[0]); - ySum += (xSums[1] * weightsY[1]); + ySum += (xSums[idxT] * weightsY[idxT]); + ySum += (xSums[idxB] * weightsY[idxB]); const xySum = (ySum >> 32); From 9899b48f16219b0bd7b4eeb343f0fc537c0ae0ed Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:02:09 +0100 Subject: [PATCH 18/38] Fix nearest neighbor algorithm --- pixmappaint.d | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 1dd5b0f..48ac5e8 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2973,10 +2973,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe Point translate(const Point dstPos) { pragma(inline, true); - const xCandidate = (() @trusted => (dstPos.x * ratios.ptr[0]).round().castTo!int)(); - const yCandidate = (() @trusted => (dstPos.y * ratios.ptr[1]).round().castTo!int)(); - const x = min(xCandidate, sourceMaxX); - const y = min(yCandidate, sourceMaxY); + const x = (dstPos.x * ratios[0]).floor().castTo!int; + const y = (dstPos.y * ratios[1]).floor().castTo!int; return Point(x, y); } From 6dc177619dc61b07c2acae308533119d32163918 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:03:14 +0100 Subject: [PATCH 19/38] Improve `UDecimal` --- pixmappaint.d | 26 ++++++++++++++++++-------- 1 file changed, 18 insertions(+), 8 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 48ac5e8..3d1007f 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -407,6 +407,16 @@ struct UDecimal { return (_value >> 32).castTo!uint; } + /// + T opCast(T : double)() const { + return (_value / double(0xFFFF_FFFF)); + } + + /// + T opCast(T : float)() const { + return (_value / float(0xFFFF_FFFF)); + } + /// public UDecimal round() const { const truncated = (_value & 0xFFFF_FFFF_0000_0000); @@ -534,49 +544,49 @@ struct UDecimal { public { /// - auto opOpAssign(string op : "+")(const uint rhs) { + UDecimal opOpAssign(string op : "+")(const uint rhs) { _value += (ulong(rhs) << 32); return this; } /// ditto - auto opOpAssign(string op : "+")(const UDecimal rhs) { + UDecimal opOpAssign(string op : "+")(const UDecimal rhs) { _value += rhs._value; return this; } /// ditto - auto opOpAssign(string op : "-")(const uint rhs) { + UDecimal opOpAssign(string op : "-")(const uint rhs) { _value -= (ulong(rhs) << 32); return this; } /// ditto - auto opOpAssign(string op : "-")(const UDecimal rhs) { + UDecimal opOpAssign(string op : "-")(const UDecimal rhs) { _value -= rhs._value; return this; } /// ditto - auto opOpAssign(string op : "*")(const uint rhs) { + UDecimal opOpAssign(string op : "*")(const uint rhs) { _value *= rhs; return this; } /// ditto - auto opOpAssign(string op : "/")(const uint rhs) { + UDecimal opOpAssign(string op : "/")(const uint rhs) { _value /= rhs; return this; } /// ditto - auto opOpAssign(string op : "<<")(const uint rhs) const { + UDecimal opOpAssign(string op : "<<")(const uint rhs) const { _value <<= rhs; return this; } /// ditto - auto opOpAssign(string op : ">>")(const uint rhs) const { + UDecimal opOpAssign(string op : ">>")(const uint rhs) const { _value >>= rhs; return this; } From 425fb918db5fed4ba33a739dc6a3b48cc8d187dc Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:04:03 +0100 Subject: [PATCH 20/38] Fix incomplete sentence --- pixmappaint.d | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index 3d1007f..1bf8368 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1068,7 +1068,8 @@ struct SubPixmap { Allocates a new Pixmap cropped to the pixel data of the subimage. See_also: - Use [extractToPixmap] for a non-allocating variant with an . + Use [extractToPixmap] for a non-allocating variant with a + target parameter. +/ Pixmap extractToNewPixmap() const { auto pm = Pixmap.makeNew(size); From 7f91abfc0ac3ecbe9d045174397d05c1aa960f55 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:12:50 +0100 Subject: [PATCH 21/38] Add further unittest for `UDecimal` --- pixmappaint.d | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 1bf8368..c8e87af 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -625,6 +625,20 @@ struct UDecimal { assert((UDecimal(10) / 5).ceil().castTo!uint == 2); } +@safe unittest { + UDecimal val; + + val = (UDecimal(1) / 2); + assert(val.roundEven().castTo!uint == 0); + assert(val.castTo!double > 0.49); + assert(val.castTo!double < 0.51); + + val = (UDecimal(3) / 2); + assert(val.roundEven().castTo!uint == 2); + assert(val.castTo!double > 1.49); + assert(val.castTo!double < 1.51); +} + @safe unittest { UDecimal val; From 31308e0777dcd5289d437adb11d5bfaf32fac7e0 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:29:45 +0100 Subject: [PATCH 22/38] Templatize bilinear up/down scaler --- pixmappaint.d | 557 ++++++++++++++++++++++++++------------------------ 1 file changed, 294 insertions(+), 263 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index c8e87af..ea74515 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3019,237 +3019,108 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ==== Bilinear ==== static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { - const Size delta = (target.size - source.size); + void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() { + auto dst = PixmapScannerRW(target); - const ScalingDirection[2] directions = [ - scalingDirectionFromDelta(delta.width), - scalingDirectionFromDelta(delta.height), - ]; + size_t y = 0; + foreach (dstLine; dst) { + foreach (x, ref pxDst; dstLine) { + const posDst = Point(x.castTo!int, y.castTo!int); - auto dst = PixmapScannerRW(target); + const UDecimal[2] posSrc = [ + posDst.x * ratios[idxX], + posDst.y * ratios[idxY], + ]; - size_t y = 0; - foreach (dstLine; dst) { - foreach (x, ref pxDst; dstLine) { - const posDst = Point(x.castTo!int, y.castTo!int); - - const UDecimal[2] posSrc = [ - posDst.x * ratios[idxX], - posDst.y * ratios[idxY], - ]; - - const int[2] posSrcX = () { - int[2] result; - if (directions[idxX] == none) { - result = [ - posSrc[idxX].castTo!int, - posSrc[idxX].castTo!int, - ]; - } else if (directions[idxX] == up) { - result = [ - min(sourceMaxX, posSrc[idxX].floor().castTo!int), - min(sourceMaxX, posSrc[idxX].ceil().castTo!int), - ]; - } else /* if (directions[0] == down) */ { - const ratioXHalf = (ratios[idxX] >> 1); - result = [ - max((posSrc[idxX] - ratioXHalf).roundEven().castTo!int, 0), - min((posSrc[idxX] + ratioXHalf).roundEven().castTo!int, sourceMaxX), - ]; - } - return result; - }(); - - const int[2] posSrcY = () { - int[2] result; - if (directions[idxY] == none) { - result = [ - posSrc[idxY].castTo!int, - posSrc[idxY].castTo!int, - ]; - } else if (directions[idxY] == up) { - result = [ - min(sourceMaxY, posSrc[idxY].floor().castTo!int), - min(sourceMaxY, posSrc[idxY].ceil().castTo!int), - ]; - } else /* if (directions[idxY] == down) */ { - const ratioHalf = (ratios[idxY] >> 1); - result = [ - max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0), - min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMaxY), - ]; - } - return result; - }(); - - enum idxL = 0, idxR = 1; - enum idxT = 0, idxB = 1; - - const Point[4] posNeighs = [ - Point(posSrcX[idxL], posSrcY[idxT]), - Point(posSrcX[idxR], posSrcY[idxT]), - Point(posSrcX[idxL], posSrcY[idxB]), - Point(posSrcX[idxR], posSrcY[idxB]), - ]; - - const Color[4] pxNeighs = [ - source.getPixel(posNeighs[0]), - source.getPixel(posNeighs[1]), - source.getPixel(posNeighs[2]), - source.getPixel(posNeighs[3]), - ]; - - enum idxTL = 0, idxTR = 1, idxBL = 2, idxBR = 3; - - // ====== Faux bilinear ====== - static if (filter == ScalingFilter.fauxLinear) { - auto pxInt = Pixel(0, 0, 0, 0); - - foreach (immutable ib, ref c; pxInt.components) { - uint sum = 0; - foreach (const pxNeigh; pxNeighs) { - sum += (() @trusted => pxNeigh.components.ptr[ib])(); + const int[2] posSrcX = () { + int[2] result; + static if (directionX == none) { + result = [ + posSrc[idxX].castTo!int, + posSrc[idxX].castTo!int, + ]; + } else static if (directionX == up) { + result = [ + min(sourceMaxX, posSrc[idxX].floor().castTo!int), + min(sourceMaxX, posSrc[idxX].ceil().castTo!int), + ]; + } else /* if (directionX == down) */ { + const ratioXHalf = (ratios[idxX] >> 1); + result = [ + max((posSrc[idxX] - ratioXHalf).roundEven().castTo!int, 0), + min((posSrc[idxX] + ratioXHalf).roundEven().castTo!int, sourceMaxX), + ]; + } + return result; + }(); + + const int[2] posSrcY = () { + int[2] result; + static if (directionY == none) { + result = [ + posSrc[idxY].castTo!int, + posSrc[idxY].castTo!int, + ]; + } else static if (directionY == up) { + result = [ + min(sourceMaxY, posSrc[idxY].floor().castTo!int), + min(sourceMaxY, posSrc[idxY].ceil().castTo!int), + ]; + } else /* if (directionY == down) */ { + const ratioHalf = (ratios[idxY] >> 1); + result = [ + max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0), + min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMaxY), + ]; + } + return result; + }(); + + enum idxL = 0, idxR = 1; + enum idxT = 0, idxB = 1; + + const Point[4] posNeighs = [ + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxR], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + Point(posSrcX[idxR], posSrcY[idxB]), + ]; + + const Color[4] pxNeighs = [ + source.getPixel(posNeighs[0]), + source.getPixel(posNeighs[1]), + source.getPixel(posNeighs[2]), + source.getPixel(posNeighs[3]), + ]; + + enum idxTL = 0, idxTR = 1, idxBL = 2, idxBR = 3; + + // ====== Faux bilinear ====== + static if (filter == ScalingFilter.fauxLinear) { + auto pxInt = Pixel(0, 0, 0, 0); + + foreach (immutable ib, ref c; pxInt.components) { + uint sum = 0; + foreach (const pxNeigh; pxNeighs) { + sum += (() @trusted => pxNeigh.components.ptr[ib])(); + } + c = (sum >> 2).castTo!ubyte; } - c = (sum >> 2).castTo!ubyte; } - } - // ====== Proper bilinear (up) + Avg (down) ====== - static if (filter == ScalingFilter.bilinear) { - auto pxInt = Pixel(0, 0, 0, 0); - foreach (immutable ib, ref c; pxInt.components) { - ulong sampleX() { - pragma(inline, true); + // ====== Proper bilinear (up) + Avg (down) ====== + static if (filter == ScalingFilter.bilinear) { + auto pxInt = Pixel(0, 0, 0, 0); + foreach (immutable ib, ref c; pxInt.components) { + ulong sampleX() { + pragma(inline, true); - if (directions[0] == none) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); - } else if (directions[0] == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); - const samplingOffset = source.scanTo(posSampling); - const srcSamples = () @trusted { - return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; - }(); - - ulong xSum = 0; - - foreach (srcSample; srcSamples) { - xSum += (() @trusted => srcSample.components.ptr[ib])(); - } - - return (xSum / nSamples); - } else /* if (directions[0] == up) */ { - ulong xSum = 0; - - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); - - () @trusted { - xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - }(); - - return (xSum >> 32); - } - } - - ulong[2] sampleXDual() { - pragma(inline, true); - - if (directions[0] == none) { - return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], - ]; - return result; - }(); - } else if (directions[0] == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const Point[2] posSampling = [ - Point(posSrcX[idxL], posSrcY[idxT]), - Point(posSrcX[idxL], posSrcY[idxB]), - ]; - - const int[2] samplingOffsets = [ - source.scanTo(posSampling[0]), - source.scanTo(posSampling[1]), - ]; - - const srcSamples2 = () @trusted { - const(const(Pixel)[])[2] result = [ - source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], - source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], - ]; - return result; - }(); - - ulong[2] xSums = [0, 0]; - - foreach (idx, srcSamples; srcSamples2) { - foreach (srcSample; srcSamples) { - () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); - } - } - - xSums[] /= nSamples; - return xSums; - } else /* if (directions[0] == up) */ { - ulong[2] xSums = [0, 0]; - - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); - - () @trusted { - xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - - xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); - xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); - }(); - - foreach (ref sum; xSums) { - sum >>= 32; - } - - return xSums; - } - } - - ulong sampleXMulti() { - pragma(inline, true); - - const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; - ulong ySum = 0; - - alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; - ulong foreachLine(scope ForeachLineCallback apply) { - ulong linesSum = 0; - foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { - const posLine = Point(posSrcX[idxL], lineY); - linesSum += apply(posLine); - } - return linesSum; - } - - if (directions[0] == none) { - ySum = foreachLine(delegate(const Point posLine) { - const pxSrc = source.getPixel(posLine); - return ulong((() @trusted => pxSrc.components.ptr[ib])()); - }); - } else if (directions[0] == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - - ySum = foreachLine(delegate(const Point posLine) { - const samplingOffset = source.scanTo(posLine); + static if (directionX == none) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } else static if (directionX == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); + const samplingOffset = source.scanTo(posSampling); const srcSamples = () @trusted { return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; }(); @@ -3260,14 +3131,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe xSum += (() @trusted => srcSample.components.ptr[ib])(); } - return xSum; - }); - - ySum /= nSamples; - } else /* if (directions[0] == up) */ { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - - ySum = foreachLine(delegate(const Point posLine) { + return (xSum / nSamples); + } else /* if (directionX == up) */ { ulong xSum = 0; const ulong[2] weightsX = () { @@ -3277,55 +3142,221 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return result; }(); - const samplingOffset = source.scanTo(posLine); - ubyte[2] pxcLR = () @trusted { - ubyte[2] result = [ - source.data.ptr[samplingOffset].components.ptr[ib], - source.data.ptr[samplingOffset + nSamples].components.ptr[ib], + () @trusted { + xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + }(); + + return (xSum >> 32); + } + } + + ulong[2] sampleXDual() { + pragma(inline, true); + + static if (directionX == none) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } else static if (directionX == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const Point[2] posSampling = [ + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + ]; + + const int[2] samplingOffsets = [ + source.scanTo(posSampling[0]), + source.scanTo(posSampling[1]), + ]; + + const srcSamples2 = () @trusted { + const(const(Pixel)[])[2] result = [ + source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], + source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], ]; return result; }(); - xSum += (pxcLR[idxL] * weightsX[idxL]); - xSum += (pxcLR[idxR] * weightsX[idxR]); + ulong[2] xSums = [0, 0]; - return (xSum >> 32); - }); + foreach (idx, srcSamples; srcSamples2) { + foreach (srcSample; srcSamples) { + () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); + } + } + + xSums[] /= nSamples; + return xSums; + } else /* if (directionX == up) */ { + ulong[2] xSums = [0, 0]; + + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + () @trusted { + xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + + xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); + xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); + }(); + + foreach (ref sum; xSums) { + sum >>= 32; + } + + return xSums; + } } - return (ySum / nLines); - } + ulong sampleXMulti() { + pragma(inline, true); - if (directions[idxY] == none) { - c = clamp255(sampleX()); - } else if (directions[idxY] == down) { - c = clamp255(sampleXMulti()); - } else /* if (directions[idxY] == up) */ { - // looks ass - const ulong[2] weightsY = () { - ulong[2] result; - result[idxB] = posSrc[1].fractionalDigits; - result[idxT] = ulong(uint.max) + 1 - result[idxB]; - return result; - }(); + const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + ulong ySum = 0; - const xSums = sampleXDual(); + alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; + ulong foreachLine(scope ForeachLineCallback apply) { + ulong linesSum = 0; + foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { + const posLine = Point(posSrcX[idxL], lineY); + linesSum += apply(posLine); + } + return linesSum; + } - ulong ySum = 0; - ySum += (xSums[idxT] * weightsY[idxT]); - ySum += (xSums[idxB] * weightsY[idxB]); + static if (directionX == none) { + ySum = foreachLine(delegate(const Point posLine) { + const pxSrc = source.getPixel(posLine); + return ulong((() @trusted => pxSrc.components.ptr[ib])()); + }); + } else static if (directionX == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const xySum = (ySum >> 32); + ySum = foreachLine(delegate(const Point posLine) { + const samplingOffset = source.scanTo(posLine); + const srcSamples = () @trusted { + return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; + }(); - c = clamp255(xySum); + ulong xSum = 0; + + foreach (srcSample; srcSamples) { + xSum += (() @trusted => srcSample.components.ptr[ib])(); + } + + return xSum; + }); + + ySum /= nSamples; + } else /* if (directionX == up) */ { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + + ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; + + const ulong[2] weightsX = () { + ulong[2] result; + result[1] = posSrc[0].fractionalDigits; + result[0] = ulong(uint.max) + 1 - result[1]; + return result; + }(); + + const samplingOffset = source.scanTo(posLine); + ubyte[2] pxcLR = () @trusted { + ubyte[2] result = [ + source.data.ptr[samplingOffset].components.ptr[ib], + source.data.ptr[samplingOffset + nSamples].components.ptr[ib], + ]; + return result; + }(); + + xSum += (pxcLR[idxL] * weightsX[idxL]); + xSum += (pxcLR[idxR] * weightsX[idxR]); + + return (xSum >> 32); + }); + } + + return (ySum / nLines); + } + + static if (directionY == none) { + c = clamp255(sampleX()); + } else static if (directionY == down) { + c = clamp255(sampleXMulti()); + } else /* if (directionY == up) */ { + // looks ass + const ulong[2] weightsY = () { + ulong[2] result; + result[idxB] = posSrc[1].fractionalDigits; + result[idxT] = ulong(uint.max) + 1 - result[idxB]; + return result; + }(); + + const xSums = sampleXDual(); + + ulong ySum = 0; + ySum += (xSums[idxT] * weightsY[idxT]); + ySum += (xSums[idxB] * weightsY[idxB]); + + const xySum = (ySum >> 32); + + c = clamp255(xySum); + } } } + + pxDst = pxInt; } - pxDst = pxInt; + ++y; } + } - ++y; + const Size delta = (target.size - source.size); + + const ScalingDirection[2] directions = [ + scalingDirectionFromDelta(delta.width), + scalingDirectionFromDelta(delta.height), + ]; + + if (directions[0] == none) { + if (directions[1] == none) { + version (none) + scaleToLinearImpl!(none, none)(); + else + return; + } else if (directions[1] == up) { + scaleToLinearImpl!(none, up)(); + } else /* if (directions[1] == down) */ { + scaleToLinearImpl!(none, down)(); + } + } else if (directions[0] == up) { + if (directions[1] == none) { + scaleToLinearImpl!(up, none)(); + } else if (directions[1] == up) { + scaleToLinearImpl!(up, up)(); + } else /* if (directions[1] == down) */ { + scaleToLinearImpl!(up, down)(); + } + } else /* if (directions[0] == down) */ { + if (directions[1] == none) { + scaleToLinearImpl!(down, none)(); + } else if (directions[1] == up) { + scaleToLinearImpl!(down, up)(); + } else /* if (directions[1] == down) */ { + scaleToLinearImpl!(down, down)(); + } } } } From 442c616baed15a62e4a4b1f014e6ece124c36b82 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 28 Jan 2025 02:42:37 +0100 Subject: [PATCH 23/38] Document function `scaleTo` --- pixmappaint.d | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index ea74515..6005bf1 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3361,7 +3361,21 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } -// TODO: Document this function +/++ + Scales a pixmap and stores the result in the provided target Pixmap. + + The size to scale the image to + is derived from the size of the target. + + --- + // This function can be used to omit a redundant size parameter + // in cases like this: + target = scale(source, target, target.size, ScalingFilter.bilinear); + + // → Instead do: + scaleTo(source, target, ScalingFilter.bilinear); + --- + +/ void scaleTo(const Pixmap source, Pixmap target, ScalingFilter filter) @nogc { import std.meta : NoDuplicates; import std.traits : EnumMembers; From c9790d0c1994502ef59cdc3e86687e534776788c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Thu, 30 Jan 2025 01:26:26 +0100 Subject: [PATCH 24/38] Fix a few things --- pixmappaint.d | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 6005bf1..8b6c50e 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -393,7 +393,7 @@ struct UDecimal { /// public this(uint initialValue) { - _value = (long(initialValue) << 32); + _value = (ulong(initialValue) << 32); } private static UDecimal make(ulong internal) { @@ -3332,10 +3332,11 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe if (directions[0] == none) { if (directions[1] == none) { - version (none) + version (none) { scaleToLinearImpl!(none, none)(); - else - return; + } else { + target.data[] = source.data[]; + } } else if (directions[1] == up) { scaleToLinearImpl!(none, up)(); } else /* if (directions[1] == down) */ { From 0b288d385f3633e55bec897bf5197ade3ceaf3b7 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 04:37:50 +0100 Subject: [PATCH 25/38] Fix upscaler --- pixmappaint.d | 446 ++++++++++++++++++++++++++++++-------------------- 1 file changed, 272 insertions(+), 174 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 8b6c50e..3d46388 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -478,6 +478,13 @@ struct UDecimal { return (_value & 0x0000_0000_FFFF_FFFF); } + public { + /// + int opCmp(const UDecimal that) const { + return ((this._value > that._value) - (this._value < that._value)); + } + } + public { /// UDecimal opBinary(string op : "+")(const uint rhs) const { @@ -2984,22 +2991,33 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe enum up = ScalingDirection.up; enum down = ScalingDirection.down; - const sourceMaxX = (source.width - 1); - const sourceMaxY = (source.height - 1); + enum udecimalHalf = UDecimal.make(0x8000_0000); + enum uint udecimalHalfFD = udecimalHalf.fractionalDigits; + + enum idxX = 0, idxY = 1; + + const int[2] sourceMax = [ + (source.width - 1), + (source.height - 1), + ]; const UDecimal[2] ratios = [ (UDecimal(source.width) / target.width), (UDecimal(source.height) / target.height), ]; - enum idxX = 0, idxY = 1; + + const UDecimal[2] ratiosHalf = [ + (ratios[idxX] >> 1), + (ratios[idxY] >> 1), + ]; // ==== Nearest Neighbor ==== static if (filter == ScalingFilter.nearest) { Point translate(const Point dstPos) { pragma(inline, true); - const x = (dstPos.x * ratios[0]).floor().castTo!int; - const y = (dstPos.y * ratios[1]).floor().castTo!int; + const x = (dstPos.x * ratios[idxX]).castTo!int; + const y = (dstPos.y * ratios[idxY]).castTo!int; return Point(x, y); } @@ -3025,30 +3043,55 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe size_t y = 0; foreach (dstLine; dst) { foreach (x, ref pxDst; dstLine) { - const posDst = Point(x.castTo!int, y.castTo!int); + const int[2] posDst = [ + x.castTo!uint, + y.castTo!uint, + ]; const UDecimal[2] posSrc = [ - posDst.x * ratios[idxX], - posDst.y * ratios[idxY], + posDst[idxX] * ratios[idxX] + ratiosHalf[idxX], + posDst[idxY] * ratios[idxY] + ratiosHalf[idxY], ]; const int[2] posSrcX = () { int[2] result; static if (directionX == none) { + const value = posSrc[idxX].castTo!int; result = [ - posSrc[idxX].castTo!int, - posSrc[idxX].castTo!int, + value, + value, ]; } else static if (directionX == up) { - result = [ - min(sourceMaxX, posSrc[idxX].floor().castTo!int), - min(sourceMaxX, posSrc[idxX].ceil().castTo!int), - ]; + if (posSrc[idxX] < udecimalHalf) { + result = [ + 0, + 0, + ]; + } else { + const floor = posSrc[idxX].castTo!uint; + if (posSrc[idxX].fractionalDigits == udecimalHalfFD) { + result = [ + floor, + floor, + ]; + } else if (posSrc[idxX].fractionalDigits > udecimalHalfFD) { + const upper = min((floor + 1), sourceMax[idxX]); + result = [ + floor, + upper, + ]; + } else { + result = [ + floor - 1, + floor, + ]; + } + } } else /* if (directionX == down) */ { const ratioXHalf = (ratios[idxX] >> 1); result = [ - max((posSrc[idxX] - ratioXHalf).roundEven().castTo!int, 0), - min((posSrc[idxX] + ratioXHalf).roundEven().castTo!int, sourceMaxX), + max((posSrc[idxX] - ratioXHalf).roundEven().castTo!uint, 0), + min((posSrc[idxX] + ratioXHalf).roundEven().castTo!uint, sourceMax[idxX]), ]; } return result; @@ -3062,15 +3105,36 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe posSrc[idxY].castTo!int, ]; } else static if (directionY == up) { - result = [ - min(sourceMaxY, posSrc[idxY].floor().castTo!int), - min(sourceMaxY, posSrc[idxY].ceil().castTo!int), - ]; + if (posSrc[idxY] < udecimalHalf) { + result = [ + 0, + 0, + ]; + } else { + const floor = posSrc[idxY].castTo!uint; + if (posSrc[idxY].fractionalDigits == udecimalHalfFD) { + result = [ + floor, + floor, + ]; + } else if (posSrc[idxY].fractionalDigits > udecimalHalfFD) { + const upper = min((floor + 1), sourceMax[idxY]); + result = [ + floor, + upper, + ]; + } else { + result = [ + floor - 1, + floor, + ]; + } + } } else /* if (directionY == down) */ { const ratioHalf = (ratios[idxY] >> 1); result = [ max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0), - min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMaxY), + min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMax[idxY]), ]; } return result; @@ -3111,139 +3175,63 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ====== Proper bilinear (up) + Avg (down) ====== static if (filter == ScalingFilter.bilinear) { auto pxInt = Pixel(0, 0, 0, 0); + + enum SamplingMode { + single, + dual, + multi, + } + foreach (immutable ib, ref c; pxInt.components) { - ulong sampleX() { + + auto sampleX(SamplingMode mode)() { pragma(inline, true); - static if (directionX == none) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); - } else static if (directionX == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); - const samplingOffset = source.scanTo(posSampling); - const srcSamples = () @trusted { - return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; - }(); + static if (mode == SamplingMode.multi) { + const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; - ulong xSum = 0; - - foreach (srcSample; srcSamples) { - xSum += (() @trusted => srcSample.components.ptr[ib])(); - } - - return (xSum / nSamples); - } else /* if (directionX == up) */ { - ulong xSum = 0; - - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); - - () @trusted { - xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - }(); - - return (xSum >> 32); - } - } - - ulong[2] sampleXDual() { - pragma(inline, true); - - static if (directionX == none) { - return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], - ]; - return result; - }(); - } else static if (directionX == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const Point[2] posSampling = [ - Point(posSrcX[idxL], posSrcY[idxT]), - Point(posSrcX[idxL], posSrcY[idxB]), - ]; - - const int[2] samplingOffsets = [ - source.scanTo(posSampling[0]), - source.scanTo(posSampling[1]), - ]; - - const srcSamples2 = () @trusted { - const(const(Pixel)[])[2] result = [ - source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], - source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], - ]; - return result; - }(); - - ulong[2] xSums = [0, 0]; - - foreach (idx, srcSamples; srcSamples2) { - foreach (srcSample; srcSamples) { - () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); + alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; + ulong foreachLine(scope ForeachLineCallback apply) { + ulong linesSum = 0; + foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { + const posLine = Point(posSrcX[idxL], lineY); + linesSum += apply(posLine); } + return linesSum; } - - xSums[] /= nSamples; - return xSums; - } else /* if (directionX == up) */ { - ulong[2] xSums = [0, 0]; - - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); - - () @trusted { - xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - - xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); - xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); - }(); - - foreach (ref sum; xSums) { - sum >>= 32; - } - - return xSums; - } - } - - ulong sampleXMulti() { - pragma(inline, true); - - const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; - ulong ySum = 0; - - alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; - ulong foreachLine(scope ForeachLineCallback apply) { - ulong linesSum = 0; - foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { - const posLine = Point(posSrcX[idxL], lineY); - linesSum += apply(posLine); - } - return linesSum; } + // ======== None ======== static if (directionX == none) { - ySum = foreachLine(delegate(const Point posLine) { - const pxSrc = source.getPixel(posLine); - return ulong((() @trusted => pxSrc.components.ptr[ib])()); - }); - } else static if (directionX == down) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + static if (mode == SamplingMode.single) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } - ySum = foreachLine(delegate(const Point posLine) { - const samplingOffset = source.scanTo(posLine); + static if (mode == SamplingMode.dual) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } + + static if (mode == SamplingMode.multi) { + const ySum = foreachLine(delegate(const Point posLine) { + const pxSrc = source.getPixel(posLine); + return ulong((() @trusted => pxSrc.components.ptr[ib])()); + }); + return (ySum / nLines); + } + } + + // ======== Down ======== + static if (directionX == down) { + static if (mode == SamplingMode.single) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); + const samplingOffset = source.scanTo(posSampling); const srcSamples = () @trusted { return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; }(); @@ -3254,56 +3242,166 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe xSum += (() @trusted => srcSample.components.ptr[ib])(); } - return xSum; - }); + return (xSum / nSamples); + } - ySum /= nSamples; - } else /* if (directionX == up) */ { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + static if (mode == SamplingMode.dual) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const Point[2] posSampling = [ + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + ]; - ySum = foreachLine(delegate(const Point posLine) { - ulong xSum = 0; + const int[2] samplingOffsets = [ + source.scanTo(posSampling[0]), + source.scanTo(posSampling[1]), + ]; - const ulong[2] weightsX = () { - ulong[2] result; - result[1] = posSrc[0].fractionalDigits; - result[0] = ulong(uint.max) + 1 - result[1]; - return result; - }(); - - const samplingOffset = source.scanTo(posLine); - ubyte[2] pxcLR = () @trusted { - ubyte[2] result = [ - source.data.ptr[samplingOffset].components.ptr[ib], - source.data.ptr[samplingOffset + nSamples].components.ptr[ib], + const srcSamples2 = () @trusted { + const(const(Pixel)[])[2] result = [ + source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], + source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], ]; return result; }(); - xSum += (pxcLR[idxL] * weightsX[idxL]); - xSum += (pxcLR[idxR] * weightsX[idxR]); + ulong[2] xSums = [0, 0]; - return (xSum >> 32); - }); + foreach (idx, srcSamples; srcSamples2) { + foreach (srcSample; srcSamples) { + () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); + } + } + + xSums[] /= nSamples; + return xSums; + } + + static if (mode == SamplingMode.multi) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + + auto ySum = foreachLine(delegate(const Point posLine) { + const samplingOffset = source.scanTo(posLine); + const srcSamples = () @trusted { + return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; + }(); + + ulong xSum = 0; + + foreach (srcSample; srcSamples) { + xSum += (() @trusted => srcSample.components.ptr[ib])(); + } + + return xSum; + }); + + ySum /= nSamples; + return (ySum / nLines); + } } - return (ySum / nLines); + // ======== Up ======== + static if (directionX == up) { + + if (posSrcX[0] == posSrcX[1]) { + static if (mode == SamplingMode.single) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } + static if (mode == SamplingMode.dual) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } + static if (mode == SamplingMode.multi) { + const ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; + const samplingOffset = source.scanTo(posLine); + return (() @trusted + => source.data.ptr[samplingOffset].components.ptr[ib] + )(); + }); + return (ySum / nLines); + } + } + + const ulong[2] weightsX = () { + ulong[2] result; + result[0] = (udecimalHalf + posSrcX[1] - posSrc[idxX]).fractionalDigits; + result[1] = ulong(uint.max) + 1 - result[0]; + return result; + }(); + + static if (mode == SamplingMode.single) { + ulong xSum = 0; + + () @trusted { + xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + }(); + + xSum >>= 32; + return xSum; + } + + static if (mode == SamplingMode.dual) { + ulong[2] xSums = [0, 0]; + + () @trusted { + xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + + xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); + xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); + }(); + + foreach (ref sum; xSums) { + sum >>= 32; + } + + return xSums; + } + + static if (mode == SamplingMode.multi) { + const ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; + + const samplingOffset = source.scanTo(posLine); + ubyte[2] pxcLR = () @trusted { + ubyte[2] result = [ + source.data.ptr[samplingOffset].components.ptr[ib], + source.data.ptr[samplingOffset + 1].components.ptr[ib], + ]; + return result; + }(); + + xSum += (pxcLR[idxL] * weightsX[idxL]); + xSum += (pxcLR[idxR] * weightsX[idxR]); + + return (xSum >> 32); + }); + + return (ySum / nLines); + } + } } static if (directionY == none) { - c = clamp255(sampleX()); + c = clamp255(sampleX!(SamplingMode.single)()); } else static if (directionY == down) { - c = clamp255(sampleXMulti()); + c = clamp255(sampleX!(SamplingMode.multi)()); } else /* if (directionY == up) */ { - // looks ass const ulong[2] weightsY = () { ulong[2] result; - result[idxB] = posSrc[1].fractionalDigits; - result[idxT] = ulong(uint.max) + 1 - result[idxB]; + result[0] = (udecimalHalf + posSrcY[1] - posSrc[idxY]).fractionalDigits; + result[1] = ulong(uint.max) + 1 - result[0]; return result; }(); - const xSums = sampleXDual(); + const xSums = sampleX!(SamplingMode.dual)(); ulong ySum = 0; ySum += (xSums[idxT] * weightsY[idxT]); From d9e1e0e84e1444a96fb4c50cc5bde6b63707a6b9 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 05:13:17 +0100 Subject: [PATCH 26/38] Refactor y foreach loop of bilinear scaler --- pixmappaint.d | 176 +++++++++++++++++++++++--------------------------- 1 file changed, 80 insertions(+), 96 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 3d46388..5a0ffc2 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2995,6 +2995,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe enum uint udecimalHalfFD = udecimalHalf.fractionalDigits; enum idxX = 0, idxY = 1; + enum idxL = 0, idxR = 1; + enum idxT = 0, idxB = 1; const int[2] sourceMax = [ (source.width - 1), @@ -3038,110 +3040,92 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ==== Bilinear ==== static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() { + + int[2] posSrcCenterToInterpolationTargets( + ScalingDirection direction, + )( + UDecimal posSrcCenter, + UDecimal ratioHalf, + int sourceMax, + ) { + int[2] result; + static if (direction == none) { + const value = posSrcCenter.castTo!int; + result = [ + value, + value, + ]; + } + + static if (direction == up) { + if (posSrcCenter < udecimalHalf) { + result = [ + 0, + 0, + ]; + } else { + const floor = posSrcCenter.castTo!uint; + if (posSrcCenter.fractionalDigits == udecimalHalfFD) { + result = [ + floor, + floor, + ]; + } else if (posSrcCenter.fractionalDigits > udecimalHalfFD) { + const upper = min((floor + 1), sourceMax); + result = [ + floor, + upper, + ]; + } else { + result = [ + floor - 1, + floor, + ]; + } + } + } + + static if (direction == down) { + result = [ + max((posSrcCenter - ratioHalf).roundEven().castTo!uint, 0), + min((posSrcCenter + ratioHalf).roundEven().castTo!uint, sourceMax), + ]; + } + return result; + }; + auto dst = PixmapScannerRW(target); size_t y = 0; foreach (dstLine; dst) { + const posDstY = y.castTo!uint; + const UDecimal posSrcCenterY = posDstY * ratios[idxY] + ratiosHalf[idxY]; + + const int[2] posSrcY = posSrcCenterToInterpolationTargets!(directionY)( + posSrcCenterY, + ratiosHalf[idxY], + sourceMax[idxY], + ); + foreach (x, ref pxDst; dstLine) { + const posDstX = x.castTo!uint; const int[2] posDst = [ - x.castTo!uint, - y.castTo!uint, + posDstX, + posDstY, ]; - const UDecimal[2] posSrc = [ - posDst[idxX] * ratios[idxX] + ratiosHalf[idxX], - posDst[idxY] * ratios[idxY] + ratiosHalf[idxY], + const posSrcCenterX = posDst[idxX] * ratios[idxX] + ratiosHalf[idxX]; + + const UDecimal[2] posSrcCenter = [ + posSrcCenterX, + posSrcCenterY, ]; - const int[2] posSrcX = () { - int[2] result; - static if (directionX == none) { - const value = posSrc[idxX].castTo!int; - result = [ - value, - value, - ]; - } else static if (directionX == up) { - if (posSrc[idxX] < udecimalHalf) { - result = [ - 0, - 0, - ]; - } else { - const floor = posSrc[idxX].castTo!uint; - if (posSrc[idxX].fractionalDigits == udecimalHalfFD) { - result = [ - floor, - floor, - ]; - } else if (posSrc[idxX].fractionalDigits > udecimalHalfFD) { - const upper = min((floor + 1), sourceMax[idxX]); - result = [ - floor, - upper, - ]; - } else { - result = [ - floor - 1, - floor, - ]; - } - } - } else /* if (directionX == down) */ { - const ratioXHalf = (ratios[idxX] >> 1); - result = [ - max((posSrc[idxX] - ratioXHalf).roundEven().castTo!uint, 0), - min((posSrc[idxX] + ratioXHalf).roundEven().castTo!uint, sourceMax[idxX]), - ]; - } - return result; - }(); - - const int[2] posSrcY = () { - int[2] result; - static if (directionY == none) { - result = [ - posSrc[idxY].castTo!int, - posSrc[idxY].castTo!int, - ]; - } else static if (directionY == up) { - if (posSrc[idxY] < udecimalHalf) { - result = [ - 0, - 0, - ]; - } else { - const floor = posSrc[idxY].castTo!uint; - if (posSrc[idxY].fractionalDigits == udecimalHalfFD) { - result = [ - floor, - floor, - ]; - } else if (posSrc[idxY].fractionalDigits > udecimalHalfFD) { - const upper = min((floor + 1), sourceMax[idxY]); - result = [ - floor, - upper, - ]; - } else { - result = [ - floor - 1, - floor, - ]; - } - } - } else /* if (directionY == down) */ { - const ratioHalf = (ratios[idxY] >> 1); - result = [ - max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0), - min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMax[idxY]), - ]; - } - return result; - }(); - - enum idxL = 0, idxR = 1; - enum idxT = 0, idxB = 1; + const int[2] posSrcX = posSrcCenterToInterpolationTargets!(directionX)( + posSrcCenterX, + ratiosHalf[idxX], + sourceMax[idxX], + ); const Point[4] posNeighs = [ Point(posSrcX[idxL], posSrcY[idxT]), @@ -3330,7 +3314,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const ulong[2] weightsX = () { ulong[2] result; - result[0] = (udecimalHalf + posSrcX[1] - posSrc[idxX]).fractionalDigits; + result[0] = (udecimalHalf + posSrcX[1] - posSrcCenter[idxX]).fractionalDigits; result[1] = ulong(uint.max) + 1 - result[0]; return result; }(); @@ -3396,7 +3380,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } else /* if (directionY == up) */ { const ulong[2] weightsY = () { ulong[2] result; - result[0] = (udecimalHalf + posSrcY[1] - posSrc[idxY]).fractionalDigits; + result[0] = (udecimalHalf + posSrcY[1] - posSrcCenter[idxY]).fractionalDigits; result[1] = ulong(uint.max) + 1 - result[0]; return result; }(); From 0108d467ad7506d7fea4d585f96ecaf9f0b535f1 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 05:16:50 +0100 Subject: [PATCH 27/38] Improve readability --- pixmappaint.d | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 5a0ffc2..2fa9435 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3168,6 +3168,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe foreach (immutable ib, ref c; pxInt.components) { + // ======== Interpolate X ======== auto sampleX(SamplingMode mode)() { pragma(inline, true); @@ -3185,7 +3186,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } - // ======== None ======== + // ========== None ========== static if (directionX == none) { static if (mode == SamplingMode.single) { return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); @@ -3210,7 +3211,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } - // ======== Down ======== + // ========== Down ========== static if (directionX == down) { static if (mode == SamplingMode.single) { const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; @@ -3284,7 +3285,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } - // ======== Up ======== + // ========== Up ========== static if (directionX == up) { if (posSrcX[0] == posSrcX[1]) { @@ -3373,11 +3374,14 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } + // ======== Interpolate Y ======== static if (directionY == none) { c = clamp255(sampleX!(SamplingMode.single)()); - } else static if (directionY == down) { + } + static if (directionY == down) { c = clamp255(sampleX!(SamplingMode.multi)()); - } else /* if (directionY == up) */ { + } + static if (directionY == up) { const ulong[2] weightsY = () { ulong[2] result; result[0] = (udecimalHalf + posSrcY[1] - posSrcCenter[idxY]).fractionalDigits; From 16994b51f64b269e22b669e06232af684edf632e Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 05:19:08 +0100 Subject: [PATCH 28/38] Move Y weight calculation --- pixmappaint.d | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 2fa9435..9dd77fe 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3107,6 +3107,15 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe sourceMax[idxY], ); + static if (directionY == up) { + const ulong[2] weightsY = () { + ulong[2] result; + result[0] = (udecimalHalf + posSrcY[1] - posSrcCenterY).fractionalDigits; + result[1] = ulong(uint.max) + 1 - result[0]; + return result; + }(); + } + foreach (x, ref pxDst; dstLine) { const posDstX = x.castTo!uint; const int[2] posDst = [ @@ -3382,13 +3391,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe c = clamp255(sampleX!(SamplingMode.multi)()); } static if (directionY == up) { - const ulong[2] weightsY = () { - ulong[2] result; - result[0] = (udecimalHalf + posSrcY[1] - posSrcCenter[idxY]).fractionalDigits; - result[1] = ulong(uint.max) + 1 - result[0]; - return result; - }(); - const xSums = sampleX!(SamplingMode.dual)(); ulong ySum = 0; From 539480a2fa69c47fee5321f6e52108a23abd43ee Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 05:48:58 +0100 Subject: [PATCH 29/38] Remove unnecessary array --- pixmappaint.d | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 9dd77fe..42b0d61 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3125,11 +3125,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const posSrcCenterX = posDst[idxX] * ratios[idxX] + ratiosHalf[idxX]; - const UDecimal[2] posSrcCenter = [ - posSrcCenterX, - posSrcCenterY, - ]; - const int[2] posSrcX = posSrcCenterToInterpolationTargets!(directionX)( posSrcCenterX, ratiosHalf[idxX], @@ -3324,7 +3319,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const ulong[2] weightsX = () { ulong[2] result; - result[0] = (udecimalHalf + posSrcX[1] - posSrcCenter[idxX]).fractionalDigits; + result[0] = (udecimalHalf + posSrcX[1] - posSrcCenterX).fractionalDigits; result[1] = ulong(uint.max) + 1 - result[0]; return result; }(); From 2804f426c49b1244d581837c41819a5642517687 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 05:59:56 +0100 Subject: [PATCH 30/38] Refactor component/channel loop of image scaler --- pixmappaint.d | 371 +++++++++++++++++++++++++------------------------- 1 file changed, 187 insertions(+), 184 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 42b0d61..ce3419d 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3092,7 +3092,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe ]; } return result; - }; + } auto dst = PixmapScannerRW(target); @@ -3170,57 +3170,105 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe multi, } - foreach (immutable ib, ref c; pxInt.components) { + // ======== Interpolate X ======== + auto sampleX(SamplingMode mode)(const size_t ib) { + pragma(inline, true); - // ======== Interpolate X ======== - auto sampleX(SamplingMode mode)() { - pragma(inline, true); + static if (mode == SamplingMode.multi) { + const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + + alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; + ulong foreachLine(scope ForeachLineCallback apply) { + ulong linesSum = 0; + foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { + const posLine = Point(posSrcX[idxL], lineY); + linesSum += apply(posLine); + } + return linesSum; + } + } + + // ========== None ========== + static if (directionX == none) { + static if (mode == SamplingMode.single) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } + + static if (mode == SamplingMode.dual) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } static if (mode == SamplingMode.multi) { - const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + const ySum = foreachLine(delegate(const Point posLine) { + const pxSrc = source.getPixel(posLine); + return ulong((() @trusted => pxSrc.components.ptr[ib])()); + }); + return (ySum / nLines); + } + } - alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; - ulong foreachLine(scope ForeachLineCallback apply) { - ulong linesSum = 0; - foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { - const posLine = Point(posSrcX[idxL], lineY); - linesSum += apply(posLine); + // ========== Down ========== + static if (directionX == down) { + static if (mode == SamplingMode.single) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); + const samplingOffset = source.scanTo(posSampling); + const srcSamples = () @trusted { + return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; + }(); + + ulong xSum = 0; + + foreach (srcSample; srcSamples) { + xSum += (() @trusted => srcSample.components.ptr[ib])(); + } + + return (xSum / nSamples); + } + + static if (mode == SamplingMode.dual) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + const Point[2] posSampling = [ + Point(posSrcX[idxL], posSrcY[idxT]), + Point(posSrcX[idxL], posSrcY[idxB]), + ]; + + const int[2] samplingOffsets = [ + source.scanTo(posSampling[0]), + source.scanTo(posSampling[1]), + ]; + + const srcSamples2 = () @trusted { + const(const(Pixel)[])[2] result = [ + source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], + source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], + ]; + return result; + }(); + + ulong[2] xSums = [0, 0]; + + foreach (idx, srcSamples; srcSamples2) { + foreach (srcSample; srcSamples) { + () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); } - return linesSum; } + + xSums[] /= nSamples; + return xSums; } - // ========== None ========== - static if (directionX == none) { - static if (mode == SamplingMode.single) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); - } + static if (mode == SamplingMode.multi) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - static if (mode == SamplingMode.dual) { - return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], - ]; - return result; - }(); - } - - static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { - const pxSrc = source.getPixel(posLine); - return ulong((() @trusted => pxSrc.components.ptr[ib])()); - }); - return (ySum / nLines); - } - } - - // ========== Down ========== - static if (directionX == down) { - static if (mode == SamplingMode.single) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); - const samplingOffset = source.scanTo(posSampling); + auto ySum = foreachLine(delegate(const Point posLine) { + const samplingOffset = source.scanTo(posLine); const srcSamples = () @trusted { return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; }(); @@ -3231,162 +3279,117 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe xSum += (() @trusted => srcSample.components.ptr[ib])(); } - return (xSum / nSamples); + return xSum; + }); + + ySum /= nSamples; + return (ySum / nLines); + } + } + + // ========== Up ========== + static if (directionX == up) { + + if (posSrcX[0] == posSrcX[1]) { + static if (mode == SamplingMode.single) { + return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + } + static if (mode == SamplingMode.dual) { + return () @trusted { + ulong[2] result = [ + pxNeighs[idxTL].components.ptr[ib], + pxNeighs[idxBL].components.ptr[ib], + ]; + return result; + }(); + } + static if (mode == SamplingMode.multi) { + const ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; + const samplingOffset = source.scanTo(posLine); + return (() @trusted + => source.data.ptr[samplingOffset].components.ptr[ib] + )(); + }); + return (ySum / nLines); + } + } + + const ulong[2] weightsX = () { + ulong[2] result; + result[0] = (udecimalHalf + posSrcX[1] - posSrcCenterX).fractionalDigits; + result[1] = ulong(uint.max) + 1 - result[0]; + return result; + }(); + + static if (mode == SamplingMode.single) { + ulong xSum = 0; + + () @trusted { + xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + }(); + + xSum >>= 32; + return xSum; + } + + static if (mode == SamplingMode.dual) { + ulong[2] xSums = [0, 0]; + + () @trusted { + xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); + xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + + xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); + xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); + }(); + + foreach (ref sum; xSums) { + sum >>= 32; } - static if (mode == SamplingMode.dual) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - const Point[2] posSampling = [ - Point(posSrcX[idxL], posSrcY[idxT]), - Point(posSrcX[idxL], posSrcY[idxB]), - ]; + return xSums; + } - const int[2] samplingOffsets = [ - source.scanTo(posSampling[0]), - source.scanTo(posSampling[1]), - ]; + static if (mode == SamplingMode.multi) { + const ySum = foreachLine(delegate(const Point posLine) { + ulong xSum = 0; - const srcSamples2 = () @trusted { - const(const(Pixel)[])[2] result = [ - source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], - source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], + const samplingOffset = source.scanTo(posLine); + ubyte[2] pxcLR = () @trusted { + ubyte[2] result = [ + source.data.ptr[samplingOffset].components.ptr[ib], + source.data.ptr[samplingOffset + 1].components.ptr[ib], ]; return result; }(); - ulong[2] xSums = [0, 0]; + xSum += (pxcLR[idxL] * weightsX[idxL]); + xSum += (pxcLR[idxR] * weightsX[idxR]); - foreach (idx, srcSamples; srcSamples2) { - foreach (srcSample; srcSamples) { - () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); - } - } + return (xSum >> 32); + }); - xSums[] /= nSamples; - return xSums; - } - - static if (mode == SamplingMode.multi) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - - auto ySum = foreachLine(delegate(const Point posLine) { - const samplingOffset = source.scanTo(posLine); - const srcSamples = () @trusted { - return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; - }(); - - ulong xSum = 0; - - foreach (srcSample; srcSamples) { - xSum += (() @trusted => srcSample.components.ptr[ib])(); - } - - return xSum; - }); - - ySum /= nSamples; - return (ySum / nLines); - } - } - - // ========== Up ========== - static if (directionX == up) { - - if (posSrcX[0] == posSrcX[1]) { - static if (mode == SamplingMode.single) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); - } - static if (mode == SamplingMode.dual) { - return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], - ]; - return result; - }(); - } - static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { - ulong xSum = 0; - const samplingOffset = source.scanTo(posLine); - return (() @trusted - => source.data.ptr[samplingOffset].components.ptr[ib] - )(); - }); - return (ySum / nLines); - } - } - - const ulong[2] weightsX = () { - ulong[2] result; - result[0] = (udecimalHalf + posSrcX[1] - posSrcCenterX).fractionalDigits; - result[1] = ulong(uint.max) + 1 - result[0]; - return result; - }(); - - static if (mode == SamplingMode.single) { - ulong xSum = 0; - - () @trusted { - xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - }(); - - xSum >>= 32; - return xSum; - } - - static if (mode == SamplingMode.dual) { - ulong[2] xSums = [0, 0]; - - () @trusted { - xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - - xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); - xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); - }(); - - foreach (ref sum; xSums) { - sum >>= 32; - } - - return xSums; - } - - static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { - ulong xSum = 0; - - const samplingOffset = source.scanTo(posLine); - ubyte[2] pxcLR = () @trusted { - ubyte[2] result = [ - source.data.ptr[samplingOffset].components.ptr[ib], - source.data.ptr[samplingOffset + 1].components.ptr[ib], - ]; - return result; - }(); - - xSum += (pxcLR[idxL] * weightsX[idxL]); - xSum += (pxcLR[idxR] * weightsX[idxR]); - - return (xSum >> 32); - }); - - return (ySum / nLines); - } + return (ySum / nLines); } } + } - // ======== Interpolate Y ======== - static if (directionY == none) { - c = clamp255(sampleX!(SamplingMode.single)()); + // ======== Interpolate Y ======== + static if (directionY == none) { + foreach (immutable ib, ref c; pxInt.components) { + c = clamp255(sampleX!(SamplingMode.single)(ib)); } - static if (directionY == down) { - c = clamp255(sampleX!(SamplingMode.multi)()); + } + static if (directionY == down) { + foreach (immutable ib, ref c; pxInt.components) { + c = clamp255(sampleX!(SamplingMode.multi)(ib)); } - static if (directionY == up) { - const xSums = sampleX!(SamplingMode.dual)(); + } + static if (directionY == up) { + foreach (immutable ib, ref c; pxInt.components) { + const xSums = sampleX!(SamplingMode.dual)(ib); ulong ySum = 0; ySum += (xSums[idxT] * weightsY[idxT]); From c3beff155c4fc361e42475e3fb5ffb46bc792b46 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 07:30:21 +0100 Subject: [PATCH 31/38] Refactor component foreach loop of bilinear scaler --- pixmappaint.d | 179 +++++++++++++++++++++++++++++++------------------- 1 file changed, 113 insertions(+), 66 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index ce3419d..83529d0 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3041,6 +3041,29 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() { + alias InterPixel = ulong[4]; + + static Pixel toPixel(const InterPixel ipx) @safe pure nothrow @nogc { + pragma(inline, true); + return Pixel( + clamp255(ipx[0]), + clamp255(ipx[1]), + clamp255(ipx[2]), + clamp255(ipx[3]), + ); + } + + static InterPixel toInterPixel(const Pixel ipx) @safe pure nothrow @nogc { + pragma(inline, true); + InterPixel result = [ + ipx.r, + ipx.g, + ipx.b, + ipx.a, + ]; + return result; + } + int[2] posSrcCenterToInterpolationTargets( ScalingDirection direction, )( @@ -3048,6 +3071,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe UDecimal ratioHalf, int sourceMax, ) { + pragma(inline, true); + int[2] result; static if (direction == none) { const value = posSrcCenter.castTo!int; @@ -3171,18 +3196,19 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } // ======== Interpolate X ======== - auto sampleX(SamplingMode mode)(const size_t ib) { + auto sampleX(SamplingMode mode)() { pragma(inline, true); static if (mode == SamplingMode.multi) { const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; - alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc; - ulong foreachLine(scope ForeachLineCallback apply) { - ulong linesSum = 0; + alias ForeachLineCallback = InterPixel delegate(const Point posLine) @safe pure nothrow @nogc; + InterPixel foreachLine(scope ForeachLineCallback apply) { + InterPixel linesSum = 0; foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { const posLine = Point(posSrcX[idxL], lineY); - linesSum += apply(posLine); + const lineValues = apply(posLine); + linesSum[] += lineValues[]; } return linesSum; } @@ -3191,25 +3217,26 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ========== None ========== static if (directionX == none) { static if (mode == SamplingMode.single) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + return pxNeighs[idxTL]; } static if (mode == SamplingMode.dual) { return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], + InterPixel[2] result = [ + toInterPixel(pxNeighs[idxTL]), + toInterPixel(pxNeighs[idxBL]), ]; return result; }(); } static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { + auto ySum = foreachLine(delegate(const Point posLine) { const pxSrc = source.getPixel(posLine); - return ulong((() @trusted => pxSrc.components.ptr[ib])()); + return toInterPixel(pxSrc); }); - return (ySum / nLines); + ySum[] /= nLines; + return ySum; } } @@ -3223,13 +3250,16 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; }(); - ulong xSum = 0; + InterPixel xSum = [0, 0, 0, 0]; foreach (srcSample; srcSamples) { - xSum += (() @trusted => srcSample.components.ptr[ib])(); + foreach (immutable ib, c; srcSample.components) { + () @trusted { xSum.ptr[ib] += c; }(); + } } - return (xSum / nSamples); + xSum[] /= nSamples; + return toPixel(xSum); } static if (mode == SamplingMode.dual) { @@ -3252,15 +3282,19 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return result; }(); - ulong[2] xSums = [0, 0]; + InterPixel[2] xSums = [[0, 0, 0, 0], [0, 0, 0, 0]]; foreach (idx, srcSamples; srcSamples2) { foreach (srcSample; srcSamples) { - () @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }(); + foreach (immutable ib, c; srcSample.components) + () @trusted { xSums.ptr[idx].ptr[ib] += c; }(); } } - xSums[] /= nSamples; + foreach (xSum; xSums) { + xSum[] /= nSamples; + } + return xSums; } @@ -3273,17 +3307,20 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; }(); - ulong xSum = 0; + InterPixel xSum = 0; foreach (srcSample; srcSamples) { - xSum += (() @trusted => srcSample.components.ptr[ib])(); + foreach (immutable ib, c; srcSample.components) { + () @trusted { xSum.ptr[ib] += c; }(); + } } return xSum; }); - ySum /= nSamples; - return (ySum / nLines); + ySum[] /= nSamples; + ySum[] /= nLines; + return ySum; } } @@ -3292,26 +3329,26 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe if (posSrcX[0] == posSrcX[1]) { static if (mode == SamplingMode.single) { - return (() @trusted => pxNeighs[idxTL].components.ptr[ib])(); + return pxNeighs[idxTL]; } static if (mode == SamplingMode.dual) { return () @trusted { - ulong[2] result = [ - pxNeighs[idxTL].components.ptr[ib], - pxNeighs[idxBL].components.ptr[ib], + InterPixel[2] result = [ + toInterPixel(pxNeighs[idxTL]), + toInterPixel(pxNeighs[idxBL]), ]; return result; }(); } static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { - ulong xSum = 0; + auto ySum = foreachLine(delegate(const Point posLine) { const samplingOffset = source.scanTo(posLine); - return (() @trusted - => source.data.ptr[samplingOffset].components.ptr[ib] - )(); + return toInterPixel( + (() @trusted => source.data.ptr[samplingOffset])() + ); }); - return (ySum / nLines); + ySum[] /= nLines; + return ySum; } } @@ -3323,80 +3360,90 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe }(); static if (mode == SamplingMode.single) { - ulong xSum = 0; + InterPixel xSum = [0, 0, 0, 0]; - () @trusted { - xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); - }(); + foreach (immutable ib, ref c; xSum) { + c += ((() @trusted => pxNeighs[idxTL].components.ptr[ib])() * weightsX[0]); + c += ((() @trusted => pxNeighs[idxTR].components.ptr[ib])() * weightsX[1]); + } - xSum >>= 32; - return xSum; + foreach (ref c; xSum) { + c >>= 32; + } + return toPixel(xSum); } static if (mode == SamplingMode.dual) { - ulong[2] xSums = [0, 0]; + InterPixel[2] xSums = [[0, 0, 0, 0], [0, 0, 0, 0]]; () @trusted { - xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]); - xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]); + foreach (immutable ib, ref c; xSums[0]) { + c += (pxNeighs[idxTL].components.ptr[ib] * weightsX[idxL]); + c += (pxNeighs[idxTR].components.ptr[ib] * weightsX[idxR]); + } - xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]); - xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]); + foreach (immutable ib, ref c; xSums[1]) { + c += (pxNeighs[idxBL].components.ptr[ib] * weightsX[idxL]); + c += (pxNeighs[idxBR].components.ptr[ib] * weightsX[idxR]); + } }(); foreach (ref sum; xSums) { - sum >>= 32; + foreach (ref c; sum) { + c >>= 32; + } } return xSums; } static if (mode == SamplingMode.multi) { - const ySum = foreachLine(delegate(const Point posLine) { - ulong xSum = 0; + auto ySum = foreachLine(delegate(const Point posLine) { + InterPixel xSum = [0, 0, 0, 0]; const samplingOffset = source.scanTo(posLine); - ubyte[2] pxcLR = () @trusted { - ubyte[2] result = [ - source.data.ptr[samplingOffset].components.ptr[ib], - source.data.ptr[samplingOffset + 1].components.ptr[ib], + Pixel[2] pxcLR = () @trusted { + Pixel[2] result = [ + source.data.ptr[samplingOffset], + source.data.ptr[samplingOffset + 1], ]; return result; }(); - xSum += (pxcLR[idxL] * weightsX[idxL]); - xSum += (pxcLR[idxR] * weightsX[idxR]); + foreach (immutable ib, ref c; xSum) { + c += ((() @trusted => pxcLR[idxL].components.ptr[ib])() * weightsX[idxL]); + c += ((() @trusted => pxcLR[idxR].components.ptr[ib])() * weightsX[idxR]); + } - return (xSum >> 32); + foreach (ref c; xSum) { + c >>= 32; + } + return xSum; }); - return (ySum / nLines); + ySum[] /= nLines; + return ySum; } } } // ======== Interpolate Y ======== static if (directionY == none) { - foreach (immutable ib, ref c; pxInt.components) { - c = clamp255(sampleX!(SamplingMode.single)(ib)); - } + const Pixel tmp = sampleX!(SamplingMode.single)(); + pxInt = tmp; } static if (directionY == down) { - foreach (immutable ib, ref c; pxInt.components) { - c = clamp255(sampleX!(SamplingMode.multi)(ib)); - } + const InterPixel tmp = sampleX!(SamplingMode.multi)(); + pxInt = toPixel(tmp); } static if (directionY == up) { + const InterPixel[2] xSums = sampleX!(SamplingMode.dual)(); foreach (immutable ib, ref c; pxInt.components) { - const xSums = sampleX!(SamplingMode.dual)(ib); - ulong ySum = 0; - ySum += (xSums[idxT] * weightsY[idxT]); - ySum += (xSums[idxB] * weightsY[idxB]); + ySum += ((() @trusted => xSums[idxT].ptr[ib])() * weightsY[idxT]); + ySum += ((() @trusted => xSums[idxB].ptr[ib])() * weightsY[idxB]); const xySum = (ySum >> 32); - c = clamp255(xySum); } } From 200524851440fc2a23f1a8c7f661883043508f92 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 1 Feb 2025 22:52:02 +0100 Subject: [PATCH 32/38] Refactor scaling code --- pixmappaint.d | 29 +++++++++++++++++------------ 1 file changed, 17 insertions(+), 12 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 83529d0..5efc1bf 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3132,6 +3132,10 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe sourceMax[idxY], ); + static if (directionY == down) { + const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + } + static if (directionY == up) { const ulong[2] weightsY = () { ulong[2] result; @@ -3156,6 +3160,10 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe sourceMax[idxX], ); + static if (directionX == down) { + const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; + } + const Point[4] posNeighs = [ Point(posSrcX[idxL], posSrcY[idxT]), Point(posSrcX[idxR], posSrcY[idxT]), @@ -3200,10 +3208,11 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe pragma(inline, true); static if (mode == SamplingMode.multi) { - const nLines = 1 + posSrcY[idxB] - posSrcY[idxT]; + alias ForeachLineCallback = + InterPixel delegate(const Point posLine) @safe pure nothrow @nogc; - alias ForeachLineCallback = InterPixel delegate(const Point posLine) @safe pure nothrow @nogc; InterPixel foreachLine(scope ForeachLineCallback apply) { + pragma(inline, true); InterPixel linesSum = 0; foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { const posLine = Point(posSrcX[idxL], lineY); @@ -3243,7 +3252,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ========== Down ========== static if (directionX == down) { static if (mode == SamplingMode.single) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); const samplingOffset = source.scanTo(posSampling); const srcSamples = () @trusted { @@ -3263,21 +3271,20 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } static if (mode == SamplingMode.dual) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; const Point[2] posSampling = [ - Point(posSrcX[idxL], posSrcY[idxT]), - Point(posSrcX[idxL], posSrcY[idxB]), + posNeighs[idxTL], + posNeighs[idxBL], ]; const int[2] samplingOffsets = [ - source.scanTo(posSampling[0]), - source.scanTo(posSampling[1]), + source.scanTo(posSampling[idxT]), + source.scanTo(posSampling[idxB]), ]; const srcSamples2 = () @trusted { const(const(Pixel)[])[2] result = [ - source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)], - source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)], + source.data.ptr[samplingOffsets[idxT] .. (samplingOffsets[idxT] + nSamples)], + source.data.ptr[samplingOffsets[idxB] .. (samplingOffsets[idxB] + nSamples)], ]; return result; }(); @@ -3299,8 +3306,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } static if (mode == SamplingMode.multi) { - const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL]; - auto ySum = foreachLine(delegate(const Point posLine) { const samplingOffset = source.scanTo(posLine); const srcSamples = () @trusted { From c65c8d462ee5ad516d47695fb24d1fee4af40f79 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 01:17:27 +0100 Subject: [PATCH 33/38] Fix downscaler --- pixmappaint.d | 13 ++----------- 1 file changed, 2 insertions(+), 11 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 5efc1bf..8e8b9ef 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3068,7 +3068,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe ScalingDirection direction, )( UDecimal posSrcCenter, - UDecimal ratioHalf, int sourceMax, ) { pragma(inline, true); @@ -3082,7 +3081,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe ]; } - static if (direction == up) { + static if (direction == up || direction == down) { if (posSrcCenter < udecimalHalf) { result = [ 0, @@ -3110,12 +3109,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } } - static if (direction == down) { - result = [ - max((posSrcCenter - ratioHalf).roundEven().castTo!uint, 0), - min((posSrcCenter + ratioHalf).roundEven().castTo!uint, sourceMax), - ]; - } return result; } @@ -3128,7 +3121,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const int[2] posSrcY = posSrcCenterToInterpolationTargets!(directionY)( posSrcCenterY, - ratiosHalf[idxY], sourceMax[idxY], ); @@ -3156,7 +3148,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe const int[2] posSrcX = posSrcCenterToInterpolationTargets!(directionX)( posSrcCenterX, - ratiosHalf[idxX], sourceMax[idxX], ); @@ -3252,7 +3243,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ========== Down ========== static if (directionX == down) { static if (mode == SamplingMode.single) { - const posSampling = Point(posSrcX[idxL], posSrcY[idxT]); + const posSampling = posNeighs[idxTL]; const samplingOffset = source.scanTo(posSampling); const srcSamples = () @trusted { return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)]; From 4ca96e723b7e650c935aef7360c495a2564ae1b2 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 01:40:17 +0100 Subject: [PATCH 34/38] Remove superfluous `SamplingMode` templating --- pixmappaint.d | 40 +++++++++++++++++----------------------- 1 file changed, 17 insertions(+), 23 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 8e8b9ef..25d8d58 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3188,17 +3188,11 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe static if (filter == ScalingFilter.bilinear) { auto pxInt = Pixel(0, 0, 0, 0); - enum SamplingMode { - single, - dual, - multi, - } - // ======== Interpolate X ======== - auto sampleX(SamplingMode mode)() { + auto sampleX() { pragma(inline, true); - static if (mode == SamplingMode.multi) { + static if (directionY == down) { alias ForeachLineCallback = InterPixel delegate(const Point posLine) @safe pure nothrow @nogc; @@ -3216,11 +3210,11 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ========== None ========== static if (directionX == none) { - static if (mode == SamplingMode.single) { + static if (directionY == none) { return pxNeighs[idxTL]; } - static if (mode == SamplingMode.dual) { + static if (directionY == up) { return () @trusted { InterPixel[2] result = [ toInterPixel(pxNeighs[idxTL]), @@ -3230,7 +3224,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe }(); } - static if (mode == SamplingMode.multi) { + static if (directionY == down) { auto ySum = foreachLine(delegate(const Point posLine) { const pxSrc = source.getPixel(posLine); return toInterPixel(pxSrc); @@ -3242,7 +3236,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ========== Down ========== static if (directionX == down) { - static if (mode == SamplingMode.single) { + static if (directionY == none) { const posSampling = posNeighs[idxTL]; const samplingOffset = source.scanTo(posSampling); const srcSamples = () @trusted { @@ -3261,7 +3255,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return toPixel(xSum); } - static if (mode == SamplingMode.dual) { + static if (directionY == up) { const Point[2] posSampling = [ posNeighs[idxTL], posNeighs[idxBL], @@ -3296,7 +3290,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return xSums; } - static if (mode == SamplingMode.multi) { + static if (directionY == down) { auto ySum = foreachLine(delegate(const Point posLine) { const samplingOffset = source.scanTo(posLine); const srcSamples = () @trusted { @@ -3324,10 +3318,10 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe static if (directionX == up) { if (posSrcX[0] == posSrcX[1]) { - static if (mode == SamplingMode.single) { + static if (directionY == none) { return pxNeighs[idxTL]; } - static if (mode == SamplingMode.dual) { + static if (directionY == up) { return () @trusted { InterPixel[2] result = [ toInterPixel(pxNeighs[idxTL]), @@ -3336,7 +3330,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return result; }(); } - static if (mode == SamplingMode.multi) { + static if (directionY == down) { auto ySum = foreachLine(delegate(const Point posLine) { const samplingOffset = source.scanTo(posLine); return toInterPixel( @@ -3355,7 +3349,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return result; }(); - static if (mode == SamplingMode.single) { + static if (directionY == none) { InterPixel xSum = [0, 0, 0, 0]; foreach (immutable ib, ref c; xSum) { @@ -3369,7 +3363,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return toPixel(xSum); } - static if (mode == SamplingMode.dual) { + static if (directionY == up) { InterPixel[2] xSums = [[0, 0, 0, 0], [0, 0, 0, 0]]; () @trusted { @@ -3393,7 +3387,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe return xSums; } - static if (mode == SamplingMode.multi) { + static if (directionY == down) { auto ySum = foreachLine(delegate(const Point posLine) { InterPixel xSum = [0, 0, 0, 0]; @@ -3425,15 +3419,15 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe // ======== Interpolate Y ======== static if (directionY == none) { - const Pixel tmp = sampleX!(SamplingMode.single)(); + const Pixel tmp = sampleX(); pxInt = tmp; } static if (directionY == down) { - const InterPixel tmp = sampleX!(SamplingMode.multi)(); + const InterPixel tmp = sampleX(); pxInt = toPixel(tmp); } static if (directionY == up) { - const InterPixel[2] xSums = sampleX!(SamplingMode.dual)(); + const InterPixel[2] xSums = sampleX(); foreach (immutable ib, ref c; pxInt.components) { ulong ySum = 0; ySum += ((() @trusted => xSums[idxT].ptr[ib])() * weightsY[idxT]); From 4d74f70ddd1139bf7c85f0182c4e98452ea9eecb Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 01:51:53 +0100 Subject: [PATCH 35/38] Fix downscaler --- pixmappaint.d | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 25d8d58..bbe2698 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3199,7 +3199,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe InterPixel foreachLine(scope ForeachLineCallback apply) { pragma(inline, true); InterPixel linesSum = 0; - foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { + foreach (const lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) { const posLine = Point(posSrcX[idxL], lineY); const lineValues = apply(posLine); linesSum[] += lineValues[]; @@ -3245,8 +3245,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe InterPixel xSum = [0, 0, 0, 0]; - foreach (srcSample; srcSamples) { - foreach (immutable ib, c; srcSample.components) { + foreach (const srcSample; srcSamples) { + foreach (immutable ib, const c; srcSample.components) { () @trusted { xSum.ptr[ib] += c; }(); } } @@ -3276,14 +3276,14 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe InterPixel[2] xSums = [[0, 0, 0, 0], [0, 0, 0, 0]]; - foreach (idx, srcSamples; srcSamples2) { - foreach (srcSample; srcSamples) { - foreach (immutable ib, c; srcSample.components) + foreach (immutable idx, const srcSamples; srcSamples2) { + foreach (const srcSample; srcSamples) { + foreach (immutable ib, const c; srcSample.components) () @trusted { xSums.ptr[idx].ptr[ib] += c; }(); } } - foreach (xSum; xSums) { + foreach (ref xSum; xSums) { xSum[] /= nSamples; } @@ -3300,7 +3300,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe InterPixel xSum = 0; foreach (srcSample; srcSamples) { - foreach (immutable ib, c; srcSample.components) { + foreach (immutable ib, const c; srcSample.components) { () @trusted { xSum.ptr[ib] += c; }(); } } From 122e60da837c15ca99d46c18ed41273057afe0fd Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 01:53:05 +0100 Subject: [PATCH 36/38] Add commented out outline of `SoftwarePixmapRenderer` --- pixmappresenter.d | 34 ++++++++++++++++++++++++++++++++++ 1 file changed, 34 insertions(+) diff --git a/pixmappresenter.d b/pixmappresenter.d index c8bab90..9cbfecd 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -838,6 +838,40 @@ final class OpenGl1PixmapRenderer : PixmapRenderer { } } +/+ +/++ + Purely software renderer + +/ +final class SoftwarePixmapRenderer : PixmapRenderer { + + private { + PresenterObjectsContainer* _poc; + } + + public WantsOpenGl wantsOpenGl() @safe pure nothrow @nogc { + return WantsOpenGl(0); + } + + public void setup(PresenterObjectsContainer* container) { + } + + public void reconfigure() { + } + + /++ + Schedules a redraw + +/ + public void redrawSchedule() { + } + + /++ + Triggers a redraw + +/ + public void redrawNow() { + } +} ++/ + /// struct LoopCtrl { int interval; /// in milliseconds From f2c15f0e3136c2d4480c2865d0329bb553809621 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 01:56:44 +0100 Subject: [PATCH 37/38] Remove janky faux-linear scaler --- pixmappaint.d | 24 +----------------------- pixmappresenter.d | 2 -- 2 files changed, 1 insertion(+), 25 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index bbe2698..b0a441a 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2957,15 +2957,6 @@ enum ScalingFilter { +/ bilinear, - /++ - Unweighted linear interpolation - - $(TIP - Visual impression: “blocky”, “pixelated” - ) - +/ - fauxLinear, - /// linear = bilinear, } @@ -3038,7 +3029,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe } // ==== Bilinear ==== - static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) { + static if (filter == ScalingFilter.bilinear) { void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() { alias InterPixel = ulong[4]; @@ -3171,19 +3162,6 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe enum idxTL = 0, idxTR = 1, idxBL = 2, idxBR = 3; - // ====== Faux bilinear ====== - static if (filter == ScalingFilter.fauxLinear) { - auto pxInt = Pixel(0, 0, 0, 0); - - foreach (immutable ib, ref c; pxInt.components) { - uint sum = 0; - foreach (const pxNeigh; pxNeighs) { - sum += (() @trusted => pxNeigh.components.ptr[ib])(); - } - c = (sum >> 2).castTo!ubyte; - } - } - // ====== Proper bilinear (up) + Avg (down) ====== static if (filter == ScalingFilter.bilinear) { auto pxInt = Pixel(0, 0, 0, 0); diff --git a/pixmappresenter.d b/pixmappresenter.d index 9cbfecd..01fed24 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -630,7 +630,6 @@ final class OpenGl3PixmapRenderer : PixmapRenderer { final switch (_poc.config.renderer.filter) with (ScalingFilter) { case nearest: - case fauxLinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; @@ -740,7 +739,6 @@ final class OpenGl1PixmapRenderer : PixmapRenderer { final switch (_poc.config.renderer.filter) with (ScalingFilter) { case nearest: - case fauxLinear: glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break; From f815c0b33697fa9ea705b23f8649eb6aff98d8a3 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 2 Feb 2025 02:21:40 +0100 Subject: [PATCH 38/38] Practise what you preach --- pixmappaint.d | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index b0a441a..778e7eb 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -3128,7 +3128,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe }(); } - foreach (x, ref pxDst; dstLine) { + foreach (const x, ref pxDst; dstLine) { const posDstX = x.castTo!uint; const int[2] posDst = [ posDstX,