1024 lines
34 KiB
Python
Raw Normal View History

2019-04-28 11:16:27 +02:00
# Copyright (c) 2016-2017 Anki, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License in the file LICENSE.txt or at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
'''Utility classes and functions.'''
# __all__ should order by constants, event classes, other classes, functions.
# (util keeps class related functions close to their associated class)
__all__ = ['Angle', 'degrees', 'radians',
'ImageBox',
'Distance', 'distance_mm', 'distance_inches', 'Matrix44',
'Pose', 'pose_quaternion', 'pose_z_angle',
'Position', 'Quaternion',
'Rotation', 'rotation_quaternion', 'rotation_z_angle',
'angle_z_to_quaternion',
'Speed', 'speed_mmps',
'Timeout', 'Vector2', 'Vector3']
import collections
import math
import time
from ._clad import _clad_to_engine_anki
class ImageBox(collections.namedtuple('ImageBox', 'top_left_x top_left_y width height')):
'''Defines a bounding box within an image frame.
This is used when objects, faces and pets are observed to denote where in
the robot's camera view the object, face or pet actually appears. It's then
used by the :mod:`cozmo.annotate` module to show an outline of a box around
the object, face or pet.
.. py:attribute:: width
float - The width of the box.
.. py:attribute:: height
float - The height of the box.
'''
__slots__ = ()
@classmethod
def _create_from_clad_rect(cls, img_rect):
return cls(img_rect.x_topLeft, img_rect.y_topLeft,
img_rect.width, img_rect.height)
@property
def left_x(self):
"""float: The x coordinate of the left of the box."""
return self.top_left_x
@property
def right_x(self):
"""float: The x coordinate of the right of the box."""
return self.top_left_x + self.width
@property
def top_y(self):
"""float: The y coordinate of the top of the box."""
return self.top_left_y
@property
def bottom_y(self):
"""float: The y coordinate of the bottom of the box."""
return self.top_left_y + self.height
@property
def center(self):
"""(float, float): The x,y coordinates of the center of the box."""
cen_x = self.top_left_x + (self.width * 0.5)
cen_y = self.top_left_y + (self.height * 0.5)
return cen_x, cen_y
def __mul__(self, other):
return ImageBox(self[0] * other, self[1] * other, self[2] * other, self[3] * other)
class Angle:
'''Represents an angle.
Use the :func:`degrees` or :func:`radians` convenience methods to generate
an Angle instance.
Args:
radians (float): The number of radians the angle should represent
(cannot be combined with ``degrees``)
degrees (float): The number of degress the angle should represent
(cannot be combined with ``radians``)
'''
__slots__ = ('_radians')
def __init__(self, radians=None, degrees=None):
if radians is None and degrees is None:
raise ValueError("Expected either the degrees or radians keyword argument")
if radians and degrees:
raise ValueError("Expected either the degrees or radians keyword argument, not both")
if degrees is not None:
radians = degrees * math.pi / 180
self._radians = float(radians)
def __repr__(self):
return "<%s %.2f radians (%.2f degrees)>" % (self.__class__.__name__, self.radians, self.degrees)
def __add__(self, other):
if not isinstance(other, Angle):
raise TypeError("Unsupported type for + expected Angle")
return radians(self.radians + other.radians)
def __sub__(self, other):
if not isinstance(other, Angle):
raise TypeError("Unsupported type for - expected Angle")
return radians(self.radians - other.radians)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported type for * expected number")
return radians(self.radians * other)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported type for / expected number")
return radians(self.radians / other)
def _cmp_int(self, other):
if not isinstance(other, Angle):
raise TypeError("Unsupported type for comparison expected Angle")
return self.radians - other.radians
def __eq__(self, other):
return self._cmp_int(other) == 0
def __ne__(self, other):
return self._cmp_int(other) != 0
def __gt__(self, other):
return self._cmp_int(other) > 0
def __lt__(self, other):
return self._cmp_int(other) < 0
def __ge__(self, other):
return self._cmp_int(other) >= 0
def __le__(self, other):
return self._cmp_int(other) <= 0
@property
def radians(self):
'''float: The angle in radians.'''
return self._radians
@property
def degrees(self):
'''float: The angle in degrees.'''
return self._radians / math.pi * 180
@property
def abs_value(self):
""":class:`cozmo.util.Angle`: The absolute value of the angle.
If the Angle is positive then it returns a copy of this Angle, otherwise it returns -Angle.
"""
return Angle(radians = abs(self._radians))
def degrees(degrees):
'''Returns an :class:`cozmo.util.Angle` instance set to the specified number of degrees.'''
return Angle(degrees=degrees)
def radians(radians):
'''Returns an :class:`cozmo.util.Angle` instance set to the specified number of radians.'''
return Angle(radians=radians)
class Distance:
'''Represents a distance.
The class allows distances to be returned in either millimeters or inches.
Use the :func:`distance_inches` or :func:`distance_mm` convenience methods to generate
a Distance instance.
Args:
distance_mm (float): The number of millimeters the distance should
represent (cannot be combined with ``distance_inches``).
distance_inches (float): The number of inches the distance should
represent (cannot be combined with ``distance_mm``).
'''
__slots__ = ('_distance_mm')
def __init__(self, distance_mm=None, distance_inches=None):
if distance_mm is None and distance_inches is None:
raise ValueError("Expected either the distance_mm or distance_inches keyword argument")
if distance_mm and distance_inches:
raise ValueError("Expected either the distance_mm or distance_inches keyword argument, not both")
if distance_inches is not None:
distance_mm = distance_inches * 25.4
self._distance_mm = distance_mm
def __repr__(self):
return "<%s %.2f mm (%.2f inches)>" % (self.__class__.__name__, self.distance_mm, self.distance_inches)
def __add__(self, other):
if not isinstance(other, Distance):
raise TypeError("Unsupported operand for + expected Distance")
return distance_mm(self.distance_mm + other.distance_mm)
def __sub__(self, other):
if not isinstance(other, Distance):
raise TypeError("Unsupported operand for - expected Distance")
return distance_mm(self.distance_mm - other.distance_mm)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for * expected number")
return distance_mm(self.distance_mm * other)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for / expected number")
return distance_mm(self.distance_mm / other)
@property
def distance_mm(self):
'''float: The distance in millimeters'''
return self._distance_mm
@property
def distance_inches(self):
'''float: The distance in inches'''
return self._distance_mm / 25.4
def distance_mm(distance_mm):
'''Returns an :class:`cozmo.util.Distance` instance set to the specified number of millimeters.'''
return Distance(distance_mm=distance_mm)
def distance_inches(distance_inches):
'''Returns an :class:`cozmo.util.Distance` instance set to the specified number of inches.'''
return Distance(distance_inches=distance_inches)
class Speed:
'''Represents a speed.
This class allows speeds to be measured in millimeters per second.
Use :func:`speed_mmps` convenience methods to generate
a Speed instance.
Args:
speed_mmps (float): The number of millimeters per second the speed
should represent.
'''
__slots__ = ('_speed_mmps')
def __init__(self, speed_mmps=None):
if speed_mmps is None:
raise ValueError("Expected speed_mmps keyword argument")
self._speed_mmps = speed_mmps
def __repr__(self):
return "<%s %.2f mmps>" % (self.__class__.__name__, self.speed_mmps)
def __add__(self, other):
if not isinstance(other, Speed):
raise TypeError("Unsupported operand for + expected Speed")
return speed_mmps(self.speed_mmps + other.speed_mmps)
def __sub__(self, other):
if not isinstance(other, Speed):
raise TypeError("Unsupported operand for - expected Speed")
return speed_mmps(self.speed_mmps - other.speed_mmps)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for * expected number")
return speed_mmps(self.speed_mmps * other)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for / expected number")
return speed_mmps(self.speed_mmps / other)
@property
def speed_mmps(self):
'''float: The speed in millimeters per second (mmps).'''
return self._speed_mmps
def speed_mmps(speed_mmps):
'''Returns an :class:`cozmo.util.Speed` instance set to the specified millimeters per second speed'''
return Speed(speed_mmps=speed_mmps)
class Pose:
'''Represents where an object is in the world.
Use the :func:'pose_quaternion' to return pose in the form of
position and rotation defined by a quaternion
Use the :func:'pose_z_angle' to return pose in the form of
position and rotation defined by rotation about the z axis
When the engine is initialized, and whenever Cozmo is de-localized (i.e.
whenever Cozmo no longer knows where he is - e.g. when he's picked up)
Cozmo creates a new pose starting at (0,0,0) with no rotation, with
origin_id incremented to show that these poses cannot be compared with
earlier ones. As Cozmo drives around, his pose (and the pose of other
objects he observes - e.g. faces, cubes etc.) is relative to this initial
position and orientation.
The coordinate space is relative to Cozmo, where Cozmo's origin is the
point on the ground between Cozmo's two front wheels:
The X axis is Cozmo's forward direction
The Y axis is to Cozmo's left
The Z axis is up
Only poses of the same origin_id can safely be compared or operated on
'''
__slots__ = ('_position', '_rotation', '_origin_id', '_is_accurate')
def __init__(self, x, y, z, q0=None, q1=None, q2=None, q3=None,
angle_z=None, origin_id=-1, is_accurate=True):
self._position = Position(x,y,z)
self._rotation = Quaternion(q0,q1,q2,q3,angle_z)
self._origin_id = origin_id
self._is_accurate = is_accurate
@classmethod
def _create_from_clad(cls, pose):
return cls(pose.x, pose.y, pose.z,
q0=pose.q0, q1=pose.q1, q2=pose.q2, q3=pose.q3,
origin_id=pose.originID)
@classmethod
def _create_default(cls):
return cls(0.0, 0.0, 0.0,
q0=1.0, q1=0.0, q2=0.0, q3=0.0,
origin_id=-1)
def __repr__(self):
return "<%s %s %s origin_id=%d>" % (self.__class__.__name__, self.position, self.rotation, self.origin_id)
def __add__(self, other):
if not isinstance(other, Pose):
raise TypeError("Unsupported operand for + expected Pose")
pos = self.position + other.position
rot = self.rotation + other.rotation
return pose_quaternion(pos.x, pos.y, pos.z, rot.q0, rot.q1, rot.q2, rot.q3)
def __sub__(self, other):
if not isinstance(other, Pose):
raise TypeError("Unsupported operand for - expected Pose")
pos = self.position - other.position
rot = self.rotation - other.rotation
return pose_quaternion(pos.x, pos.y, pos.z, rot.q0, rot.q1, rot.q2, rot.q3)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for * expected number")
pos = self.position * other
rot = self.rotation * other
return pose_quaternion(pos.x, pos.y, pos.z, rot.q0, rot.q1, rot.q2, rot.q3)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for / expected number")
pos = self.position / other
rot = self.rotation / other
return pose_quaternion(pos.x, pos.y, pos.z, rot.q0, rot.q1, rot.q2, rot.q3)
def define_pose_relative_this(self, new_pose):
'''Creates a new pose such that new_pose's origin is now at the location of this pose.
Args:
new_pose (:class:`cozmo.util.Pose`): The pose which origin is being changed.
Returns:
A :class:`cozmo.util.pose` object for which the origin was this pose's origin.
'''
if not isinstance(new_pose, Pose):
raise TypeError("Unsupported type for new_origin, must be of type Pose")
x,y,z = self.position.x_y_z
angle_z = self.rotation.angle_z
new_x,new_y,new_z = new_pose.position.x_y_z
new_angle_z = new_pose.rotation.angle_z
cos_angle = math.cos(angle_z.radians)
sin_angle = math.sin(angle_z.radians)
res_x = x + (cos_angle * new_x) - (sin_angle * new_y)
res_y = y + (sin_angle * new_x) + (cos_angle * new_y)
res_z = z + new_z
res_angle = angle_z + new_angle_z
return Pose(res_x, res_y, res_z, angle_z=res_angle, origin_id=self._origin_id)
def encode_pose(self):
x, y, z = self.position.x_y_z
q0, q1, q2, q3 = self.rotation.q0_q1_q2_q3
return _clad_to_engine_anki.PoseStruct3d(x, y, z, q0, q1, q2, q3, self.origin_id)
def invalidate(self):
'''Mark this pose as being invalid (unusable)'''
self._origin_id = -1
def is_comparable(self, other_pose):
'''Are these two poses comparable.
Poses are comparable if they're valid and having matching origin IDs.
Args:
other_pose (:class:`cozmo.util.Pose`): The other pose to compare against.
Returns:
bool: True if the two poses are comparable, False otherwise.
'''
return (self.is_valid and other_pose.is_valid and
(self.origin_id == other_pose.origin_id))
@property
def is_valid(self):
'''bool: Returns True if this is a valid, usable pose.'''
return self.origin_id >= 0
@property
def position(self):
''':class:`cozmo.util.Position`: The position component of this pose.'''
return self._position
@property
def rotation(self):
''':class:`cozmo.util.Rotation`: The rotation component of this pose.'''
return self._rotation
def to_matrix(self):
"""Convert the Pose to a Matrix44.
Returns:
:class:`cozmo.util.Matrix44`: A matrix representing this Pose's
position and rotation.
"""
return self.rotation.to_matrix(*self.position.x_y_z)
@property
def origin_id(self):
'''int: An ID maintained by the engine which represents which coordinate frame this pose is in.'''
return self._origin_id
@origin_id.setter
def origin_id(self, value):
'''Allows this to be changed later in case it was not originally defined.'''
if not isinstance(value, int):
raise TypeError("The type of origin_id must be int")
self._origin_id = value
@property
def is_accurate(self):
'''bool: Returns True if this pose is valid and accurate.
Poses are marked as inaccurate if we detect movement via accelerometer,
or if they were observed from far enough away that we're less certain
of the exact pose.
'''
return self.is_valid and self._is_accurate
def pose_quaternion(x, y, z, q0, q1, q2, q3, origin_id=0):
'''Returns a :class:`cozmo.util.Pose` instance set to the pose given in quaternion format.'''
return Pose(x, y, z, q0=q0, q1=q1, q2=q2, q3=q3, origin_id=origin_id)
def pose_z_angle(x, y, z, angle_z, origin_id=0):
'''Returns a :class:`cozmo.util.Pose` instance set to the pose given in z angle format.'''
return Pose(x, y, z, angle_z=angle_z, origin_id=origin_id)
class Matrix44:
"""A 4x4 Matrix for representing the rotation and/or position of an object in the world.
Can be generated from a :class:`Quaternion` for a pure rotation matrix, or
combined with a position for a full translation matrix, as done by
:meth:`Pose.to_matrix`.
"""
__slots__ = ('m00', 'm10', 'm20', 'm30',
'm01', 'm11', 'm21', 'm31',
'm02', 'm12', 'm22', 'm32',
'm03', 'm13', 'm23', 'm33')
def __init__(self,
m00, m10, m20, m30,
m01, m11, m21, m31,
m02, m12, m22, m32,
m03, m13, m23, m33):
self.m00 = m00
self.m10 = m10
self.m20 = m20
self.m30 = m30
self.m01 = m01
self.m11 = m11
self.m21 = m21
self.m31 = m31
self.m02 = m02
self.m12 = m12
self.m22 = m22
self.m32 = m32
self.m03 = m03
self.m13 = m13
self.m23 = m23
self.m33 = m33
def __repr__(self):
return ("<%s: "
"%.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f "
"%.1f %.1f %.1f %.1f %.1f %.1f %.1f %.1f>" % (
self.__class__.__name__, *self.in_row_order))
@property
def tabulated_string(self):
"""str: A multi-line string formatted with tabs to show the matrix contents."""
return ("%.1f\t%.1f\t%.1f\t%.1f\n"
"%.1f\t%.1f\t%.1f\t%.1f\n"
"%.1f\t%.1f\t%.1f\t%.1f\n"
"%.1f\t%.1f\t%.1f\t%.1f" % self.in_row_order)
@property
def in_row_order(self):
"""tuple of 16 floats: The contents of the matrix in row order."""
return self.m00, self.m01, self.m02, self.m03,\
self.m10, self.m11, self.m12, self.m13,\
self.m20, self.m21, self.m22, self.m23,\
self.m30, self.m31, self.m32, self.m33
@property
def in_column_order(self):
"""tuple of 16 floats: The contents of the matrix in column order."""
return self.m00, self.m10, self.m20, self.m30,\
self.m01, self.m11, self.m21, self.m31,\
self.m02, self.m12, self.m22, self.m32,\
self.m03, self.m13, self.m23, self.m33
@property
def forward_xyz(self):
"""tuple of 3 floats: The x,y,z components representing the matrix's forward vector."""
return self.m00, self.m01, self.m02
@property
def left_xyz(self):
"""tuple of 3 floats: The x,y,z components representing the matrix's left vector."""
return self.m10, self.m11, self.m12
@property
def up_xyz(self):
"""tuple of 3 floats: The x,y,z components representing the matrix's up vector."""
return self.m20, self.m21, self.m22
@property
def pos_xyz(self):
"""tuple of 3 floats: The x,y,z components representing the matrix's position vector."""
return self.m30, self.m31, self.m32
def set_forward(self, x, y, z):
"""Set the x,y,z components representing the matrix's forward vector.
Args:
x (float): The X component.
y (float): The Y component.
z (float): The Z component.
"""
self.m00 = x
self.m01 = y
self.m02 = z
def set_left(self, x, y, z):
"""Set the x,y,z components representing the matrix's left vector.
Args:
x (float): The X component.
y (float): The Y component.
z (float): The Z component.
"""
self.m10 = x
self.m11 = y
self.m12 = z
def set_up(self, x, y, z):
"""Set the x,y,z components representing the matrix's up vector.
Args:
x (float): The X component.
y (float): The Y component.
z (float): The Z component.
"""
self.m20 = x
self.m21 = y
self.m22 = z
def set_pos(self, x, y, z):
"""Set the x,y,z components representing the matrix's position vector.
Args:
x (float): The X component.
y (float): The Y component.
z (float): The Z component.
"""
self.m30 = x
self.m31 = y
self.m32 = z
class Quaternion:
'''Represents the rotation of an object in the world. Can be generated with
quaternion to define its rotation in 3d space, or with only a z axis rotation
to define things limited to the x-y plane like Cozmo.
Use the :func:`rotation_quaternion` to return rotation defined by a quaternion.
Use the :func:`rotation_angle_z` to return rotation defined by an angle in the z axis.
'''
__slots__ = ('_q0', '_q1', '_q2', '_q3')
def __init__(self, q0=None, q1=None, q2=None, q3=None, angle_z=None):
is_quaternion = not (q0 is None) and not (q1 is None) and not (q2 is None) and not (q3 is None)
if not is_quaternion and angle_z is None:
raise ValueError("Expected either the q0 q1 q2 and q3 or angle_z keyword arguments")
if is_quaternion and angle_z:
raise ValueError("Expected either the q0 q1 q2 and q3 or angle_z keyword argument, not both")
if angle_z is not None:
if not isinstance(angle_z, Angle):
raise TypeError("Unsupported type for angle_z expected Angle")
q0,q1,q2,q3 = angle_z_to_quaternion(angle_z)
self._q0, self._q1, self._q2, self._q3 = q0, q1, q2, q3
def __repr__(self):
return ("<%s q0: %.2f q1: %.2f q2: %.2f q3: %.2f (angle_z: %s)>" %
(self.__class__.__name__, self.q0, self.q1, self.q2, self.q3, self.angle_z))
def to_matrix(self, pos_x=0.0, pos_y=0.0, pos_z=0.0):
"""Convert the Quaternion to a 4x4 matrix representing this rotation.
A position can also be provided to generate a full translation matrix.
Args:
pos_x (float): The x component for the position.
pos_y (float): The y component for the position.
pos_z (float): The z component for the position.
Returns:
:class:`cozmo.util.Matrix44`: A matrix representing this Quaternion's
rotation, with the provided position (which defaults to 0,0,0).
"""
# See https://en.wikipedia.org/wiki/Quaternions_and_spatial_rotation
q0q0 = self.q0 * self.q0
q1q1 = self.q1 * self.q1
q2q2 = self.q2 * self.q2
q3q3 = self.q3 * self.q3
q0x2 = self.q0 * 2.0 # saves 2 multiplies
q0q1x2 = q0x2 * self.q1
q0q2x2 = q0x2 * self.q2
q0q3x2 = q0x2 * self.q3
q1x2 = self.q1 * 2.0 # saves 1 multiply
q1q2x2 = q1x2 * self.q2
q1q3x2 = q1x2 * self.q3
q2q3x2 = 2.0 * self.q2 * self.q3
m00 = (q0q0 + q1q1 - q2q2 - q3q3)
m01 = (q1q2x2 + q0q3x2)
m02 = (q1q3x2 - q0q2x2)
m10 = (q1q2x2 - q0q3x2)
m11 = (q0q0 - q1q1 + q2q2 - q3q3)
m12 = (q0q1x2 + q2q3x2)
m20 = (q0q2x2 + q1q3x2)
m21 = (q2q3x2 - q0q1x2)
m22 = (q0q0 - q1q1 - q2q2 + q3q3)
return Matrix44(m00, m10, m20, pos_x,
m01, m11, m21, pos_y,
m02, m12, m22, pos_z,
0.0, 0.0, 0.0, 1.0)
#These are only for angle_z because quaternion addition/subtraction is not relevant here
def __add__(self, other):
if not isinstance(other, Quaternion):
raise TypeError("Unsupported operand for + expected Quaternion")
return rotation_z_angle(self.angle_z + other.angle_z)
def __sub__(self, other):
if not isinstance(other, Quaternion):
raise TypeError("Unsupported operand for - expected Quaternion")
return rotation_z_angle(self.angle_z - other.angle_z)
def __mul__(self, other):
if not isinstance(other, (int,float)):
raise TypeError("Unsupported operand for * expected number")
return rotation_z_angle(self.angle_z * other)
def __truediv__(self, other):
if not isinstance(other, (int,float)):
raise TypeError("Unsupported operand for / expected number")
return rotation_z_angle(self.angle_z / other)
@property
def q0(self):
'''float: The q0 (w) value of the quaternion.'''
return self._q0
@property
def q1(self):
'''float: The q1 (i) value of the quaternion.'''
return self._q1
@property
def q2(self):
'''float: The q2 (j) value of the quaternion.'''
return self._q2
@property
def q3(self):
'''float: The q3 (k) value of the quaternion.'''
return self._q3
@property
def q0_q1_q2_q3(self):
'''tuple of float: Contains all elements of the quaternion (q0,q1,q2,q3)'''
return self._q0,self._q1,self._q2,self._q3
@property
def angle_z(self):
'''class:`Angle`: The z Euler component of the object's rotation.
Defined as the rotation in the z axis.
'''
q0,q1,q2,q3 = self.q0_q1_q2_q3
return Angle(radians=math.atan2(2*(q1*q2+q0*q3), 1-2*(q2**2+q3**2)))
@property
def euler_angles(self):
'''tuple of float: Euler angles of an object.
Returns the pitch, yaw, roll Euler components of the object's
rotation defined as rotations in the x, y, and z axis respectively.
It interprets the rotations performed in the order: Z, Y, X
'''
# convert to matrix
matrix = self.to_matrix()
# normalize the magnitudes of cos(roll)*sin(pitch) (i.e. m12) and
# cos(roll)*cos(pitch) (ie. m22), to isolate cos(roll) to be compared
# against -sin(roll) (m02). Unfortunately, this omits results with an
# absolute angle larger than 90 degrees on roll.
absolute_cos_roll = math.sqrt(matrix.m12*matrix.m12+matrix.m22*matrix.m22)
near_gimbal_lock = absolute_cos_roll < 1e-6
if not near_gimbal_lock:
# general case euler decomposition
pitch = math.atan2(matrix.m22, matrix.m12)
yaw = math.atan2(matrix.m00, matrix.m01)
roll = math.atan2(absolute_cos_roll, -matrix.m02)
else:
# special case euler angle decomposition near gimbal lock
pitch = math.atan2(matrix.m11, -matrix.m21)
yaw = 0
roll = math.atan2(absolute_cos_roll, -matrix.m02)
# adjust roll to be consistent with how we orient the device
roll = math.pi * 0.5 - roll
if roll > math.pi:
roll -= math.pi * 2
return pitch, yaw, roll
class Rotation(Quaternion):
'''An alias for :class:`Quaternion`'''
__slots__ = ()
def rotation_quaternion(q0, q1, q2, q3):
'''Returns a :class:`Rotation` instance set by a quaternion.'''
return Quaternion(q0=q0, q1=q1, q2=q2, q3=q3)
def rotation_z_angle(angle_z):
'''Returns a class:`Rotation` instance set by an angle in the z axis'''
return Quaternion(angle_z=angle_z)
def angle_z_to_quaternion(angle_z):
'''This function converts an angle in the z axis (Euler angle z component) to a quaternion.
Args:
angle_z (:class:`cozmo.util.Angle`): The z axis angle.
Returns:
q0,q1,q2,q3 (float, float, float, float): A tuple with all the members
of a quaternion defined by angle_z.
'''
#Define the quaternion to be converted from a Euler angle (x,y,z) of 0,0,angle_z
#These equations have their original equations above, and simplified implemented
# q0 = cos(x/2)*cos(y/2)*cos(z/2) + sin(x/2)*sin(y/2)*sin(z/2)
q0 = math.cos(angle_z.radians/2)
# q1 = sin(x/2)*cos(y/2)*cos(z/2) - cos(x/2)*sin(y/2)*sin(z/2)
q1 = 0
# q2 = cos(x/2)*sin(y/2)*cos(z/2) + sin(x/2)*cos(y/2)*sin(z/2)
q2 = 0
# q3 = cos(x/2)*cos(y/2)*sin(z/2) - sin(x/2)*sin(y/2)*cos(z/2)
q3 = math.sin(angle_z.radians/2)
return q0,q1,q2,q3
class Vector2:
'''Represents a 2D Vector (type/units aren't specified)
Args:
x (float): X component
y (float): Y component
'''
__slots__ = ('_x', '_y')
def __init__(self, x, y):
self._x = x
self._y = y
def set_to(self, rhs):
"""Copy the x and y components of the given vector.
Args:
rhs (:class:`Vector2`): The right-hand-side of this assignment - the
source vector to copy into this vector.
"""
self._x = rhs.x
self._y = rhs.y
@property
def x(self):
'''float: The x component.'''
return self._x
@property
def y(self):
'''float: The y component.'''
return self._y
@property
def x_y(self):
'''tuple (float, float): The X, Y elements of the Vector2 (x,y)'''
return self._x, self._y
def __repr__(self):
return "<%s x: %.2f y: %.2f>" % (self.__class__.__name__, self.x, self.y)
def __add__(self, other):
if not isinstance(other, Vector2):
raise TypeError("Unsupported operand for + expected Vector2")
return Vector2(self.x + other.x, self.y + other.y)
def __sub__(self, other):
if not isinstance(other, Vector2):
raise TypeError("Unsupported operand for - expected Vector2")
return Vector2(self.x - other.x, self.y - other.y)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for * expected number")
return Vector2(self.x * other, self.y * other)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for / expected number")
return Vector2(self.x / other, self.y / other)
class Vector3:
'''Represents a 3D Vector (type/units aren't specified)
Args:
x (float): X component
y (float): Y component
z (float): Z component
'''
__slots__ = ('_x', '_y', '_z')
def __init__(self, x, y, z):
self._x = x
self._y = y
self._z = z
def set_to(self, rhs):
"""Copy the x, y and z components of the given vector.
Args:
rhs (:class:`Vector3`): The right-hand-side of this assignment - the
source vector to copy into this vector.
"""
self._x = rhs.x
self._y = rhs.y
self._z = rhs.z
@property
def x(self):
'''float: The x component.'''
return self._x
@property
def y(self):
'''float: The y component.'''
return self._y
@property
def z(self):
'''float: The z component.'''
return self._z
@property
def x_y_z(self):
'''tuple (float, float, float): The X, Y, Z elements of the Vector3 (x,y,z)'''
return self._x, self._y, self._z
def __repr__(self):
return "<%s x: %.2f y: %.2f z: %.2f>" % (self.__class__.__name__, self.x, self.y, self.z)
def __add__(self, other):
if not isinstance(other, Vector3):
raise TypeError("Unsupported operand for + expected Vector3")
return Vector3(self.x + other.x, self.y + other.y, self.z + other.z)
def __sub__(self, other):
if not isinstance(other, Vector3):
raise TypeError("Unsupported operand for - expected Vector3")
return Vector3(self.x - other.x, self.y - other.y, self.z - other.z)
def __mul__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for * expected number")
return Vector3(self.x * other, self.y * other, self.z * other)
def __truediv__(self, other):
if not isinstance(other, (int, float)):
raise TypeError("Unsupported operand for / expected number")
return Vector3(self.x / other, self.y / other, self.z / other)
class Position(Vector3):
'''Represents the position of an object in the world.
A position consists of its x, y and z values in millimeters.
Args:
x (float): X position in millimeters
y (float): Y position in millimeters
z (float): Z position in millimeters
'''
__slots__ = ()
class Timeout:
'''Utility class to keep track of a timeout condition.
This measures a timeout from the point in time that the class
is instantiated.
Args:
timeout (float): Amount of time (in seconds) allotted to pass before
considering the timeout condition to be met.
use_inf (bool): If True, then :attr:`remaining` will return
:const:`math.inf` if `timeout` is None, else it will return
`None`.
'''
def __init__(self, timeout=None, use_inf=False):
self.start = time.time()
self.timeout = timeout
self.use_inf = use_inf
@property
def is_timed_out(self):
'''bool: True if the timeout has expired.'''
if self.timeout is None:
return False
return time.time() - self.start > self.timeout
@property
def remaining(self):
'''float: The number of seconds remaining before reaching the timeout.
Will return a number of zero or higher, even if the timer has
since expired (it will never return a negative value).
Will return None or math.inf (if ``use_inf`` was passed as ``True``
to the constructor) if the original timeout was ``None``.
'''
if self.timeout is None:
return math.inf if self.use_inf else None
return max(0, self.timeout - (time.time() - self.start))