Source code for phenotypic.correction._grid_aligner
from __future__ import annotations
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from phenotypic import GridImage
import numpy as np
from scipy.spatial.distance import euclidean
from scipy.optimize import minimize_scalar
from phenotypic.abc_ import GridCorrector
from phenotypic.tools.constants_ import OBJECT, BBOX, GRID
[docs]
class GridAligner(GridCorrector):
"""Calculates the optimal gridding orientation based on the alignment of the objects in the image and rotates the image accordingly.
This class inherits from `GridCorrector` and is designed to calculate the optimal gridding orientation. This is used to align the image,
and helps to improve the quality of automated gridding results. It's highly recommended to redetect objects in the image after alignment.
"""
def __init__(self, axis: int = 0, mode: str = "edge"):
self.axis = axis
self.mode = mode
def _operate(self, image: GridImage):
"""Calculates the optimal rotation angle and applies it to a grid image for alignment along the specified axis.
The method performs alignment of a `GridImage` object along either nrows or columns based on the specified
axis. It calculates the linear regression slope and intercept for the axis, determines geometric properties of the grid
vertices, and computes rotation angles needed to align the image. The optimal angle is found by minimizing the error
across all computed angles, and the image is rotated accordingly.
Raises:
ValueError: If the axis is not 0 (row-wise) or 1 (column-wise).
Args:
image (ImageGridHandler): The arr grid image object to be aligned.
Returns:
ImageGridHandler: The rotated grid image object after alignment.
"""
if self.axis == 0:
# If performing row-wise alignment, the x other_image is the cc other_image
x_group = str(GRID.ROW_NUM)
x_val = str(BBOX.CENTER_CC)
elif self.axis == 1:
# If performing column-wise alignment, the x other_image is the rr other_image
x_group = str(GRID.COL_NUM)
x_val = str(BBOX.CENTER_RR)
else:
raise ValueError("Axis must be either 0 or 1")
# Find the slope info along the axis
m, b = image.grid.get_centroid_alignment_info(axis=self.axis)
grid_info = image.grid.info()
# Collect the X position of the vertices
x_min = grid_info.groupby(x_group, observed=True)[x_val].min().to_numpy()
y_0 = (
x_min * m
) + b # Find the corresponding y-other_image at the above x values
# Find the x other_image of the upper ray
x_max = grid_info.groupby(x_group, observed=True)[x_val].max().to_numpy()
y_1 = (
x_max * m
) + b # Find the corresponding y-other_image at the above x values
# Collect opening angle ray coordinate info
xy_vertices = np.vstack(
[x_min, y_0]
).T # An array containing the x & y coordinates of the vertices
xy_upper_ray = np.vstack(
[x_max, y_1]
).T # An array containing the x & y coordinates of the upper ray endpoint
# Function to find the euclidead distance between two points within two xy arrays stacked column-wise
# Get the size of each hypotenuse
hyp_dist = np.apply_along_axis(
func1d=self._find_hyp_dist,
axis=1,
arr=np.column_stack([xy_vertices, xy_upper_ray]),
)
adj_dist = x_max - x_min
adj_over_hyp = np.divide(
adj_dist, hyp_dist, where=(hyp_dist != 0) | (adj_dist != 0)
)
# Find the angle of rotation from horizon in degrees
theta = np.arccos(adj_over_hyp) * (180.0 / np.pi)
# Adds the correct orientation to the angle
theta_sign = y_0 - y_1
theta = theta * (np.divide(theta_sign, abs(theta_sign), where=theta_sign != 0))
def find_angle_of_rot(x):
new_theta = theta + x
err = np.mean(new_theta**2)
return err
largest_angle = np.abs(theta).max()
optimal_angle = minimize_scalar(
fun=find_angle_of_rot,
bounds=(-largest_angle, largest_angle),
)
image.rotate(angle_of_rotation=optimal_angle.x, mode=self.mode)
return image
@staticmethod
def _find_hyp_dist(row):
return euclidean(u=[row[0], row[1]], v=[row[2], row[3]])
# Set the documentation to match for sphinx. This is unavoidable due to sphinx statically resolving.
GridAligner.apply.__doc__ = GridAligner._operate.__doc__