# 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