mirror of https://github.com/adamdruppe/arsd.git
Refactor y foreach loop of bilinear scaler
This commit is contained in:
parent
0b288d385f
commit
d9e1e0e84e
176
pixmappaint.d
176
pixmappaint.d
|
@ -2995,6 +2995,8 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
enum uint udecimalHalfFD = udecimalHalf.fractionalDigits;
|
enum uint udecimalHalfFD = udecimalHalf.fractionalDigits;
|
||||||
|
|
||||||
enum idxX = 0, idxY = 1;
|
enum idxX = 0, idxY = 1;
|
||||||
|
enum idxL = 0, idxR = 1;
|
||||||
|
enum idxT = 0, idxB = 1;
|
||||||
|
|
||||||
const int[2] sourceMax = [
|
const int[2] sourceMax = [
|
||||||
(source.width - 1),
|
(source.width - 1),
|
||||||
|
@ -3038,110 +3040,92 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
// ==== Bilinear ====
|
// ==== Bilinear ====
|
||||||
static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) {
|
static if ((filter == ScalingFilter.bilinear) || (filter == ScalingFilter.fauxLinear)) {
|
||||||
void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() {
|
void scaleToLinearImpl(ScalingDirection directionX, ScalingDirection directionY)() {
|
||||||
|
|
||||||
|
int[2] posSrcCenterToInterpolationTargets(
|
||||||
|
ScalingDirection direction,
|
||||||
|
)(
|
||||||
|
UDecimal posSrcCenter,
|
||||||
|
UDecimal ratioHalf,
|
||||||
|
int sourceMax,
|
||||||
|
) {
|
||||||
|
int[2] result;
|
||||||
|
static if (direction == none) {
|
||||||
|
const value = posSrcCenter.castTo!int;
|
||||||
|
result = [
|
||||||
|
value,
|
||||||
|
value,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
static if (direction == up) {
|
||||||
|
if (posSrcCenter < udecimalHalf) {
|
||||||
|
result = [
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
const floor = posSrcCenter.castTo!uint;
|
||||||
|
if (posSrcCenter.fractionalDigits == udecimalHalfFD) {
|
||||||
|
result = [
|
||||||
|
floor,
|
||||||
|
floor,
|
||||||
|
];
|
||||||
|
} else if (posSrcCenter.fractionalDigits > udecimalHalfFD) {
|
||||||
|
const upper = min((floor + 1), sourceMax);
|
||||||
|
result = [
|
||||||
|
floor,
|
||||||
|
upper,
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
result = [
|
||||||
|
floor - 1,
|
||||||
|
floor,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static if (direction == down) {
|
||||||
|
result = [
|
||||||
|
max((posSrcCenter - ratioHalf).roundEven().castTo!uint, 0),
|
||||||
|
min((posSrcCenter + ratioHalf).roundEven().castTo!uint, sourceMax),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
auto dst = PixmapScannerRW(target);
|
auto dst = PixmapScannerRW(target);
|
||||||
|
|
||||||
size_t y = 0;
|
size_t y = 0;
|
||||||
foreach (dstLine; dst) {
|
foreach (dstLine; dst) {
|
||||||
|
const posDstY = y.castTo!uint;
|
||||||
|
const UDecimal posSrcCenterY = posDstY * ratios[idxY] + ratiosHalf[idxY];
|
||||||
|
|
||||||
|
const int[2] posSrcY = posSrcCenterToInterpolationTargets!(directionY)(
|
||||||
|
posSrcCenterY,
|
||||||
|
ratiosHalf[idxY],
|
||||||
|
sourceMax[idxY],
|
||||||
|
);
|
||||||
|
|
||||||
foreach (x, ref pxDst; dstLine) {
|
foreach (x, ref pxDst; dstLine) {
|
||||||
|
const posDstX = x.castTo!uint;
|
||||||
const int[2] posDst = [
|
const int[2] posDst = [
|
||||||
x.castTo!uint,
|
posDstX,
|
||||||
y.castTo!uint,
|
posDstY,
|
||||||
];
|
];
|
||||||
|
|
||||||
const UDecimal[2] posSrc = [
|
const posSrcCenterX = posDst[idxX] * ratios[idxX] + ratiosHalf[idxX];
|
||||||
posDst[idxX] * ratios[idxX] + ratiosHalf[idxX],
|
|
||||||
posDst[idxY] * ratios[idxY] + ratiosHalf[idxY],
|
const UDecimal[2] posSrcCenter = [
|
||||||
|
posSrcCenterX,
|
||||||
|
posSrcCenterY,
|
||||||
];
|
];
|
||||||
|
|
||||||
const int[2] posSrcX = () {
|
const int[2] posSrcX = posSrcCenterToInterpolationTargets!(directionX)(
|
||||||
int[2] result;
|
posSrcCenterX,
|
||||||
static if (directionX == none) {
|
ratiosHalf[idxX],
|
||||||
const value = posSrc[idxX].castTo!int;
|
sourceMax[idxX],
|
||||||
result = [
|
);
|
||||||
value,
|
|
||||||
value,
|
|
||||||
];
|
|
||||||
} else static if (directionX == up) {
|
|
||||||
if (posSrc[idxX] < udecimalHalf) {
|
|
||||||
result = [
|
|
||||||
0,
|
|
||||||
0,
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
const floor = posSrc[idxX].castTo!uint;
|
|
||||||
if (posSrc[idxX].fractionalDigits == udecimalHalfFD) {
|
|
||||||
result = [
|
|
||||||
floor,
|
|
||||||
floor,
|
|
||||||
];
|
|
||||||
} else if (posSrc[idxX].fractionalDigits > udecimalHalfFD) {
|
|
||||||
const upper = min((floor + 1), sourceMax[idxX]);
|
|
||||||
result = [
|
|
||||||
floor,
|
|
||||||
upper,
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
result = [
|
|
||||||
floor - 1,
|
|
||||||
floor,
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else /* if (directionX == down) */ {
|
|
||||||
const ratioXHalf = (ratios[idxX] >> 1);
|
|
||||||
result = [
|
|
||||||
max((posSrc[idxX] - ratioXHalf).roundEven().castTo!uint, 0),
|
|
||||||
min((posSrc[idxX] + ratioXHalf).roundEven().castTo!uint, sourceMax[idxX]),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
|
|
||||||
const int[2] posSrcY = () {
|
|
||||||
int[2] result;
|
|
||||||
static if (directionY == none) {
|
|
||||||
result = [
|
|
||||||
posSrc[idxY].castTo!int,
|
|
||||||
posSrc[idxY].castTo!int,
|
|
||||||
];
|
|
||||||
} else static if (directionY == up) {
|
|
||||||
if (posSrc[idxY] < udecimalHalf) {
|
|
||||||
result = [
|
|
||||||
0,
|
|
||||||
0,
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
const floor = posSrc[idxY].castTo!uint;
|
|
||||||
if (posSrc[idxY].fractionalDigits == udecimalHalfFD) {
|
|
||||||
result = [
|
|
||||||
floor,
|
|
||||||
floor,
|
|
||||||
];
|
|
||||||
} else if (posSrc[idxY].fractionalDigits > udecimalHalfFD) {
|
|
||||||
const upper = min((floor + 1), sourceMax[idxY]);
|
|
||||||
result = [
|
|
||||||
floor,
|
|
||||||
upper,
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
result = [
|
|
||||||
floor - 1,
|
|
||||||
floor,
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else /* if (directionY == down) */ {
|
|
||||||
const ratioHalf = (ratios[idxY] >> 1);
|
|
||||||
result = [
|
|
||||||
max((posSrc[idxY] - ratioHalf).roundEven().castTo!int, 0),
|
|
||||||
min((posSrc[idxY] + ratioHalf).roundEven().castTo!int, sourceMax[idxY]),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}();
|
|
||||||
|
|
||||||
enum idxL = 0, idxR = 1;
|
|
||||||
enum idxT = 0, idxB = 1;
|
|
||||||
|
|
||||||
const Point[4] posNeighs = [
|
const Point[4] posNeighs = [
|
||||||
Point(posSrcX[idxL], posSrcY[idxT]),
|
Point(posSrcX[idxL], posSrcY[idxT]),
|
||||||
|
@ -3330,7 +3314,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
|
|
||||||
const ulong[2] weightsX = () {
|
const ulong[2] weightsX = () {
|
||||||
ulong[2] result;
|
ulong[2] result;
|
||||||
result[0] = (udecimalHalf + posSrcX[1] - posSrc[idxX]).fractionalDigits;
|
result[0] = (udecimalHalf + posSrcX[1] - posSrcCenter[idxX]).fractionalDigits;
|
||||||
result[1] = ulong(uint.max) + 1 - result[0];
|
result[1] = ulong(uint.max) + 1 - result[0];
|
||||||
return result;
|
return result;
|
||||||
}();
|
}();
|
||||||
|
@ -3396,7 +3380,7 @@ private void scaleToImpl(ScalingFilter filter)(const Pixmap source, Pixmap targe
|
||||||
} else /* if (directionY == up) */ {
|
} else /* if (directionY == up) */ {
|
||||||
const ulong[2] weightsY = () {
|
const ulong[2] weightsY = () {
|
||||||
ulong[2] result;
|
ulong[2] result;
|
||||||
result[0] = (udecimalHalf + posSrcY[1] - posSrc[idxY]).fractionalDigits;
|
result[0] = (udecimalHalf + posSrcY[1] - posSrcCenter[idxY]).fractionalDigits;
|
||||||
result[1] = ulong(uint.max) + 1 - result[0];
|
result[1] = ulong(uint.max) + 1 - result[0];
|
||||||
return result;
|
return result;
|
||||||
}();
|
}();
|
||||||
|
|
Loading…
Reference in New Issue