mod arrows2d;
mod arrows3d;
mod assets3d;
mod boxes2d;
mod boxes3d;
mod cameras;
mod capsules3d;
mod depth_images;
mod ellipsoids;
mod encoded_image;
mod images;
mod lines2d;
mod lines3d;
mod meshes;
mod points2d;
mod points3d;
mod segmentation_images;
mod transform3d_arrows;
mod utilities;
mod videos;
pub use cameras::CamerasVisualizer;
pub use depth_images::DepthImageVisualizer;
pub use transform3d_arrows::{add_axis_arrows, AxisLengthDetector, Transform3DArrowsVisualizer};
pub use utilities::{
entity_iterator, process_labels_3d, textured_rect_from_image, SpatialViewVisualizerData,
UiLabel, UiLabelStyle, UiLabelTarget,
};
#[derive(Clone, Copy, Debug)]
pub struct LoadingSpinner {
pub center: glam::Vec3,
pub half_extent_u: glam::Vec3,
pub half_extent_v: glam::Vec3,
}
use ahash::HashMap;
use re_entity_db::EntityPath;
use re_types::datatypes::{KeypointId, KeypointPair};
use re_viewer_context::{
auto_color_egui, ApplicableEntities, IdentifiedViewSystem, ViewClassRegistryError,
ViewSystemExecutionError, ViewSystemIdentifier, ViewSystemRegistrator, VisualizableEntities,
VisualizableFilterContext, VisualizerCollection,
};
use re_view::clamped_or_nothing;
use crate::view_2d::VisualizableFilterContext2D;
use crate::view_3d::VisualizableFilterContext3D;
use super::contexts::SpatialSceneEntityContext;
pub type Keypoints = HashMap<(re_types::components::ClassId, i64), HashMap<KeypointId, glam::Vec3>>;
pub fn register_2d_spatial_visualizers(
system_registry: &mut ViewSystemRegistrator<'_>,
) -> Result<(), ViewClassRegistryError> {
system_registry.register_visualizer::<arrows2d::Arrows2DVisualizer>()?;
system_registry.register_visualizer::<arrows3d::Arrows3DVisualizer>()?;
system_registry.register_visualizer::<assets3d::Asset3DVisualizer>()?;
system_registry.register_visualizer::<boxes2d::Boxes2DVisualizer>()?;
system_registry.register_visualizer::<boxes3d::Boxes3DVisualizer>()?;
system_registry.register_visualizer::<depth_images::DepthImageVisualizer>()?;
system_registry.register_visualizer::<encoded_image::EncodedImageVisualizer>()?;
system_registry.register_visualizer::<images::ImageVisualizer>()?;
system_registry.register_visualizer::<lines2d::Lines2DVisualizer>()?;
system_registry.register_visualizer::<lines3d::Lines3DVisualizer>()?;
system_registry.register_visualizer::<meshes::Mesh3DVisualizer>()?;
system_registry.register_visualizer::<points2d::Points2DVisualizer>()?;
system_registry.register_visualizer::<points3d::Points3DVisualizer>()?;
system_registry.register_visualizer::<segmentation_images::SegmentationImageVisualizer>()?;
system_registry.register_visualizer::<transform3d_arrows::AxisLengthDetector>()?;
system_registry.register_visualizer::<transform3d_arrows::Transform3DArrowsVisualizer>()?;
system_registry.register_visualizer::<videos::VideoFrameReferenceVisualizer>()?;
Ok(())
}
pub fn register_3d_spatial_visualizers(
system_registry: &mut ViewSystemRegistrator<'_>,
) -> Result<(), ViewClassRegistryError> {
system_registry.register_visualizer::<arrows2d::Arrows2DVisualizer>()?;
system_registry.register_visualizer::<arrows3d::Arrows3DVisualizer>()?;
system_registry.register_visualizer::<assets3d::Asset3DVisualizer>()?;
system_registry.register_visualizer::<boxes2d::Boxes2DVisualizer>()?;
system_registry.register_visualizer::<boxes3d::Boxes3DVisualizer>()?;
system_registry.register_visualizer::<capsules3d::Capsules3DVisualizer>()?;
system_registry.register_visualizer::<cameras::CamerasVisualizer>()?;
system_registry.register_visualizer::<depth_images::DepthImageVisualizer>()?;
system_registry.register_visualizer::<encoded_image::EncodedImageVisualizer>()?;
system_registry.register_visualizer::<images::ImageVisualizer>()?;
system_registry.register_visualizer::<lines2d::Lines2DVisualizer>()?;
system_registry.register_visualizer::<lines3d::Lines3DVisualizer>()?;
system_registry.register_visualizer::<meshes::Mesh3DVisualizer>()?;
system_registry.register_visualizer::<points2d::Points2DVisualizer>()?;
system_registry.register_visualizer::<points3d::Points3DVisualizer>()?;
system_registry.register_visualizer::<segmentation_images::SegmentationImageVisualizer>()?;
system_registry.register_visualizer::<ellipsoids::Ellipsoids3DVisualizer>()?;
system_registry.register_visualizer::<transform3d_arrows::AxisLengthDetector>()?;
system_registry.register_visualizer::<transform3d_arrows::Transform3DArrowsVisualizer>()?;
system_registry.register_visualizer::<videos::VideoFrameReferenceVisualizer>()?;
Ok(())
}
pub fn visualizers_processing_draw_order() -> impl Iterator<Item = ViewSystemIdentifier> {
[
arrows2d::Arrows2DVisualizer::identifier(),
boxes2d::Boxes2DVisualizer::identifier(),
depth_images::DepthImageVisualizer::identifier(),
encoded_image::EncodedImageVisualizer::identifier(),
images::ImageVisualizer::identifier(),
lines2d::Lines2DVisualizer::identifier(),
points2d::Points2DVisualizer::identifier(),
segmentation_images::SegmentationImageVisualizer::identifier(),
]
.into_iter()
}
pub fn collect_ui_labels(visualizers: &VisualizerCollection) -> Vec<UiLabel> {
visualizers
.iter_visualizer_data::<SpatialViewVisualizerData>()
.flat_map(|data| data.ui_labels.iter().cloned())
.collect()
}
pub fn process_radius_slice(
entity_path: &EntityPath,
num_instances: usize,
radii: &[re_types::components::Radius],
fallback_radius: re_types::components::Radius,
) -> Vec<re_renderer::Size> {
re_tracing::profile_function!();
if let Some(last_radius) = radii.last() {
if radii.len() == num_instances {
radii
.iter()
.map(|radius| process_radius(entity_path, *radius))
.collect()
} else if radii.len() == 1 {
let last_radius = process_radius(entity_path, *last_radius);
vec![last_radius; num_instances]
} else {
clamped_or_nothing(radii, num_instances)
.map(|radius| process_radius(entity_path, *radius))
.collect()
}
} else {
vec![re_renderer::Size(*fallback_radius.0); num_instances]
}
}
fn process_radius(
entity_path: &EntityPath,
radius: re_types::components::Radius,
) -> re_renderer::Size {
if radius.0.is_infinite() {
re_log::warn_once!("Found infinite radius in entity {entity_path}");
} else if radius.0.is_nan() {
re_log::warn_once!("Found NaN radius in entity {entity_path}");
}
re_renderer::Size(*radius.0)
}
pub fn load_keypoint_connections(
line_builder: &mut re_renderer::LineDrawableBuilder<'_>,
ent_context: &SpatialSceneEntityContext<'_>,
ent_path: &re_entity_db::EntityPath,
keypoints: &Keypoints,
) -> Result<(), ViewSystemExecutionError> {
re_tracing::profile_function!();
let max_num_connections = keypoints
.iter()
.map(|((class_id, _time), _keypoints_in_class)| {
ent_context
.annotations
.resolved_class_description(Some(*class_id))
.class_description
.map_or(0, |d| d.keypoint_connections.len())
})
.sum();
if max_num_connections == 0 {
return Ok(());
}
line_builder.reserve_strips(max_num_connections)?;
line_builder.reserve_vertices(max_num_connections * 2)?;
let world_from_obj = ent_context.transform_info.single_entity_transform_silent();
let mut line_batch = line_builder
.batch("keypoint connections")
.world_from_obj(world_from_obj)
.picking_object_id(re_renderer::PickingLayerObjectId(ent_path.hash64()));
let line_radius = re_renderer::Size(*re_types::components::Radius::default().0);
for ((class_id, _time), keypoints_in_class) in keypoints {
let resolved_class_description = ent_context
.annotations
.resolved_class_description(Some(*class_id));
let Some(class_description) = resolved_class_description.class_description else {
continue;
};
let color = class_description.info.color.map_or_else(
|| auto_color_egui(class_description.info.id),
|color| color.into(),
);
for KeypointPair {
keypoint0: a,
keypoint1: b,
} in &class_description.keypoint_connections
{
let (Some(a), Some(b)) = (keypoints_in_class.get(a), keypoints_in_class.get(b)) else {
re_log::warn_once!(
"Keypoint connection from index {:?} to {:?} could not be resolved in object {:?}",
a, b, ent_path
);
continue;
};
line_batch
.add_segment(*a, *b)
.radius(line_radius)
.color(color)
.flags(re_renderer::renderer::LineStripFlags::FLAG_COLOR_GRADIENT)
.picking_instance_id(re_renderer::PickingLayerInstanceId(
re_log_types::Instance::ALL.get(),
));
}
}
Ok(())
}
pub fn image_view_coordinates() -> re_types::components::ViewCoordinates {
re_types::components::ViewCoordinates::RDF
}
fn filter_visualizable_2d_entities(
entities: ApplicableEntities,
context: &dyn VisualizableFilterContext,
) -> VisualizableEntities {
if let Some(context) = context
.as_any()
.downcast_ref::<VisualizableFilterContext2D>()
{
VisualizableEntities(
context
.entities_in_main_2d_space
.intersection(&entities.0)
.cloned()
.collect(),
)
} else if let Some(context) = context
.as_any()
.downcast_ref::<VisualizableFilterContext3D>()
{
VisualizableEntities(
context
.entities_under_pinholes
.intersection(&entities.0)
.cloned()
.collect(),
)
} else {
VisualizableEntities(entities.0)
}
}
fn filter_visualizable_3d_entities(
entities: ApplicableEntities,
context: &dyn VisualizableFilterContext,
) -> VisualizableEntities {
if let Some(context) = context
.as_any()
.downcast_ref::<VisualizableFilterContext2D>()
{
VisualizableEntities(
context
.reprojectable_3d_entities
.intersection(&entities.0)
.cloned()
.collect(),
)
} else if let Some(context) = context
.as_any()
.downcast_ref::<VisualizableFilterContext3D>()
{
VisualizableEntities(
context
.entities_in_main_3d_space
.intersection(&entities.0)
.cloned()
.collect(),
)
} else {
VisualizableEntities(entities.0)
}
}