From 0e79aea97346c336102df22a922bcef8f80d1395 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 01:02:03 +0200 Subject: [PATCH 01/25] Move Pixmap functionality into its own module --- core.d | 11 +++++ dub.json | 16 +++++- pixmappaint.d | 122 ++++++++++++++++++++++++++++++++++++++++++++++ pixmappresenter.d | 122 ++-------------------------------------------- 4 files changed, 153 insertions(+), 118 deletions(-) create mode 100644 pixmappaint.d diff --git a/core.d b/core.d index e22e5dc..3078f12 100644 --- a/core.d +++ b/core.d @@ -181,6 +181,17 @@ version(Posix) { ========================= +/ +/++ + Casts value `v` to type `T`. + + $(TIP + This is a helper function for readability purposes. + ) + +/ +auto ref T typeCast(T, S)(auto ref S v) { + return cast(T) v; +} + // enum stringz : const(char)* { init = null } /++ diff --git a/dub.json b/dub.json index ca5268c..b8d057a 100644 --- a/dub.json +++ b/dub.json @@ -681,13 +681,27 @@ "arsd-official:color_base":"*" } }, + { + "name": "pixmappaint", + "description": "2D drawing companion library of Pixmap Presenter.", + "targetType": "library", + "sourceFiles": ["pixmappaint.d"], + "dependencies": { + "arsd-official:color_base":"*", + "arsd-official:core":"*" + }, + "dflags-dmd": ["-mv=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"], + "dflags-ldc": ["--mv=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"], + "dflags-gdc": ["-fmodule-file=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"] + }, { "name": "pixmappresenter", "description": "High-level display library. Designed to blit fully-rendered frames to the screen.", "targetType": "library", "sourceFiles": ["pixmappresenter.d"], "dependencies": { - "arsd-official:color_base":"*", + "arsd-official:core":"*", + "arsd-official:pixmappaint":"*", "arsd-official:simpledisplay":"*" }, "dflags-dmd": ["-mv=arsd.pixmappresenter=$PACKAGE_DIR/pixmappresenter.d"], diff --git a/pixmappaint.d b/pixmappaint.d new file mode 100644 index 0000000..ad32f0d --- /dev/null +++ b/pixmappaint.d @@ -0,0 +1,122 @@ +/+ + == pixmappaint == + Copyright Elias Batek (0xEAB) 2024. + Distributed under the Boost Software License, Version 1.0. + +/ +module arsd.pixmappaint; + +import arsd.color; +import arsd.core; + +/// +alias Pixel = Color; + +/// +alias ColorF = arsd.color.ColorF; + +/// +alias Size = arsd.color.Size; + +/// +alias Point = arsd.color.Point; + +// verify assumption(s) +static assert(Pixel.sizeof == uint.sizeof); + +@safe pure nothrow @nogc { + /// + Pixel rgba(ubyte r, ubyte g, ubyte b, ubyte a = 0xFF) { + return Pixel(r, g, b, a); + } + + /// + Pixel rgb(ubyte r, ubyte g, ubyte b) { + return rgba(r, g, b, 0xFF); + } +} + +/++ + Pixel data container + +/ +struct Pixmap { + + /// Pixel data + Pixel[] data; + + /// Pixel per row + int width; + +@safe pure nothrow: + + /// + this(Size size) { + this.size = size; + } + + /// + this(Pixel[] data, int width) @nogc + in (data.length % width == 0) { + this.data = data; + this.width = width; + } + + // undocumented: really shouldn’t be used. + // carries the risks of `length` and `width` getting out of sync accidentally. + deprecated("Use `size` instead.") + void length(int value) { + data.length = value; + } + + /++ + Changes the size of the buffer + + Reallocates the underlying pixel array. + +/ + void size(Size value) { + data.length = value.area; + width = value.width; + } + + /// ditto + void size(int totalPixels, int width) + in (totalPixels % width == 0) { + data.length = totalPixels; + this.width = width; + } + +@safe pure nothrow @nogc: + + /// Height of the buffer, i.e. the number of lines + int height() inout { + if (width == 0) { + return 0; + } + + return typeCast!int(data.length / width); + } + + /// Rectangular size of the buffer + Size size() inout { + return Size(width, height); + } + + /// Length of the buffer, i.e. the number of pixels + int length() inout { + return typeCast!int(data.length); + } + + /++ + Number of bytes per line + + Returns: + width × Pixel.sizeof + +/ + int pitch() inout { + return (width * int(Pixel.sizeof)); + } + + /// Clears the buffer’s contents (by setting each pixel to the same color) + void clear(Pixel value) { + data[] = value; + } +} diff --git a/pixmappresenter.d b/pixmappresenter.d index 18b935d..21394be 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -157,9 +157,12 @@ +/ module arsd.pixmappresenter; -import arsd.color; +import arsd.core; import arsd.simpledisplay; +/// +public import arsd.pixmappaint; + /* ## TODO @@ -176,129 +179,14 @@ import arsd.simpledisplay; */ /// -alias Pixel = Color; - -/// -alias ColorF = arsd.color.ColorF; - -/// -alias Size = arsd.color.Size; - -/// -alias Point = arsd.color.Point; +alias Pixmap = arsd.pixmappaint.Pixmap; /// alias WindowResizedCallback = void delegate(Size); -// verify assumption(s) -static assert(Pixel.sizeof == uint.sizeof); - // is the Timer class available on this platform? private enum hasTimer = is(Timer == class); -/// casts value `v` to type `T` -auto ref T typeCast(T, S)(auto ref S v) { - return cast(T) v; -} - -@safe pure nothrow @nogc { - /// - Pixel rgba(ubyte r, ubyte g, ubyte b, ubyte a = 0xFF) { - return Pixel(r, g, b, a); - } - - /// - Pixel rgb(ubyte r, ubyte g, ubyte b) { - return rgba(r, g, b, 0xFF); - } -} - -/++ - Pixel data container - +/ -struct Pixmap { - - /// Pixel data - Pixel[] data; - - /// Pixel per row - int width; - -@safe pure nothrow: - - /// - this(Size size) { - this.size = size; - } - - /// - this(Pixel[] data, int width) @nogc - in (data.length % width == 0) { - this.data = data; - this.width = width; - } - - // undocumented: really shouldn’t be used. - // carries the risks of `length` and `width` getting out of sync accidentally. - deprecated("Use `size` instead.") - void length(int value) { - data.length = value; - } - - /++ - Changes the size of the buffer - - Reallocates the underlying pixel array. - +/ - void size(Size value) { - data.length = value.area; - width = value.width; - } - - /// ditto - void size(int totalPixels, int width) - in (totalPixels % width == 0) { - data.length = totalPixels; - this.width = width; - } - -@safe pure nothrow @nogc: - - /// Height of the buffer, i.e. the number of lines - int height() inout { - if (width == 0) { - return 0; - } - - return typeCast!int(data.length / width); - } - - /// Rectangular size of the buffer - Size size() inout { - return Size(width, height); - } - - /// Length of the buffer, i.e. the number of pixels - int length() inout { - return typeCast!int(data.length); - } - - /++ - Number of bytes per line - - Returns: - width × Pixel.sizeof - +/ - int pitch() inout { - return (width * int(Pixel.sizeof)); - } - - /// Clears the buffer’s contents (by setting each pixel to the same color) - void clear(Pixel value) { - data[] = value; - } -} - // viewport math private @safe pure nothrow @nogc { From dc1cc02cde5c6ea93910b8517966ecde1a9b769f Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 01:09:57 +0200 Subject: [PATCH 02/25] Update changelog + TODO lists --- README.md | 2 +- pixmappresenter.d | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/README.md b/README.md index a5ec1db..73f740c 100644 --- a/README.md +++ b/README.md @@ -28,7 +28,7 @@ Future release, likely May 2024 or later. Nothing is planned for it at this time. -arsd.pixmappresenter was added. +arsd.pixmappresenter and arsd.pixmappaint were added. ## 11.0 diff --git a/pixmappresenter.d b/pixmappresenter.d index 21394be..486996d 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -169,7 +169,6 @@ public import arsd.pixmappaint; - More comprehensive documentation - Additional renderer implementations: - a `ScreenPainter`-based renderer - - a legacy OpenGL renderer (maybe) - Is there something in arsd that serves a similar purpose to `Pixmap`? - Can we convert to/from it? - Minimum window size From 77765756efe362e0c187e9001db7117aba67ea96 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 02:07:24 +0200 Subject: [PATCH 03/25] Add typeCast examples --- core.d | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/core.d b/core.d index 3078f12..f971c60 100644 --- a/core.d +++ b/core.d @@ -186,7 +186,28 @@ version(Posix) { $(TIP This is a helper function for readability purposes. + The idea is to make type-casting as accessible as `to()` from `std.conv`. ) + + --- + int i = cast(int)(foo * bar); + int i = typeCast!int(foo * bar); + + int j = cast(int) round(floatValue); + int j = round(floatValue).typeCast!int; + + int k = cast(int) floatValue + foobar; + int k = floatValue.typeCast!int + foobar; + + auto m = Point( + cast(int) calc(a.x, b.x), + cast(int) calc(a.y, b.y), + ); + auto m = Point( + calc(a.x, b.x).typeCast!int, + calc(a.y, b.y).typeCast!int, + ); + --- +/ auto ref T typeCast(T, S)(auto ref S v) { return cast(T) v; From 762280aa4cfbfe8733d39a0f0d4befa3dee1d46d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 02:23:41 +0200 Subject: [PATCH 04/25] Implement a couple of drawing functions Copied over from older codebases of mine. --- pixmappaint.d | 180 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 180 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index ad32f0d..1d91e2c 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -115,8 +115,188 @@ struct Pixmap { return (width * int(Pixel.sizeof)); } + /++ + 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 { + immutable size_t offset = linearOffset(width, pos); + immutable size_t end = (offset + n); + return data[offset .. end]; + } + /// Clears the buffer’s contents (by setting each pixel to the same color) void clear(Pixel value) { data[] = value; } } + +// Alpha-blending functions +@safe pure nothrow @nogc { + + /// + public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted + in (source.length == target.length) { + foreach (immutable idx, ref pxtarget; target) { + alphaBlend(pxtarget, source.ptr[idx]); + } + } + + /// + public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { + pragma(inline, true); + + immutable alphaSource = (pxSource.a | (pxSource.a << 8)); + immutable alphaTarget = (0xFFFF - alphaSource); + + foreach (immutable ib, ref px; pxTarget.components) { + immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); + immutable s = cast(ubyte)(((pxSource.components.ptr[ib] * alphaSource) + 0x8080) >> 16); + px = cast(ubyte)(d + s); + } + } +} + +// Drawing functions +@safe pure nothrow @nogc { + + private { + struct OriginRectangle { + Size size; + + @safe pure nothrow @nogc: + + int left() const => 0; + int top() const => 0; + int right() const => size.width; + int bottom() const => size.height; + + bool intersect(const Rectangle b) const { + // dfmt off + return ( + (b.right > 0 ) && + (b.left < this.right ) && + (b.bottom > 0 ) && + (b.top < this.bottom) + ); + // dfmt on + } + } + + Point pos(Rectangle r) => r.upperLeft; + } + + /++ + Draws a single pixel + +/ + void drawPixel(Pixmap target, Point pos, Pixel color) { + immutable size_t offset = linearOffset(target.width, pos); + target.data[offset] = color; + } + + /++ + Draws a rectangle + +/ + void drawRectangle(Pixmap target, Rectangle rectangle, Pixel color) { + alias r = rectangle; + + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + // out of bounds? + if (!tRect.intersect(r)) { + return; + } + + immutable drawingTarget = Point( + (r.pos.x >= 0) ? r.pos.x : 0, + (r.pos.y >= 0) ? r.pos.y : 0, + ); + + immutable drawingEnd = Point( + (r.right < tRect.right) ? r.right : tRect.right, + (r.bottom < tRect.bottom) ? r.bottom : tRect.bottom, + ); + + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; + + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = color; + } + } + + /++ + Draws a line + +/ + void drawLine(Pixmap target, Point a, Point b, Pixel color) { + import std.math : round, sqrt; + + float deltaX = b.x - a.x; + float deltaY = b.y - a.y; + int steps = sqrt(deltaX * deltaX + deltaY * deltaY).typeCast!int; + + float[2] step = [ + (deltaX / steps), + (deltaY / steps), + ]; + + foreach (i; 0 .. steps) { + // dfmt off + immutable Point p = a + Point( + round(step[0] * i).typeCast!int, + round(step[1] * i).typeCast!int, + ); + // dfmt on + + immutable offset = linearOffset(p, target.width); + target.data[offset] = color; + } + + immutable offsetEnd = linearOffset(b, target.width); + target.data[offsetEnd] = color; + } + + /++ + Draws an image (a source pixmap) on a target pixmap + + Params: + target = target pixmap to draw on + image = source pixmap + pos = top-left destination position (on the target pixmap) + +/ + void drawPixmap(Pixmap target, Pixmap image, Point pos) { + alias source = image; + + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + immutable sRect = Rectangle(pos, source.size); + + // out of bounds? + if (!tRect.intersect(sRect)) { + return; + } + + immutable drawingTarget = Point( + (pos.x >= 0) ? pos.x : 0, + (pos.y >= 0) ? pos.y : 0, + ); + + immutable drawingEnd = Point( + (sRect.right < tRect.right) ? sRect.right : tRect.right, + (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, + ); + + immutable drawingSource = Point(drawingTarget.x, 0) - Point(sRect.pos.x, sRect.pos.y); + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; + + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = + source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); + } + } +} From 9bc0dd74ea60701b62302cc6284b5306af95f17d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 02:34:28 +0200 Subject: [PATCH 05/25] Add docs, comments and Pixmap.clone() --- core.d | 3 +++ pixmappaint.d | 13 +++++++++++++ 2 files changed, 16 insertions(+) diff --git a/core.d b/core.d index f971c60..9998e0f 100644 --- a/core.d +++ b/core.d @@ -208,6 +208,9 @@ version(Posix) { calc(a.y, b.y).typeCast!int, ); --- + + History: + Added April 24, 2024 +/ auto ref T typeCast(T, S)(auto ref S v) { return cast(T) v; diff --git a/pixmappaint.d b/pixmappaint.d index 1d91e2c..bcc040b 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -60,6 +60,16 @@ struct Pixmap { this.width = width; } + /++ + Creates a $(I deep clone) of the Pixmap + +/ + Pixmap clone() const { + auto c = Pixmap(); + c.width = this.width; + c.data = this.data.dup; + return c; + } + // undocumented: really shouldn’t be used. // carries the risks of `length` and `width` getting out of sync accidentally. deprecated("Use `size` instead.") @@ -234,6 +244,9 @@ struct Pixmap { void drawLine(Pixmap target, Point a, Point b, Pixel color) { import std.math : round, sqrt; + // TODO: line width + // TODO: anti-aliasing (looks awful without it!) + float deltaX = b.x - a.x; float deltaY = b.y - a.y; int steps = sqrt(deltaX * deltaX + deltaY * deltaY).typeCast!int; From 23e3d59538a79a948dff42feb1f09e47e85c6835 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 02:57:30 +0200 Subject: [PATCH 06/25] Add sprite-sheet functionality --- pixmappaint.d | 146 +++++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 120 insertions(+), 26 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index bcc040b..66f9538 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -143,6 +143,88 @@ struct Pixmap { } } +/// +struct SpriteSheet { + private { + Pixmap _pixmap; + Size _spriteDimensions; + Size _layout; // pre-computed upon construction + } + +@safe pure nothrow @nogc: + + /// + public this(Pixmap pixmap, Size spriteSize) { + _pixmap = pixmap; + _spriteDimensions = spriteSize; + + _layout = Size( + _pixmap.width / _spriteDimensions.width, + _pixmap.height / _spriteDimensions.height, + ); + } + + /// + inout(Pixmap) pixmap() inout { + return _pixmap; + } + + /// + Size spriteSize() inout { + return _spriteDimensions; + } + + /// + Size layout() inout { + return _layout; + } + + /// + Point getSpriteColumn(int index) inout { + immutable x = index % layout.width; + immutable y = (index - x) / layout.height; + return Point(x, y); + } + + /// + Point getSpritePixelOffset2D(int index) inout { + immutable col = this.getSpriteColumn(index); + return Point( + col.x * _spriteDimensions.width, + col.y * _spriteDimensions.height, + ); + } +} + +// Silly micro-optimization +private struct OriginRectangle { + Size size; + +@safe pure nothrow @nogc: + + int left() const => 0; + int top() const => 0; + int right() const => size.width; + int bottom() const => size.height; + + bool intersect(const Rectangle b) const { + // dfmt off + return ( + (b.right > 0 ) && + (b.left < this.right ) && + (b.bottom > 0 ) && + (b.top < this.bottom) + ); + // dfmt on + } +} + +private { + +@safe pure nothrow @nogc: + Point pos(Rectangle r) => r.upperLeft; +} + // Alpha-blending functions @safe pure nothrow @nogc { @@ -172,32 +254,6 @@ struct Pixmap { // Drawing functions @safe pure nothrow @nogc { - private { - struct OriginRectangle { - Size size; - - @safe pure nothrow @nogc: - - int left() const => 0; - int top() const => 0; - int right() const => size.width; - int bottom() const => size.height; - - bool intersect(const Rectangle b) const { - // dfmt off - return ( - (b.right > 0 ) && - (b.left < this.right ) && - (b.bottom > 0 ) && - (b.top < this.bottom) - ); - // dfmt on - } - } - - Point pos(Rectangle r) => r.upperLeft; - } - /++ Draws a single pixel +/ @@ -312,4 +368,42 @@ struct Pixmap { source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); } } + + /++ + Draws a sprite from a spritesheet + +/ + void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos) { + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + immutable spriteOffset = sheet.getSpritePixelOffset2D(spriteIndex); + immutable sRect = Rectangle(pos, sheet.spriteSize); + + // out of bounds? + if (!tRect.intersect(sRect)) { + return; + } + + immutable drawingTarget = Point( + (pos.x >= 0) ? pos.x : 0, + (pos.y >= 0) ? pos.y : 0, + ); + + immutable drawingEnd = Point( + (sRect.right < tRect.right) ? sRect.right : tRect.right, + (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, + ); + + immutable drawingSource = + spriteOffset + + Point(drawingTarget.x, 0) + - Point(sRect.pos.x, sRect.pos.y); + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; + + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] + = sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); + } + } } From 486195859c0aff6558d5bb827f7dcda31f483bd1 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 03:00:47 +0200 Subject: [PATCH 07/25] Refactor blocks --- pixmappaint.d | 325 +++++++++++++++++++++++++------------------------- 1 file changed, 161 insertions(+), 164 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 66f9538..a4eb1e6 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -219,191 +219,188 @@ private struct OriginRectangle { } } -private { - @safe pure nothrow @nogc: + +// misc +private { Point pos(Rectangle r) => r.upperLeft; } -// Alpha-blending functions -@safe pure nothrow @nogc { +// ==== Alpha-blending functions ==== - /// - public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted - in (source.length == target.length) { - foreach (immutable idx, ref pxtarget; target) { - alphaBlend(pxtarget, source.ptr[idx]); - } - } - - /// - public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { - pragma(inline, true); - - immutable alphaSource = (pxSource.a | (pxSource.a << 8)); - immutable alphaTarget = (0xFFFF - alphaSource); - - foreach (immutable ib, ref px; pxTarget.components) { - immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); - immutable s = cast(ubyte)(((pxSource.components.ptr[ib] * alphaSource) + 0x8080) >> 16); - px = cast(ubyte)(d + s); - } +/// +public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted +in (source.length == target.length) { + foreach (immutable idx, ref pxtarget; target) { + alphaBlend(pxtarget, source.ptr[idx]); } } -// Drawing functions -@safe pure nothrow @nogc { +/// +public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { + pragma(inline, true); - /++ - Draws a single pixel - +/ - void drawPixel(Pixmap target, Point pos, Pixel color) { - immutable size_t offset = linearOffset(target.width, pos); + immutable alphaSource = (pxSource.a | (pxSource.a << 8)); + immutable alphaTarget = (0xFFFF - alphaSource); + + foreach (immutable ib, ref px; pxTarget.components) { + immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); + immutable s = cast(ubyte)(((pxSource.components.ptr[ib] * alphaSource) + 0x8080) >> 16); + px = cast(ubyte)(d + s); + } +} + +// ==== Drawing functions ==== + +/++ + Draws a single pixel + +/ +void drawPixel(Pixmap target, Point pos, Pixel color) { + immutable size_t offset = linearOffset(target.width, pos); + target.data[offset] = color; +} + +/++ + Draws a rectangle + +/ +void drawRectangle(Pixmap target, Rectangle rectangle, Pixel color) { + alias r = rectangle; + + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + // out of bounds? + if (!tRect.intersect(r)) { + return; + } + + immutable drawingTarget = Point( + (r.pos.x >= 0) ? r.pos.x : 0, + (r.pos.y >= 0) ? r.pos.y : 0, + ); + + immutable drawingEnd = Point( + (r.right < tRect.right) ? r.right : tRect.right, + (r.bottom < tRect.bottom) ? r.bottom : tRect.bottom, + ); + + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; + + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = color; + } +} + +/++ + Draws a line + +/ +void drawLine(Pixmap target, Point a, Point b, Pixel color) { + import std.math : round, sqrt; + + // TODO: line width + // TODO: anti-aliasing (looks awful without it!) + + float deltaX = b.x - a.x; + float deltaY = b.y - a.y; + int steps = sqrt(deltaX * deltaX + deltaY * deltaY).typeCast!int; + + float[2] step = [ + (deltaX / steps), + (deltaY / steps), + ]; + + foreach (i; 0 .. steps) { + // dfmt off + immutable Point p = a + Point( + round(step[0] * i).typeCast!int, + round(step[1] * i).typeCast!int, + ); + // dfmt on + + immutable offset = linearOffset(p, target.width); target.data[offset] = color; } - /++ - Draws a rectangle - +/ - void drawRectangle(Pixmap target, Rectangle rectangle, Pixel color) { - alias r = rectangle; + immutable offsetEnd = linearOffset(b, target.width); + target.data[offsetEnd] = color; +} - immutable tRect = OriginRectangle( - Size(target.width, target.height), - ); +/++ + Draws an image (a source pixmap) on a target pixmap - // out of bounds? - if (!tRect.intersect(r)) { - return; - } + Params: + target = target pixmap to draw on + image = source pixmap + pos = top-left destination position (on the target pixmap) + +/ +void drawPixmap(Pixmap target, Pixmap image, Point pos) { + alias source = image; - immutable drawingTarget = Point( - (r.pos.x >= 0) ? r.pos.x : 0, - (r.pos.y >= 0) ? r.pos.y : 0, - ); + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); - immutable drawingEnd = Point( - (r.right < tRect.right) ? r.right : tRect.right, - (r.bottom < tRect.bottom) ? r.bottom : tRect.bottom, - ); + immutable sRect = Rectangle(pos, source.size); - immutable int drawingWidth = drawingEnd.x - drawingTarget.x; - - foreach (y; drawingTarget.y .. drawingEnd.y) { - target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = color; - } + // out of bounds? + if (!tRect.intersect(sRect)) { + return; } - /++ - Draws a line - +/ - void drawLine(Pixmap target, Point a, Point b, Pixel color) { - import std.math : round, sqrt; + immutable drawingTarget = Point( + (pos.x >= 0) ? pos.x : 0, + (pos.y >= 0) ? pos.y : 0, + ); - // TODO: line width - // TODO: anti-aliasing (looks awful without it!) + immutable drawingEnd = Point( + (sRect.right < tRect.right) ? sRect.right : tRect.right, + (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, + ); - float deltaX = b.x - a.x; - float deltaY = b.y - a.y; - int steps = sqrt(deltaX * deltaX + deltaY * deltaY).typeCast!int; + immutable drawingSource = Point(drawingTarget.x, 0) - Point(sRect.pos.x, sRect.pos.y); + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; - float[2] step = [ - (deltaX / steps), - (deltaY / steps), - ]; - - foreach (i; 0 .. steps) { - // dfmt off - immutable Point p = a + Point( - round(step[0] * i).typeCast!int, - round(step[1] * i).typeCast!int, - ); - // dfmt on - - immutable offset = linearOffset(p, target.width); - target.data[offset] = color; - } - - immutable offsetEnd = linearOffset(b, target.width); - target.data[offsetEnd] = color; - } - - /++ - Draws an image (a source pixmap) on a target pixmap - - Params: - target = target pixmap to draw on - image = source pixmap - pos = top-left destination position (on the target pixmap) - +/ - void drawPixmap(Pixmap target, Pixmap image, Point pos) { - alias source = image; - - immutable tRect = OriginRectangle( - Size(target.width, target.height), - ); - - immutable sRect = Rectangle(pos, source.size); - - // out of bounds? - if (!tRect.intersect(sRect)) { - return; - } - - immutable drawingTarget = Point( - (pos.x >= 0) ? pos.x : 0, - (pos.y >= 0) ? pos.y : 0, - ); - - immutable drawingEnd = Point( - (sRect.right < tRect.right) ? sRect.right : tRect.right, - (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, - ); - - immutable drawingSource = Point(drawingTarget.x, 0) - Point(sRect.pos.x, sRect.pos.y); - immutable int drawingWidth = drawingEnd.x - drawingTarget.x; - - foreach (y; drawingTarget.y .. drawingEnd.y) { - target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = - source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); - } - } - - /++ - Draws a sprite from a spritesheet - +/ - void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos) { - immutable tRect = OriginRectangle( - Size(target.width, target.height), - ); - - immutable spriteOffset = sheet.getSpritePixelOffset2D(spriteIndex); - immutable sRect = Rectangle(pos, sheet.spriteSize); - - // out of bounds? - if (!tRect.intersect(sRect)) { - return; - } - - immutable drawingTarget = Point( - (pos.x >= 0) ? pos.x : 0, - (pos.y >= 0) ? pos.y : 0, - ); - - immutable drawingEnd = Point( - (sRect.right < tRect.right) ? sRect.right : tRect.right, - (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, - ); - - immutable drawingSource = - spriteOffset - + Point(drawingTarget.x, 0) - - Point(sRect.pos.x, sRect.pos.y); - immutable int drawingWidth = drawingEnd.x - drawingTarget.x; - - foreach (y; drawingTarget.y .. drawingEnd.y) { - target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] - = sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); - } + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = + source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); + } +} + +/++ + Draws a sprite from a spritesheet + +/ +void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos) { + immutable tRect = OriginRectangle( + Size(target.width, target.height), + ); + + immutable spriteOffset = sheet.getSpritePixelOffset2D(spriteIndex); + immutable sRect = Rectangle(pos, sheet.spriteSize); + + // out of bounds? + if (!tRect.intersect(sRect)) { + return; + } + + immutable drawingTarget = Point( + (pos.x >= 0) ? pos.x : 0, + (pos.y >= 0) ? pos.y : 0, + ); + + immutable drawingEnd = Point( + (sRect.right < tRect.right) ? sRect.right : tRect.right, + (sRect.bottom < tRect.bottom) ? sRect.bottom : tRect.bottom, + ); + + immutable drawingSource = + spriteOffset + + Point(drawingTarget.x, 0) + - Point(sRect.pos.x, sRect.pos.y); + immutable int drawingWidth = drawingEnd.x - drawingTarget.x; + + foreach (y; drawingTarget.y .. drawingEnd.y) { + target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] + = sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); } } From e12846e996e79323635d64405248854ac66b08bc Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 22:17:06 +0200 Subject: [PATCH 08/25] Publicly import simpledisplay from pixmappresenter Looks like a convenience feature, *but* - more imporantly - this prevents certain symbol name clashes from occurring in common user-code. --- pixmappresenter.d | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/pixmappresenter.d b/pixmappresenter.d index 486996d..8949637 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -158,7 +158,18 @@ module arsd.pixmappresenter; import arsd.core; -import arsd.simpledisplay; + +/++ + While publicly importing `arsd.simpledisplay` is not actually necessary, + most real-world code would eventually import said module as well anyway. + + More importantly, this public import prevents users from facing certain + symbol clashes in their code that would occur in modules importing both + `pixmappresenter` and `simpledisplay`. + For instance both of these modules happen to define different types + as `Pixmap`. + +/ +public import arsd.simpledisplay; /// public import arsd.pixmappaint; From 44dfc72d78028151effc723b2388e6cd9e5da1bf Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Wed, 24 Apr 2024 22:25:52 +0200 Subject: [PATCH 09/25] Cleanup --- pixmappaint.d | 11 ++++++++--- pixmappresenter.d | 1 - 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index a4eb1e6..07b1c63 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -8,18 +8,23 @@ module arsd.pixmappaint; import arsd.color; import arsd.core; -/// -alias Pixel = Color; +alias Color = arsd.color.Color; /// alias ColorF = arsd.color.ColorF; /// -alias Size = arsd.color.Size; +alias Pixel = Color; /// alias Point = arsd.color.Point; +/// +alias Rectangle = arsd.color.Rectangle; + +/// +alias Size = arsd.color.Size; + // verify assumption(s) static assert(Pixel.sizeof == uint.sizeof); diff --git a/pixmappresenter.d b/pixmappresenter.d index 8949637..15f87fa 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -87,7 +87,6 @@ --- import arsd.pixmappresenter; - import arsd.simpledisplay : MouseEvent; int main() { // Internal resolution of the images (“frames”) we will render. From 7c3d511a337e90a19eacd518a9b1f8637d5574f8 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 6 May 2024 00:48:49 +0200 Subject: [PATCH 10/25] Prepare implementation of blend modes --- pixmappaint.d | 201 +++++++++++++++++++++++++++++++++++++++++++++++--- 1 file changed, 192 insertions(+), 9 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 07b1c63..52846c5 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -34,6 +34,12 @@ static assert(Pixel.sizeof == uint.sizeof); return Pixel(r, g, b, a); } + /// + Pixel rgba(ubyte r, ubyte g, ubyte b, float aPct) + in (aPct >= 0 && aPct <= 1) { + return Pixel(r, g, b, typeCast!ubyte(aPct * 255)); + } + /// Pixel rgb(ubyte r, ubyte g, ubyte b) { return rgba(r, g, b, 0xFF); @@ -58,6 +64,13 @@ struct Pixmap { this.size = size; } + /// + this(int width, int height) + in (width > 0) + in (height > 0) { + this(Size(width, height)); + } + /// this(Pixel[] data, int width) @nogc in (data.length % width == 0) { @@ -231,13 +244,86 @@ private { Point pos(Rectangle r) => r.upperLeft; } +/++ + Fast 8-bit “percentage” function + + This function optimizes its runtime performance by substituting + the division by 255 with an approximation using bitshifts. + + Nonetheless, the its result are as accurate as a floating point + division with 64-bit precision. + + Params: + nPercentage = percentage as the number of 255ths (“two hundred fifty-fifths”) + value = base value (“total”) + + Returns: + `round(value * nPercentage / 255.0)` + +/ +ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { + immutable factor = (nPercentage | (nPercentage << 8)); + return (((value * factor) + 0x8080) >> 16); +} + +@safe unittest { + // Accuracy verification + + static ubyte n255thsOfFP64(const ubyte nPercentage, const ubyte value) { + import std.math : round; + + return (value * nPercentage / 255.0).round().typeCast!ubyte(); + } + + for (int value = ubyte.min; value <= ubyte.max; ++value) { + for (int percent = ubyte.min; percent <= ubyte.max; ++percent) { + immutable v = cast(ubyte) value; + immutable p = cast(ubyte) percent; + + immutable approximated = n255thsOf(p, v); + immutable precise = n255thsOfFP64(p, v); + assert(approximated == precise); + } + } +} + +/++ + Sets the opacity of a [Pixmap]. + + This lossy operation updates the alpha-channel value of each pixel. + → `alpha *= opacity` + + See_Also: + Use [opacityF] with opacity values in percent (%). + +/ +void opacity(ref Pixmap pixmap, const ubyte opacity) { + foreach (ref px; pixmap.data) { + px.a = opacity.n255thsOf(px.a); + } +} + +/++ + Sets the opacity of a [Pixmap]. + + This lossy operation updates the alpha-channel value of each pixel. + → `alpha *= opacity` + + See_Also: + Use [opacity] with 8-bit integer opacity values (in 255ths). + +/ +void opacityF(ref Pixmap pixmap, const float opacity) +in (opacity >= 0) +in (opacity <= 1.0) { + immutable opacity255 = typeCast!ubyte(opacity * 255); + pixmap.opacity = opacity255; +} + // ==== Alpha-blending functions ==== /// public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted in (source.length == target.length) { - foreach (immutable idx, ref pxtarget; target) { - alphaBlend(pxtarget, source.ptr[idx]); + foreach (immutable idx, ref pxTarget; target) { + alphaBlend(pxTarget, source.ptr[idx]); } } @@ -255,6 +341,97 @@ public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { } } +// ==== Blending functions ==== + +enum BlendMode { + none = 0, + overwrite = none, + normal = 1, + alpha = normal, + + multiply, + screen, + + darken, + lighten, +} + +/// +alias Blend = BlendMode; + +// undocumented +enum blendNormal = BlendMode.normal; + +/++ + Blends pixel `source` into pixel `target`. + +/ +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == BlendMode.overwrite) { + target = source; +} + +/// ditto +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.alpha) { + return alphaBlend(target, source); +} + +/// ditto +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.multiply) { + function(ref Pixel target, const Pixel source) @trusted { + foreach (immutable ib, ref ch; target.components) { + ch = n255thsOf(source.components.ptr[ib], ch); + } + }(target, source); +} + +/// ditto +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.screen) { + assert(false, "TODO"); +} + +/// ditto +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.darken) { + assert(false, "TODO"); +} + +/// ditto +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.lighten) { + assert(false, "TODO"); +} + +/++ + Blends the pixel data of `source` into `target`. + + `source` and `target` MUST have the same length. + +/ +void blendPixels(BlendMode mode)(scope Pixel[] target, scope const Pixel[] source) @trusted +in (source.length == target.length) { + static if (mode == BlendMode.overwrite) { + target.ptr[0 .. target.length] = source.ptr[0 .. target.length]; + } else { + // better error message in case it’s not implemented + static if (!is(typeof(blendPixel!mode))) { + static assert(false, "Missing `blendPixel!(" ~ mode.stringof ~ ")`."); + } + + foreach (immutable idx, ref pxTarget; target) { + blendPixel!mode(pxTarget, source.ptr[idx]); + } + } +} + +/// ditto +void blendPixels(scope Pixel[] target, scope const Pixel[] source, BlendMode mode) { + import std.meta : NoDuplicates; + import std.traits : EnumMembers; + + final switch (mode) with (BlendMode) { + static foreach (m; NoDuplicates!(EnumMembers!BlendMode)) { + case m: + return blendPixels!m(target, source); + } + } +} + // ==== Drawing functions ==== /++ @@ -339,7 +516,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) { +void drawPixmap(Pixmap target, Pixmap image, Point pos, Blend blend = blendNormal) { alias source = image; immutable tRect = OriginRectangle( @@ -367,15 +544,18 @@ void drawPixmap(Pixmap target, Pixmap image, Point pos) { immutable int drawingWidth = drawingEnd.x - drawingTarget.x; foreach (y; drawingTarget.y .. drawingEnd.y) { - target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] = - source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); + blendPixels( + target.sliceAt(Point(drawingTarget.x, y), drawingWidth), + source.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth), + blend, + ); } } /++ - Draws a sprite from a spritesheet + Draws a sprite from a spritesheet +/ -void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos) { +void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point pos, Blend blend = blendNormal) { immutable tRect = OriginRectangle( Size(target.width, target.height), ); @@ -405,7 +585,10 @@ void drawSprite(Pixmap target, const SpriteSheet sheet, int spriteIndex, Point p immutable int drawingWidth = drawingEnd.x - drawingTarget.x; foreach (y; drawingTarget.y .. drawingEnd.y) { - target.sliceAt(Point(drawingTarget.x, y), drawingWidth)[] - = sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth); + blendPixels( + target.sliceAt(Point(drawingTarget.x, y), drawingWidth), + sheet.pixmap.sliceAt(Point(drawingSource.x, y + drawingSource.y), drawingWidth), + blend, + ); } } From 9d0c6e1d0c5b3d709feeab33447d5a4f14bf020c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 6 May 2024 00:49:20 +0200 Subject: [PATCH 11/25] Add functions for loading image-data from arsd.color --- pixmappaint.d | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 52846c5..1cc9701 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -112,6 +112,26 @@ struct Pixmap { this.width = width; } + static { + /++ + Creates a Pixmap wrapping the pixel data from the provided `TrueColorImage`. + + Interoperability function: `arsd.color` + +/ + Pixmap fromTrueColorImage(TrueColorImage source) @nogc { + return Pixmap(source.imageData.colors, source.width); + } + + /++ + Creates a Pixmap wrapping the pixel data from the provided `MemoryImage`. + + Interoperability function: `arsd.color` + +/ + Pixmap fromMemoryImage(MemoryImage source) { + return fromTrueColorImage(source.getAsTrueColorImage()); + } + } + @safe pure nothrow @nogc: /// Height of the buffer, i.e. the number of lines From 511ec3ed6ddd087b62c1046543382707e7c99c59 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Mon, 6 May 2024 01:06:18 +0200 Subject: [PATCH 12/25] Rename BlendMode.overwrite to BlendMode.replace --- pixmappaint.d | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 1cc9701..05b823a 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -365,7 +365,7 @@ public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { enum BlendMode { none = 0, - overwrite = none, + replace = none, normal = 1, alpha = normal, @@ -385,7 +385,7 @@ enum blendNormal = BlendMode.normal; /++ Blends pixel `source` into pixel `target`. +/ -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == BlendMode.overwrite) { +void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == BlendMode.replace) { target = source; } @@ -425,7 +425,7 @@ void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode = +/ void blendPixels(BlendMode mode)(scope Pixel[] target, scope const Pixel[] source) @trusted in (source.length == target.length) { - static if (mode == BlendMode.overwrite) { + static if (mode == BlendMode.replace) { target.ptr[0 .. target.length] = source.ptr[0 .. target.length]; } else { // better error message in case it’s not implemented From 5c2817e563951ea21d2a4c722acebc60b5d5f2e3 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 24 May 2024 04:44:44 +0200 Subject: [PATCH 13/25] Fix alpha-blending with color-mix/blend functions --- pixmappaint.d | 30 +++++++++++++++++++++--------- 1 file changed, 21 insertions(+), 9 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 05b823a..e5ed2f1 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -7,6 +7,7 @@ module arsd.pixmappaint; import arsd.color; import arsd.core; +import std.math : round; alias Color = arsd.color.Color; @@ -289,8 +290,6 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { // Accuracy verification static ubyte n255thsOfFP64(const ubyte nPercentage, const ubyte value) { - import std.math : round; - return (value * nPercentage / 255.0).round().typeCast!ubyte(); } @@ -333,7 +332,7 @@ void opacity(ref Pixmap pixmap, const ubyte opacity) { void opacityF(ref Pixmap pixmap, const float opacity) in (opacity >= 0) in (opacity <= 1.0) { - immutable opacity255 = typeCast!ubyte(opacity * 255); + immutable opacity255 = round(opacity * 255).typeCast!ubyte; pixmap.opacity = opacity255; } @@ -361,6 +360,23 @@ public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { } } +/// +public void alphaBlend( + ubyte function(const ubyte, const ubyte) blend +)(ref Pixel pxTarget, const Pixel pxSource) @trusted { + pragma(inline, true); + + immutable alphaSource = (pxSource.a | (pxSource.a << 8)); + immutable alphaTarget = (0xFFFF - alphaSource); + + foreach (immutable ib, ref px; pxTarget.components) { + immutable b = blend(px, pxSource.components.ptr[ib]); + immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); + immutable s = cast(ubyte)(((b * alphaSource) + 0x8080) >> 16); + px = cast(ubyte)(d + s); + } +} + // ==== Blending functions ==== enum BlendMode { @@ -396,11 +412,7 @@ void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode = /// ditto void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.multiply) { - function(ref Pixel target, const Pixel source) @trusted { - foreach (immutable ib, ref ch; target.components) { - ch = n255thsOf(source.components.ptr[ib], ch); - } - }(target, source); + return alphaBlend!((a, b) => n255thsOf(a, b))(target, source); } /// ditto @@ -498,7 +510,7 @@ void drawRectangle(Pixmap target, Rectangle rectangle, Pixel color) { Draws a line +/ void drawLine(Pixmap target, Point a, Point b, Pixel color) { - import std.math : round, sqrt; + import std.math : sqrt; // TODO: line width // TODO: anti-aliasing (looks awful without it!) From 033770a6c6c23cdec842a2c31716664d951e534c Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 24 May 2024 04:45:48 +0200 Subject: [PATCH 14/25] Turn clumsy `static assert` into a `pragma(msg)` --- pixmappaint.d | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pixmappaint.d b/pixmappaint.d index e5ed2f1..42994e4 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -442,7 +442,7 @@ in (source.length == target.length) { } else { // better error message in case it’s not implemented static if (!is(typeof(blendPixel!mode))) { - static assert(false, "Missing `blendPixel!(" ~ mode.stringof ~ ")`."); + pragma(msg, "Hint: Missing or bad `blendPixel!(" ~ mode.stringof ~ ")`."); } foreach (immutable idx, ref pxTarget; target) { From bd502fb91204de400b46607702cbde222e7bd297 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 24 May 2024 04:56:36 +0200 Subject: [PATCH 15/25] Rename `typeCast!T()` to `castTo!T()` --- core.d | 22 +++++++++++++--------- pixmappaint.d | 16 ++++++++-------- pixmappresenter.d | 16 ++++++++-------- 3 files changed, 29 insertions(+), 25 deletions(-) diff --git a/core.d b/core.d index 9998e0f..6c0588f 100644 --- a/core.d +++ b/core.d @@ -190,32 +190,36 @@ version(Posix) { ) --- - int i = cast(int)(foo * bar); - int i = typeCast!int(foo * bar); + int i = cast(int)(foo * bar); + int i = castTo!int(foo * bar); int j = cast(int) round(floatValue); - int j = round(floatValue).typeCast!int; + int j = round(floatValue).castTo!int; - int k = cast(int) floatValue + foobar; - int k = floatValue.typeCast!int + foobar; + int k = cast(int) floatValue + foobar; + int k = floatValue.castTo!int + foobar; auto m = Point( cast(int) calc(a.x, b.x), cast(int) calc(a.y, b.y), ); auto m = Point( - calc(a.x, b.x).typeCast!int, - calc(a.y, b.y).typeCast!int, + calc(a.x, b.x).castTo!int, + calc(a.y, b.y).castTo!int, ); --- History: - Added April 24, 2024 + Added on April 24, 2024. + Renamed from `typeCast` to `castTo` on May 24, 2024. +/ -auto ref T typeCast(T, S)(auto ref S v) { +auto ref T castTo(T, S)(auto ref S v) { return cast(T) v; } +/// +alias typeCast = castTo; + // enum stringz : const(char)* { init = null } /++ diff --git a/pixmappaint.d b/pixmappaint.d index 42994e4..7c7b789 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -38,7 +38,7 @@ 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, typeCast!ubyte(aPct * 255)); + return Pixel(r, g, b, castTo!ubyte(aPct * 255)); } /// @@ -141,7 +141,7 @@ struct Pixmap { return 0; } - return typeCast!int(data.length / width); + return castTo!int(data.length / width); } /// Rectangular size of the buffer @@ -151,7 +151,7 @@ struct Pixmap { /// Length of the buffer, i.e. the number of pixels int length() inout { - return typeCast!int(data.length); + return castTo!int(data.length); } /++ @@ -290,7 +290,7 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { // Accuracy verification static ubyte n255thsOfFP64(const ubyte nPercentage, const ubyte value) { - return (value * nPercentage / 255.0).round().typeCast!ubyte(); + return (value * nPercentage / 255.0).round().castTo!ubyte(); } for (int value = ubyte.min; value <= ubyte.max; ++value) { @@ -332,7 +332,7 @@ void opacity(ref Pixmap pixmap, const ubyte opacity) { void opacityF(ref Pixmap pixmap, const float opacity) in (opacity >= 0) in (opacity <= 1.0) { - immutable opacity255 = round(opacity * 255).typeCast!ubyte; + immutable opacity255 = round(opacity * 255).castTo!ubyte; pixmap.opacity = opacity255; } @@ -517,7 +517,7 @@ void drawLine(Pixmap target, Point a, Point b, Pixel color) { float deltaX = b.x - a.x; float deltaY = b.y - a.y; - int steps = sqrt(deltaX * deltaX + deltaY * deltaY).typeCast!int; + int steps = sqrt(deltaX * deltaX + deltaY * deltaY).castTo!int; float[2] step = [ (deltaX / steps), @@ -527,8 +527,8 @@ void drawLine(Pixmap target, Point a, Point b, Pixel color) { foreach (i; 0 .. steps) { // dfmt off immutable Point p = a + Point( - round(step[0] * i).typeCast!int, - round(step[1] * i).typeCast!int, + round(step[0] * i).castTo!int, + round(step[1] * i).castTo!int, ); // dfmt on diff --git a/pixmappresenter.d b/pixmappresenter.d index 15f87fa..719cdfc 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -238,7 +238,7 @@ private @safe pure nothrow @nogc { Point offsetCenter(const Size drawing, const Size canvas) { auto delta = canvas.deltaPerimeter(drawing); - return (typeCast!Point(delta) >> 1); + return (castTo!Point(delta) >> 1); } } @@ -278,8 +278,8 @@ Viewport calculateViewport(const ref PresenterConfig config) @safe pure nothrow case Scaling.contain: const float scaleF = karContainScalingFactorF(config.renderer.resolution, config.window.size); size = Size( - typeCast!int(scaleF * config.renderer.resolution.width), - typeCast!int(scaleF * config.renderer.resolution.height), + castTo!int(scaleF * config.renderer.resolution.width), + castTo!int(scaleF * config.renderer.resolution.height), ); break; @@ -297,8 +297,8 @@ Viewport calculateViewport(const ref PresenterConfig config) @safe pure nothrow case Scaling.cover: const float fillF = karCoverScalingFactorF(config.renderer.resolution, config.window.size); size = Size( - typeCast!int(fillF * config.renderer.resolution.width), - typeCast!int(fillF * config.renderer.resolution.height), + castTo!int(fillF * config.renderer.resolution.width), + castTo!int(fillF * config.renderer.resolution.height), ); break; } @@ -549,7 +549,7 @@ final class OpenGl3PixmapRenderer : PixmapRenderer { 0, 0, _poc.config.renderer.resolution.width, _poc.config.renderer.resolution.height, GL_RGBA, GL_UNSIGNED_BYTE, - typeCast!(void*)(_poc.framebuffer.data.ptr) + castTo!(void*)(_poc.framebuffer.data.ptr) ); glUseProgram(_shader.shaderProgram); @@ -604,7 +604,7 @@ final class OpenGl3PixmapRenderer : PixmapRenderer { glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 4 * GLfloat.sizeof, null); glEnableVertexAttribArray(0); - glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4 * GLfloat.sizeof, typeCast!(void*)(2 * GLfloat.sizeof)); + glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, 4 * GLfloat.sizeof, castTo!(void*)(2 * GLfloat.sizeof)); glEnableVertexAttribArray(1); } @@ -785,7 +785,7 @@ final class OpenGl1PixmapRenderer : PixmapRenderer { 0, 0, _poc.config.renderer.resolution.width, _poc.config.renderer.resolution.height, GL_RGBA, GL_UNSIGNED_BYTE, - typeCast!(void*)(_poc.framebuffer.data.ptr) + castTo!(void*)(_poc.framebuffer.data.ptr) ); glBegin(GL_QUADS); From 6072e6d8fd364dab87a97fa1c2782340c3fe01d4 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Fri, 24 May 2024 04:59:42 +0200 Subject: [PATCH 16/25] Update docs --- pixmappaint.d | 9 +++++++++ pixmappresenter.d | 2 -- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 7c7b789..9d5ae99 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -2,6 +2,15 @@ == pixmappaint == Copyright Elias Batek (0xEAB) 2024. Distributed under the Boost Software License, Version 1.0. + + $(WARNING + $(B Early Technology Preview.) + ) + + $(PITFALL + This module is $(B work in progress). + API is subject to changes until further notice. + ) +/ module arsd.pixmappaint; diff --git a/pixmappresenter.d b/pixmappresenter.d index 719cdfc..21311ef 100644 --- a/pixmappresenter.d +++ b/pixmappresenter.d @@ -179,8 +179,6 @@ public import arsd.pixmappaint; - More comprehensive documentation - Additional renderer implementations: - a `ScreenPainter`-based renderer - - Is there something in arsd that serves a similar purpose to `Pixmap`? - - Can we convert to/from it? - Minimum window size - to ensure `Scaling.integer` doesn’t break “unexpectedly” - More control over timing From 6e469c27bd459f655c90db7ad49f6b25e539fbaa Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 May 2024 00:39:10 +0200 Subject: [PATCH 17/25] Refactor pixel-blending --- pixmappaint.d | 172 ++++++++++++++++++++++++++++++++++++++++---------- 1 file changed, 138 insertions(+), 34 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 9d5ae99..18e16c3 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -274,6 +274,14 @@ private { Point pos(Rectangle r) => r.upperLeft; } +/++ + Limits a value to a maximum 0xFF (= 255). + +/ +ubyte clamp255(Tint)(const Tint value) { + pragma(inline, true); + return (value < 0xFF) ? value.castTo!ubyte : 0xFF; +} + /++ Fast 8-bit “percentage” function @@ -347,43 +355,96 @@ in (opacity <= 1.0) { // ==== Alpha-blending functions ==== +/++ + Alpha-blending accuracy level + + $(TIP + This primarily exists for performance reasons. + In my tests LLVM manages to auto-vectorize the RGB-only codepath significantly better, + while the codegen for the accurate RGBA path is pretty conservative. + + This provides an optimization opportunity for use-cases + that don’t require an alpha-channel on the result. + ) + +/ +enum BlendAccuracy { + /++ + Only RGB channels will have the correct result. + + A(lpha) channel can contain any value. + + Suitable for blending into non-transparent targets (e.g. framebuffer, canvas) + where the resulting alpha-channel (opacity) value does not matter. + +/ + rgb = false, + + /++ + All RGBA channels will have the correct result. + + Suitable for blending into transparent targets (e.g. images) + where the resulting alpha-channel (opacity) value matters. + + Use this mode for image manipulation. + +/ + rgba = true, +} + /// -public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted +public void alphaBlend( + BlendAccuracy accuracy, + ubyte function(const ubyte, const ubyte) pure blend = null, +)( + scope Pixel[] target, + scope const Pixel[] source, +) @trusted in (source.length == target.length) { foreach (immutable idx, ref pxTarget; target) { alphaBlend(pxTarget, source.ptr[idx]); } } -/// -public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { - pragma(inline, true); - - immutable alphaSource = (pxSource.a | (pxSource.a << 8)); - immutable alphaTarget = (0xFFFF - alphaSource); - - foreach (immutable ib, ref px; pxTarget.components) { - immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); - immutable s = cast(ubyte)(((pxSource.components.ptr[ib] * alphaSource) + 0x8080) >> 16); - px = cast(ubyte)(d + s); - } +/// ditto +public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @safe { + return alphaBlend!(BlendAccuracy.rgba, null)(target, source); } /// public void alphaBlend( - ubyte function(const ubyte, const ubyte) blend -)(ref Pixel pxTarget, const Pixel pxSource) @trusted { + BlendAccuracy accuracy, + ubyte function(const ubyte, const ubyte) blend = null, +)( + ref Pixel pxTarget, + const Pixel pxSource, +) @trusted { pragma(inline, true); + static if (accuracy) { + immutable alphaResult = clamp255(pxSource.a + n255thsOf(pxTarget.a, (0xFF - pxSource.a))); + //immutable alphaResult = clamp255(pxTarget.a + n255thsOf(pxSource.a, (0xFF - pxTarget.a))); + } + immutable alphaSource = (pxSource.a | (pxSource.a << 8)); immutable alphaTarget = (0xFFFF - alphaSource); foreach (immutable ib, ref px; pxTarget.components) { - immutable b = blend(px, pxSource.components.ptr[ib]); + static if (blend !is null) { + immutable bx = blend(px, pxSource.components.ptr[ib]); + } else { + immutable bx = pxSource.components.ptr[ib]; + } immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); - immutable s = cast(ubyte)(((b * alphaSource) + 0x8080) >> 16); + immutable s = cast(ubyte)(((bx * alphaSource) + 0x8080) >> 16); px = cast(ubyte)(d + s); } + + static if (accuracy) { + pxTarget.a = alphaResult; + } +} + +/// ditto +public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @safe { + return alphaBlend!(BlendAccuracy.rgba, null)(pxTarget, pxSource); } // ==== Blending functions ==== @@ -410,32 +471,55 @@ enum blendNormal = BlendMode.normal; /++ Blends pixel `source` into pixel `target`. +/ -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == BlendMode.replace) { +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == BlendMode.replace) { target = source; } /// ditto -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.alpha) { - return alphaBlend(target, source); +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.alpha) { + return alphaBlend!accuracy(target, source); } /// ditto -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.multiply) { - return alphaBlend!((a, b) => n255thsOf(a, b))(target, source); +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.multiply) { + return alphaBlend!(accuracy, + (a, b) => n255thsOf(a, b) + )(target, source); } /// ditto -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.screen) { +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.screen) { + + return alphaBlend!(accuracy, + (a, b) => castTo!ubyte(0xFF - n255thsOf((0xFF - a), (0xFF - b))) + )(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.darken) { assert(false, "TODO"); } /// ditto -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.darken) { - assert(false, "TODO"); -} - -/// ditto -void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode == Blend.lighten) { +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.lighten) { assert(false, "TODO"); } @@ -444,35 +528,55 @@ void blendPixel(BlendMode mode)(ref Pixel target, const Pixel source) if (mode = `source` and `target` MUST have the same length. +/ -void blendPixels(BlendMode mode)(scope Pixel[] target, scope const Pixel[] source) @trusted +void blendPixels(BlendMode mode, BlendAccuracy accuracy)(scope Pixel[] target, scope const Pixel[] source) @trusted in (source.length == target.length) { static if (mode == BlendMode.replace) { + // explicit optimization target.ptr[0 .. target.length] = source.ptr[0 .. target.length]; } else { + // better error message in case it’s not implemented - static if (!is(typeof(blendPixel!mode))) { + static if (!is(typeof(blendPixel!(mode, accuracy)))) { pragma(msg, "Hint: Missing or bad `blendPixel!(" ~ mode.stringof ~ ")`."); } foreach (immutable idx, ref pxTarget; target) { - blendPixel!mode(pxTarget, source.ptr[idx]); + blendPixel!(mode, accuracy)(pxTarget, source.ptr[idx]); } } } /// ditto -void blendPixels(scope Pixel[] target, scope const Pixel[] source, BlendMode mode) { +void blendPixels(BlendAccuracy accuracy = BlendAccuracy.rgba)( + scope Pixel[] target, + scope const Pixel[] source, + BlendMode mode, +) { import std.meta : NoDuplicates; import std.traits : EnumMembers; final switch (mode) with (BlendMode) { static foreach (m; NoDuplicates!(EnumMembers!BlendMode)) { case m: - return blendPixels!m(target, source); + return blendPixels!(m, accuracy)(target, source); } } } +/// ditto +void blendPixels( + scope Pixel[] target, + scope const Pixel[] source, + BlendMode mode, + BlendAccuracy accuracy = BlendAccuracy.rgba, +) { + if (accuracy == BlendAccuracy.rgb) { + return blendPixels!(BlendAccuracy.rgb)(target, source, mode); + } else { + return blendPixels!(BlendAccuracy.rgba)(target, source, mode); + } +} + // ==== Drawing functions ==== /++ From 9a4f8467f8300e885151c531f93ca2beb676a7a2 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 May 2024 00:58:05 +0200 Subject: [PATCH 18/25] Implement "lighten" + "darken" blend-modes --- pixmappaint.d | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index 18e16c3..98e75c9 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -272,6 +272,9 @@ private struct OriginRectangle { // misc private { Point pos(Rectangle r) => r.upperLeft; + + T max(T)(T a, T b) => (a >= b) ? a : b; + T min(T)(T a, T b) => (a <= b) ? a : b; } /++ @@ -491,6 +494,7 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( ref Pixel target, const Pixel source, ) if (mode == Blend.multiply) { + return alphaBlend!(accuracy, (a, b) => n255thsOf(a, b) )(target, source); @@ -512,7 +516,10 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( ref Pixel target, const Pixel source, ) if (mode == Blend.darken) { - assert(false, "TODO"); + + return alphaBlend!(accuracy, + (a, b) => min(a, b) + )(target, source); } /// ditto @@ -520,7 +527,10 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( ref Pixel target, const Pixel source, ) if (mode == Blend.lighten) { - assert(false, "TODO"); + + return alphaBlend!(accuracy, + (a, b) => max(a, b) + )(target, source); } /++ From ca6dd1bb90153dbe03195ca246a1c541ec7f1aac Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 May 2024 01:58:36 +0200 Subject: [PATCH 19/25] Implement "divide" blend-mode --- pixmappaint.d | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 98e75c9..aff06c8 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -463,6 +463,8 @@ enum BlendMode { darken, lighten, + + divide, } /// @@ -533,6 +535,17 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( )(target, source); } +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.divide) { + + return alphaBlend!(accuracy, + (b, f) => (f == 0) ? ubyte(0xFF) : clamp255(0xFF * b / f) + )(target, source); +} + /++ Blends the pixel data of `source` into `target`. From e4551548cdac0e0643bd4d6bd7b259f11b773a91 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 25 May 2024 01:58:55 +0200 Subject: [PATCH 20/25] Fix typos and clean up --- pixmappaint.d | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index aff06c8..66262ef 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -278,7 +278,7 @@ private { } /++ - Limits a value to a maximum 0xFF (= 255). + Limits a value to a maximum of 0xFF (= 255). +/ ubyte clamp255(Tint)(const Tint value) { pragma(inline, true); @@ -291,7 +291,7 @@ ubyte clamp255(Tint)(const Tint value) { This function optimizes its runtime performance by substituting the division by 255 with an approximation using bitshifts. - Nonetheless, the its result are as accurate as a floating point + Nonetheless, its result are as accurate as a floating point division with 64-bit precision. Params: @@ -310,7 +310,7 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { // Accuracy verification static ubyte n255thsOfFP64(const ubyte nPercentage, const ubyte value) { - return (value * nPercentage / 255.0).round().castTo!ubyte(); + return (double(value) * double(nPercentage) / 255.0).round().castTo!ubyte(); } for (int value = ubyte.min; value <= ubyte.max; ++value) { @@ -395,7 +395,7 @@ enum BlendAccuracy { /// public void alphaBlend( BlendAccuracy accuracy, - ubyte function(const ubyte, const ubyte) pure blend = null, + ubyte function(const ubyte, const ubyte) blend = null, )( scope Pixel[] target, scope const Pixel[] source, From bd4ec9b45ae154fd58accec0110a9dfdfc72e078 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 26 May 2024 03:10:39 +0200 Subject: [PATCH 21/25] Implement overlay + hard-light + soft-light blend modes --- pixmappaint.d | 574 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 574 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index 66262ef..d245dab 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -18,6 +18,17 @@ import arsd.color; import arsd.core; import std.math : round; +/* + ## TODO + + - Refactoring the template-mess of blendPixel() & co. + - A bunch more blend modes + - Scaling + - Rotating + - Skewing + */ + +/// alias Color = arsd.color.Color; /// @@ -277,6 +288,479 @@ private { T min(T)(T a, T b) => (a <= b) ? a : b; } +/++ + Integer square root + +/ +ubyte intSqrt(const ubyte value) @safe pure nothrow @nogc { + switch (value) { + default: + // unreachable + assert(false, "ubyte != uint8"); + case 0: + return 0; + case 1: .. case 2: + return 1; + case 3: .. case 6: + return 2; + case 7: .. case 12: + return 3; + case 13: .. case 20: + return 4; + case 21: .. case 30: + return 5; + case 31: .. case 42: + return 6; + case 43: .. case 56: + return 7; + case 57: .. case 72: + return 8; + case 73: .. case 90: + return 9; + case 91: .. case 110: + return 10; + case 111: .. case 132: + return 11; + case 133: .. case 156: + return 12; + case 157: .. case 182: + return 13; + case 183: .. case 210: + return 14; + case 211: .. case 240: + return 15; + case 241: .. case 255: + return 16; + } +} + +/// +unittest { + assert(intSqrt(4) == 2); + assert(intSqrt(9) == 3); + assert(intSqrt(10) == 3); +} + +unittest { + import std.math : round, sqrt; + + foreach (n; ubyte.min .. ubyte.max + 1) { + ubyte fp = sqrt(float(n)).round().castTo!ubyte; + ubyte i8 = intSqrt(n.castTo!ubyte); + assert(fp == i8); + } +} + +/++ + Square root of a normalized integer + + Normalization: + `[0x00 .. 0xFF]` → `[0.0 .. 1.0]` + + Returns: + sqrt(value / 255f) * 255 + +/ +ubyte intNormalizedSqrt(const ubyte value) { + switch (value) { + default: + // unreachable + assert(false, "ubyte != uint8"); + case 0x00: + return 0x00; + case 0x01: + return 0x10; + case 0x02: + return 0x17; + case 0x03: + return 0x1C; + case 0x04: + return 0x20; + case 0x05: + return 0x24; + case 0x06: + return 0x27; + case 0x07: + return 0x2A; + case 0x08: + return 0x2D; + case 0x09: + return 0x30; + case 0x0A: + return 0x32; + case 0x0B: + return 0x35; + case 0x0C: + return 0x37; + case 0x0D: + return 0x3A; + case 0x0E: + return 0x3C; + case 0x0F: + return 0x3E; + case 0x10: + return 0x40; + case 0x11: + return 0x42; + case 0x12: + return 0x44; + case 0x13: + return 0x46; + case 0x14: + return 0x47; + case 0x15: + return 0x49; + case 0x16: + return 0x4B; + case 0x17: + return 0x4D; + case 0x18: + return 0x4E; + case 0x19: + return 0x50; + case 0x1A: + return 0x51; + case 0x1B: + return 0x53; + case 0x1C: + return 0x54; + case 0x1D: + return 0x56; + case 0x1E: + return 0x57; + case 0x1F: + return 0x59; + case 0x20: + return 0x5A; + case 0x21: + return 0x5C; + case 0x22: + return 0x5D; + case 0x23: + return 0x5E; + case 0x24: + return 0x60; + case 0x25: + return 0x61; + case 0x26: + return 0x62; + case 0x27: + return 0x64; + case 0x28: + return 0x65; + case 0x29: + return 0x66; + case 0x2A: + return 0x67; + case 0x2B: + return 0x69; + case 0x2C: + return 0x6A; + case 0x2D: + return 0x6B; + case 0x2E: + return 0x6C; + case 0x2F: + return 0x6D; + case 0x30: + return 0x6F; + case 0x31: + return 0x70; + case 0x32: + return 0x71; + case 0x33: + return 0x72; + case 0x34: + return 0x73; + case 0x35: + return 0x74; + case 0x36: + return 0x75; + case 0x37: + return 0x76; + case 0x38: + return 0x77; + case 0x39: + return 0x79; + case 0x3A: + return 0x7A; + case 0x3B: + return 0x7B; + case 0x3C: + return 0x7C; + case 0x3D: + return 0x7D; + case 0x3E: + return 0x7E; + case 0x3F: + return 0x7F; + case 0x40: + return 0x80; + case 0x41: + return 0x81; + case 0x42: + return 0x82; + case 0x43: + return 0x83; + case 0x44: + return 0x84; + case 0x45: + return 0x85; + case 0x46: + return 0x86; + case 0x47: .. case 0x48: + return 0x87; + case 0x49: + return 0x88; + case 0x4A: + return 0x89; + case 0x4B: + return 0x8A; + case 0x4C: + return 0x8B; + case 0x4D: + return 0x8C; + case 0x4E: + return 0x8D; + case 0x4F: + return 0x8E; + case 0x50: + return 0x8F; + case 0x51: + return 0x90; + case 0x52: .. case 0x53: + return 0x91; + case 0x54: + return 0x92; + case 0x55: + return 0x93; + case 0x56: + return 0x94; + case 0x57: + return 0x95; + case 0x58: + return 0x96; + case 0x59: .. case 0x5A: + return 0x97; + case 0x5B: + return 0x98; + case 0x5C: + return 0x99; + case 0x5D: + return 0x9A; + case 0x5E: + return 0x9B; + case 0x5F: .. case 0x60: + return 0x9C; + case 0x61: + return 0x9D; + case 0x62: + return 0x9E; + case 0x63: + return 0x9F; + case 0x64: .. case 0x65: + return 0xA0; + case 0x66: + return 0xA1; + case 0x67: + return 0xA2; + case 0x68: + return 0xA3; + case 0x69: .. case 0x6A: + return 0xA4; + case 0x6B: + return 0xA5; + case 0x6C: + return 0xA6; + case 0x6D: .. case 0x6E: + return 0xA7; + case 0x6F: + return 0xA8; + case 0x70: + return 0xA9; + case 0x71: .. case 0x72: + return 0xAA; + case 0x73: + return 0xAB; + case 0x74: + return 0xAC; + case 0x75: .. case 0x76: + return 0xAD; + case 0x77: + return 0xAE; + case 0x78: + return 0xAF; + case 0x79: .. case 0x7A: + return 0xB0; + case 0x7B: + return 0xB1; + case 0x7C: + return 0xB2; + case 0x7D: .. case 0x7E: + return 0xB3; + case 0x7F: + return 0xB4; + case 0x80: .. case 0x81: + return 0xB5; + case 0x82: + return 0xB6; + case 0x83: .. case 0x84: + return 0xB7; + case 0x85: + return 0xB8; + case 0x86: + return 0xB9; + case 0x87: .. case 0x88: + return 0xBA; + case 0x89: + return 0xBB; + case 0x8A: .. case 0x8B: + return 0xBC; + case 0x8C: + return 0xBD; + case 0x8D: .. case 0x8E: + return 0xBE; + case 0x8F: + return 0xBF; + case 0x90: .. case 0x91: + return 0xC0; + case 0x92: + return 0xC1; + case 0x93: .. case 0x94: + return 0xC2; + case 0x95: + return 0xC3; + case 0x96: .. case 0x97: + return 0xC4; + case 0x98: + return 0xC5; + case 0x99: .. case 0x9A: + return 0xC6; + case 0x9B: .. case 0x9C: + return 0xC7; + case 0x9D: + return 0xC8; + case 0x9E: .. case 0x9F: + return 0xC9; + case 0xA0: + return 0xCA; + case 0xA1: .. case 0xA2: + return 0xCB; + case 0xA3: .. case 0xA4: + return 0xCC; + case 0xA5: + return 0xCD; + case 0xA6: .. case 0xA7: + return 0xCE; + case 0xA8: + return 0xCF; + case 0xA9: .. case 0xAA: + return 0xD0; + case 0xAB: .. case 0xAC: + return 0xD1; + case 0xAD: + return 0xD2; + case 0xAE: .. case 0xAF: + return 0xD3; + case 0xB0: .. case 0xB1: + return 0xD4; + case 0xB2: + return 0xD5; + case 0xB3: .. case 0xB4: + return 0xD6; + case 0xB5: .. case 0xB6: + return 0xD7; + case 0xB7: + return 0xD8; + case 0xB8: .. case 0xB9: + return 0xD9; + case 0xBA: .. case 0xBB: + return 0xDA; + case 0xBC: + return 0xDB; + case 0xBD: .. case 0xBE: + return 0xDC; + case 0xBF: .. case 0xC0: + return 0xDD; + case 0xC1: .. case 0xC2: + return 0xDE; + case 0xC3: + return 0xDF; + case 0xC4: .. case 0xC5: + return 0xE0; + case 0xC6: .. case 0xC7: + return 0xE1; + case 0xC8: .. case 0xC9: + return 0xE2; + case 0xCA: + return 0xE3; + case 0xCB: .. case 0xCC: + return 0xE4; + case 0xCD: .. case 0xCE: + return 0xE5; + case 0xCF: .. case 0xD0: + return 0xE6; + case 0xD1: .. case 0xD2: + return 0xE7; + case 0xD3: + return 0xE8; + case 0xD4: .. case 0xD5: + return 0xE9; + case 0xD6: .. case 0xD7: + return 0xEA; + case 0xD8: .. case 0xD9: + return 0xEB; + case 0xDA: .. case 0xDB: + return 0xEC; + case 0xDC: .. case 0xDD: + return 0xED; + case 0xDE: .. case 0xDF: + return 0xEE; + case 0xE0: + return 0xEF; + case 0xE1: .. case 0xE2: + return 0xF0; + case 0xE3: .. case 0xE4: + return 0xF1; + case 0xE5: .. case 0xE6: + return 0xF2; + case 0xE7: .. case 0xE8: + return 0xF3; + case 0xE9: .. case 0xEA: + return 0xF4; + case 0xEB: .. case 0xEC: + return 0xF5; + case 0xED: .. case 0xEE: + return 0xF6; + case 0xEF: .. case 0xF0: + return 0xF7; + case 0xF1: .. case 0xF2: + return 0xF8; + case 0xF3: .. case 0xF4: + return 0xF9; + case 0xF5: .. case 0xF6: + return 0xFA; + case 0xF7: .. case 0xF8: + return 0xFB; + case 0xF9: .. case 0xFA: + return 0xFC; + case 0xFB: .. case 0xFC: + return 0xFD; + case 0xFD: .. case 0xFE: + return 0xFE; + case 0xFF: + return 0xFF; + } +} + +unittest { + import std.math : round, sqrt; + + foreach (n; ubyte.min .. ubyte.max + 1) { + ubyte fp = (sqrt(n / 255.0f) * 255).round().castTo!ubyte; + ubyte i8 = intNormalizedSqrt(n.castTo!ubyte); + assert(fp == i8); + } +} + /++ Limits a value to a maximum of 0xFF (= 255). +/ @@ -452,18 +936,46 @@ public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @safe { // ==== Blending functions ==== +/++ + Blend modes + + $(NOTE + As blending operations are implemented as integer calculations, + results may be slightly less precise than those from image manipulation + programs using floating-point math. + ) + + See_Also: + + +/ enum BlendMode { + /// none = 0, + /// replace = none, + /// normal = 1, + /// alpha = normal, + /// multiply, + /// screen, + /// + overlay, + /// + hardLight, + /// + softLight, + + /// darken, + /// lighten, + /// divide, } @@ -535,6 +1047,68 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( )(target, source); } +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.overlay) { + + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b < 0x80) { + return n255thsOf((2 * b).castTo!ubyte, f); + } + return castTo!ubyte( + 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - b)), (0xFF - f)) + ); + })(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.hardLight) { + + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (f < 0x80) { + return n255thsOf(castTo!ubyte(2 * f), b); + } + return castTo!ubyte( + 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - f)), (0xFF - b)) + ); + })(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.softLight) { + + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (f < 0x80) { + // dfmt off + return castTo!ubyte( + b - n255thsOf( + n255thsOf((0xFF - 2 * f).castTo!ubyte, b), + (0xFF - b), + ) + ); + // dfmt on + } + + // dfmt off + immutable ubyte d = (b < 0x40) + ? castTo!ubyte((b * (0x3FC + (((16 * b - 0xBF4) * b) / 255))) / 255) + : intNormalizedSqrt(b); + //dfmt on + + return castTo!ubyte( + b + n255thsOf((2 * f - 0xFF).castTo!ubyte, (d - b).castTo!ubyte) + ); + })(target, source); +} + /// ditto void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( ref Pixel target, From 5d839f34c2f736acdd483c81df60af1c05fc279d Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 26 May 2024 14:58:41 +0200 Subject: [PATCH 22/25] Implement color-burn, color-dodge, difference, exclusion and subtract blend modes --- pixmappaint.d | 88 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 88 insertions(+) diff --git a/pixmappaint.d b/pixmappaint.d index d245dab..aba8e9a 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -24,8 +24,10 @@ import std.math : round; - Refactoring the template-mess of blendPixel() & co. - A bunch more blend modes - Scaling + - Cropping - Rotating - Skewing + - HSL */ /// @@ -975,6 +977,17 @@ enum BlendMode { /// lighten, + /// + colorDodge, + /// + colorBurn, + + /// + difference, + /// + exclusion, + /// + subtract, /// divide, } @@ -1109,6 +1122,81 @@ void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( })(target, source); } +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.colorDodge) { + + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b == 0x00) { + return ubyte(0x00); + } + if (f == 0xFF) { + return ubyte(0xFF); + } + return min( + ubyte(0xFF), + clamp255((255 * b) / (0xFF - f)) + ); + })(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.colorBurn) { + + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b == 0xFF) { + return ubyte(0xFF); + } + if (f == 0x00) { + return ubyte(0x00); + } + + immutable m = min( + ubyte(0xFF), + clamp255(((0xFF - b) * 255) / f) + ); + return castTo!ubyte(0xFF - m); + })(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.difference) { + + return alphaBlend!(accuracy, + (b, f) => (b > f) ? castTo!ubyte(b - f) : castTo!ubyte(f - b) + )(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.exclusion) { + + return alphaBlend!(accuracy, + (b, f) => castTo!ubyte(b + f - (2 * n255thsOf(f, b))) + )(target, source); +} + +/// ditto +void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( + ref Pixel target, + const Pixel source, +) if (mode == Blend.subtract) { + + return alphaBlend!(accuracy, + (b, f) => (b > f) ? castTo!ubyte(b - f) : ubyte(0) + )(target, source); +} + /// ditto void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( ref Pixel target, From 1aae3674dcc2acfa9f12ba0c9485431d2547de29 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 26 May 2024 15:07:11 +0200 Subject: [PATCH 23/25] Improve documentation --- pixmappaint.d | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index aba8e9a..ee79997 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -291,7 +291,9 @@ private { } /++ - Integer square root + Calculates the square root + of an integer number + as an integer number. +/ ubyte intSqrt(const ubyte value) @safe pure nothrow @nogc { switch (value) { @@ -353,7 +355,9 @@ unittest { } /++ - Square root of a normalized integer + Calculates the square root + of the normalized value + representated by the input integer number. Normalization: `[0x00 .. 0xFF]` → `[0.0 .. 1.0]` From 0b2481bab64ccc04193c5fd2e0b99d8203da8261 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 26 May 2024 16:26:09 +0200 Subject: [PATCH 24/25] Refactor blending functions --- pixmappaint.d | 532 +++++++++++++++++++++++++------------------------- 1 file changed, 269 insertions(+), 263 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index ee79997..a5f1fd3 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -846,7 +846,7 @@ in (opacity <= 1.0) { pixmap.opacity = opacity255; } -// ==== Alpha-blending functions ==== +// ==== Blending functions ==== /++ Alpha-blending accuracy level @@ -882,66 +882,6 @@ enum BlendAccuracy { rgba = true, } -/// -public void alphaBlend( - BlendAccuracy accuracy, - ubyte function(const ubyte, const ubyte) blend = null, -)( - scope Pixel[] target, - scope const Pixel[] source, -) @trusted -in (source.length == target.length) { - foreach (immutable idx, ref pxTarget; target) { - alphaBlend(pxTarget, source.ptr[idx]); - } -} - -/// ditto -public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @safe { - return alphaBlend!(BlendAccuracy.rgba, null)(target, source); -} - -/// -public void alphaBlend( - BlendAccuracy accuracy, - ubyte function(const ubyte, const ubyte) blend = null, -)( - ref Pixel pxTarget, - const Pixel pxSource, -) @trusted { - pragma(inline, true); - - static if (accuracy) { - immutable alphaResult = clamp255(pxSource.a + n255thsOf(pxTarget.a, (0xFF - pxSource.a))); - //immutable alphaResult = clamp255(pxTarget.a + n255thsOf(pxSource.a, (0xFF - pxTarget.a))); - } - - immutable alphaSource = (pxSource.a | (pxSource.a << 8)); - immutable alphaTarget = (0xFFFF - alphaSource); - - foreach (immutable ib, ref px; pxTarget.components) { - static if (blend !is null) { - immutable bx = blend(px, pxSource.components.ptr[ib]); - } else { - immutable bx = pxSource.components.ptr[ib]; - } - immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); - immutable s = cast(ubyte)(((bx * alphaSource) + 0x8080) >> 16); - px = cast(ubyte)(d + s); - } - - static if (accuracy) { - pxTarget.a = alphaResult; - } -} - -/// ditto -public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @safe { - return alphaBlend!(BlendAccuracy.rgba, null)(pxTarget, pxSource); -} - -// ==== Blending functions ==== - /++ Blend modes @@ -1002,222 +942,292 @@ alias Blend = BlendMode; // undocumented enum blendNormal = BlendMode.normal; +/// +alias BlendFn = ubyte function(const ubyte background, const ubyte foreground) pure nothrow @nogc; + /++ - Blends pixel `source` into pixel `target`. + Blends `source` into `target` + with respect to the opacity of the source image (as stored in the alpha channel). + + See_Also: + [alphaBlendRGBA] and [alphaBlendRGB] are shorthand functions + in cases where no special blending algorithm is needed. +/ -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == BlendMode.replace) { - target = source; -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.alpha) { - return alphaBlend!accuracy(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.multiply) { - - return alphaBlend!(accuracy, - (a, b) => n255thsOf(a, b) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.screen) { - - return alphaBlend!(accuracy, - (a, b) => castTo!ubyte(0xFF - n255thsOf((0xFF - a), (0xFF - b))) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.darken) { - - return alphaBlend!(accuracy, - (a, b) => min(a, b) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.lighten) { - - return alphaBlend!(accuracy, - (a, b) => max(a, b) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.overlay) { - - return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { - if (b < 0x80) { - return n255thsOf((2 * b).castTo!ubyte, f); +template alphaBlend(BlendFn blend = null, BlendAccuracy accuracy = BlendAccuracy.rgba) { + /// ditto + public void alphaBlend(scope Pixel[] target, scope const Pixel[] source) @trusted + in (source.length == target.length) { + foreach (immutable idx, ref pxTarget; target) { + alphaBlend(pxTarget, source.ptr[idx]); } - return castTo!ubyte( - 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - b)), (0xFF - f)) - ); - })(target, source); -} + } -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.hardLight) { + /// ditto + public void alphaBlend(ref Pixel pxTarget, const Pixel pxSource) @trusted { + pragma(inline, true); - return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { - if (f < 0x80) { - return n255thsOf(castTo!ubyte(2 * f), b); - } - return castTo!ubyte( - 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - f)), (0xFF - b)) - ); - })(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.softLight) { - - return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { - if (f < 0x80) { - // dfmt off - return castTo!ubyte( - b - n255thsOf( - n255thsOf((0xFF - 2 * f).castTo!ubyte, b), - (0xFF - b), - ) - ); - // dfmt on + static if (accuracy == BlendAccuracy.rgba) { + immutable alphaResult = clamp255(pxSource.a + n255thsOf(pxTarget.a, (0xFF - pxSource.a))); + //immutable alphaResult = clamp255(pxTarget.a + n255thsOf(pxSource.a, (0xFF - pxTarget.a))); } - // dfmt off - immutable ubyte d = (b < 0x40) - ? castTo!ubyte((b * (0x3FC + (((16 * b - 0xBF4) * b) / 255))) / 255) - : intNormalizedSqrt(b); - //dfmt on + immutable alphaSource = (pxSource.a | (pxSource.a << 8)); + immutable alphaTarget = (0xFFFF - alphaSource); - return castTo!ubyte( - b + n255thsOf((2 * f - 0xFF).castTo!ubyte, (d - b).castTo!ubyte) - ); - })(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.colorDodge) { - - return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { - if (b == 0x00) { - return ubyte(0x00); - } - if (f == 0xFF) { - return ubyte(0xFF); - } - return min( - ubyte(0xFF), - clamp255((255 * b) / (0xFF - f)) - ); - })(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.colorBurn) { - - return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { - if (b == 0xFF) { - return ubyte(0xFF); - } - if (f == 0x00) { - return ubyte(0x00); + foreach (immutable ib, ref px; pxTarget.components) { + static if (blend !is null) { + immutable bx = blend(px, pxSource.components.ptr[ib]); + } else { + immutable bx = pxSource.components.ptr[ib]; + } + immutable d = cast(ubyte)(((px * alphaTarget) + 0x8080) >> 16); + immutable s = cast(ubyte)(((bx * alphaSource) + 0x8080) >> 16); + px = cast(ubyte)(d + s); } - immutable m = min( - ubyte(0xFF), - clamp255(((0xFF - b) * 255) / f) - ); - return castTo!ubyte(0xFF - m); - })(target, source); + static if (accuracy == BlendAccuracy.rgba) { + pxTarget.a = alphaResult; + } + } } /// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.difference) { - - return alphaBlend!(accuracy, - (b, f) => (b > f) ? castTo!ubyte(b - f) : castTo!ubyte(f - b) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.exclusion) { - - return alphaBlend!(accuracy, - (b, f) => castTo!ubyte(b + f - (2 * n255thsOf(f, b))) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.subtract) { - - return alphaBlend!(accuracy, - (b, f) => (b > f) ? castTo!ubyte(b - f) : ubyte(0) - )(target, source); -} - -/// ditto -void blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba)( - ref Pixel target, - const Pixel source, -) if (mode == Blend.divide) { - - return alphaBlend!(accuracy, - (b, f) => (f == 0) ? ubyte(0xFF) : clamp255(0xFF * b / f) - )(target, source); +template alphaBlend(BlendAccuracy accuracy, BlendFn blend = null) { + alias alphaBlend = alphaBlend!(blend, accuracy); } /++ - Blends the pixel data of `source` into `target`. + Blends `source` into `target` + with respect to the opacity of the source image (as stored in the alpha channel). + + This variant is $(slower than) [alphaBlendRGB], + but calculates the correct alpha-channel value of the target. + See [BlendAccuracy] for further explanation. + +/ +public void alphaBlendRGBA(scope Pixel[] target, scope const Pixel[] source) @safe { + return alphaBlend!(null, BlendAccuracy.rgba)(target, source); +} + +/// ditto +public void alphaBlendRGBA(ref Pixel pxTarget, const Pixel pxSource) @safe { + return alphaBlend!(null, BlendAccuracy.rgba)(pxTarget, pxSource); +} + +/++ + Blends `source` into `target` + with respect to the opacity of the source image (as stored in the alpha channel). + + This variant is $(B faster than) [alphaBlendRGBA], + but leads to a wrong alpha-channel value in the target. + Useful because of the performance advantage in cases where the resulting + alpha does not matter. + See [BlendAccuracy] for further explanation. + +/ +public void alphaBlendRGB(scope Pixel[] target, scope const Pixel[] source) @safe { + return alphaBlend!(null, BlendAccuracy.rgb)(target, source); +} + +/// ditto +public void alphaBlendRGB(ref Pixel pxTarget, const Pixel pxSource) @safe { + return alphaBlend!(null, BlendAccuracy.rgb)(pxTarget, pxSource); +} + +/++ + Blends pixel `source` into pixel `target` + using the requested $(B blending mode). + +/ +template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) { + + static if (mode == BlendMode.replace) { + /// ditto + void blendPixel(ref Pixel target, const Pixel source) { + target = source; + } + } + + static if (mode == BlendMode.alpha) { + /// ditto + void blendPixel(ref Pixel target, const Pixel source) { + return alphaBlend!accuracy(target, source); + } + } + + static if (mode == BlendMode.multiply) { + /// ditto + void blendPixel(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (a, b) => n255thsOf(a, b) + )(target, source); + } + } + + static if (mode == BlendMode.screen) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (a, b) => castTo!ubyte(0xFF - n255thsOf((0xFF - a), (0xFF - b))) + )(target, source); + } + } + + static if (mode == BlendMode.darken) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (a, b) => min(a, b) + )(target, source); + } + } + static if (mode == BlendMode.lighten) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (a, b) => max(a, b) + )(target, source); + } + } + + static if (mode == BlendMode.overlay) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b < 0x80) { + return n255thsOf((2 * b).castTo!ubyte, f); + } + return castTo!ubyte( + 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - b)), (0xFF - f)) + ); + })(target, source); + } + } + + static if (mode == BlendMode.hardLight) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (f < 0x80) { + return n255thsOf(castTo!ubyte(2 * f), b); + } + return castTo!ubyte( + 0xFF - n255thsOf(castTo!ubyte(2 * (0xFF - f)), (0xFF - b)) + ); + })(target, source); + } + } + + static if (mode == BlendMode.softLight) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (f < 0x80) { + // dfmt off + return castTo!ubyte( + b - n255thsOf( + n255thsOf((0xFF - 2 * f).castTo!ubyte, b), + (0xFF - b), + ) + ); + // dfmt on + } + + // dfmt off + immutable ubyte d = (b < 0x40) + ? castTo!ubyte((b * (0x3FC + (((16 * b - 0xBF4) * b) / 255))) / 255) + : intNormalizedSqrt(b); + //dfmt on + + return castTo!ubyte( + b + n255thsOf((2 * f - 0xFF).castTo!ubyte, (d - b).castTo!ubyte) + ); + })(target, source); + } + } + + static if (mode == BlendMode.colorDodge) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b == 0x00) { + return ubyte(0x00); + } + if (f == 0xFF) { + return ubyte(0xFF); + } + return min( + ubyte(0xFF), + clamp255((255 * b) / (0xFF - f)) + ); + })(target, source); + } + } + + static if (mode == BlendMode.colorBurn) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, function(const ubyte b, const ubyte f) { + if (b == 0xFF) { + return ubyte(0xFF); + } + if (f == 0x00) { + return ubyte(0x00); + } + + immutable m = min( + ubyte(0xFF), + clamp255(((0xFF - b) * 255) / f) + ); + return castTo!ubyte(0xFF - m); + })(target, source); + } + } + + static if (mode == BlendMode.difference) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (b, f) => (b > f) ? castTo!ubyte(b - f) : castTo!ubyte(f - b) + )(target, source); + } + } + + static if (mode == BlendMode.exclusion) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (b, f) => castTo!ubyte(b + f - (2 * n255thsOf(f, b))) + )(target, source); + } + } + + static if (mode == BlendMode.subtract) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (b, f) => (b > f) ? castTo!ubyte(b - f) : ubyte(0) + )(target, source); + } + } + + static if (mode == BlendMode.divide) { + /// ditto + void blendPixel()(ref Pixel target, const Pixel source) { + return alphaBlend!(accuracy, + (b, f) => (f == 0) ? ubyte(0xFF) : clamp255(0xFF * b / f) + )(target, source); + } + } +} + +/++ + Blends the pixel data of `source` into `target` + using the requested $(B blending mode). `source` and `target` MUST have the same length. +/ -void blendPixels(BlendMode mode, BlendAccuracy accuracy)(scope Pixel[] target, scope const Pixel[] source) @trusted +void blendPixels( + BlendMode mode, + BlendAccuracy accuracy, +)(scope Pixel[] target, scope const Pixel[] source) @trusted in (source.length == target.length) { static if (mode == BlendMode.replace) { // explicit optimization @@ -1236,11 +1246,7 @@ in (source.length == target.length) { } /// ditto -void blendPixels(BlendAccuracy accuracy = BlendAccuracy.rgba)( - scope Pixel[] target, - scope const Pixel[] source, - BlendMode mode, -) { +void blendPixels(BlendAccuracy accuracy)(scope Pixel[] target, scope const Pixel[] source, BlendMode mode) { import std.meta : NoDuplicates; import std.traits : EnumMembers; From cc64d0c10480b8baf7a664ba3692dce1d4bed715 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sun, 26 May 2024 16:56:45 +0200 Subject: [PATCH 25/25] Add color-inversion functions --- pixmappaint.d | 38 ++++++++++++++++++++++++++++++++++---- 1 file changed, 34 insertions(+), 4 deletions(-) diff --git a/pixmappaint.d b/pixmappaint.d index a5f1fd3..48ad2db 100644 --- a/pixmappaint.d +++ b/pixmappaint.d @@ -19,15 +19,15 @@ import arsd.core; import std.math : round; /* - ## TODO + ## TODO: - Refactoring the template-mess of blendPixel() & co. - - A bunch more blend modes - Scaling - Cropping - Rotating - Skewing - HSL + - Advanced blend modes (maybe) */ /// @@ -824,7 +824,7 @@ ubyte n255thsOf(const ubyte nPercentage, const ubyte value) { See_Also: Use [opacityF] with opacity values in percent (%). +/ -void opacity(ref Pixmap pixmap, const ubyte opacity) { +void opacity(Pixmap pixmap, const ubyte opacity) { foreach (ref px; pixmap.data) { px.a = opacity.n255thsOf(px.a); } @@ -839,13 +839,38 @@ void opacity(ref Pixmap pixmap, const ubyte opacity) { See_Also: Use [opacity] with 8-bit integer opacity values (in 255ths). +/ -void opacityF(ref Pixmap pixmap, const float opacity) +void opacityF(Pixmap pixmap, const float opacity) in (opacity >= 0) in (opacity <= 1.0) { immutable opacity255 = round(opacity * 255).castTo!ubyte; pixmap.opacity = opacity255; } +/++ + Inverts a color (to its negative color). + +/ +Pixel invert(const Pixel color) { + return Pixel( + 0xFF - color.r, + 0xFF - color.g, + 0xFF - color.b, + color.a, + ); +} + +/++ + Inverts all colors to produce a $(B negative image). + + $(TIP + Develops a positive image when applied to a negative one. + ) + +/ +void invert(Pixmap pixmap) { + foreach (ref px; pixmap.data) { + px = invert(px); + } +} + // ==== Blending functions ==== /++ @@ -1130,6 +1155,7 @@ template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) // dfmt on } + // TODO: optimize if possible // dfmt off immutable ubyte d = (b < 0x40) ? castTo!ubyte((b * (0x3FC + (((16 * b - 0xBF4) * b) / 255))) / 255) @@ -1216,6 +1242,10 @@ template blendPixel(BlendMode mode, BlendAccuracy accuracy = BlendAccuracy.rgba) )(target, source); } } + + //else { + // static assert(false, "Missing `blendPixel()` implementation for `BlendMode`.`" ~ mode ~ "`."); + //} } /++