Skip to content

Commit

Permalink
Add rot_to_rad function to visu3d.math.rotation_utils
Browse files Browse the repository at this point in the history
PiperOrigin-RevId: 628363278
  • Loading branch information
The visu3d Authors committed Apr 26, 2024
1 parent 3cdb8fb commit 0abeafc
Show file tree
Hide file tree
Showing 4 changed files with 47 additions and 1 deletion.
3 changes: 2 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,8 @@ Changelog follow https://keepachangelog.com/ format.
## [Unreleased]

* Drop Python 3.10 support
* Add `hash()` support (e.g. after `etree.spec_like`).
* Add `hash()` support (e.g. after `etree.spec_like`).
* Added: `visu3d.math.rotation_utils.rot_to_rad`.

## [1.5.4] - 2023-11-23

Expand Down
1 change: 1 addition & 0 deletions visu3d/math/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
from visu3d.math.rotation_utils import is_rot
from visu3d.math.rotation_utils import RAD2DEG
from visu3d.math.rotation_utils import rot_to_euler
from visu3d.math.rotation_utils import rot_to_rad
from visu3d.math.rotation_utils import rot_x
from visu3d.math.rotation_utils import rot_y
from visu3d.math.rotation_utils import rot_z
Expand Down
34 changes: 34 additions & 0 deletions visu3d/math/rotation_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -204,6 +204,40 @@ def rot_to_euler(
return theta_x, theta_y, theta_z


@enp.check_and_normalize_arrays
def rot_to_rad(
rot: FloatArray['*B 3 3'],
*,
xnp: enp.NpModule = ...,
) -> FloatArray['*B']:
"""Compute the absolute angle of rotation in radians of a 3x3 rotation matrix.
With a change of coordinate frame, any rotation matrix can be expressed as:
[[cos(t), -sin(t), 0], [sin(t), cos(t), 0], [0, 0, 1]], which gives
tr(A) = 2 * cos(t) + 1. This expression still holds for the original matrix as
trace(SAS^{-1})=trace(A).
```python
rads = acos(0.5 * (trace(R) - 1))
```
See:
https://en.wikipedia.org/wiki/Rotation_matrix#Determining_the_angle
Args:
rot: Rotation matrix
xnp: Np module used (jnp, tnp, np,...)
Returns:
rads: The absolute angle in radians represented by rot
"""
trace = xnp.trace(rot, axis1=-2, axis2=-1)
# Because tr(A) = 2 * cos(t) + 1, it should lie in the interval [-1, 3]
# In case this doesn't happen due to numerical errors, we apply clipping.
trace = xnp.clip(trace, -1, 3)
return xnp.arccos(0.5 * (trace - 1))


@enp.check_and_normalize_arrays
def is_orth(
rot: FloatArray['3 3'],
Expand Down
10 changes: 10 additions & 0 deletions visu3d/math/rotation_utils_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,16 @@
enable_tf_np_mode = enp.testing.set_tnp


@enp.testing.parametrize_xnp(skip=['torch'])
def test_is_rot_to_rad(xnp: enp.NpModule):
rot = xnp.asarray([
[-1.0, 0.0, 0.0],
[0.0, -1.0, 0.0],
[0.0, 0.0, 1.0],
])
np.testing.assert_allclose(v3d.math.rot_to_rad(rot), np.pi, atol=1e-6)


@enp.testing.parametrize_xnp()
def test_is_rotation_matrix(xnp: enp.NpModule):
assert v3d.math.is_rot(xnp.eye(3))
Expand Down

0 comments on commit 0abeafc

Please sign in to comment.