net.algart.maps.pyramids.io.api.sources.RotatingPlanePyramidSource Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scichains-maps Show documentation
Show all versions of scichains-maps Show documentation
Open-source libraries, providing the base functions for SciChains product in area of computer vision.
/*
* The MIT License (MIT)
*
* Copyright (c) 2017-2024 Daniel Alievsky, AlgART Laboratory (http://algart.net)
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package net.algart.maps.pyramids.io.api.sources;
import net.algart.arrays.Arrays;
import net.algart.arrays.*;
import net.algart.maps.pyramids.io.api.PlanePyramidSource;
import net.algart.math.IPoint;
import net.algart.math.IRectangularArea;
import net.algart.math.functions.Func;
import net.algart.math.functions.LinearOperator;
import java.nio.channels.NotYetConnectedException;
import java.util.*;
public final class RotatingPlanePyramidSource implements PlanePyramidSource {
private static final boolean DEBUG_MODE = false;
private static final System.Logger LOG = System.getLogger(RotatingPlanePyramidSource.class.getName());
public enum RotationMode {
NONE(1, 0, 0, 0, false, 0),
CLOCKWISE_90(0, 1, 0, 1, true, 90),
CLOCKWISE_180(-1, 0, 1, 1, false, 180),
CLOCKWISE_270(0, -1, 1, 0, true, 270);
final int rotationInDegrees;
final long cos; // long field necessary for precise calculation of the bounds of the rectangular area
final long sin;
final long bX;
final long bY;
final boolean switchWidthAndHeight;
RotationMode(
long cos,
long sin,
long bX,
long bY,
boolean switchWidthAndHeight,
int rotationInDegrees) {
assert bX == 0 || bX == 1 || bY == 0 || bY == 1;
this.cos = cos;
this.sin = sin;
this.bX = bX;
this.bY = bY;
this.switchWidthAndHeight = switchWidthAndHeight;
this.rotationInDegrees = rotationInDegrees;
}
public static RotationMode valueOf(int clockwiseRotationInDegrees) {
return switch (clockwiseRotationInDegrees) {
case 0 -> NONE;
case 90 -> CLOCKWISE_90;
case 180 -> CLOCKWISE_180;
case 270 -> CLOCKWISE_270;
default -> throw new IllegalArgumentException(RotationMode.class.getCanonicalName()
+ " does not support rotation by " + clockwiseRotationInDegrees + " degree");
};
}
public static boolean isAngleSupported(int clockwiseRotationInDegrees) {
return switch (clockwiseRotationInDegrees) {
case 0, 90, 180, 270 -> true;
default -> false;
};
}
public int rotationInDegrees() {
return rotationInDegrees;
}
public RotationMode reverse() {
return rotationInDegrees == 0 ? this : valueOf(360 - rotationInDegrees);
}
public LinearOperator operator(long width, long height) {
return LinearOperator.getInstance(a(), b(width, height));
}
public boolean isSwitchWidthAndHeight() {
return switchWidthAndHeight;
}
public long[] correctDimensions(long[] dimensionsToCorrect) {
if (switchWidthAndHeight) {
dimensionsToCorrect = dimensionsToCorrect.clone();
long temp = dimensionsToCorrect[2];
dimensionsToCorrect[2] = dimensionsToCorrect[1];
dimensionsToCorrect[1] = temp;
}
return dimensionsToCorrect;
}
public int[] correctDimensions(int[] dimensionsToCorrect) {
if (switchWidthAndHeight) {
dimensionsToCorrect = dimensionsToCorrect.clone();
int temp = dimensionsToCorrect[2];
dimensionsToCorrect[2] = dimensionsToCorrect[1];
dimensionsToCorrect[1] = temp;
}
return dimensionsToCorrect;
}
public long[] correctWidthAndHeight(long[] widthAndHeightToCorrect) {
if (switchWidthAndHeight) {
widthAndHeightToCorrect = widthAndHeightToCorrect.clone();
long temp = widthAndHeightToCorrect[1];
widthAndHeightToCorrect[1] = widthAndHeightToCorrect[0];
widthAndHeightToCorrect[0] = temp;
}
return widthAndHeightToCorrect;
}
public int[] correctWidthAndHeight(int[] widthAndHeightToCorrect) {
if (switchWidthAndHeight) {
widthAndHeightToCorrect = widthAndHeightToCorrect.clone();
int temp = widthAndHeightToCorrect[1];
widthAndHeightToCorrect[1] = widthAndHeightToCorrect[0];
widthAndHeightToCorrect[0] = temp;
}
return widthAndHeightToCorrect;
}
// Note that parentImageWidth/parentImageHeight are in ROTATED coordinate system (parent source),
// if we consider from/to as arguments of readSubMatrix of the required virtual image (this source).
public long[] correctFromAndTo(
long parentImageWidth, long parentImageHeight, long fromX, long fromY, long toX, long toY) {
// We recommend to understand the following algorithm in floating-point coordinates, it will be more clear
long rotatedFromX = cos * fromX + sin * fromY + bX * parentImageWidth;
long rotatedFromY = -sin * fromX + cos * fromY + bY * parentImageHeight;
long rotatedSizeX = cos * (toX - fromX) + sin * (toY - fromY);
long rotatedSizeY = -sin * (toX - fromX) + cos * (toY - fromY);
// System.out.printf("Rotated = %d,%d; %dx%d%n", rotatedFromX, rotatedFromY, rotatedSizeX, rotatedSizeY);
long newFromX = rotatedSizeX >= 0 ? rotatedFromX : rotatedFromX + rotatedSizeX;
long newFromY = rotatedSizeY >= 0 ? rotatedFromY : rotatedFromY + rotatedSizeY;
long newToX = newFromX + Math.abs(rotatedSizeX);
long newToY = newFromY + Math.abs(rotatedSizeY);
// System.out.printf("Result = %d,%d; %d,%d%n", newFromX, newFromY, newToX, newToY);
return new long[]{newFromX, newFromY, newToX, newToY};
}
public IPoint correctPoint(
long parentImageWidth, long parentImageHeight, IPoint point) {
long rotatedX = cos * point.x() + sin * point.y() + bX * parentImageWidth;
long rotatedY = -sin * point.x() + cos * point.y() + bY * parentImageHeight;
return IPoint.valueOf(rotatedX, rotatedY);
}
public IRectangularArea correctRectangle(
long parentImageWidth, long parentImageHeight, IRectangularArea rectangle) {
long fromX = rectangle.min(0);
long fromY = rectangle.min(1);
long toX = rectangle.max(0) + 1;
long toY = rectangle.max(1) + 1;
long[] fromAndTo = correctFromAndTo(parentImageWidth, parentImageHeight, fromX, fromY, toX, toY);
return IRectangularArea.valueOf(
IPoint.valueOf(fromAndTo[0], fromAndTo[1]),
IPoint.valueOf(fromAndTo[2] - 1, fromAndTo[3] - 1));
}
public Matrix extends PArray> asRotated(Matrix extends PArray> m) {
final long[] newDimensions = correctDimensions(m.dimensions());
if (Arrays.isNCopies(m.array())) {
assert Arrays.longMul(newDimensions) == m.size();
// - it is convenient that it is so, because allows simply to use the same array
return Matrices.matrix(m.array(), newDimensions);
}
final LinearOperator operator = operator(m.dim(1), m.dim(2));
final Func source = Matrices.asInterpolationFunc(m, Matrices.InterpolationMethod.STEP_FUNCTION, DEBUG_MODE);
final Func rotated = operator.apply(source);
return Matrices.asCoordFuncMatrix(rotated, m.type(PArray.class), newDimensions);
}
private double[] a() {
return new double[]{
1.0, 0.0, 0.0, // coordinate #0 is 0,1,2 for R,G,B
0.0, cos, sin,
0.0, -sin, cos,
};
}
private double[] b(long width, long height) {
return new double[]{
0.0,
bX * (width - 1),
bY * (height - 1)
};
}
}
private final PlanePyramidSource parent;
private final RotationMode rotationMode;
private MemoryModel memoryModel = Arrays.SMM;
private RotatingPlanePyramidSource(PlanePyramidSource parent, RotationMode rotationMode) {
Objects.requireNonNull(parent, "Null parent");
Objects.requireNonNull(rotationMode, "Null rotationMode");
assert rotationMode != RotationMode.NONE;
this.parent = parent;
this.rotationMode = rotationMode;
}
public static PlanePyramidSource newInstance(PlanePyramidSource parent, RotationMode rotationMode) {
Objects.requireNonNull(parent, "Null parent");
if (rotationMode == RotationMode.NONE) {
return parent;
}
return new RotatingPlanePyramidSource(parent, rotationMode);
}
public MemoryModel getMemoryModel() {
return memoryModel;
}
public RotatingPlanePyramidSource setMemoryModel(MemoryModel memoryModel) {
this.memoryModel = Objects.requireNonNull(memoryModel, "Null memoryModel");
return this;
}
public int numberOfResolutions() {
return parent.numberOfResolutions();
}
public int compression() {
return parent.compression();
}
public int bandCount() {
return parent.bandCount();
}
public boolean isResolutionLevelAvailable(int resolutionLevel) {
return parent.isResolutionLevelAvailable(resolutionLevel);
}
public boolean[] getResolutionLevelsAvailability() {
return parent.getResolutionLevelsAvailability();
}
public long[] dimensions(int resolutionLevel) {
return rotationMode.correctDimensions(parent.dimensions(resolutionLevel));
}
public boolean isElementTypeSupported() {
return parent.isElementTypeSupported();
}
public Class> elementType() throws UnsupportedOperationException {
return parent.elementType();
}
@Override
public OptionalDouble pixelSizeInMicrons() {
return parent.pixelSizeInMicrons();
}
@Override
public OptionalDouble magnification() {
return parent.magnification();
}
public List zeroLevelActualRectangles() {
final List parentRectangles = parent.zeroLevelActualRectangles();
if (parentRectangles == null) {
return null;
}
final long[] rotatedDim = this.dimensions(0);
final long rotatedWidth = rotatedDim[DIM_WIDTH];
final long rotatedHeight = rotatedDim[DIM_HEIGHT];
final RotationMode reverseRotation = rotationMode.reverse();
// Note: here we have reverse task in comparison with readSubMatrix
final List result = new ArrayList(parentRectangles.size());
for (IRectangularArea parentRectangle : parentRectangles) {
final IRectangularArea rotatedRectangle = reverseRotation.correctRectangle(
rotatedWidth, rotatedHeight, parentRectangle);
LOG.log(System.Logger.Level.DEBUG, () -> String.format(Locale.US,
"Rotating zero-level actual rectangle %s by %d degree to %s inside %dx%d",
parentRectangle, rotationMode.rotationInDegrees, rotatedRectangle, rotatedWidth, rotatedHeight));
result.add(rotatedRectangle);
}
return result;
}
@Override
public List>> zeroLevelActualAreaBoundaries() {
final List>> boundaries = parent.zeroLevelActualAreaBoundaries();
if (boundaries == null) {
return null;
}
final long[] rotatedDim = this.dimensions(0);
final long rotatedWidth = rotatedDim[DIM_WIDTH];
final long rotatedHeight = rotatedDim[DIM_HEIGHT];
final RotationMode reverseRotation = rotationMode.reverse();
final List>> result = new ArrayList>>();
for (List> area : boundaries) {
final List> rotatedArea = new ArrayList>();
for (List boundary : area) {
final List rotatedBoundary = new ArrayList();
for (IPoint p : boundary) {
rotatedBoundary.add(reverseRotation.correctPoint(rotatedWidth, rotatedHeight, p));
}
rotatedArea.add(rotatedBoundary);
}
result.add(rotatedArea);
}
return result;
}
public Matrix extends PArray> readSubMatrix(int resolutionLevel, long fromX, long fromY, long toX, long toY) {
long t1 = System.nanoTime();
long[] parentDim = parent.dimensions(resolutionLevel);
long[] fromAndTo = rotationMode.correctFromAndTo(
parentDim[DIM_WIDTH], parentDim[DIM_HEIGHT], fromX, fromY, toX, toY);
Matrix extends PArray> parentSubMatrix = parent.readSubMatrix(
resolutionLevel, fromAndTo[0], fromAndTo[1], fromAndTo[2], fromAndTo[3]);
long t2 = System.nanoTime();
Matrix extends PArray> rotated = rotated(parentSubMatrix);
long t3 = System.nanoTime();
LOG.log(System.Logger.Level.DEBUG, () -> String.format(Locale.US,
"%s completed reading %d..%d x %d..%d, level %d in %.3f ms " +
"(%.3f parent reading + %.3f rotation)",
((Object) this).getClass().getSimpleName(), fromX, toX, fromY, toY, resolutionLevel,
(t3 - t1) * 1e-6, (t2 - t1) * 1e-6, (t3 - t2) * 1e-6));
return rotated;
}
public boolean isFullMatrixSupported() {
return parent.isFullMatrixSupported();
}
public Matrix extends PArray> readFullMatrix(int resolutionLevel)
throws UnsupportedOperationException {
return rotated(parent.readFullMatrix(resolutionLevel));
}
public boolean isSpecialMatrixSupported(SpecialImageKind kind) {
return parent.isSpecialMatrixSupported(kind);
}
public Optional> readSpecialMatrix(SpecialImageKind kind)
throws NotYetConnectedException {
if (kind == SpecialImageKind.NONE) {
return Optional.empty();
}
return parent.readSpecialMatrix(kind).map(this::rotated);
}
public boolean isDataReady() {
return parent.isDataReady();
}
public Optional metadata() {
return parent.metadata();
}
public void loadResources() {
parent.loadResources();
}
public void freeResources(FlushMode flushMode) {
parent.freeResources(flushMode);
}
@Override
public String toString() {
return "RotatingPlanePyramidSource-" + rotationMode.rotationInDegrees() + ", based on " + parent;
}
private Matrix extends PArray> rotated(Matrix extends PArray> m) {
final long[] newDimensions = rotationMode.correctDimensions(m.dimensions());
if (Arrays.isNCopies(m.array())) {
assert Arrays.longMul(newDimensions) == m.size();
// - it is convenient, because allows simply to use the same array
return Matrices.matrix(m.array(), newDimensions);
}
final Matrix extends PArray> lazy = rotationMode.asRotated(m);
final Matrix extends UpdatablePArray> actual = memoryModel.newMatrix(
Arrays.SystemSettings.maxTempJavaMemory(),
UpdatablePArray.class,
lazy.elementType(),
lazy.dimensions());
Matrices.copy(null, actual, lazy, 0, false);
return actual;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy