Source code for astropy.coordinates.matrix_utilities

# Licensed under a 3-clause BSD style license - see LICENSE.rst

"""
Utililies used for constructing and inspecting rotation matrices.
"""

__all__ = ["is_rotation_or_reflection", "rotation_matrix"]

import numpy as np

from astropy import units as u
from astropy.utils.compat import COPY_IF_NEEDED
from astropy.utils.decorators import deprecated

from .angles import Angle


def matrix_transpose(matrix):
    """Transpose a matrix or stack of matrices by swapping the last two axes.

    This function mostly exists for readability; seeing ``.swapaxes(-2, -1)``
    it is not that obvious that one does a transpose.

    Note that one cannot use `~numpy.ndarray.T`, as this transposes all axes
    and thus does not work for stacks of matrices.  We also avoid
    ``np.matrix_transpose`` (new in numpy 2.0), since it is slower, as it
    first ensures the input is an array, while we ducktype, assuming the
    input has a ``.swapaxes`` method.

    """
    return matrix.swapaxes(-2, -1)


[docs] def rotation_matrix(angle, axis="z", unit=None): """ Generate matrices for rotation by some angle around some axis. Parameters ---------- angle : angle-like The amount of rotation the matrices should represent. Can be an array. axis : str or array-like Either ``'x'``, ``'y'``, ``'z'``, or a (x,y,z) specifying the axis to rotate about. If ``'x'``, ``'y'``, or ``'z'``, the rotation sense is counterclockwise looking down the + axis (e.g. positive rotations obey left-hand-rule). If given as an array, the last dimension should be 3; it will be broadcast against ``angle``. unit : unit-like, optional If ``angle`` does not have associated units, they are in this unit. If neither are provided, it is assumed to be degrees. Returns ------- rmat : `numpy.matrix` A unitary rotation matrix. """ if not isinstance(angle, u.Quantity): angle = Angle(angle, unit=unit or u.deg, copy=COPY_IF_NEEDED) angle_in_rad = angle.to_value(u.rad) s = np.sin(angle_in_rad) c = np.cos(angle_in_rad) # use optimized implementations for x/y/z try: i = "xyz".index(axis) except TypeError: axis = np.asarray(axis) axis = axis / np.sqrt((axis * axis).sum(axis=-1, keepdims=True)) R = ( axis[..., np.newaxis] * axis[..., np.newaxis, :] * (1.0 - c)[..., np.newaxis, np.newaxis] ) for i in range(3): R[..., i, i] += c a1 = (i + 1) % 3 a2 = (i + 2) % 3 R[..., a1, a2] += axis[..., i] * s R[..., a2, a1] -= axis[..., i] * s else: a1 = (i + 1) % 3 a2 = (i + 2) % 3 R = np.zeros(angle_in_rad.shape + (3, 3)) R[..., i, i] = 1.0 R[..., a1, a1] = c R[..., a1, a2] = s R[..., a2, a1] = -s R[..., a2, a2] = c return R
@deprecated(since="7.2") def angle_axis(matrix): """ Angle of rotation and rotation axis for a given rotation matrix. Parameters ---------- matrix : array-like A 3 x 3 unitary rotation matrix (or stack of matrices). Returns ------- angle : `~astropy.coordinates.Angle` The angle of rotation. axis : array The (normalized) axis of rotation (with last dimension 3). """ m = np.asanyarray(matrix) if m.shape[-2:] != (3, 3): raise ValueError("matrix is not 3x3") axis = np.zeros(m.shape[:-1]) axis[..., 0] = m[..., 2, 1] - m[..., 1, 2] axis[..., 1] = m[..., 0, 2] - m[..., 2, 0] axis[..., 2] = m[..., 1, 0] - m[..., 0, 1] r = np.sqrt((axis * axis).sum(-1, keepdims=True)) angle = np.arctan2(r[..., 0], m[..., 0, 0] + m[..., 1, 1] + m[..., 2, 2] - 1.0) return Angle(angle, u.radian), -axis / r
[docs] def is_rotation_or_reflection(matrix, atol=None): """Check whether a matrix describes rotation or reflection (or both). Proper and improper rotations (i.e. rotations without and with a reflection) could be distinguished by the sign of the determinant, but this function does not bother with that because both preserve lengths of vectors. Parameters ---------- matrix : numpy.ndarray The check is performed along the last two axes, which must have equal sizes. atol : float, optional The allowed absolute difference. If `None` it defaults to 1e-15 or 5 * epsilon of the matrix's dtype, if floating. Returns ------- np.ndarray, bool If the matrix has more than two axes, the check is performed on slices along the last two axes -- (M, N, N) => (M, ) bool array. """ if atol is None: atol = ( 5 * np.finfo(matrix.dtype).eps if np.issubdtype(matrix.dtype, np.floating) else 1e-15 ) return np.isclose( matrix @ matrix.swapaxes(-2, -1), np.identity(matrix.shape[-1]), atol=atol ).all(axis=(-2, -1))
@deprecated(since="7.2", alternative="is_rotation_or_reflection") def is_O3(matrix, atol=None): """Check whether a matrix is in the length-preserving group O(3). Parameters ---------- matrix : (..., N, N) array-like Must have attribute ``.shape`` and method ``.swapaxes()`` and not error when using `~numpy.isclose`. atol : float, optional The allowed absolute difference. If `None` it defaults to 1e-15 or 5 * epsilon of the matrix's dtype, if floating. .. versionadded:: 5.3 Returns ------- is_o3 : bool or array of bool If the matrix has more than two axes, the O(3) check is performed on slices along the last two axes -- (M, N, N) => (M, ) bool array. Notes ----- The orthogonal group O(3) preserves lengths, but is not guaranteed to keep orientations. Rotations and reflections are in this group. For more information, see https://en.wikipedia.org/wiki/Orthogonal_group """ # matrix is in O(3) (rotations, proper and improper). return is_rotation_or_reflection(matrix, atol) @deprecated(since="7.2") def is_rotation(matrix, allow_improper=False, atol=None): """Check whether a matrix is a rotation, proper or improper. Parameters ---------- matrix : (..., N, N) array-like Must have attribute ``.shape`` and method ``.swapaxes()`` and not error when using `~numpy.isclose` and `~numpy.linalg.det`. allow_improper : bool, optional Whether to restrict check to the SO(3), the group of proper rotations, or also allow improper rotations (with determinant -1). The default (False) is only SO(3). atol : float, optional The allowed absolute difference. If `None` it defaults to 1e-15 or 5 * epsilon of the matrix's dtype, if floating. .. versionadded:: 5.3 Returns ------- isrot : bool or array of bool If the matrix has more than two axes, the checks are performed on slices along the last two axes -- (M, N, N) => (M, ) bool array. See Also -------- astopy.coordinates.matrix_utilities.is_O3 : For the less restrictive check that a matrix is in the group O(3). Notes ----- The group SO(3) is the rotation group. It is O(3), with determinant 1. Rotations with determinant -1 are improper rotations, combining both a rotation and a reflection. For more information, see https://en.wikipedia.org/wiki/Orthogonal_group """ if atol is None: if np.issubdtype(matrix.dtype, np.floating): atol = np.finfo(matrix.dtype).eps * 5 else: atol = 1e-15 # matrix is in O(3). is_o3 = is_rotation_or_reflection(matrix, atol=atol) # determinant checks for rotation (proper and improper) if allow_improper: # determinant can be +/- 1 is_det1 = np.isclose(np.abs(np.linalg.det(matrix)), 1.0, atol=atol) else: # restrict to SO(3) is_det1 = np.isclose(np.linalg.det(matrix), 1.0, atol=atol) return is_o3 & is_det1