1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
use std::iter;
use itertools::{izip, Either};
use re_entity_db::InstancePathHash;
use re_log_types::{EntityPath, Instance};
use re_types::components::{ShowLabels, Text};
use re_types::Component;
use re_viewer_context::ResolvedAnnotationInfos;
#[cfg(doc)]
use re_viewer_context::ComponentFallbackProvider;
use crate::visualizers::entity_iterator::clamped_or;
#[derive(Clone)]
pub enum UiLabelTarget {
/// Labels a given rect (in scene coordinates)
Rect(egui::Rect),
/// Labels a given point (in scene coordinates)
Point2D(egui::Pos2),
/// A point in space.
Position3D(glam::Vec3),
}
#[derive(Clone)]
pub enum UiLabelStyle {
Color(egui::Color32),
/// Style it like an error message
Error,
}
impl From<egui::Color32> for UiLabelStyle {
fn from(color: egui::Color32) -> Self {
Self::Color(color)
}
}
#[derive(Clone)]
pub struct UiLabel {
pub text: String,
pub style: UiLabelStyle,
/// The shape/position being labeled.
pub target: UiLabelTarget,
/// What is hovered if this label is hovered.
pub labeled_instance: InstancePathHash,
}
/// Inputs for [`process_labels()`], defining the label(s) of a single [batch].
///
/// * `P` is the type of the _positions_ of the labels, which might be 3D or 2D.
/// * `I` is the type of the iterator over positions.
///
/// [batch]: https://rerun.io/docs/concepts/batches
pub struct LabeledBatch<'a, P: 'a, I: Iterator<Item = P> + 'a> {
pub entity_path: &'a EntityPath,
/// `num_instances` should be equal to the length of `instance_positions`.
pub num_instances: usize,
/// The position where a single shared label will be displayed if it is.
/// This is typically the center of the bounding box of the entity.
pub overall_position: P,
/// Note: If we find a reason to make this data type usable more than once,
/// replace this `Iterator` with `IntoIterator`.
pub instance_positions: I,
/// Label data from the batch.
///
/// Length 1 is treated as a label for the whole batch.
///
/// The number of per-instance labels actually drawn is the minimum of the lengths of
/// `instance_positions` and `labels`.
pub labels: &'a [re_types::ArrowString],
/// Colors from the batch to apply to the labels.
///
/// Length 1 is treated as a color for the whole batch.
pub colors: &'a [egui::Color32],
/// The [`ShowLabels`] component value.
///
/// If no value is available from the data, use [`show_labels_fallback`] to obtain it.
pub show_labels: re_types::components::ShowLabels,
pub annotation_infos: &'a ResolvedAnnotationInfos,
}
/// Maximum number of labels after which we stop displaying labels for that entity all together,
/// unless overridden by a [`ShowLabels`] component.
const MAX_NUM_LABELS_PER_ENTITY: usize = 30;
/// Given a visualizer’s query context, compute its [`ShowLabels`] fallback value
/// (used when neither the logged data nor the blueprint provides a value).
///
/// Assumes that the visualizer reads the [`Text`] component for components.
/// The type parameter `C` must be the component type that defines the number of instances
/// in the batch.
///
// TODO(kpreid): This component type (or the length directly) should be gotten from some kind of
// general mechanism of "how big is this batch?" rather than requiring the caller to specify it,
// possibly incorrectly.
///
/// This function is normally used to implement the [`ComponentFallbackProvider`]
/// that will be used in a [`LabeledBatch`].
pub fn show_labels_fallback<C: Component>(ctx: &re_viewer_context::QueryContext<'_>) -> ShowLabels {
let results =
ctx.recording()
.latest_at(ctx.query, ctx.target_entity_path, [C::name(), Text::name()]);
let num_instances = results
.component_batch_raw(&C::name())
.map_or(0, |array| array.len());
let num_labels = results
.component_batch_raw(&Text::name())
.map_or(0, |array| array.len());
ShowLabels::from(num_labels == 1 || num_instances < MAX_NUM_LABELS_PER_ENTITY)
}
/// Produces 3D ui labels from component data.
///
/// See [`process_labels()`] for further documentation.
pub fn process_labels_3d<'a>(
batch: LabeledBatch<'a, glam::Vec3, impl Iterator<Item = glam::Vec3> + 'a>,
world_from_obj: glam::Affine3A,
) -> impl Iterator<Item = UiLabel> + 'a {
process_labels(batch, move |position| {
UiLabelTarget::Position3D(world_from_obj.transform_point3(position))
})
}
/// Produces 2D ui labels from component data.
///
/// See [`process_labels()`] for further documentation.
pub fn process_labels_2d<'a>(
batch: LabeledBatch<'a, glam::Vec2, impl Iterator<Item = glam::Vec2> + 'a>,
world_from_obj: glam::Affine3A,
) -> impl Iterator<Item = UiLabel> + 'a {
process_labels(batch, move |position| {
let point = world_from_obj.transform_point3(position.extend(0.0));
UiLabelTarget::Point2D(egui::pos2(point.x, point.y))
})
}
/// Produces ui labels from component data, allowing the caller to produce [`UiLabelTarget`]s
/// as they see fit.
///
/// Implements policy for displaying a single label vs. per-instance labels, or hiding labels.
pub fn process_labels<'a, P: 'a>(
batch: LabeledBatch<'a, P, impl Iterator<Item = P> + 'a>,
target_from_position: impl Fn(P) -> UiLabelTarget + 'a,
) -> impl Iterator<Item = UiLabel> + 'a {
let LabeledBatch {
entity_path,
num_instances,
overall_position,
instance_positions,
labels,
colors,
show_labels,
annotation_infos,
} = batch;
let show_labels = bool::from(show_labels.0);
if !show_labels {
return Either::Left(iter::empty());
}
// If there's many instances but only a single label, place the single label at the
// overall_position (which is usually a bounding box center).
// TODO(andreas): A smoothed over time (+ discontinuity detection) bounding box would be great.
let label_positions = if labels.len() == 1 && num_instances > 1 {
Either::Left(std::iter::once(overall_position))
} else {
Either::Right(instance_positions)
};
let labels = izip!(
annotation_infos.iter(),
labels.iter().map(Some).chain(std::iter::repeat(None))
)
.map(|(annotation_info, label)| annotation_info.label(label.map(|l| l.as_str())));
let colors = clamped_or(colors, &egui::Color32::WHITE);
Either::Right(
itertools::izip!(label_positions, labels, colors)
.enumerate()
.filter_map(move |(i, (position, label, color))| {
label.map(|label| UiLabel {
text: label,
style: (*color).into(),
target: target_from_position(position),
labeled_instance: InstancePathHash::instance(
entity_path,
Instance::from(i as u64),
),
})
}),
)
}