From 011abf026e1e6b10fb320dabf892e05caa6a0689 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 5 Oct 2024 23:59:55 +0200 Subject: [PATCH 01/38] Add SubPixmap support to PixmapPaint --- color.d | 20 ++ pixmappaint.d | 595 +++++++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 612 insertions(+), 3 deletions(-) diff --git a/color.d b/color.d index 6e5f1bf..d2a0b83 100644 --- a/color.d +++ b/color.d @@ -1906,6 +1906,23 @@ struct Point { Size opCast(T : Size)() inout @nogc { return Size(x, y); } + + /++ + Calculates the point of linear offset in a rectangle. + + `Offset = 0` is assumed to be equivalent to `Point(0,0)`. + + See_also: + [linearOffset] is the inverse function. + + History: + Added October 05, 2024. + +/ + static Point fromLinearOffset(int linearOffset, int width) @nogc { + const y = (linearOffset / width); + const x = (linearOffset % width); + return Point(x, y); + } } /// @@ -1959,6 +1976,9 @@ struct Size { Returns: `y * width + x` + See_also: + [Point.fromLinearOffset] is the inverse function. + History: Added December 19, 2023 (dub v11.4) +/ diff --git a/pixmappaint.d b/pixmappaint.d index 9440603..4d55d6c 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -106,7 +106,7 @@ struct Pixmap { } /// - this(Pixel[] data, int width) @nogc + this(inout(Pixel)[] data, int width) inout @nogc in (data.length % width == 0) { this.data = data; this.width = width; @@ -197,24 +197,551 @@ struct Pixmap { return (width * int(Pixel.sizeof)); } + /++ + Calculates the index (linear offset) of the requested position + within the pixmap data. + +/ + int scanTo(Point pos) inout { + return linearOffset(width, pos); + } + + /++ + Accesses the pixel at the requested position within the pixmap data. + +/ + ref inout(Pixel) scan(Point pos) inout { + return data[scanTo(pos)]; + } + /++ Retrieves a linear slice of the pixmap. Returns: `n` pixels starting at the top-left position `pos`. +/ - inout(Pixel)[] sliceAt(Point pos, int n) inout { + inout(Pixel)[] scan(Point pos, int n) inout { immutable size_t offset = linearOffset(width, pos); immutable size_t end = (offset + n); return data[offset .. end]; } + /// ditto + inout(Pixel)[] sliceAt(Point pos, int n) inout { + return scan(pos, n); + } + + /++ + Retrieves a rectangular subimage of the pixmap. + +/ + inout(SubPixmap) scan2D(Point pos, Size size) inout { + return inout(SubPixmap)(this, size, pos); + } + + /++ + Retrieves the first line of the Pixmap. + + See_also: + Check out [PixmapScanner] for more useful scanning functionality. + +/ + inout(Pixel)[] scanLine() inout { + return data[0 .. width]; + } + /// Clears the buffer’s contents (by setting each pixel to the same color) void clear(Pixel value) { data[] = value; } } +/++ + A subpixmap represents a subimage of a [Pixmap]. + + This wrapper provides convenient access to a rectangular slice of a Pixmap. + + ``` + ╔═════════════╗ + ║ Pixmap ║ + ║ ║ + ║ ┌───┐ ║ + ║ │Sub│ ║ + ║ └───┘ ║ + ╚═════════════╝ + ``` + +/ +struct SubPixmap { + + /++ + Source image referenced by the subimage + +/ + Pixmap source; + + /++ + Size of the subimage + +/ + Size size; + + /++ + 2D offset of the subimage + +/ + Point offset; + + public @safe pure nothrow @nogc { + /// + this(inout Pixmap source, Size size = Size(0, 0), Point offset = Point(0, 0)) inout { + this.source = source; + this.size = size; + this.offset = offset; + } + + /// + this(inout Pixmap source, Point offset, Size size = Size(0, 0)) inout { + this(source, size, offset); + } + } + +@safe pure nothrow @nogc: + + public { + /++ + Width of the subimage. + +/ + int width() const { + return size.width; + } + + /// ditto + void width(int value) { + size.width = value; + } + + /++ + Height of the subimage. + +/ + int height() const { + return size.height; + } + + /// height + void height(int value) { + size.height = value; + } + } + + public { + /++ + Linear offset of the subimage within the source image. + + Calculates the index of the “first pixel of the subimage” + in the “pixel data of the source image”. + +/ + int sourceOffsetLinear() const { + return linearOffset(offset, source.width); + } + + /// ditto + void sourceOffsetLinear(int value) { + this.offset = Point.fromLinearOffset(value, source.width); + } + + /++ + $(I Advanced functionality.) + + Offset of the bottom right corner of the subimage + from the top left corner the source image. + +/ + Point sourceOffsetEnd() const { + return (offset + castTo!Point(size)); + } + + /++ + Linear offset of the subimage within the source image. + + Calculates the index of the “first pixel of the subimage” + in the “pixel data of the source image”. + +/ + int sourceOffsetLinearEnd() const { + return linearOffset(sourceOffsetEnd, source.width); + } + } + + /++ + Determines whether the area of the subimage + lies within the source image + and does not overflow its lines. + + $(TIP + If the offset and/or size of a subimage are off, two issues can occur: + + $(LIST + * The resulting subimage will look displaced. + (As if the lines were shifted.) + This indicates that one scanline of the subimage spans over + two ore more lines of the source image. + (Happens when `(subimage.offset.x + subimage.size.width) > source.size.width`.) + * When accessing the pixel data, bounds checks will fail. + This suggests that the area of the subimage extends beyond + the bottom end (and optionally also beyond the right end) of + the source. + ) + + Both defects could indicate a non-sound subimage. + Use this function to verify the SubPixmap. + ) + +/ + bool isSound() const { + return ( + (sourceMarginLeft >= 0) + && (sourceMarginTop >= 0) + && (sourceMarginBottom >= 0) + && (sourceMarginRight >= 0) + ); + } + + public inout { + /++ + Retrieves the pixel at the requested position of the subimage. + +/ + ref inout(Pixel) scan(Point pos) { + return source.scan(offset + pos); + } + + /++ + Retrieves the first line of the subimage. + +/ + inout(Pixel)[] scanLine() { + const lo = linearOffset(offset, size.width); + return source.data[lo .. size.width]; + } + } + + public void xferTo(SubPixmap target, Blend blend = blendNormal) const { + auto src = SubPixmapScanner(this); + auto dst = SubPixmapScannerRW(target); + + foreach (dstLine; dst) { + blendPixels(dstLine, src.front, blend); + src.popFront(); + } + } + + // opposite offset + public const { + /++ + $(I Advanced functionality.) + + Offset of the bottom right corner of the source image + to the bottom right corner of the subimage. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ │ │ ║ + ║ └───┘ ║ + ║ ↘ ║ + ╚═══════════╝ + ``` + +/ + Point oppositeOffset() { + return Point(oppositeOffsetX, oppositeOffsetY); + } + + /++ + $(I Advanced functionality.) + + Offset of the right edge of the source image + to the right edge of the subimage. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ │ S │ → ║ + ║ └───┘ ║ + ║ ║ + ╚═══════════╝ + ``` + +/ + int oppositeOffsetX() { + return (offset.x + size.width); + } + + /++ + $(I Advanced functionality.) + + Offset of the bottom edge of the source image + to the bottom edge of the subimage. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ │ S │ ║ + ║ └───┘ ║ + ║ ↓ ║ + ╚═══════════╝ + ``` + +/ + int oppositeOffsetY() { + return (offset.y + size.height); + } + + } + + // source-image margins + public const { + /++ + $(I Advanced functionality.) + + X-axis margin (left + right) of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ ↔ │ S │ ↔ ║ + ║ └───┘ ║ + ║ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginX() { + return (source.width - size.width); + } + + /++ + $(I Advanced functionality.) + + Y-axis margin (top + bottom) of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ↕ ║ + ║ ┌───┐ ║ + ║ │ S │ ║ + ║ └───┘ ║ + ║ ↕ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginY() { + return (source.height - size.height); + } + + /++ + $(I Advanced functionality.) + + Top margin of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ↕ ║ + ║ ┌───┐ ║ + ║ │ S │ ║ + ║ └───┘ ║ + ║ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginTop() { + return offset.y; + } + + /++ + $(I Advanced functionality.) + + Right margin of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ │ S │ ↔ ║ + ║ └───┘ ║ + ║ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginRight() { + return (sourceMarginX - sourceMarginLeft); + } + + /++ + $(I Advanced functionality.) + + Bottom margin of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ │ S │ ║ + ║ └───┘ ║ + ║ ↕ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginBottom() { + return (sourceMarginY - sourceMarginTop); + } + + /++ + $(I Advanced functionality.) + + Left margin of the subimage within the source image. + + ``` + ╔═══════════╗ + ║ ║ + ║ ┌───┐ ║ + ║ ↔ │ S │ ║ + ║ └───┘ ║ + ║ ║ + ╚═══════════╝ + ``` + +/ + int sourceMarginLeft() { + return offset.x; + } + } + + public const { + /++ + $(I Advanced functionality.) + + Calculates the linear offset of the provided point in the subimage + relative to the source image. + +/ + int sourceOffsetOf(Point pos) { + pos = (pos + offset); + debug { + import std.stdio : writeln; + + try { + writeln(pos); + } catch (Exception) { + } + } + return linearOffset(pos, source.width); + } + } +} + +/++ + Wrapper for scanning a [Pixmap] line by line. + +/ +struct PixmapScanner { + private { + const(Pixel)[] _data; + int _width; + } + +@safe pure nothrow @nogc: + + /// + public this(const(Pixmap) pixmap) { + _data = pixmap.data; + _width = pixmap.width; + } + + /// + bool empty() const { + return (_data.length == 0); + } + + /// + const(Pixel)[] front() const { + return _data[0 .. _width]; + } + + /// + void popFront() { + _data = _data[_width .. $]; + } +} + +/++ + Wrapper for scanning a [Pixmap] line by line. + +/ +struct SubPixmapScanner { + private { + const(Pixel)[] _data; + int _width; + int _feed; + } + +@safe pure nothrow @nogc: + + /// + public this(const(SubPixmap) subPixmap) { + _data = subPixmap.source.data[subPixmap.sourceOffsetLinear .. subPixmap.sourceOffsetLinearEnd]; + _width = subPixmap.size.width; + _feed = subPixmap.source.width; + } + + /// + bool empty() const { + return (_data.length == 0); + } + + /// + const(Pixel)[] front() const { + return _data[0 .. _width]; + } + + /// + void popFront() { + if (_data.length < _feed) { + _data.length = 0; + return; + } + + _data = _data[_feed .. $]; + } +} + +/++ + Wrapper for scanning a [Pixmap] line by line. + + See_also: + Unlike [SubPixmapScanner], this does not work with `const(Pixmap)`. + +/ +struct SubPixmapScannerRW { + private { + Pixel[] _data; + int _width; + int _feed; + } + +@safe pure nothrow @nogc: + + /// + public this(SubPixmap subPixmap) { + _data = subPixmap.source.data[subPixmap.sourceOffsetLinear .. subPixmap.sourceOffsetLinearEnd]; + _width = subPixmap.size.width; + _feed = subPixmap.source.width; + } + + /// + bool empty() const { + return (_data.length == 0); + } + + /// + Pixel[] front() { + return _data[0 .. _width]; + } + + /// + void popFront() { + if (_data.length < _feed) { + _data.length = 0; + return; + } + + _data = _data[_feed .. $]; + } +} + /// struct SpriteSheet { private { @@ -1397,7 +1924,7 @@ void drawLine(Pixmap target, Point a, Point b, Pixel color) { image = source pixmap pos = top-left destination position (on the target pixmap) +/ -void drawPixmap(Pixmap target, Pixmap image, Point pos, Blend blend = blendNormal) { +void drawPixmap(Pixmap target, const Pixmap image, Point pos, Blend blend = blendNormal) { alias source = image; immutable tRect = OriginRectangle( @@ -1433,6 +1960,68 @@ void drawPixmap(Pixmap target, Pixmap image, Point pos, Blend blend = blendNorma } } +/++ + Draws an image (a subimage from a source pixmap) on a target pixmap + + Params: + target = target pixmap to draw on + image = source subpixmap + pos = top-left destination position (on the target pixmap) + +/ +void drawPixmap(Pixmap target, const SubPixmap image, Point pos, Blend blend = blendNormal) { + alias source = image; + + debug assert(source.isSound); + + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + immutable sRect = Rectangle(pos, source.size); + + // out of bounds? + if (!tRect.intersect(sRect)) { + return; + } + + Point sourceOffset = source.offset; + Point drawingTarget; + Size drawingSize = source.size; + + if (pos.x <= 0) { + sourceOffset.x -= pos.x; + drawingTarget.x = 0; + drawingSize.width += pos.x; + } else { + drawingTarget.x = pos.x; + } + + if (pos.y <= 0) { + sourceOffset.y -= pos.y; + drawingTarget.y = 0; + drawingSize.height += pos.y; + } else { + drawingTarget.y = pos.y; + } + + Point drawingEnd = drawingTarget + drawingSize.castTo!Point(); + if (drawingEnd.x >= source.width) { + drawingSize.width -= (drawingEnd.x - source.width); + } + if (drawingEnd.y >= source.height) { + drawingSize.height -= (drawingEnd.y - source.height); + } + + auto dst = SubPixmap(target, drawingTarget, drawingSize); + auto src = const(SubPixmap)( + source.source, + drawingSize, + sourceOffset, + ); + + src.xferTo(dst, blend); +} + /++ Draws a sprite from a spritesheet +/ From d83ae009829b59cf6ccd38345ec872de1b03d993 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 00:00:42 +0200 Subject: [PATCH 02/38] Make the pure round()ing function private --- pixmappaint.d | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 4d55d6c..3750ff1 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -20,13 +20,16 @@ module arsd.pixmappaint; import arsd.color; import arsd.core; -private float hackyRound(float f) { +private float roundImpl(float f) { import std.math : round; + return round(f); } -float round(float f) pure @nogc nothrow @trusted { - return (cast(float function(float) pure @nogc nothrow) &hackyRound)(f); +// `pure` rounding function. +// std.math.round() isn’t pure on all targets. +private float round(float f) pure @nogc nothrow @trusted { + return (castTo!(float function(float) pure @nogc nothrow)(&roundImpl))(f); } /* From 1d0822c89a9f2ea2a65160b28fe288834c27f9fe Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 00:03:45 +0200 Subject: [PATCH 03/38] Rename SubPixmap.isSound() to SubPixmap.isValid() --- pixmappaint.d | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 3750ff1..6900bd8 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -386,11 +386,11 @@ struct SubPixmap { the source. ) - Both defects could indicate a non-sound subimage. + Both defects could indicate an invalid subimage. Use this function to verify the SubPixmap. ) +/ - bool isSound() const { + bool isValid() const { return ( (sourceMarginLeft >= 0) && (sourceMarginTop >= 0) @@ -1974,7 +1974,7 @@ void drawPixmap(Pixmap target, const Pixmap image, Point pos, Blend blend = blen void drawPixmap(Pixmap target, const SubPixmap image, Point pos, Blend blend = blendNormal) { alias source = image; - debug assert(source.isSound); + debug assert(source.isValid); immutable tRect = OriginRectangle( Size(target.width, target.height), From 38301f1507b22d5f5ee6b9b99e0a64f73ebf1e57 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 00:19:32 +0200 Subject: [PATCH 04/38] Document SubPixmap.xferTo() and improve docs slightly. --- pixmappaint.d | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 6900bd8..83b5c0e 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -416,7 +416,17 @@ struct SubPixmap { } } + /++ + Blends the pixels of this subimage into a target image. + + The target MUST have the same size. + + See_also: + Usually you’ll want to use [drawPixmap] instead. + +/ public void xferTo(SubPixmap target, Blend blend = blendNormal) const { + debug assert(target.size == this.size); + auto src = SubPixmapScanner(this); auto dst = SubPixmapScannerRW(target); @@ -630,6 +640,8 @@ struct SubPixmap { } /++ + $(I Advanced functionality.) + Wrapper for scanning a [Pixmap] line by line. +/ struct PixmapScanner { @@ -663,6 +675,8 @@ struct PixmapScanner { } /++ + $(I Advanced functionality.) + Wrapper for scanning a [Pixmap] line by line. +/ struct SubPixmapScanner { @@ -703,6 +717,8 @@ struct SubPixmapScanner { } /++ + $(I Advanced functionality.) + Wrapper for scanning a [Pixmap] line by line. See_also: From a391b8dad9949a70c5684dfc3b70fc0d6dee4caf Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 01:34:18 +0200 Subject: [PATCH 05/38] Remove leftover debug{writeln();} --- pixmappaint.d | 8 -------- 1 file changed, 8 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 83b5c0e..d7d4363 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -626,14 +626,6 @@ struct SubPixmap { +/ int sourceOffsetOf(Point pos) { pos = (pos + offset); - debug { - import std.stdio : writeln; - - try { - writeln(pos); - } catch (Exception) { - } - } return linearOffset(pos, source.width); } } From 0bdcc43a57b6feb89a8066047c9d3399d5804b78 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 01:34:51 +0200 Subject: [PATCH 06/38] Fix SubPixmap --- pixmappaint.d | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index d7d4363..d496d7d 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -348,11 +348,13 @@ struct SubPixmap { /++ $(I Advanced functionality.) - Offset of the bottom right corner of the subimage - from the top left corner the source image. + Offset of the pixel following the bottom right corner of the subimage. + + (`Point(O, 0)` is the top left corner of the source image.) +/ Point sourceOffsetEnd() const { - return (offset + castTo!Point(size)); + auto vec = Point(size.x, (size.y - 1)); + return (offset + vec); } /++ @@ -2016,11 +2018,11 @@ void drawPixmap(Pixmap target, const SubPixmap image, Point pos, Blend blend = b } Point drawingEnd = drawingTarget + drawingSize.castTo!Point(); - if (drawingEnd.x >= source.width) { - drawingSize.width -= (drawingEnd.x - source.width); + if (drawingEnd.x >= target.width) { + drawingSize.width -= (drawingEnd.x - target.width); } - if (drawingEnd.y >= source.height) { - drawingSize.height -= (drawingEnd.y - source.height); + if (drawingEnd.y >= target.height) { + drawingSize.height -= (drawingEnd.y - target.height); } auto dst = SubPixmap(target, drawingTarget, drawingSize); From 9418cbaa2410d7e14f3c36e7b582db1024aec747 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 01:35:08 +0200 Subject: [PATCH 07/38] Document the coordinate system used by PixmapPaint --- pixmappaint.d | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index d496d7d..3aaea4f 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -14,6 +14,22 @@ This module is $(B work in progress). API is subject to changes until further notice. ) + + ### The coordinate system + + The top left corner of a pixmap is its $(B origin) `(0,0)`. + + The $(horizontal axis) is called `x`. + Its corresponding length/dimension is known as `width`. + + The letter `y` is used to describe the $(B vertical axis). + Its corresponding length/dimension is known as `height`. + + ``` + 0 → x + ↓ + x + ``` +/ module arsd.pixmappaint; From 33cd84552b179acf5fc430a0e1b951a6ed12746d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 01:43:20 +0200 Subject: [PATCH 08/38] Fix refucktoring regression --- pixmappaint.d | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index 3aaea4f..7b0a4a4 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -369,7 +369,7 @@ struct SubPixmap { (`Point(O, 0)` is the top left corner of the source image.) +/ Point sourceOffsetEnd() const { - auto vec = Point(size.x, (size.y - 1)); + auto vec = Point(size.width, (size.height - 1)); return (offset + vec); } From 1d2f907d3cb108c2b92320a8be0434b865f24682 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 03:54:47 +0200 Subject: [PATCH 09/38] Implement Pixmap cropping --- pixmappaint.d | 175 ++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 163 insertions(+), 12 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 7b0a4a4..b52bcf8 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -53,7 +53,6 @@ private float round(float f) pure @nogc nothrow @trusted { - Refactoring the template-mess of blendPixel() & co. - Scaling - - Cropping - Rotating - Skewing - HSL @@ -317,6 +316,70 @@ struct SubPixmap { } } +@safe pure nothrow: + + public { + /++ + Allocates a new Pixmap cropped to the pixel data of the subimage. + + See_also: + Use [extractToPixmap] for a non-allocating variant with an . + +/ + Pixmap extractToNewPixmap() const { + auto pm = Pixmap(size); + this.extractToPixmap(pm); + return pm; + } + + /++ + Copies the pixel data – cropped to the subimage region – + into the target Pixmap. + + Returns: + A size-adjusted shallow copy of the input Pixmap overwritten + with the image data of the SubPixmap. + + $(PITFALL + While the returned Pixmap utilizes the buffer provided by the input, + the returned Pixmap might not exactly match the input. + + Always use the returned Pixmap structure. + + --- + // Same buffer, but new structure: + auto pixmap2 = subPixmap.extractToPixmap(pixmap); + + // Alternatively, replace the old structure: + pixmap = subPixmap.extractToPixmap(pixmap); + --- + ) + +/ + Pixmap extractToPixmap(Pixmap target) @nogc const { + // Length adjustment + const l = this.length; + if (target.data.length < l) { + assert(false, "The target Pixmap is too small."); + } else if (target.data.length > l) { + target.data = target.data[0 .. l]; + } + + target.width = this.width; + + extractToPixmapCopyImpl(target); + return target; + } + + private void extractToPixmapCopyImpl(Pixmap target) @nogc const { + auto src = SubPixmapScanner(this); + auto dst = PixmapScannerRW(target); + + foreach (dstLine; dst) { + dstLine[] = src.front[]; + src.popFront(); + } + } + } + @safe pure nothrow @nogc: public { @@ -343,6 +406,13 @@ struct SubPixmap { void height(int value) { size.height = value; } + + /++ + Number of pixels in the subimage. + +/ + int length() const { + return size.area; + } } public { @@ -434,15 +504,35 @@ struct SubPixmap { } } + /++ + Copies the pixels of this subimage to a target image. + + The target MUST have the same size. + + See_also: + Usually you’ll want to use [extractToPixmap] or [drawPixmap] instead. + +/ + public void xferTo(SubPixmap target) const { + debug assert(target.size == this.size); + + auto src = SubPixmapScanner(this); + auto dst = SubPixmapScannerRW(target); + + foreach (dstLine; dst) { + dstLine[] = src.front[]; + src.popFront(); + } + } + /++ Blends the pixels of this subimage into a target image. The target MUST have the same size. See_also: - Usually you’ll want to use [drawPixmap] instead. + Usually you’ll want to use [extractToPixmap] or [drawPixmap] instead. +/ - public void xferTo(SubPixmap target, Blend blend = blendNormal) const { + public void xferTo(SubPixmap target, Blend blend) const { debug assert(target.size == this.size); auto src = SubPixmapScanner(this); @@ -684,6 +774,44 @@ struct PixmapScanner { } } +/++ + $(I Advanced functionality.) + + Wrapper for scanning a [Pixmap] line by line. + + See_also: + Unlike [PixmapScanner], this does not work with `const(Pixmap)`. + +/ +struct PixmapScannerRW { + private { + Pixel[] _data; + int _width; + } + +@safe pure nothrow @nogc: + + /// + public this(Pixmap pixmap) { + _data = pixmap.data; + _width = pixmap.width; + } + + /// + bool empty() const { + return (_data.length == 0); + } + + /// + Pixel[] front() { + return _data[0 .. _width]; + } + + /// + void popFront() { + _data = _data[_width .. $]; + } +} + /++ $(I Advanced functionality.) @@ -847,10 +975,10 @@ private struct OriginRectangle { } } -@safe pure nothrow @nogc: +@safe pure nothrow: // misc -private { +private @nogc { Point pos(Rectangle r) => r.upperLeft; T max(T)(T a, T b) => (a >= b) ? a : b; @@ -932,7 +1060,7 @@ unittest { Returns: sqrt(value / 255f) * 255 +/ -ubyte intNormalizedSqrt(const ubyte value) { +ubyte intNormalizedSqrt(const ubyte value) @nogc { switch (value) { default: // unreachable @@ -1337,7 +1465,7 @@ unittest { /++ Limits a value to a maximum of 0xFF (= 255). +/ -ubyte clamp255(Tint)(const Tint value) { +ubyte clamp255(Tint)(const Tint value) @nogc { pragma(inline, true); return (value < 0xFF) ? value.castTo!ubyte : 0xFF; } @@ -1358,7 +1486,7 @@ ubyte clamp255(Tint)(const Tint value) { Returns: `round(value * nPercentage / 255.0)` +/ -ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { +ubyte n255thsOf(const ubyte nPercentage, const ubyte value) @nogc { immutable factor = (nPercentage | (nPercentage << 8)); return (((value * factor) + 0x8080) >> 16); } @@ -1382,6 +1510,8 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { } } +// ==== Image manipulation functions ==== + /++ Sets the opacity of a [Pixmap]. @@ -1391,7 +1521,7 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { See_Also: Use [opacityF] with opacity values in percent (%). +/ -void opacity(Pixmap pixmap, const ubyte opacity) { +void opacity(Pixmap pixmap, const ubyte opacity) @nogc { foreach (ref px; pixmap.data) { px.a = opacity.n255thsOf(px.a); } @@ -1406,7 +1536,7 @@ void opacity(Pixmap pixmap, const ubyte opacity) { See_Also: Use [opacity] with 8-bit integer opacity values (in 255ths). +/ -void opacityF(Pixmap pixmap, const float opacity) +void opacityF(Pixmap pixmap, const float opacity) @nogc in (opacity >= 0) in (opacity <= 1.0) { immutable opacity255 = round(opacity * 255).castTo!ubyte; @@ -1416,7 +1546,7 @@ in (opacity <= 1.0) { /++ Inverts a color (to its negative color). +/ -Pixel invert(const Pixel color) { +Pixel invert(const Pixel color) @nogc { return Pixel( 0xFF - color.r, 0xFF - color.g, @@ -1432,12 +1562,33 @@ Pixel invert(const Pixel color) { Develops a positive image when applied to a negative one. ) +/ -void invert(Pixmap pixmap) { +void invert(Pixmap pixmap) @nogc { foreach (ref px; pixmap.data) { px = invert(px); } } +/// +void crop(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nogc { + auto src = const(SubPixmap)(source, target.size, offset); + src.extractToPixmapCopyImpl(target); +} + +/// +Pixmap cropNew(const Pixmap source, Size targetSize, Point offset = Point(0, 0)) { + auto target = Pixmap(targetSize); + crop(source, target, offset); + return target; +} + +/// +void cropInplace(ref Pixmap source, Size targetSize, Point offset = Point(0, 0)) @nogc { + auto src = const(SubPixmap)(source, targetSize, offset); + source = src.extractToPixmap(source); +} + +@safe pure nothrow @nogc: + // ==== Blending functions ==== /++ From eade7a375475dbaeeab1260f2832a1d3d0d164df Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 03:55:04 +0200 Subject: [PATCH 10/38] Fix and improve documentation of PixmapPaint --- pixmappaint.d | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index b52bcf8..66d6790 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -402,7 +402,7 @@ struct SubPixmap { return size.height; } - /// height + /// ditto void height(int value) { size.height = value; } @@ -477,6 +477,15 @@ struct SubPixmap { Both defects could indicate an invalid subimage. Use this function to verify the SubPixmap. ) + + $(WARNING + Do not use invalid SubPixmaps. + The library assumes that the SubPixmaps it receives are always valid. + + Non-valid SubPixmaps are not meant to be used for creative effects + or similar either. Such uses might lead to unexpected quirks or + crashes eventually. + ) +/ bool isValid() const { return ( @@ -860,7 +869,7 @@ struct SubPixmapScanner { Wrapper for scanning a [Pixmap] line by line. See_also: - Unlike [SubPixmapScanner], this does not work with `const(Pixmap)`. + Unlike [SubPixmapScanner], this does not work with `const(SubPixmap)`. +/ struct SubPixmapScannerRW { private { From 8bf54227ae68f448f241d894af68f4e6719b5725 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 20:31:01 +0200 Subject: [PATCH 11/38] Add documentation for the crop functions --- pixmappaint.d | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 66d6790..8c14359 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1577,20 +1577,32 @@ void invert(Pixmap pixmap) @nogc { } } -/// +/++ + Crops an image and stores the result in the provided target Pixmap. + + The size of the area to crop the image to + is derived from the size of the target. + +/ void crop(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nogc { auto src = const(SubPixmap)(source, target.size, offset); src.extractToPixmapCopyImpl(target); } -/// +/++ + Crops an image and stores the result in a newly allocated Pixmap. + +/ Pixmap cropNew(const Pixmap source, Size targetSize, Point offset = Point(0, 0)) { auto target = Pixmap(targetSize); crop(source, target, offset); return target; } -/// +/++ + Crops an image and stores the result in the source buffer. + + The source pixmap structure is passed by ref and gets with a size-adjusted + structure using a slice of the same underlying memory. + +/ void cropInplace(ref Pixmap source, Size targetSize, Point offset = Point(0, 0)) @nogc { auto src = const(SubPixmap)(source, targetSize, offset); source = src.extractToPixmap(source); From d60426e83319eef77150936ba0e436d3a00a1f0c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 6 Oct 2024 21:52:39 +0200 Subject: [PATCH 12/38] Add Pixmap copy function --- pixmappaint.d | 40 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 40 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 8c14359..2ac1603 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -140,6 +140,46 @@ struct Pixmap { return c; } + /++ + Copies the pixel data to the target Pixmap. + + Returns: + A size-adjusted shallow copy of the input Pixmap overwritten + with the image data of the SubPixmap. + + $(PITFALL + While the returned Pixmap utilizes the buffer provided by the input, + the returned Pixmap might not exactly match the input. + + Always use the returned Pixmap structure. + + --- + // Same buffer, but new structure: + auto pixmap2 = source.copyTo(pixmap); + + // Alternatively, replace the old structure: + pixmap = source.copyTo(pixmap); + --- + ) + +/ + Pixmap copyTo(Pixmap target) const { + // Length adjustment + const l = this.length; + if (target.data.length < l) { + assert(false, "The target Pixmap is too small."); + } else if (target.data.length > l) { + target.data = target.data[0 .. l]; + } + + copyToImpl(target); + + return target; + } + + private void copyToImpl(Pixmap target) const { + target.data[] = this.data[]; + } + // undocumented: really shouldn’t be used. // carries the risks of `length` and `width` getting out of sync accidentally. deprecated("Use `size` instead.") From 6978d1f2dde91ec4865ea9731a20931362363450 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 00:17:36 +0200 Subject: [PATCH 13/38] Implement clockwise rotation --- pixmappaint.d | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 2ac1603..20314f3 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1648,6 +1648,32 @@ void cropInplace(ref Pixmap source, Size targetSize, Point offset = Point(0, 0)) source = src.extractToPixmap(source); } +/++ + Rotates an image by 90° clockwise. + + $(PITFALL + This function does not work in place. + Do not attempt to pass Pixmaps sharing the same buffer for both source + and target. Such would lead to a bad result with heavy artifacts. + ) + +/ +void rotateClockwise(const Pixmap source, Pixmap target) @nogc { + debug assert(source.data.length == target.data.length); + + const area = source.data.length; + const rowLength = source.size.height; + ptrdiff_t cursor = -1; + + foreach (px; source.data) { + cursor += rowLength; + if (cursor > area) { + cursor -= (area + 1); + } + + target.data[cursor] = px; + } +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From fc346c3ec29b36da3eaa54f87267db7221b739e7 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 01:48:04 +0200 Subject: [PATCH 14/38] Fix cropInPlace() --- pixmappaint.d | 50 +++++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 43 insertions(+), 7 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 20314f3..fbdcd9a 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -375,13 +375,16 @@ struct SubPixmap { Copies the pixel data – cropped to the subimage region – into the target Pixmap. - Returns: - A size-adjusted shallow copy of the input Pixmap overwritten - with the image data of the SubPixmap. + $(PITFALL + Do not attempt to extract a subimage back into the source pixmap. + This will fail in cases where source and target regions overlap + and potentially crash the program. + ) $(PITFALL While the returned Pixmap utilizes the buffer provided by the input, the returned Pixmap might not exactly match the input. + The dimensions (width and height) and the length might have changed. Always use the returned Pixmap structure. @@ -393,6 +396,10 @@ struct SubPixmap { pixmap = subPixmap.extractToPixmap(pixmap); --- ) + + Returns: + A size-adjusted shallow copy of the input Pixmap overwritten + with the image data of the SubPixmap. +/ Pixmap extractToPixmap(Pixmap target) @nogc const { // Length adjustment @@ -418,6 +425,19 @@ struct SubPixmap { src.popFront(); } } + + private void extractToPixmapCopyPixelByPixelImpl(Pixmap target) @nogc const { + auto src = SubPixmapScanner(this); + auto dst = PixmapScannerRW(target); + + foreach (dstLine; dst) { + const srcLine = src.front; + foreach (idx, ref px; dstLine) { + px = srcLine[idx]; + } + src.popFront(); + } + } } @safe pure nothrow @nogc: @@ -1640,12 +1660,18 @@ Pixmap cropNew(const Pixmap source, Size targetSize, Point offset = Point(0, 0)) /++ Crops an image and stores the result in the source buffer. - The source pixmap structure is passed by ref and gets with a size-adjusted - structure using a slice of the same underlying memory. + The source pixmap structure is passed by value. + A size-adjusted structure using a slice of the same underlying memory is + returned. +/ -void cropInplace(ref Pixmap source, Size targetSize, Point offset = Point(0, 0)) @nogc { +Pixmap cropInPlace(Pixmap source, Size targetSize, Point offset = Point(0, 0)) @nogc { + Pixmap target = source; + target.width = targetSize.width; + target.data = target.data[0 .. targetSize.area]; + auto src = const(SubPixmap)(source, targetSize, offset); - source = src.extractToPixmap(source); + src.extractToPixmapCopyPixelByPixelImpl(target); + return target; } /++ @@ -1674,6 +1700,16 @@ void rotateClockwise(const Pixmap source, Pixmap target) @nogc { } } +/++ + Rotates an image by 90° clockwise. + Stores the result in a newly allocated Pixmap. + +/ +Pixmap rotateClockwiseNew(const Pixmap source) { + auto target = Pixmap(Size(source.height, source.width)); + source.rotateClockwise(target); + return target; +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From e5841da630cb1bc441a83f006c7306e1286e1024 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 01:49:00 +0200 Subject: [PATCH 15/38] Add further general documentation to PixmapPaint --- pixmappaint.d | 53 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index fbdcd9a..52e3326 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -15,6 +15,12 @@ API is subject to changes until further notice. ) + ### Colors + + Colors are stored in an RGBA format with 8 bit per channel. + See [arsd.color.Color|Pixel] for details. + + ### The coordinate system The top left corner of a pixmap is its $(B origin) `(0,0)`. @@ -30,6 +36,53 @@ ↓ x ``` + + + ### Pixmaps + + A [Pixmap] consist of two fields: + $(LIST + * a slice (of an array of [Pixel|Pixels]) + * a width + ) + + This design comes with many advantages. + First and foremost it brings simplicity. + + Pixel data buffers can be reused across pixmaps, + even when those have different sizes. + Simply slice the buffer to fit just enough pixels for the new pixmap. + + Memory management can also happen outside of the pixmap. + It is possible to use a buffer allocated elsewhere. (Such a one shouldn’t + be mixed with the built-in memory management facilities of the pixmap type. + Otherwise one will end up with GC-allocated copies.) + + The most important downside is that it makes pixmaps basically a reference + type. + + Copying a pixmap creates a shallow copy still poiting to the same pixel + data that is also used by the source pixmap. + This implies that manipulating the source pixels also manipulates the + pixels of the copy – and vice versa. + + The issues implied by this become an apparent when one of the references + modifies the pixel data in a way that also affects the dimensions of the + image; such as cropping. + + Pixmaps describe how pixel data stored in a 1-dimensional memory space is + meant to be interpreted as a 2-dimensional image. + + A notable implication of this 1D ↔ 2D mapping is, that slicing the 1D data + leads to non-sensical results in the 2D space when the 1D-slice is + reinterpreted as 2D-image. + + Especially slicing across scanlines (→ horizontal rows of an image) is + 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.) +/ module arsd.pixmappaint; From 9fdaf00197158421c3358704ea8c25fcaaebc85e Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 01:51:01 +0200 Subject: [PATCH 16/38] Fix the coordinate system diagram MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit To quote one of my IN(formation)SY(stems) teachers, “Sorry, this is just a copy-paste mistake.” --- pixmappaint.d | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index 52e3326..0a23cf6 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -34,7 +34,7 @@ ``` 0 → x ↓ - x + y ``` From 65ba2793cb1add82640743846a9f0624eca1ebe3 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 02:09:33 +0200 Subject: [PATCH 17/38] Move that back Why did I move this in the first place? --- pixmappaint.d | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 0a23cf6..c900e0b 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -434,6 +434,10 @@ struct SubPixmap { and potentially crash the program. ) + Returns: + A size-adjusted shallow copy of the input Pixmap overwritten + with the image data of the SubPixmap. + $(PITFALL While the returned Pixmap utilizes the buffer provided by the input, the returned Pixmap might not exactly match the input. @@ -449,10 +453,6 @@ struct SubPixmap { pixmap = subPixmap.extractToPixmap(pixmap); --- ) - - Returns: - A size-adjusted shallow copy of the input Pixmap overwritten - with the image data of the SubPixmap. +/ Pixmap extractToPixmap(Pixmap target) @nogc const { // Length adjustment From 1fbdcab94846f3ce4309acdfd8c37c1843101e2c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 03:23:36 +0200 Subject: [PATCH 18/38] Implement horizontal flipping --- pixmappaint.d | 51 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 51 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index c900e0b..2e73cee 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1763,6 +1763,57 @@ Pixmap rotateClockwiseNew(const Pixmap source) { return target; } +/++ + Flips an image horizontally. + + ``` + ╔═══╗ ╔═══╗ + ║!. ║ → ║ .!║ + ╚═══╝ ╚═══╝ + ``` + +/ +void flipHorizontally(const Pixmap source, Pixmap target) @nogc { + debug assert(source.size == target.size); + + auto src = PixmapScanner(source); + auto dst = PixmapScannerRW(target); + + foreach (srcLine; src) { + auto dstLine = dst.front; + foreach (idxSrc, px; srcLine) { + const idxDst = (dstLine.length - (idxSrc + 1)); + dstLine[idxDst] = px; + } + + dst.popFront(); + } +} + +/// ditto +Pixmap flipHorizontallyNew(const Pixmap source) { + auto target = Pixmap(source.size); + source.flipHorizontally(target); + return target; +} + +/// ditto +void flipHorizontallyInPlace(Pixmap source) @nogc { + auto scanner = PixmapScannerRW(source); + + foreach (line; scanner) { + const idxMiddle = (1 + (line.length >> 1)); + auto halfA = line[0 .. idxMiddle]; + + foreach (idxA, ref px; halfA) { + const idxB = (line.length - (idxA + 1)); + Pixel tmp = line[idxB]; + // swap + line[idxB] = px; + px = tmp; + } + } +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From db6b6d1f74bb2470155942822729e12c6f237174 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 03:24:22 +0200 Subject: [PATCH 19/38] Slightly improve docs and usability --- pixmappaint.d | 46 ++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 44 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 2e73cee..adff89c 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -343,10 +343,13 @@ struct Pixmap { /++ Retrieves a rectangular subimage of the pixmap. +/ - inout(SubPixmap) scan2D(Point pos, Size size) inout { + inout(SubPixmap) scanSubPixmap(Point pos, Size size) inout { return inout(SubPixmap)(this, size, pos); } + /// TODO: remove + deprecated alias scan2D = scanSubPixmap; + /++ Retrieves the first line of the Pixmap. @@ -357,6 +360,42 @@ struct Pixmap { return data[0 .. width]; } + public { + /++ + Provides access to a single pixel at the requested 2D-position. + + See_also: + Accessing pixels through the [data] array will be more useful, + usually. + +/ + ref inout(Pixel) accessPixel(Point pos) inout @system { + const idx = linearOffset(pos, this.width); + return this.data[idx]; + } + + /// ditto + Pixel getPixel(Point pos) const { + const idx = linearOffset(pos, this.width); + return this.data[idx]; + } + + /// ditto + Pixel getPixel(int x, int y) const { + return this.getPixel(Point(x, y)); + } + + /// ditto + void setPixel(Point pos, Pixel value) { + const idx = linearOffset(pos, this.width); + this.data[idx] = value; + } + + /// ditto + void setPixel(int x, int y, Pixel value) { + return this.setPixel(Point(x, y), value); + } + } + /// Clears the buffer’s contents (by setting each pixel to the same color) void clear(Pixel value) { data[] = value; @@ -1733,7 +1772,10 @@ Pixmap cropInPlace(Pixmap source, Size targetSize, Point offset = Point(0, 0)) @ $(PITFALL This function does not work in place. Do not attempt to pass Pixmaps sharing the same buffer for both source - and target. Such would lead to a bad result with heavy artifacts. + and target. Such would lead to bad results with heavy artifacts. + + Do not use the artifacts produced by this as a creative effect. + Those are an implementation detail. ) +/ void rotateClockwise(const Pixmap source, Pixmap target) @nogc { From a2f437d4ad7570f0d5c6ac1fdd0a3cef9eeedd0f Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 03:43:44 +0200 Subject: [PATCH 20/38] Make PixmapScanner types bidirectional ranges --- pixmappaint.d | 60 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 60 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index adff89c..3c9165a 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -919,6 +919,11 @@ struct PixmapScanner { _width = pixmap.width; } + /// + typeof(this) save() { + return this; + } + /// bool empty() const { return (_data.length == 0); @@ -933,6 +938,16 @@ struct PixmapScanner { void popFront() { _data = _data[_width .. $]; } + + /// + const(Pixel)[] back() const { + return _data[($ - _width) .. $]; + } + + /// + void popBack() { + _data = _data[0 .. ($ - _width)]; + } } /++ @@ -957,6 +972,11 @@ struct PixmapScannerRW { _width = pixmap.width; } + /// + typeof(this) save() { + return this; + } + /// bool empty() const { return (_data.length == 0); @@ -971,6 +991,16 @@ struct PixmapScannerRW { void popFront() { _data = _data[_width .. $]; } + + /// + Pixel[] back() { + return _data[($ - _width) .. $]; + } + + /// + void popBack() { + _data = _data[0 .. ($ - _width)]; + } } /++ @@ -994,6 +1024,11 @@ struct SubPixmapScanner { _feed = subPixmap.source.width; } + /// + typeof(this) save() { + return this; + } + /// bool empty() const { return (_data.length == 0); @@ -1013,6 +1048,16 @@ struct SubPixmapScanner { _data = _data[_feed .. $]; } + + /// + const(Pixel)[] back() const { + return _data[($ - _width) .. $]; + } + + /// + void popBack() { + _data = _data[0 .. ($ - _width)]; + } } /++ @@ -1039,6 +1084,11 @@ struct SubPixmapScannerRW { _feed = subPixmap.source.width; } + /// + typeof(this) save() { + return this; + } + /// bool empty() const { return (_data.length == 0); @@ -1058,6 +1108,16 @@ struct SubPixmapScannerRW { _data = _data[_feed .. $]; } + + /// + Pixel[] back() { + return _data[($ - _width) .. $]; + } + + /// + void popBack() { + _data = _data[0 .. ($ - _width)]; + } } /// From 582e47c13b32ebf55223818300e67af86fc006e5 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 04:06:31 +0200 Subject: [PATCH 21/38] Implement vertical flipping --- pixmappaint.d | 62 +++++++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 60 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 3c9165a..57a442d 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1870,7 +1870,7 @@ Pixmap rotateClockwiseNew(const Pixmap source) { ``` ╔═══╗ ╔═══╗ - ║!. ║ → ║ .!║ + ║#-.║ → ║.-#║ ╚═══╝ ╚═══╝ ``` +/ @@ -1908,7 +1908,7 @@ void flipHorizontallyInPlace(Pixmap source) @nogc { foreach (idxA, ref px; halfA) { const idxB = (line.length - (idxA + 1)); - Pixel tmp = line[idxB]; + const tmp = line[idxB]; // swap line[idxB] = px; px = tmp; @@ -1916,6 +1916,64 @@ void flipHorizontallyInPlace(Pixmap source) @nogc { } } +/++ + Flips an image vertically. + + ``` + ╔═══╗ ╔═══╗ + ║## ║ ║ -║ + ║ -║ → ║## ║ + ╚═══╝ ╚═══╝ + ``` + +/ +void flipVertically(const Pixmap source, Pixmap target) @nogc { + debug assert(source.size == target.size); + + auto src = PixmapScanner(source); + auto dst = PixmapScannerRW(target); + + foreach (srcLine; src) { + auto dstLine = dst.back; + foreach (idxSrc, px; srcLine) { + const idxDst = (dstLine.length - (idxSrc + 1)); + dstLine[idxDst] = px; + } + + dst.popBack(); + } +} + +/// ditto +Pixmap flipVerticallyNew(const Pixmap source) { + auto target = Pixmap(source.size); + source.flipVertically(target); + return target; +} + +/// ditto +void flipVerticallyInPlace(Pixmap source) { + auto scanner = PixmapScannerRW(source); + + while (!scanner.empty) { + auto a = scanner.front; + auto b = scanner.back; + + // middle line? (odd number of lines) + if (a.ptr is b.ptr) { + break; + } + + foreach (idx, ref pxA; a) { + const tmp = pxA; + pxA = b[idx]; + b[idx] = tmp; + } + + scanner.popFront(); + scanner.popBack(); + } +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From cf2e084a707c5ea8f8e1c19314713d3ac209f2a7 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 04:17:10 +0200 Subject: [PATCH 22/38] Fix flipVertically() --- pixmappaint.d | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 57a442d..0b5c9ef 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1933,12 +1933,7 @@ void flipVertically(const Pixmap source, Pixmap target) @nogc { auto dst = PixmapScannerRW(target); foreach (srcLine; src) { - auto dstLine = dst.back; - foreach (idxSrc, px; srcLine) { - const idxDst = (dstLine.length - (idxSrc + 1)); - dstLine[idxDst] = px; - } - + dst.back[] = srcLine[]; dst.popBack(); } } @@ -1951,7 +1946,7 @@ Pixmap flipVerticallyNew(const Pixmap source) { } /// ditto -void flipVerticallyInPlace(Pixmap source) { +void flipVerticallyInPlace(Pixmap source) @nogc { auto scanner = PixmapScannerRW(source); while (!scanner.empty) { From 8aaaa2a3c2bfa78db11c91bedac88e18c7cedf78 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 7 Oct 2024 04:50:28 +0200 Subject: [PATCH 23/38] =?UTF-8?q?Implement=20180=C2=B0=20rotation=20functi?= =?UTF-8?q?on?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pixmappaint.d | 52 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 0b5c9ef..72c6ce0 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1865,6 +1865,58 @@ Pixmap rotateClockwiseNew(const Pixmap source) { return target; } +/++ + Rotates an image by 180°. + +/ +void rotate180deg(const Pixmap source, Pixmap target) @nogc { + debug assert(source.size == target.size); + + // Technically, this is implemented as flip vertical + flip horizontal. + auto src = PixmapScanner(source); + auto dst = PixmapScannerRW(target); + + foreach (srcLine; src) { + auto dstLine = dst.back; + foreach (idxSrc, px; srcLine) { + const idxDst = (dstLine.length - (idxSrc + 1)); + dstLine[idxDst] = px; + } + dst.popBack(); + } +} + +/// ditto +Pixmap rotate180degNew(const Pixmap source) { + auto target = Pixmap(source.size); + source.rotate180deg(target); + return target; +} + +/// ditto +void rotate180degInPlace(Pixmap source) @nogc { + auto scanner = PixmapScannerRW(source); + + while (!scanner.empty) { + auto a = scanner.front; + auto b = scanner.back; + + // middle line? (odd number of lines) + if (a.ptr is b.ptr) { + break; + } + + foreach (idxSrc, ref pxA; a) { + const idxDst = (b.length - (idxSrc + 1)); + const tmp = pxA; + pxA = b[idxDst]; + b[idxDst] = tmp; + } + + scanner.popFront(); + scanner.popBack(); + } +} + /++ Flips an image horizontally. From 0bdea48b9b67d2ad816304b130920d59b8e7172d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 8 Oct 2024 00:21:53 +0200 Subject: [PATCH 24/38] Fix bidirectional features of SubPixmapScanner --- pixmappaint.d | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 72c6ce0..91ccefc 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -1056,7 +1056,12 @@ struct SubPixmapScanner { /// void popBack() { - _data = _data[0 .. ($ - _width)]; + if (_data.length < _feed) { + _data.length = 0; + return; + } + + _data = _data[0 .. ($ - _feed)]; } } @@ -1116,7 +1121,12 @@ struct SubPixmapScannerRW { /// void popBack() { - _data = _data[0 .. ($ - _width)]; + if (_data.length < _feed) { + _data.length = 0; + return; + } + + _data = _data[0 .. ($ - _feed)]; } } From 7d3908685781e7759390515ddbf48e323889a1c8 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Tue, 8 Oct 2024 00:22:35 +0200 Subject: [PATCH 25/38] Rename scanSubPixmap() to scanArea() --- pixmappaint.d | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 91ccefc..44a9825 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -343,12 +343,15 @@ struct Pixmap { /++ Retrieves a rectangular subimage of the pixmap. +/ - inout(SubPixmap) scanSubPixmap(Point pos, Size size) inout { + inout(SubPixmap) scanArea(Point pos, Size size) inout { return inout(SubPixmap)(this, size, pos); } /// TODO: remove - deprecated alias scan2D = scanSubPixmap; + deprecated alias scanSubPixmap = scanArea; + + /// TODO: remove + deprecated alias scan2D = scanArea; /++ Retrieves the first line of the Pixmap. From 31a247c4c9dc72ac7d5bbc15c66a44a532d2eed3 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 12 Oct 2024 23:00:39 +0200 Subject: [PATCH 26/38] Implement PixmapBlueprints --- pixmappaint.d | 91 +++++++++++++++++++++++++++++++++++++++++++++-- pixmappresenter.d | 2 +- 2 files changed, 89 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 44a9825..f61f128 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -151,6 +151,61 @@ static assert(Pixel.sizeof == uint.sizeof); } } +/++ + Meta data for the construction a Pixmap + +/ +struct PixmapBlueprint { + /++ + Total number of pixels stored in a Pixmap. + +/ + size_t length; + + /++ + Width of a Pixmap. + +/ + int width; + +@safe pure nothrow @nogc: + + /// + public static PixmapBlueprint fromSize(const Size size) { + return PixmapBlueprint( + size.area, + size.width, + ); + } + + /// + public static PixmapBlueprint fromPixmap(const Pixmap pixmap) { + return PixmapBlueprint( + pixmap.length, + pixmap.width, + ); + } + + /++ + Determines whether the blueprint is plausible. + +/ + bool isValid() const { + return ((length % width) == 0); + } + + /++ + Height of a Pixmap. + + See_also: + This is the counterpart to the dimension known as [width]. + +/ + int height() const { + return castTo!int(length / width); + } + + /// + Size size() const { + return Size(width, height); + } +} + /++ Pixel data container +/ @@ -165,11 +220,13 @@ struct Pixmap { @safe pure nothrow: /// + deprecated("Do `Pixmap.makeNew(size)` instead.") this(Size size) { this.size = size; } /// + deprecated("Do `Pixmap.makeNew(Size(width, height))` instead.") this(int width, int height) in (width > 0) in (height > 0) { @@ -183,6 +240,17 @@ struct Pixmap { this.width = width; } + /// + static Pixmap makeNew(PixmapBlueprint blueprint) { + auto data = new Pixel[](blueprint.length); + return Pixmap(data, blueprint.width); + } + + /// + static Pixmap makeNew(Size size) { + return Pixmap.makeNew(PixmapBlueprint.fromSize(size)); + } + /++ Creates a $(I deep clone) of the Pixmap +/ @@ -215,7 +283,7 @@ struct Pixmap { --- ) +/ - Pixmap copyTo(Pixmap target) const { + Pixmap copyTo(Pixmap target) @nogc const { // Length adjustment const l = this.length; if (target.data.length < l) { @@ -229,7 +297,7 @@ struct Pixmap { return target; } - private void copyToImpl(Pixmap target) const { + private void copyToImpl(Pixmap target) @nogc const { target.data[] = this.data[]; } @@ -308,6 +376,23 @@ struct Pixmap { return (width * int(Pixel.sizeof)); } + /++ + Adjusts the Pixmap according to the provided blueprint. + + The blueprint must not be larger than the data buffer of the pixmap. + + This function does not reallocate the pixel data buffer. + + If the blueprint is larger than the data buffer of the pixmap, + this will result in a bounds-check error if applicable. + +/ + void adjustTo(PixmapBlueprint blueprint) { + debug assert(this.data.length >= blueprint.length); + debug assert(blueprint.isValid); + this.data = this.data[0 .. blueprint.length]; + this.width = blueprint.width; + } + /++ Calculates the index (linear offset) of the requested position within the pixmap data. @@ -461,7 +546,7 @@ struct SubPixmap { Use [extractToPixmap] for a non-allocating variant with an . +/ Pixmap extractToNewPixmap() const { - auto pm = Pixmap(size); + auto pm = Pixmap.makeNew(size); this.extractToPixmap(pm); return pm; } diff --git a/pixmappresenter.d b/pixmappresenter.d index fa9269a..d6218b1 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -902,7 +902,7 @@ final class PixmapPresenter { _renderer = renderer; // create software framebuffer - auto framebuffer = Pixmap(config.renderer.resolution); + auto framebuffer = Pixmap.makeNew(config.renderer.resolution); // OpenGL? auto openGlOptions = OpenGlOptions.no; From dea6de12e06722c7541c5f33679ac7e8ef68d489 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 12 Oct 2024 23:02:27 +0200 Subject: [PATCH 27/38] Improve and add image manipulation functions --- pixmappaint.d | 400 ++++++++++++++++++++++++++++++++++++++++++-------- 1 file changed, 338 insertions(+), 62 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index f61f128..b4c81e2 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -37,6 +37,11 @@ y ``` + Furthermore, $(B length) refers to the areal size of a pixmap. + It represents the total number of pixels in a pixmap. + It follows from the foregoing that the term $(I long) usually refers to + the length (not the width). + ### Pixmaps @@ -83,6 +88,173 @@ (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.) + + + ### Image manipulation + + The term “image manipulation function” here refers to functions that + manipulate (e.g. transform) an image as a whole. + + Image manipulation functions in this library are provided in up to three + flavors: + + $(LIST + * a “source to target” function + * a “source to newly allocated target” wrapper + * $(I optionally) an “in-place” adaption + ) + + Additionally, a “compute dimensions of target” function is provided. + + #### Source to Target + + The regular “source to target” function takes (at least) two parameters: + A source [Pixmap] and a target [Pixmap]. + + (Additional operation-specific arguments may be required as well.) + + The target pixmap usually needs to be able to fit at least the same number + of pixels as the source holds. + Use the corresponding “compute size of target function” to calculate the + required size when needed. + (A notable exception would be cropping, where to target pixmap must be only + at least long enough to hold the area of the size to crop to.) + + The data stored in the buffer of the target pixmap is overwritten by the + operation. + + A modified Pixmap structure with adjusted dimensions is returned. + + These functions are named plain and simple after the respective operation + they perform; e.g. [flipHorizontally] or [crop]. + + --- + // Allocate a new target Pixmap. + Pixmap target = Pixmap.makeNew( + flipHorizontallyCalcDims(sourceImage) + ); + + // Flip the image horizontally and store the updated structure. + // (Note: As a horizontal flip does not affect the dimensions of a Pixmap, + // storing the updated structure would not be necessary + // in this specific scenario.) + target = sourceImage.flipHorizontally(target); + --- + + --- + const cropOffset = Point(0, 0); + const cropSize = Size(100, 100); + + // Allocate a new target Pixmap. + Pixmap target = Pixmap.makeNew( + cropCalcDims(sourceImage, cropSize, cropOffset) + ); + + // Crop the Pixmap. + target = sourceImage.crop(target, cropSize, cropOffset); + --- + + $(PITFALL + “Source to target” functions do not work in place. + Do not attempt to pass Pixmaps sharing the same buffer for both source + and target. Such would lead to bad results with heavy artifacts. + + Use the “in-place” variant of the operation instead. + + Moreover: + Do not use the artifacts produced by this as a creative effect. + Those are an implementation detail (and may change at any point). + ) + + #### Source to New Target + + The “source to newly allocated target” wrapper allocates a new buffer to + hold the manipulated target. + + These wrappers are provided for user convenience. + + They are identified by the suffix `-New` that is appended to the name of + the corresponding “source to target” function; + e.g. [flipHorizontallyNew] or [cropNew]. + + --- + // Create a new flipped Pixmap. + Pixmap target = sourceImage.flipHorizontallyNew(); + --- + + --- + const cropOffset = Point(0, 0); + const cropSize = Size(100, 100); + + // Create a new cropped Pixmap. + Pixmap target = sourceImage.cropNew(cropSize, cropOffset); + --- + + #### In-Place + + For selected image manipulation functions a special adaption is provided + that stores the result in the source pixel data buffer. + + 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. + 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 + be performed in-place. + + Certain in-place manipulation functions return a shallow-copy of the + source structure with dimensions adjusted accordingly. + This is behavior is not streamlined consistently as the lack of an + in-place option for certain operations makes them a special case anyway. + + These function are suffixed with `-InPlace`; + e.g. [flipHorizontallyInPlace] or [cropInPlace]. + + $(TIP + Manipulating the source image directly can lead to unexpected results + when the source image is used in multiple places. + ) + + $(NOTE + Users are usually better off to utilize the regular “source to target” + functions with a reused pixel data buffer. + + These functions do not serve as a performance optimization. + Some of them might perform significantly worse than their regular + variant. Always benchmark and profile. + ) + + --- + image.flipHorizontallyInPlace(); + --- + + --- + const cropOffset = Point(0, 0); + const cropSize = Size(100, 100); + + image = image.cropInPlace(cropSize, cropOffset); + --- + + + #### Compute size of target + + Functions to “compute (the) dimensions of (a) target” are primarily meant + to be utilized to calculate the size for allocating new pixmaps to be used + as a target for manipulation functions. + + They are provided for all manipulation functions even in cases where they + are provide little to no benefit. This is for consistency and to ease + development. + + Such functions are identified by a `-CalcDims` suffix; + e.g. [flipHorizontallyCalcDims] or [cropCalcDims]. + + They receive the same parameters as their corresponding “source to new + target” function. For consistency reasons this also applies in cases where + certain parameters are irrelevant for the computation of the target size. +/ module arsd.pixmappaint; @@ -1892,53 +2064,88 @@ void invert(Pixmap pixmap) @nogc { The size of the area to crop 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 = crop(source, target, target.size, offset); + + // → Instead do: + cropTo(source, target, offset); + --- +/ -void crop(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nogc { +void cropTo(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nogc { auto src = const(SubPixmap)(source, target.size, offset); src.extractToPixmapCopyImpl(target); } /++ - Crops an image and stores the result in a newly allocated Pixmap. + Crops an image to the provided size with the requested offset. + + The target Pixmap must be big enough in length to hold the cropped image. +/ -Pixmap cropNew(const Pixmap source, Size targetSize, Point offset = Point(0, 0)) { - auto target = Pixmap(targetSize); - crop(source, target, offset); +Pixmap crop(const Pixmap source, Pixmap target, Size cropToSize, Point offset = Point(0, 0)) @nogc { + target.adjustTo(source.cropCalcDims(cropToSize, offset)); + cropTo(source, target, offset); return target; } -/++ - Crops an image and stores the result in the source buffer. +/// ditto +Pixmap cropNew(const Pixmap source, Size cropToSize, Point offset = Point(0, 0)) { + auto target = Pixmap.makeNew(cropToSize); + cropTo(source, target, offset); + return target; +} - The source pixmap structure is passed by value. - A size-adjusted structure using a slice of the same underlying memory is - returned. - +/ -Pixmap cropInPlace(Pixmap source, Size targetSize, Point offset = Point(0, 0)) @nogc { +/// ditto +Pixmap cropInPlace(Pixmap source, Size cropToSize, Point offset = Point(0, 0)) @nogc { Pixmap target = source; - target.width = targetSize.width; - target.data = target.data[0 .. targetSize.area]; + target.width = cropToSize.width; + target.data = target.data[0 .. cropToSize.area]; - auto src = const(SubPixmap)(source, targetSize, offset); + auto src = const(SubPixmap)(source, cropToSize, offset); src.extractToPixmapCopyPixelByPixelImpl(target); return target; } +/// ditto +PixmapBlueprint cropCalcDims(const Pixmap source, Size cropToSize, Point offset = Point(0, 0)) @nogc { + return PixmapBlueprint.fromSize(cropToSize); +} + +private void transposeTo(const Pixmap source, Pixmap target) @nogc { + foreach (y; 0 .. target.width) { + foreach (x; 0 .. source.width) { + const idxSrc = linearOffset(Point(x, y), source.width); + const idxDst = linearOffset(Point(y, x), target.width); + + target.data[idxDst] = source.data[idxSrc]; + } + } +} + /++ - Rotates an image by 90° clockwise. - - $(PITFALL - This function does not work in place. - Do not attempt to pass Pixmaps sharing the same buffer for both source - and target. Such would lead to bad results with heavy artifacts. - - Do not use the artifacts produced by this as a creative effect. - Those are an implementation detail. - ) + Transposes an image. +/ -void rotateClockwise(const Pixmap source, Pixmap target) @nogc { - debug assert(source.data.length == target.data.length); +Pixmap transpose(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.transposeCalcDims()); + source.transposeTo(target); + return target; +} +/// ditto +Pixmap transposeNew(const Pixmap source) { + auto target = Pixmap.makeNew(source.transposeCalcDims()); + source.transposeTo(target); + return target; +} + +/// ditto +PixmapBlueprint transposeCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint(source.length, source.height); +} + +private void rotateClockwiseTo(const Pixmap source, Pixmap target) @nogc { const area = source.data.length; const rowLength = source.size.height; ptrdiff_t cursor = -1; @@ -1955,20 +2162,53 @@ void rotateClockwise(const Pixmap source, Pixmap target) @nogc { /++ Rotates an image by 90° clockwise. - Stores the result in a newly allocated Pixmap. +/ -Pixmap rotateClockwiseNew(const Pixmap source) { - auto target = Pixmap(Size(source.height, source.width)); - source.rotateClockwise(target); +Pixmap rotateClockwise(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.rotateClockwiseCalcDims()); + source.rotateClockwiseTo(target); return target; } -/++ - Rotates an image by 180°. - +/ -void rotate180deg(const Pixmap source, Pixmap target) @nogc { - debug assert(source.size == target.size); +/// ditto +Pixmap rotateClockwiseNew(const Pixmap source) { + auto target = Pixmap.makeNew(source.rotateClockwiseCalcDims()); + source.rotateClockwiseTo(target); + return target; +} +/// ditto +PixmapBlueprint rotateClockwiseCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint(source.length, source.height); +} + +private void rotateCounterClockwiseTo(const Pixmap source, Pixmap target) @nogc { + // TODO: can this be optimized? + target = transpose(source, target); + target.flipVerticallyInPlace(); +} + +/++ + Rotates an image by 90° counter-clockwise. + +/ +Pixmap rotateCounterClockwise(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.rotateCounterClockwiseCalcDims()); + source.rotateCounterClockwiseTo(target); + return target; +} + +/// ditto +Pixmap rotateCounterClockwiseNew(const Pixmap source) { + auto target = Pixmap.makeNew(source.rotateCounterClockwiseCalcDims()); + source.rotateCounterClockwiseTo(target); + return target; +} + +/// ditto +PixmapBlueprint rotateCounterClockwiseCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint(source.length, source.height); +} + +private void rotate180degTo(const Pixmap source, Pixmap target) @nogc { // Technically, this is implemented as flip vertical + flip horizontal. auto src = PixmapScanner(source); auto dst = PixmapScannerRW(target); @@ -1983,10 +2223,19 @@ void rotate180deg(const Pixmap source, Pixmap target) @nogc { } } +/++ + Rotates an image by 180°. + +/ +Pixmap rotate180deg(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.rotate180degCalcDims()); + source.rotate180degTo(target); + return target; +} + /// ditto Pixmap rotate180degNew(const Pixmap source) { - auto target = Pixmap(source.size); - source.rotate180deg(target); + auto target = Pixmap.makeNew(source.size); + source.rotate180degTo(target); return target; } @@ -1994,6 +2243,9 @@ Pixmap rotate180degNew(const Pixmap source) { void rotate180degInPlace(Pixmap source) @nogc { auto scanner = PixmapScannerRW(source); + // Technically, this is implemented as a flip vertical + flip horizontal + // combo, i.e. the image is flipped vertically line by line, but the lines + // are overwritten in a horizontally flipped way. while (!scanner.empty) { auto a = scanner.front; auto b = scanner.back; @@ -2015,18 +2267,12 @@ void rotate180degInPlace(Pixmap source) @nogc { } } -/++ - Flips an image horizontally. - - ``` - ╔═══╗ ╔═══╗ - ║#-.║ → ║.-#║ - ╚═══╝ ╚═══╝ - ``` - +/ -void flipHorizontally(const Pixmap source, Pixmap target) @nogc { - debug assert(source.size == target.size); +/// +PixmapBlueprint rotate180degCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint.fromPixmap(source); +} +private void flipHorizontallyTo(const Pixmap source, Pixmap target) @nogc { auto src = PixmapScanner(source); auto dst = PixmapScannerRW(target); @@ -2041,10 +2287,25 @@ void flipHorizontally(const Pixmap source, Pixmap target) @nogc { } } +/++ + Flips an image horizontally. + + ``` + ╔═══╗ ╔═══╗ + ║#-.║ → ║.-#║ + ╚═══╝ ╚═══╝ + ``` + +/ +Pixmap flipHorizontally(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.flipHorizontallyCalcDims()); + source.flipHorizontallyTo(target); + return target; +} + /// ditto Pixmap flipHorizontallyNew(const Pixmap source) { - auto target = Pixmap(source.size); - source.flipHorizontally(target); + auto target = Pixmap.makeNew(source.size); + source.flipHorizontallyTo(target); return target; } @@ -2066,6 +2327,21 @@ void flipHorizontallyInPlace(Pixmap source) @nogc { } } +/// ditto +PixmapBlueprint flipHorizontallyCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint.fromPixmap(source); +} + +private void flipVerticallyTo(const Pixmap source, Pixmap target) @nogc { + auto src = PixmapScanner(source); + auto dst = PixmapScannerRW(target); + + foreach (srcLine; src) { + dst.back[] = srcLine[]; + dst.popBack(); + } +} + /++ Flips an image vertically. @@ -2076,22 +2352,17 @@ void flipHorizontallyInPlace(Pixmap source) @nogc { ╚═══╝ ╚═══╝ ``` +/ -void flipVertically(const Pixmap source, Pixmap target) @nogc { - debug assert(source.size == target.size); +Pixmap flipVertically(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.flipVerticallyCalcDims()); - auto src = PixmapScanner(source); - auto dst = PixmapScannerRW(target); - - foreach (srcLine; src) { - dst.back[] = srcLine[]; - dst.popBack(); - } + flipVerticallyTo(source, target); + return target; } /// ditto Pixmap flipVerticallyNew(const Pixmap source) { - auto target = Pixmap(source.size); - source.flipVertically(target); + auto target = Pixmap.makeNew(source.flipVerticallyCalcDims()); + source.flipVerticallyTo(target); return target; } @@ -2119,6 +2390,11 @@ void flipVerticallyInPlace(Pixmap source) @nogc { } } +/// ditto +PixmapBlueprint flipVerticallyCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint.fromPixmap(source); +} + @safe pure nothrow @nogc: // ==== Blending functions ==== From ebdfcaf799749da919abf44b865e57b07c98d0a8 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 12 Oct 2024 23:04:46 +0200 Subject: [PATCH 28/38] Add further documentation --- pixmappaint.d | 45 +++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 43 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index b4c81e2..01df2d8 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -15,6 +15,47 @@ API is subject to changes until further notice. ) + Pixmap refers to raster graphics, a subset of “bitmap” graphics. + A pixmap is an array of pixels and the corresponding meta data to describe + how an image if formed from those pixels. + In the case of this library, a “width” field is used to map a specified + number of pixels to a row of an image. + + ``` + pixels := [ 0, 1, 2, 3 ] + width := 2 + + pixmap(pixels, width) + => [ + [ 0, 1 ] + [ 2, 3 ] + ] + ``` + + ``` + pixels := [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 ] + width := 3 + + pixmap(pixels, width) + => [ + [ 0, 1, 2 ] + [ 3, 4, 5 ] + [ 6, 7, 8 ] + [ 9, 10, 11 ] + ] + ``` + + ``` + pixels := [ 0, 1, 2, 3, 4, 5, 6, 7 ] + width := 4 + + pixmap(pixels, width) + => [ + [ 0, 1, 2, 3 ] + [ 4, 5, 6, 7 ] + ] + ``` + ### Colors Colors are stored in an RGBA format with 8 bit per channel. @@ -63,8 +104,8 @@ be mixed with the built-in memory management facilities of the pixmap type. Otherwise one will end up with GC-allocated copies.) - The most important downside is that it makes pixmaps basically a reference - type. + The most important downside is that it makes pixmaps basically a partial + reference type. Copying a pixmap creates a shallow copy still poiting to the same pixel data that is also used by the source pixmap. From 75f77176d7762754d603d609973a738892230ad8 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 12 Oct 2024 23:19:00 +0200 Subject: [PATCH 29/38] Improve PixmapPaint docs --- pixmappaint.d | 32 ++++++++++++++++++++++++++++++-- 1 file changed, 30 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 01df2d8..909657f 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -433,13 +433,13 @@ struct Pixmap { @safe pure nothrow: /// - deprecated("Do `Pixmap.makeNew(size)` instead.") + deprecated("Use `Pixmap.makeNew(size)` instead.") this(Size size) { this.size = size; } /// - deprecated("Do `Pixmap.makeNew(Size(width, height))` instead.") + deprecated("Use `Pixmap.makeNew(Size(width, height))` instead.") this(int width, int height) in (width > 0) in (height > 0) { @@ -2167,6 +2167,13 @@ private void transposeTo(const Pixmap source, Pixmap target) @nogc { /++ Transposes an image. + + ``` + ╔══╗ ╔══╗ + ║# ║ ║#+║ + ║+x║ → ║ x║ + ╚══╝ ╚══╝ + ``` +/ Pixmap transpose(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.transposeCalcDims()); @@ -2203,6 +2210,13 @@ private void rotateClockwiseTo(const Pixmap source, Pixmap target) @nogc { /++ Rotates an image by 90° clockwise. + + ``` + ╔══╗ ╔══╗ + ║# ║ ║+#║ + ║+x║ → ║x ║ + ╚══╝ ╚══╝ + ``` +/ Pixmap rotateClockwise(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotateClockwiseCalcDims()); @@ -2230,6 +2244,13 @@ private void rotateCounterClockwiseTo(const Pixmap source, Pixmap target) @nogc /++ Rotates an image by 90° counter-clockwise. + + ``` + ╔══╗ ╔══╗ + ║# ║ ║ x║ + ║+x║ → ║#+║ + ╚══╝ ╚══╝ + ``` +/ Pixmap rotateCounterClockwise(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotateCounterClockwiseCalcDims()); @@ -2266,6 +2287,13 @@ private void rotate180degTo(const Pixmap source, Pixmap target) @nogc { /++ Rotates an image by 180°. + + ``` + ╔═══╗ ╔═══╗ + ║#- ║ ║%~~║ + ║~~%║ → ║ -#║ + ╚═══╝ ╚═══╝ + ``` +/ Pixmap rotate180deg(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotate180degCalcDims()); From c21d14664c329057b096c6ddf49e0458541182f0 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 17:52:57 +0200 Subject: [PATCH 30/38] Refactor invert() to match other image manip functions --- pixmappaint.d | 28 +++++++++++++++++++++++++++- 1 file changed, 27 insertions(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index 909657f..ab7d2f2 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2087,6 +2087,13 @@ Pixel invert(const Pixel color) @nogc { ); } +private void invertTo(const Pixmap source, Pixmap target) @trusted @nogc { + debug assert(source.length == target.length); + foreach (idx, ref px; target.data) { + px = invert(source.data.ptr[idx]); + } +} + /++ Inverts all colors to produce a $(B negative image). @@ -2094,12 +2101,31 @@ Pixel invert(const Pixel color) @nogc { Develops a positive image when applied to a negative one. ) +/ -void invert(Pixmap pixmap) @nogc { +Pixmap invert(const Pixmap source, Pixmap target) @nogc { + target.adjustTo(source.invertCalcDims()); + source.invertTo(target); + return target; +} + +/// ditto +Pixmap invertNew(const Pixmap source) { + auto target = Pixmap.makeNew(source.invertCalcDims()); + source.invertTo(target); + return target; +} + +/// ditto +void invertInPlace(Pixmap pixmap) @nogc { foreach (ref px; pixmap.data) { px = invert(px); } } +/// ditto +PixmapBlueprint invertCalcDims(const Pixmap source) @nogc { + return PixmapBlueprint.fromPixmap(source); +} + /++ Crops an image and stores the result in the provided target Pixmap. From ac7f4c9889ffae063aa5bd9d517e971ec6a09218 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 18:47:06 +0200 Subject: [PATCH 31/38] Refactor opacity() to decreaseOpacity() & friends --- pixmappaint.d | 120 +++++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 104 insertions(+), 16 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index ab7d2f2..53c5e85 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -310,6 +310,7 @@ private float roundImpl(float f) { // `pure` rounding function. // std.math.round() isn’t pure on all targets. +// → private float round(float f) pure @nogc nothrow @trusted { return (castTo!(float function(float) pure @nogc nothrow)(&roundImpl))(f); } @@ -2042,37 +2043,124 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) @nogc { } } +/// +ubyte percentageDecimalToUInt8(const float decimal) @nogc +in (decimal >= 0) +in (decimal <= 1) { + return round(decimal * 255).castTo!ubyte; +} + +/// +float percentageUInt8ToDecimal(const ubyte n255ths) @nogc { + return (float(n255ths) / 255.0f); +} + // ==== Image manipulation functions ==== /++ - Sets the opacity of a [Pixmap]. + Lowers the opacity of a Pixel. - This lossy operation updates the alpha-channel value of each pixel. - → `alpha *= opacity` + This function multiplies the opacity of the input + with the given percentage. See_Also: - Use [opacityF] with opacity values in percent (%). + Use [decreaseOpacityF] with decimal opacity values in percent (%). +/ -void opacity(Pixmap pixmap, const ubyte opacity) @nogc { - foreach (ref px; pixmap.data) { - px.a = opacity.n255thsOf(px.a); - } +Pixel decreaseOpacity(const Pixel source, ubyte opacityPercentage) @nogc { + return Pixel( + source.r, + source.g, + source.b, + opacityPercentage.n255thsOf(source.a), + ); } /++ - Sets the opacity of a [Pixmap]. + Lowers the opacity of a Pixel. - This lossy operation updates the alpha-channel value of each pixel. - → `alpha *= opacity` + This function multiplies the opacity of the input + with the given percentage. + + Value Range: + 0.0 = 0% + 1.0 = 100% See_Also: Use [opacity] with 8-bit integer opacity values (in 255ths). +/ -void opacityF(Pixmap pixmap, const float opacity) @nogc -in (opacity >= 0) -in (opacity <= 1.0) { - immutable opacity255 = round(opacity * 255).castTo!ubyte; - pixmap.opacity = opacity255; +Pixel decreaseOpacityF(const Pixel source, float opacityPercentage) @nogc { + return decreaseOpacity(source, percentageDecimalToUInt8(opacityPercentage)); +} + +// Don’t get fooled by the name of this function. +// It’s called like that for consistency reasons. +private void decreaseOpacityTo(const Pixmap source, Pixmap target, ubyte opacityPercentage) @trusted @nogc { + debug assert(source.data.length == target.data.length); + foreach (idx, ref px; target.data) { + px = decreaseOpacity(source.data.ptr[idx], opacityPercentage); + } +} + +/++ + Lowers the opacity of a [Pixmap]. + + This operation updates the alpha-channel value of each pixel. + → `alpha *= opacity` + + See_Also: + Use [decreaseOpacityF] with decimal opacity values in percent (%). + +/ +Pixmap decreaseOpacity(const Pixmap source, Pixmap target, ubyte opacityPercentage) @nogc { + target.adjustTo(source.decreaseOpacityCalcDims(opacityPercentage)); + source.decreaseOpacityTo(target, opacityPercentage); + return target; +} + +/// ditto +Pixmap decreaseOpacityNew(const Pixmap source, ubyte opacityPercentage) { + auto target = Pixmap.makeNew(source.decreaseOpacityCalcDims(opacityPercentage)); + source.decreaseOpacityTo(target, opacityPercentage); + return target; +} + +/// ditto +void decreaseOpacityInPlace(Pixmap source, ubyte opacityPercentage) @nogc { + foreach (ref px; source.data) { + px.a = opacityPercentage.n255thsOf(px.a); + } +} + +/// ditto +PixmapBlueprint decreaseOpacityCalcDims(const Pixmap source, ubyte opacity) @nogc { + return PixmapBlueprint.fromPixmap(source); +} + +/++ + Adjusts the opacity of a [Pixmap]. + + This operation updates the alpha-channel value of each pixel. + → `alpha *= opacity` + + See_Also: + Use [decreaseOpacity] with 8-bit integer opacity values (in 255ths). + +/ +Pixmap decreaseOpacityF(const Pixmap source, Pixmap target, float opacityPercentage) @nogc { + return source.decreaseOpacity(target, percentageDecimalToUInt8(opacityPercentage)); +} + +/// ditto +Pixmap decreaseOpacityFNew(const Pixmap source, float opacityPercentage) { + return source.decreaseOpacityNew(percentageDecimalToUInt8(opacityPercentage)); +} + +/// ditto +void decreaseOpacityFInPlace(Pixmap source, const float opacityPercentage) @nogc { + return source.decreaseOpacityInPlace(percentageDecimalToUInt8(opacityPercentage)); +} + +/// ditto +PixmapBlueprint decreaseOpacityF(Pixmap source, const float opacityPercentage) @nogc { + return PixmapBlueprint.fromPixmap(source); } /++ From c23f7116c7f79fc1520e0359f5a25dea68677339 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 18:55:43 +0200 Subject: [PATCH 32/38] =?UTF-8?q?Rename=20"=E2=80=A6To"=20image=20manipula?= =?UTF-8?q?tion=20functions=20to=20"=E2=80=A6Into"?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pixmappaint.d | 56 ++++++++++++++++++++++++++------------------------- 1 file changed, 29 insertions(+), 27 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 53c5e85..8ff33cf 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2094,7 +2094,7 @@ Pixel decreaseOpacityF(const Pixel source, float opacityPercentage) @nogc { // Don’t get fooled by the name of this function. // It’s called like that for consistency reasons. -private void decreaseOpacityTo(const Pixmap source, Pixmap target, ubyte opacityPercentage) @trusted @nogc { +private void decreaseOpacityInto(const Pixmap source, Pixmap target, ubyte opacityPercentage) @trusted @nogc { debug assert(source.data.length == target.data.length); foreach (idx, ref px; target.data) { px = decreaseOpacity(source.data.ptr[idx], opacityPercentage); @@ -2112,14 +2112,14 @@ private void decreaseOpacityTo(const Pixmap source, Pixmap target, ubyte opacity +/ Pixmap decreaseOpacity(const Pixmap source, Pixmap target, ubyte opacityPercentage) @nogc { target.adjustTo(source.decreaseOpacityCalcDims(opacityPercentage)); - source.decreaseOpacityTo(target, opacityPercentage); + source.decreaseOpacityInto(target, opacityPercentage); return target; } /// ditto Pixmap decreaseOpacityNew(const Pixmap source, ubyte opacityPercentage) { auto target = Pixmap.makeNew(source.decreaseOpacityCalcDims(opacityPercentage)); - source.decreaseOpacityTo(target, opacityPercentage); + source.decreaseOpacityInto(target, opacityPercentage); return target; } @@ -2175,7 +2175,7 @@ Pixel invert(const Pixel color) @nogc { ); } -private void invertTo(const Pixmap source, Pixmap target) @trusted @nogc { +private void invertInto(const Pixmap source, Pixmap target) @trusted @nogc { debug assert(source.length == target.length); foreach (idx, ref px; target.data) { px = invert(source.data.ptr[idx]); @@ -2191,14 +2191,14 @@ private void invertTo(const Pixmap source, Pixmap target) @trusted @nogc { +/ Pixmap invert(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.invertCalcDims()); - source.invertTo(target); + source.invertInto(target); return target; } /// ditto Pixmap invertNew(const Pixmap source) { auto target = Pixmap.makeNew(source.invertCalcDims()); - source.invertTo(target); + source.invertInto(target); return target; } @@ -2234,6 +2234,9 @@ void cropTo(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nog src.extractToPixmapCopyImpl(target); } +// consistency +private alias cropInto = cropTo; + /++ Crops an image to the provided size with the requested offset. @@ -2241,14 +2244,14 @@ void cropTo(const Pixmap source, Pixmap target, Point offset = Point(0, 0)) @nog +/ Pixmap crop(const Pixmap source, Pixmap target, Size cropToSize, Point offset = Point(0, 0)) @nogc { target.adjustTo(source.cropCalcDims(cropToSize, offset)); - cropTo(source, target, offset); + cropInto(source, target, offset); return target; } /// ditto Pixmap cropNew(const Pixmap source, Size cropToSize, Point offset = Point(0, 0)) { auto target = Pixmap.makeNew(cropToSize); - cropTo(source, target, offset); + cropInto(source, target, offset); return target; } @@ -2268,7 +2271,7 @@ PixmapBlueprint cropCalcDims(const Pixmap source, Size cropToSize, Point offset return PixmapBlueprint.fromSize(cropToSize); } -private void transposeTo(const Pixmap source, Pixmap target) @nogc { +private void transposeInto(const Pixmap source, Pixmap target) @nogc { foreach (y; 0 .. target.width) { foreach (x; 0 .. source.width) { const idxSrc = linearOffset(Point(x, y), source.width); @@ -2291,14 +2294,14 @@ private void transposeTo(const Pixmap source, Pixmap target) @nogc { +/ Pixmap transpose(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.transposeCalcDims()); - source.transposeTo(target); + source.transposeInto(target); return target; } /// ditto Pixmap transposeNew(const Pixmap source) { auto target = Pixmap.makeNew(source.transposeCalcDims()); - source.transposeTo(target); + source.transposeInto(target); return target; } @@ -2307,7 +2310,7 @@ PixmapBlueprint transposeCalcDims(const Pixmap source) @nogc { return PixmapBlueprint(source.length, source.height); } -private void rotateClockwiseTo(const Pixmap source, Pixmap target) @nogc { +private void rotateClockwiseInto(const Pixmap source, Pixmap target) @nogc { const area = source.data.length; const rowLength = source.size.height; ptrdiff_t cursor = -1; @@ -2334,14 +2337,14 @@ private void rotateClockwiseTo(const Pixmap source, Pixmap target) @nogc { +/ Pixmap rotateClockwise(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotateClockwiseCalcDims()); - source.rotateClockwiseTo(target); + source.rotateClockwiseInto(target); return target; } /// ditto Pixmap rotateClockwiseNew(const Pixmap source) { auto target = Pixmap.makeNew(source.rotateClockwiseCalcDims()); - source.rotateClockwiseTo(target); + source.rotateClockwiseInto(target); return target; } @@ -2350,7 +2353,7 @@ PixmapBlueprint rotateClockwiseCalcDims(const Pixmap source) @nogc { return PixmapBlueprint(source.length, source.height); } -private void rotateCounterClockwiseTo(const Pixmap source, Pixmap target) @nogc { +private void rotateCounterClockwiseInto(const Pixmap source, Pixmap target) @nogc { // TODO: can this be optimized? target = transpose(source, target); target.flipVerticallyInPlace(); @@ -2368,14 +2371,14 @@ private void rotateCounterClockwiseTo(const Pixmap source, Pixmap target) @nogc +/ Pixmap rotateCounterClockwise(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotateCounterClockwiseCalcDims()); - source.rotateCounterClockwiseTo(target); + source.rotateCounterClockwiseInto(target); return target; } /// ditto Pixmap rotateCounterClockwiseNew(const Pixmap source) { auto target = Pixmap.makeNew(source.rotateCounterClockwiseCalcDims()); - source.rotateCounterClockwiseTo(target); + source.rotateCounterClockwiseInto(target); return target; } @@ -2384,7 +2387,7 @@ PixmapBlueprint rotateCounterClockwiseCalcDims(const Pixmap source) @nogc { return PixmapBlueprint(source.length, source.height); } -private void rotate180degTo(const Pixmap source, Pixmap target) @nogc { +private void rotate180degInto(const Pixmap source, Pixmap target) @nogc { // Technically, this is implemented as flip vertical + flip horizontal. auto src = PixmapScanner(source); auto dst = PixmapScannerRW(target); @@ -2411,14 +2414,14 @@ private void rotate180degTo(const Pixmap source, Pixmap target) @nogc { +/ Pixmap rotate180deg(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.rotate180degCalcDims()); - source.rotate180degTo(target); + source.rotate180degInto(target); return target; } /// ditto Pixmap rotate180degNew(const Pixmap source) { auto target = Pixmap.makeNew(source.size); - source.rotate180degTo(target); + source.rotate180degInto(target); return target; } @@ -2455,7 +2458,7 @@ PixmapBlueprint rotate180degCalcDims(const Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } -private void flipHorizontallyTo(const Pixmap source, Pixmap target) @nogc { +private void flipHorizontallyInto(const Pixmap source, Pixmap target) @nogc { auto src = PixmapScanner(source); auto dst = PixmapScannerRW(target); @@ -2481,14 +2484,14 @@ private void flipHorizontallyTo(const Pixmap source, Pixmap target) @nogc { +/ Pixmap flipHorizontally(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.flipHorizontallyCalcDims()); - source.flipHorizontallyTo(target); + source.flipHorizontallyInto(target); return target; } /// ditto Pixmap flipHorizontallyNew(const Pixmap source) { auto target = Pixmap.makeNew(source.size); - source.flipHorizontallyTo(target); + source.flipHorizontallyInto(target); return target; } @@ -2515,7 +2518,7 @@ PixmapBlueprint flipHorizontallyCalcDims(const Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } -private void flipVerticallyTo(const Pixmap source, Pixmap target) @nogc { +private void flipVerticallyInto(const Pixmap source, Pixmap target) @nogc { auto src = PixmapScanner(source); auto dst = PixmapScannerRW(target); @@ -2537,15 +2540,14 @@ private void flipVerticallyTo(const Pixmap source, Pixmap target) @nogc { +/ Pixmap flipVertically(const Pixmap source, Pixmap target) @nogc { target.adjustTo(source.flipVerticallyCalcDims()); - - flipVerticallyTo(source, target); + flipVerticallyInto(source, target); return target; } /// ditto Pixmap flipVerticallyNew(const Pixmap source) { auto target = Pixmap.makeNew(source.flipVerticallyCalcDims()); - source.flipVerticallyTo(target); + source.flipVerticallyInto(target); return target; } From a23b05682286b897fab38188f564c7d8f62e5f91 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:00:12 +0200 Subject: [PATCH 33/38] Blank lines are fun and good for you --- pixmappaint.d | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 8ff33cf..8d5b00d 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -56,6 +56,9 @@ ] ``` + + + ### Colors Colors are stored in an RGBA format with 8 bit per channel. @@ -84,6 +87,8 @@ the length (not the width). + + ### Pixmaps A [Pixmap] consist of two fields: @@ -131,6 +136,8 @@ the aforementioned issues.) + + ### Image manipulation The term “image manipulation function” here refers to functions that @@ -147,6 +154,7 @@ Additionally, a “compute dimensions of target” function is provided. + #### Source to Target The regular “source to target” function takes (at least) two parameters: @@ -207,6 +215,7 @@ Those are an implementation detail (and may change at any point). ) + #### Source to New Target The “source to newly allocated target” wrapper allocates a new buffer to @@ -231,6 +240,7 @@ Pixmap target = sourceImage.cropNew(cropSize, cropOffset); --- + #### In-Place For selected image manipulation functions a special adaption is provided From c4485e3f889bc00db8cd22cf5c250af2b49ed22c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:04:25 +0200 Subject: [PATCH 34/38] =?UTF-8?q?Allow=20"=E2=80=A6CalcDims"=20functions?= =?UTF-8?q?=20to=20be=20less=20consistent?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Not all consistency is good. --- pixmappaint.d | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 8d5b00d..e7556bd 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -303,8 +303,8 @@ Such functions are identified by a `-CalcDims` suffix; e.g. [flipHorizontallyCalcDims] or [cropCalcDims]. - They receive the same parameters as their corresponding “source to new - target” function. For consistency reasons this also applies in cases where + They usually take the same parameters as their corresponding + “source to new target” function. This does not apply in cases where certain parameters are irrelevant for the computation of the target size. +/ module arsd.pixmappaint; @@ -2121,14 +2121,14 @@ private void decreaseOpacityInto(const Pixmap source, Pixmap target, ubyte opaci Use [decreaseOpacityF] with decimal opacity values in percent (%). +/ Pixmap decreaseOpacity(const Pixmap source, Pixmap target, ubyte opacityPercentage) @nogc { - target.adjustTo(source.decreaseOpacityCalcDims(opacityPercentage)); + target.adjustTo(source.decreaseOpacityCalcDims()); source.decreaseOpacityInto(target, opacityPercentage); return target; } /// ditto Pixmap decreaseOpacityNew(const Pixmap source, ubyte opacityPercentage) { - auto target = Pixmap.makeNew(source.decreaseOpacityCalcDims(opacityPercentage)); + auto target = Pixmap.makeNew(source.decreaseOpacityCalcDims()); source.decreaseOpacityInto(target, opacityPercentage); return target; } @@ -2141,7 +2141,7 @@ void decreaseOpacityInPlace(Pixmap source, ubyte opacityPercentage) @nogc { } /// ditto -PixmapBlueprint decreaseOpacityCalcDims(const Pixmap source, ubyte opacity) @nogc { +PixmapBlueprint decreaseOpacityCalcDims(const Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } @@ -2164,12 +2164,12 @@ Pixmap decreaseOpacityFNew(const Pixmap source, float opacityPercentage) { } /// ditto -void decreaseOpacityFInPlace(Pixmap source, const float opacityPercentage) @nogc { +void decreaseOpacityFInPlace(Pixmap source, float opacityPercentage) @nogc { return source.decreaseOpacityInPlace(percentageDecimalToUInt8(opacityPercentage)); } /// ditto -PixmapBlueprint decreaseOpacityF(Pixmap source, const float opacityPercentage) @nogc { +PixmapBlueprint decreaseOpacityF(Pixmap source) @nogc { return PixmapBlueprint.fromPixmap(source); } @@ -2253,7 +2253,7 @@ private alias cropInto = cropTo; The target Pixmap must be big enough in length to hold the cropped image. +/ Pixmap crop(const Pixmap source, Pixmap target, Size cropToSize, Point offset = Point(0, 0)) @nogc { - target.adjustTo(source.cropCalcDims(cropToSize, offset)); + target.adjustTo(cropCalcDims(cropToSize)); cropInto(source, target, offset); return target; } @@ -2277,7 +2277,7 @@ Pixmap cropInPlace(Pixmap source, Size cropToSize, Point offset = Point(0, 0)) @ } /// ditto -PixmapBlueprint cropCalcDims(const Pixmap source, Size cropToSize, Point offset = Point(0, 0)) @nogc { +PixmapBlueprint cropCalcDims(Size cropToSize) @nogc { return PixmapBlueprint.fromSize(cropToSize); } From b9098844b13be73c8247a20475e65b0a93204e64 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:15:03 +0200 Subject: [PATCH 35/38] Refactor rgba(ubyte,ubyte,ubyte,float) --- pixmappaint.d | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index e7556bd..71fae49 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -364,9 +364,8 @@ static assert(Pixel.sizeof == uint.sizeof); } /// - Pixel rgba(ubyte r, ubyte g, ubyte b, float aPct) - in (aPct >= 0 && aPct <= 1) { - return Pixel(r, g, b, castTo!ubyte(aPct * 255)); + Pixel rgba(ubyte r, ubyte g, ubyte b, float aPct) { + return Pixel(r, g, b, percentageDecimalToUInt8(aPct)); } /// From 195d1b228c38f489504032fbf2109f547151d38b Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:16:49 +0200 Subject: [PATCH 36/38] Improve docs of PixmapBlueprint --- pixmappaint.d | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index 71fae49..dd58e90 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -375,7 +375,9 @@ static assert(Pixel.sizeof == uint.sizeof); } /++ - Meta data for the construction a Pixmap + $(I Advanced functionality.) + + Meta data for the construction of a Pixmap. +/ struct PixmapBlueprint { /++ From 647c19997ddc6c8f7afab48405c96b3f6566cf03 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:17:26 +0200 Subject: [PATCH 37/38] Improve Pixmap.clone() --- pixmappaint.d | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index dd58e90..180b133 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -477,13 +477,13 @@ struct Pixmap { } /++ - Creates a $(I deep clone) of the Pixmap + Creates a $(I deep copy) of the Pixmap +/ Pixmap clone() const { - auto c = Pixmap(); - c.width = this.width; - c.data = this.data.dup; - return c; + return Pixmap( + this.data.dup, + this.width, + ); } /++ From 4979085a4c12b4d20b87077b73d94864c5b9dd1f Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 13 Oct 2024 19:21:11 +0200 Subject: [PATCH 38/38] Get rid of bold text in the first paragraph of function doc comments --- pixmappaint.d | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 180b133..15acaa0 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2194,7 +2194,7 @@ private void invertInto(const Pixmap source, Pixmap target) @trusted @nogc { } /++ - Inverts all colors to produce a $(B negative image). + Inverts all colors to produce a $(I negative image). $(TIP Develops a positive image when applied to a negative one. @@ -2781,7 +2781,7 @@ public void alphaBlendRGB(ref Pixel pxTarget, const Pixel pxSource) @safe { /++ Blends pixel `source` into pixel `target` - using the requested $(B blending mode). + using the requested [BlendMode|blending mode]. +/ template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) { @@ -2972,7 +2972,7 @@ template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) /++ Blends the pixel data of `source` into `target` - using the requested $(B blending mode). + using the requested [BlendMode|blending mode]. `source` and `target` MUST have the same length. +/