use egui::{color_picker, Vec2};
use itertools::Itertools;
use re_types::components::AnnotationContext;
use re_types::datatypes::{
AnnotationInfo, ClassDescription, ClassDescriptionMapElem, KeypointId, KeypointPair,
};
use re_ui::{DesignTokens, UiExt as _};
use re_viewer_context::{auto_color_egui, UiLayout, ViewerContext};
use super::DataUi;
impl crate::EntityDataUi for re_types::components::ClassId {
fn entity_data_ui(
&self,
ctx: &ViewerContext<'_>,
ui: &mut egui::Ui,
ui_layout: UiLayout,
entity_path: &re_log_types::EntityPath,
_row_id: Option<re_chunk_store::RowId>,
query: &re_chunk_store::LatestAtQuery,
_db: &re_entity_db::EntityDb,
) {
let annotations = crate::annotations(ctx, query, entity_path);
let class = annotations
.resolved_class_description(Some(*self))
.class_description;
if let Some(class) = class {
let response = ui.horizontal(|ui| {
small_color_ui(ui, &class.info);
let mut text = format!("{}", self.0);
if let Some(label) = &class.info.label {
text.push(' ');
text.push_str(label.as_str());
}
ui_layout.label(ui, text);
});
let id = self.0;
if ui_layout.is_single_line() {
if !class.keypoint_connections.is_empty() || !class.keypoint_annotations.is_empty()
{
response.response.on_hover_ui(|ui| {
class_description_ui(ui, UiLayout::Tooltip, class, id);
});
}
} else {
ui.separator();
class_description_ui(ui, ui_layout, class, id);
}
} else {
ui_layout.label(ui, format!("{}", self.0));
}
}
}
impl crate::EntityDataUi for re_types::components::KeypointId {
fn entity_data_ui(
&self,
ctx: &ViewerContext<'_>,
ui: &mut egui::Ui,
ui_layout: UiLayout,
entity_path: &re_log_types::EntityPath,
_row_id: Option<re_chunk_store::RowId>,
query: &re_chunk_store::LatestAtQuery,
_db: &re_entity_db::EntityDb,
) {
if let Some(info) = annotation_info(ctx, entity_path, query, self.0) {
ui.horizontal(|ui| {
small_color_ui(ui, &info);
let mut text = format!("{}", self.0);
if let Some(label) = &info.label {
text.push(' ');
text.push_str(label.as_str());
}
ui_layout.data_label(ui, text);
});
} else {
ui_layout.data_label(ui, format!("{}", self.0));
}
}
}
fn annotation_info(
ctx: &re_viewer_context::ViewerContext<'_>,
entity_path: &re_log_types::EntityPath,
query: &re_chunk_store::LatestAtQuery,
keypoint_id: KeypointId,
) -> Option<AnnotationInfo> {
let (_, class_id) = ctx
.recording()
.latest_at_component_quiet::<re_types::components::ClassId>(entity_path, query)?;
let annotations = crate::annotations(ctx, query, entity_path);
let class = annotations.resolved_class_description(Some(class_id));
class.keypoint_map?.get(&keypoint_id).cloned()
}
impl DataUi for AnnotationContext {
fn data_ui(
&self,
_ctx: &ViewerContext<'_>,
ui: &mut egui::Ui,
ui_layout: UiLayout,
_query: &re_chunk_store::LatestAtQuery,
_db: &re_entity_db::EntityDb,
) {
match ui_layout {
UiLayout::List | UiLayout::Tooltip => {
let text = if self.0.len() == 1 {
let descr = &self.0[0].class_description;
format!(
"One class containing {} keypoints and {} connections",
descr.keypoint_annotations.len(),
descr.keypoint_connections.len()
)
} else {
format!("{} classes", self.0.len())
};
ui_layout.label(ui, text);
}
UiLayout::SelectionPanel => {
ui.vertical(|ui| {
ui.maybe_collapsing_header(true, "Classes", true, |ui| {
let annotation_infos = self
.0
.iter()
.map(|class| &class.class_description.info)
.sorted_by_key(|info| info.id)
.collect_vec();
annotation_info_table_ui(ui, ui_layout, &annotation_infos);
});
for ClassDescriptionMapElem {
class_id,
class_description,
} in &self.0
{
class_description_ui(ui, ui_layout, class_description, *class_id);
}
});
}
}
}
}
fn class_description_ui(
ui: &mut egui::Ui,
ui_layout: UiLayout,
class: &ClassDescription,
id: re_types::datatypes::ClassId,
) {
if class.keypoint_connections.is_empty() && class.keypoint_annotations.is_empty() {
return;
}
re_tracing::profile_function!();
let use_collapsible = ui_layout == UiLayout::SelectionPanel;
let row_height = DesignTokens::table_line_height();
if !class.keypoint_annotations.is_empty() {
ui.maybe_collapsing_header(
use_collapsible,
&format!("Keypoints Annotation for Class {}", id.0),
true,
|ui| {
let annotation_infos = class
.keypoint_annotations
.iter()
.sorted_by_key(|annotation| annotation.id)
.collect_vec();
ui.push_id(format!("keypoint_annotations_{}", id.0), |ui| {
annotation_info_table_ui(ui, ui_layout, &annotation_infos);
});
},
);
}
if !class.keypoint_connections.is_empty() {
ui.maybe_collapsing_header(
use_collapsible,
&format!("Keypoint Connections for Class {}", id.0),
true,
|ui| {
use egui_extras::Column;
let table = ui_layout
.table(ui)
.id_salt(("keypoints_connections", id))
.cell_layout(egui::Layout::left_to_right(egui::Align::Center))
.column(Column::auto().clip(true).at_least(40.0))
.column(Column::auto().clip(true).at_least(40.0));
table
.header(DesignTokens::table_header_height(), |mut header| {
DesignTokens::setup_table_header(&mut header);
header.col(|ui| {
ui.strong("From");
});
header.col(|ui| {
ui.strong("To");
});
})
.body(|mut body| {
DesignTokens::setup_table_body(&mut body);
let keypoint_map: ahash::HashMap<KeypointId, AnnotationInfo> = {
re_tracing::profile_scope!("build_annotation_map");
class
.keypoint_annotations
.iter()
.map(|kp| (kp.id.into(), kp.clone()))
.collect()
};
body.rows(row_height, class.keypoint_connections.len(), |mut row| {
let pair = &class.keypoint_connections[row.index()];
let KeypointPair {
keypoint0,
keypoint1,
} = pair;
for id in [keypoint0, keypoint1] {
row.col(|ui| {
ui.label(
keypoint_map
.get(id)
.and_then(|info| info.label.as_ref())
.map_or_else(
|| format!("id {}", id.0),
|label| label.to_string(),
),
);
});
}
});
});
},
);
}
}
fn annotation_info_table_ui(
ui: &mut egui::Ui,
ui_layout: UiLayout,
annotation_infos: &[&AnnotationInfo],
) {
re_tracing::profile_function!();
let row_height = DesignTokens::table_line_height();
ui.spacing_mut().item_spacing.x = 20.0; use egui_extras::Column;
let table = ui_layout
.table(ui)
.cell_layout(egui::Layout::left_to_right(egui::Align::Center))
.column(Column::auto()) .column(Column::auto().clip(true).at_least(40.0)) .column(Column::auto()); table
.header(DesignTokens::table_header_height(), |mut header| {
DesignTokens::setup_table_header(&mut header);
header.col(|ui| {
ui.strong("Class Id");
});
header.col(|ui| {
ui.strong("Label");
});
header.col(|ui| {
ui.strong("Color");
});
})
.body(|mut body| {
DesignTokens::setup_table_body(&mut body);
body.rows(row_height, annotation_infos.len(), |mut row| {
let info = &annotation_infos[row.index()];
row.col(|ui| {
ui.label(info.id.to_string());
});
row.col(|ui| {
let label = if let Some(label) = &info.label {
label.as_str()
} else {
""
};
ui.label(label);
});
row.col(|ui| {
color_ui(ui, info, Vec2::new(64.0, row_height));
});
});
});
}
fn color_ui(ui: &mut egui::Ui, info: &AnnotationInfo, size: Vec2) {
ui.horizontal(|ui| {
ui.spacing_mut().item_spacing.x = 8.0;
let color = info
.color
.map_or_else(|| auto_color_egui(info.id), |color| color.into());
color_picker::show_color(ui, color, size);
if info.color.is_none() {
ui.weak("(auto)")
.on_hover_text("Color chosen automatically, since it was not logged");
}
});
}
fn small_color_ui(ui: &mut egui::Ui, info: &AnnotationInfo) {
let size = egui::Vec2::splat(DesignTokens::table_line_height());
let color = info
.color
.map_or_else(|| auto_color_egui(info.id), |color| color.into());
let response = color_picker::show_color(ui, color, size);
if info.color.is_none() {
response.on_hover_text("Color chosen automatically, since it was not logged");
}
}