mirror of https://github.com/adamdruppe/arsd.git
Finish downscaler implementation
This commit is contained in:
parent
b031783e84
commit
68a94f03c3
333
pixmappaint.d
333
pixmappaint.d
|
@ -334,8 +334,7 @@ private float round(float f) pure @nogc nothrow @trusted {
|
||||||
## TODO:
|
## TODO:
|
||||||
|
|
||||||
- Refactoring the template-mess of blendPixel() & co.
|
- Refactoring the template-mess of blendPixel() & co.
|
||||||
- Scaling
|
- Rotating (by arbitrary angles)
|
||||||
- Rotating
|
|
||||||
- Skewing
|
- Skewing
|
||||||
- HSL
|
- HSL
|
||||||
- Advanced blend modes (maybe)
|
- Advanced blend modes (maybe)
|
||||||
|
@ -422,6 +421,29 @@ struct UDecimal {
|
||||||
return UDecimal.make(rounded);
|
return UDecimal.make(rounded);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
|
public UDecimal roundEven() const {
|
||||||
|
const truncated = (_value & 0xFFFF_FFFF_0000_0000);
|
||||||
|
const delta = _value - truncated;
|
||||||
|
|
||||||
|
ulong rounded;
|
||||||
|
|
||||||
|
if (delta == 0x8000_0000) {
|
||||||
|
const bool floorIsOdd = ((truncated & 0x1_0000_0000) != 0);
|
||||||
|
// dfmt off
|
||||||
|
rounded = (floorIsOdd)
|
||||||
|
? truncated + 0x1_0000_0000 // ceil
|
||||||
|
: truncated; // floor
|
||||||
|
// dfmt on
|
||||||
|
} else if (delta > 0x8000_0000) {
|
||||||
|
rounded = truncated + 0x1_0000_0000;
|
||||||
|
} else {
|
||||||
|
rounded = truncated;
|
||||||
|
}
|
||||||
|
|
||||||
|
return UDecimal.make(rounded);
|
||||||
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
public UDecimal floor() const {
|
public UDecimal floor() const {
|
||||||
const truncated = (_value & 0xFFFF_FFFF_0000_0000);
|
const truncated = (_value & 0xFFFF_FFFF_0000_0000);
|
||||||
|
@ -2895,6 +2917,8 @@ enum ScalingFilter {
|
||||||
/++
|
/++
|
||||||
Bilinear interpolation
|
Bilinear interpolation
|
||||||
|
|
||||||
|
(Uses arithmetic mean for downscaling.)
|
||||||
|
|
||||||
$(TIP
|
$(TIP
|
||||||
Visual impression: “smooth”, “blurred”
|
Visual impression: “smooth”, “blurred”
|
||||||
)
|
)
|
||||||
|
@ -2931,7 +2955,6 @@ private static ScalingDirection scalingDirectionFromDelta(const int delta) @nogc
|
||||||
}
|
}
|
||||||
|
|
||||||
private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap target) @nogc {
|
private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap target) @nogc {
|
||||||
|
|
||||||
enum none = ScalingDirection.none;
|
enum none = ScalingDirection.none;
|
||||||
enum up = ScalingDirection.up;
|
enum up = ScalingDirection.up;
|
||||||
enum down = ScalingDirection.down;
|
enum down = ScalingDirection.down;
|
||||||
|
@ -2943,6 +2966,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
(UDecimal(source.width) / target.width),
|
(UDecimal(source.width) / target.width),
|
||||||
(UDecimal(source.height) / target.height),
|
(UDecimal(source.height) / target.height),
|
||||||
];
|
];
|
||||||
|
enum idxX = 0, idxY = 1;
|
||||||
|
|
||||||
// ==== Nearest Neighbor ====
|
// ==== Nearest Neighbor ====
|
||||||
static if (filter == ScalingFilter.nearest) {
|
static if (filter == ScalingFilter.nearest) {
|
||||||
|
@ -2987,27 +3011,27 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
const posDst = Point(x.castTo!int, y.castTo!int);
|
const posDst = Point(x.castTo!int, y.castTo!int);
|
||||||
|
|
||||||
const UDecimal[2] posSrc = [
|
const UDecimal[2] posSrc = [
|
||||||
(() @trusted => posDst.x * ratios.ptr[0])(),
|
posDst.x * ratios[idxX],
|
||||||
(() @trusted => posDst.y * ratios.ptr[1])(),
|
posDst.y * ratios[idxY],
|
||||||
];
|
];
|
||||||
|
|
||||||
const int[2] posSrcX = () {
|
const int[2] posSrcX = () {
|
||||||
int[2] result;
|
int[2] result;
|
||||||
if (directions[0] == none) {
|
if (directions[idxX] == none) {
|
||||||
result = [
|
result = [
|
||||||
posSrc[0].castTo!int,
|
posSrc[idxX].castTo!int,
|
||||||
posSrc[0].castTo!int,
|
posSrc[idxX].castTo!int,
|
||||||
];
|
];
|
||||||
} else if (directions[0] == up) {
|
} else if (directions[idxX] == up) {
|
||||||
result = [
|
result = [
|
||||||
min(sourceMaxX, posSrc[0].floor().castTo!int),
|
min(sourceMaxX, posSrc[idxX].floor().castTo!int),
|
||||||
min(sourceMaxX, posSrc[0].ceil().castTo!int),
|
min(sourceMaxX, posSrc[idxX].ceil().castTo!int),
|
||||||
];
|
];
|
||||||
} else {
|
} else /* if (directions[0] == down) */ {
|
||||||
const ratioXHalf = (ratios[0] >> 1);
|
const ratioXHalf = (ratios[idxX] >> 1);
|
||||||
result = [
|
result = [
|
||||||
max((posSrc[0] - ratioXHalf).round().castTo!int, 0),
|
max((posSrc[idxX] - ratioXHalf).roundEven().castTo!int, 0),
|
||||||
min((posSrc[0] + ratioXHalf).round().castTo!int, sourceMaxX),
|
min((posSrc[idxX] + ratioXHalf).roundEven().castTo!int, sourceMaxX),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
@ -3015,31 +3039,34 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
|
|
||||||
const int[2] posSrcY = () {
|
const int[2] posSrcY = () {
|
||||||
int[2] result;
|
int[2] result;
|
||||||
if (directions[1] == none) {
|
if (directions[idxY] == none) {
|
||||||
result = [
|
result = [
|
||||||
posSrc[1].castTo!int,
|
posSrc[idxY].castTo!int,
|
||||||
posSrc[1].castTo!int,
|
posSrc[idxY].castTo!int,
|
||||||
];
|
];
|
||||||
} else if (directions[1] == up) {
|
} else if (directions[idxY] == up) {
|
||||||
result = [
|
result = [
|
||||||
min(sourceMaxY, posSrc[1].floor().castTo!int),
|
min(sourceMaxY, posSrc[idxY].floor().castTo!int),
|
||||||
min(sourceMaxY, posSrc[1].ceil().castTo!int),
|
min(sourceMaxY, posSrc[idxY].ceil().castTo!int),
|
||||||
];
|
];
|
||||||
} else {
|
} else /* if (directions[idxY] == down) */ {
|
||||||
const ratioHalf = (ratios[1] >> 1);
|
const ratioHalf = (ratios[idxY] >> 1);
|
||||||
result = [
|
result = [
|
||||||
max((posSrc[1] - ratioHalf).round().castTo!int, 0),
|
max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0),
|
||||||
min((posSrc[1] + ratioHalf).round().castTo!int, sourceMaxY),
|
min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMaxY),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}();
|
}();
|
||||||
|
|
||||||
|
enum idxL = 0, idxR = 1;
|
||||||
|
enum idxT = 0, idxB = 1;
|
||||||
|
|
||||||
const Point[4] posNeighs = [
|
const Point[4] posNeighs = [
|
||||||
Point(posSrcX[0], posSrcY[0]),
|
Point(posSrcX[idxL], posSrcY[idxT]),
|
||||||
Point(posSrcX[1], posSrcY[0]),
|
Point(posSrcX[idxR], posSrcY[idxT]),
|
||||||
Point(posSrcX[0], posSrcY[1]),
|
Point(posSrcX[idxL], posSrcY[idxB]),
|
||||||
Point(posSrcX[1], posSrcY[1]),
|
Point(posSrcX[idxR], posSrcY[idxB]),
|
||||||
];
|
];
|
||||||
|
|
||||||
const Color[4] pxNeighs = [
|
const Color[4] pxNeighs = [
|
||||||
|
@ -3049,6 +3076,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
source.getPixel(posNeighs[3]),
|
source.getPixel(posNeighs[3]),
|
||||||
];
|
];
|
||||||
|
|
||||||
|
enum idxTL = 0, idxTR = 1, idxBL = 2, idxBR = 3;
|
||||||
|
|
||||||
// ====== Faux bilinear ======
|
// ====== Faux bilinear ======
|
||||||
static if (filter == ScalingFilter.fauxLinear) {
|
static if (filter == ScalingFilter.fauxLinear) {
|
||||||
auto pxInt = Pixel(0, 0, 0, 0);
|
auto pxInt = Pixel(0, 0, 0, 0);
|
||||||
|
@ -3062,96 +3091,206 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ====== Proper bilinear ======
|
// ====== Proper bilinear (up) + Avg (down) ======
|
||||||
static if (filter == ScalingFilter.bilinear) {
|
static if (filter == ScalingFilter.bilinear) {
|
||||||
// TODO: Downscaling looks bad as-is.
|
|
||||||
auto pxInt = Pixel(0, 0, 0, 0);
|
auto pxInt = Pixel(0, 0, 0, 0);
|
||||||
foreach (immutable ib, ref c; pxInt.components) {
|
foreach (immutable ib, ref c; pxInt.components) {
|
||||||
ulong[2] xSums;
|
ulong sampleX() {
|
||||||
|
pragma(inline, true);
|
||||||
|
|
||||||
// ======== X ========
|
if (directions[0] == none) {
|
||||||
if (directions[0] == none) {
|
return (() @trusted => pxNeighs[idxTL].components.ptr[ib])();
|
||||||
xSums = () @trusted {
|
} else if (directions[0] == down) {
|
||||||
ulong[2] result = [
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
pxNeighs[0].components.ptr[ib],
|
const posSampling = Point(posSrcX[idxL], posSrcY[idxT]);
|
||||||
pxNeighs[2].components.ptr[ib],
|
const samplingOffset = source.scanTo(posSampling);
|
||||||
];
|
const srcSamples = () @trusted {
|
||||||
return result;
|
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
||||||
}();
|
}();
|
||||||
} else if (directions[1] == down) {
|
|
||||||
xSums = [0, 0];
|
|
||||||
|
|
||||||
const UDecimal[2] deltasX = [
|
ulong xSum = 0;
|
||||||
posSrc[0] - posSrcX[0],
|
|
||||||
posSrcX[1] - posSrc[0],
|
|
||||||
];
|
|
||||||
|
|
||||||
const deltasXSum = (deltasX[0] + deltasX[1]).round().castTo!uint;
|
foreach (srcSample; srcSamples) {
|
||||||
const UDecimal[2] weightsX = [
|
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
||||||
deltasX[0] / deltasXSum,
|
}
|
||||||
deltasX[1] / deltasXSum,
|
|
||||||
];
|
|
||||||
|
|
||||||
() @trusted {
|
return (xSum / nSamples);
|
||||||
xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]).round().castTo!uint;
|
} else /* if (directions[0] == up) */ {
|
||||||
xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]).round().castTo!uint;
|
ulong xSum = 0;
|
||||||
|
|
||||||
xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]).round().castTo!uint;
|
const ulong[2] weightsX = () {
|
||||||
xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]).round().castTo!uint;
|
ulong[2] result;
|
||||||
}();
|
result[1] = posSrc[0].fractionalDigits;
|
||||||
} else {
|
result[0] = ulong(uint.max) + 1 - result[1];
|
||||||
xSums = [0, 0];
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
const ulong[2] weightsX = () {
|
() @trusted {
|
||||||
ulong[2] result;
|
xSum += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
||||||
result[1] = posSrc[0].fractionalDigits;
|
xSum += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
||||||
result[0] = ulong(uint.max) + 1 - result[1];
|
}();
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
|
|
||||||
() @trusted {
|
return (xSum >> 32);
|
||||||
xSums[0] += (pxNeighs[0].components.ptr[ib] * weightsX[0]);
|
|
||||||
xSums[0] += (pxNeighs[1].components.ptr[ib] * weightsX[1]);
|
|
||||||
|
|
||||||
xSums[1] += (pxNeighs[2].components.ptr[ib] * weightsX[0]);
|
|
||||||
xSums[1] += (pxNeighs[3].components.ptr[ib] * weightsX[1]);
|
|
||||||
}();
|
|
||||||
foreach (ref sum; xSums) {
|
|
||||||
sum >>= 32;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ======== Y ========
|
ulong[2] sampleXDual() {
|
||||||
if (directions[1] == none) {
|
pragma(inline, true);
|
||||||
c = clamp255(xSums[0]);
|
|
||||||
} else if (directions[1] == down) {
|
|
||||||
const UDecimal[2] deltasY = [
|
|
||||||
posSrc[1] - posSrcY[0],
|
|
||||||
posSrcY[1] - posSrc[1],
|
|
||||||
];
|
|
||||||
|
|
||||||
const deltasYSum = (deltasY[0] + deltasY[1]).round().castTo!uint;
|
if (directions[0] == none) {
|
||||||
const UDecimal[2] weightsY = [
|
return () @trusted {
|
||||||
deltasY[0] / deltasYSum,
|
ulong[2] result = [
|
||||||
deltasY[1] / deltasYSum,
|
pxNeighs[idxTL].components.ptr[ib],
|
||||||
];
|
pxNeighs[idxBL].components.ptr[ib],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
} else if (directions[0] == down) {
|
||||||
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
|
const Point[2] posSampling = [
|
||||||
|
Point(posSrcX[idxL], posSrcY[idxT]),
|
||||||
|
Point(posSrcX[idxL], posSrcY[idxB]),
|
||||||
|
];
|
||||||
|
|
||||||
auto ySum = UDecimal(0);
|
const int[2] samplingOffsets = [
|
||||||
ySum += ((xSums[0] & 0xFFFF_FFFF) * weightsY[0]);
|
source.scanTo(posSampling[0]),
|
||||||
ySum += ((xSums[1] & 0xFFFF_FFFF) * weightsY[1]);
|
source.scanTo(posSampling[1]),
|
||||||
|
];
|
||||||
|
|
||||||
c = clamp255(ySum.round().castTo!uint);
|
const srcSamples2 = () @trusted {
|
||||||
} else {
|
const(const(Pixel)[])[2] result = [
|
||||||
|
source.data.ptr[samplingOffsets[0] .. (samplingOffsets[0] + nSamples)],
|
||||||
|
source.data.ptr[samplingOffsets[1] .. (samplingOffsets[1] + nSamples)],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
ulong[2] xSums = [0, 0];
|
||||||
|
|
||||||
|
foreach (idx, srcSamples; srcSamples2) {
|
||||||
|
foreach (srcSample; srcSamples) {
|
||||||
|
() @trusted { xSums.ptr[idx] += srcSample.components.ptr[ib]; }();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xSums[] /= nSamples;
|
||||||
|
return xSums;
|
||||||
|
} else /* if (directions[0] == up) */ {
|
||||||
|
ulong[2] xSums = [0, 0];
|
||||||
|
|
||||||
|
const ulong[2] weightsX = () {
|
||||||
|
ulong[2] result;
|
||||||
|
result[1] = posSrc[0].fractionalDigits;
|
||||||
|
result[0] = ulong(uint.max) + 1 - result[1];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
() @trusted {
|
||||||
|
xSums[0] += (pxNeighs[idxTL].components.ptr[ib] * weightsX[0]);
|
||||||
|
xSums[0] += (pxNeighs[idxTR].components.ptr[ib] * weightsX[1]);
|
||||||
|
|
||||||
|
xSums[1] += (pxNeighs[idxBL].components.ptr[ib] * weightsX[0]);
|
||||||
|
xSums[1] += (pxNeighs[idxBR].components.ptr[ib] * weightsX[1]);
|
||||||
|
}();
|
||||||
|
|
||||||
|
foreach (ref sum; xSums) {
|
||||||
|
sum >>= 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
return xSums;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ulong sampleXMulti() {
|
||||||
|
pragma(inline, true);
|
||||||
|
|
||||||
|
const nLines = 1 + posSrcY[idxB] - posSrcY[idxT];
|
||||||
|
ulong ySum = 0;
|
||||||
|
|
||||||
|
alias ForeachLineCallback = ulong delegate(const Point posLine) @safe pure nothrow @nogc;
|
||||||
|
ulong foreachLine(scope ForeachLineCallback apply) {
|
||||||
|
ulong linesSum = 0;
|
||||||
|
foreach (lineY; posSrcY[idxT] .. (1 + posSrcY[idxB])) {
|
||||||
|
const posLine = Point(posSrcX[idxL], lineY);
|
||||||
|
linesSum += apply(posLine);
|
||||||
|
}
|
||||||
|
return linesSum;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (directions[0] == none) {
|
||||||
|
ySum = foreachLine(delegate(const Point posLine) {
|
||||||
|
const pxSrc = source.getPixel(posLine);
|
||||||
|
return ulong((() @trusted => pxSrc.components.ptr[ib])());
|
||||||
|
});
|
||||||
|
} else if (directions[0] == down) {
|
||||||
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
|
|
||||||
|
ySum = foreachLine(delegate(const Point posLine) {
|
||||||
|
const samplingOffset = source.scanTo(posLine);
|
||||||
|
const srcSamples = () @trusted {
|
||||||
|
return source.data.ptr[samplingOffset .. (samplingOffset + nSamples)];
|
||||||
|
}();
|
||||||
|
|
||||||
|
ulong xSum = 0;
|
||||||
|
|
||||||
|
foreach (srcSample; srcSamples) {
|
||||||
|
xSum += (() @trusted => srcSample.components.ptr[ib])();
|
||||||
|
}
|
||||||
|
|
||||||
|
return xSum;
|
||||||
|
});
|
||||||
|
|
||||||
|
ySum /= nSamples;
|
||||||
|
} else /* if (directions[0] == up) */ {
|
||||||
|
const nSamples = 1 + posSrcX[idxR] - posSrcX[idxL];
|
||||||
|
|
||||||
|
ySum = foreachLine(delegate(const Point posLine) {
|
||||||
|
ulong xSum = 0;
|
||||||
|
|
||||||
|
const ulong[2] weightsX = () {
|
||||||
|
ulong[2] result;
|
||||||
|
result[1] = posSrc[0].fractionalDigits;
|
||||||
|
result[0] = ulong(uint.max) + 1 - result[1];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
const samplingOffset = source.scanTo(posLine);
|
||||||
|
ubyte[2] pxcLR = () @trusted {
|
||||||
|
ubyte[2] result = [
|
||||||
|
source.data.ptr[samplingOffset].components.ptr[ib],
|
||||||
|
source.data.ptr[samplingOffset + nSamples].components.ptr[ib],
|
||||||
|
];
|
||||||
|
return result;
|
||||||
|
}();
|
||||||
|
|
||||||
|
xSum += (pxcLR[idxL] * weightsX[idxL]);
|
||||||
|
xSum += (pxcLR[idxR] * weightsX[idxR]);
|
||||||
|
|
||||||
|
return (xSum >> 32);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return (ySum / nLines);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (directions[idxY] == none) {
|
||||||
|
c = clamp255(sampleX());
|
||||||
|
} else if (directions[idxY] == down) {
|
||||||
|
c = clamp255(sampleXMulti());
|
||||||
|
} else /* if (directions[idxY] == up) */ {
|
||||||
|
// looks ass
|
||||||
const ulong[2] weightsY = () {
|
const ulong[2] weightsY = () {
|
||||||
ulong[2] result;
|
ulong[2] result;
|
||||||
result[1] = posSrc[1].fractionalDigits;
|
result[idxB] = posSrc[1].fractionalDigits;
|
||||||
result[0] = ulong(uint.max) + 1 - result[1];
|
result[idxT] = ulong(uint.max) + 1 - result[idxB];
|
||||||
return result;
|
return result;
|
||||||
}();
|
}();
|
||||||
|
|
||||||
|
const xSums = sampleXDual();
|
||||||
|
|
||||||
ulong ySum = 0;
|
ulong ySum = 0;
|
||||||
ySum += (xSums[0] * weightsY[0]);
|
ySum += (xSums[idxT] * weightsY[idxT]);
|
||||||
ySum += (xSums[1] * weightsY[1]);
|
ySum += (xSums[idxB] * weightsY[idxB]);
|
||||||
|
|
||||||
const xySum = (ySum >> 32);
|
const xySum = (ySum >> 32);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue