from typing import Optional, TypeVar, Callable, List
import weakref
import numpy as np
import ouster.sdk.util.pose_util as pu
from ouster.sdk._bindings.viz import PointViz, Cloud, Label, WindowCtx
T = TypeVar('T')
class BoundMethod:
"""Used to wrap a bound method's instance with a weak ref,
which is necessary to use the method as a PointViz event callback
while allowing the bound method's instance to be deleted.
"""
# TODO[tws] type annotation
def __init__(self, method):
self._instance = weakref.ref(method.__self__)
self._function = method.__func__
def __call__(self, *args, **kwargs):
instance = self._instance()
if instance:
return self._function(instance, *args, **kwargs)
def push_point_viz_handler(
viz: PointViz, arg: T, handler: Callable[[T, WindowCtx, int, int],
bool]) -> None:
"""Add a key handler with extra context without keeping it alive.
It's often useful to add a key callback that calls a method of an object
that wraps a PointViz instance. In this case it's necessary to take some
extra care to avoid a reference cycle; holding onto self in the callback
passed to native code would cause a memory leak.
Args:
viz: The PointViz instance.
arg: The extra context to pass to handler; often `self`.
handler: Key handler callback taking an extra argument
"""
weakarg = weakref.ref(arg)
def handle_keys(ctx: WindowCtx, key: int, mods: int) -> bool:
arg = weakarg()
if arg is not None:
return handler(arg, ctx, key, mods)
return True
viz.push_key_handler(handle_keys)
def push_point_viz_fb_handler(
viz: PointViz, arg: T, handler: Callable[[T, List, int, int],
bool]) -> None:
"""Add a frame buffer handler with extra context without keeping it alive.
See docs for `push_point_viz_handler()` method above for details.
Args:
viz: The PointViz instance.
arg: The extra context to pass to handler; often `self`.
handler: Frame buffer handler callback taking an extra argument
"""
weakarg = weakref.ref(arg)
def handle_fb_data(fb_data: List, fb_width: int, fb_height: int) -> bool:
arg = weakarg()
if arg is not None:
return handler(arg, fb_data, fb_width, fb_height)
return True
viz.push_frame_buffer_handler(handle_fb_data)
def _cloud_axis_points(axis_length: float = 1.0) -> np.ndarray:
"""Generate coordinate axis point cloud."""
# basis vectors
x_ = np.array([1, 0, 0]).reshape((-1, 1))
y_ = np.array([0, 1, 0]).reshape((-1, 1))
z_ = np.array([0, 0, 1]).reshape((-1, 1))
axis_n = 100
line = np.linspace(0, axis_length, axis_n).reshape((1, -1))
# basis vector to point cloud
axis_points = np.hstack(
(x_ @ line,
y_ @ line,
z_ @ line)).transpose()
return axis_points
def _make_cloud_axis(axis_points) -> Cloud:
"""Create viz.Cloud object with colors from coordinate axis points"""
axis_n = int(axis_points.shape[0] / 3)
# colors for basis vectors
axis_color_mask = np.vstack((np.full(
(axis_n, 4), [1, 0.1, 0.1, 1]), np.full((axis_n, 4), [0.1, 1, 0.1, 1]),
np.full((axis_n, 4), [0.1, 0.1, 1, 1])))
cloud_axis = Cloud(axis_points.shape[0])
cloud_axis.set_xyz(axis_points)
cloud_axis.set_key(np.full(axis_points.shape[0], 0.5))
# TODO[pb]: To use set_key(rgb) instead of set_mask() for colors
cloud_axis.set_mask(axis_color_mask)
cloud_axis.set_point_size(3)
return cloud_axis
[docs]class AxisWithLabel:
"""Coordinate axis with a text label."""
def __init__(self,
point_viz: PointViz,
*,
pose: pu.PoseH = np.eye(4),
label: str = "",
length: float = 1.0,
thickness: int = 3,
label_scale: Optional[float] = None,
enabled: bool = True):
self._viz = point_viz
self._pose = pose
self._label = label
self._axis_cloud = _make_cloud_axis(_cloud_axis_points(length))
self._axis_cloud.set_point_size(thickness)
self._axis_cloud.set_pose(self._pose)
self._axis_label = Label(self._label, *pose[:3, 3])
if label_scale:
self._axis_label.set_scale(label_scale)
self._enabled = False
if enabled:
self.enable()
@property
def enabled(self) -> bool:
"""True if label is added to the viz"""
return self._enabled
[docs] def enable(self) -> None:
"""Enable the label and make it added to the viz"""
if not self._enabled:
self._viz.add(self._axis_cloud)
self._viz.add(self._axis_label)
self._enabled = True
[docs] def disable(self) -> None:
"""Disable the label and remove it from the viz"""
if self._enabled:
self._viz.remove(self._axis_cloud)
self._viz.remove(self._axis_label)
self._enabled = False
[docs] def toggle(self) -> bool:
"""Toggle the label visibility (i.e. presence in the viz)"""
if not self._enabled:
self.enable()
else:
self.disable()
return self._enabled
@property
def pose(self) -> np.ndarray:
"""Label pose, 4x4 matrix"""
return self._pose
@pose.setter
def pose(self, pose: np.ndarray):
"""Set label pose, 4x4 matrix, and update internal states"""
self._pose = pose
self.update()
@property
def label(self) -> str:
"""Label text, 4x4 matrix"""
return self._label
@label.setter
def label(self, label_text: str):
"""Set label text, and update internal states"""
self._label = label_text
self.update()
[docs] def update(self) -> None:
"""Update label component viz states."""
self._axis_cloud.set_pose(self._pose)
self._axis_label.set_position(*self._pose[:3, 3])
self._axis_label.set_text(self._label)