Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update UI for static components #6101

Merged
merged 6 commits into from
Apr 25, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

46 changes: 46 additions & 0 deletions crates/re_data_ui/src/component.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,52 @@ impl DataUi for EntityLatestAtResults {
UiVerbosity::LimitHeight | UiVerbosity::Full => num_instances,
};

// Display log time and additional diagnostic information for static components.
abey79 marked this conversation as resolved.
Show resolved Hide resolved
if verbosity != UiVerbosity::Small {
ui.label(format!(
"Timestamp: {}",
query
.timeline()
.typ()
.format(self.results.index().0, ctx.app_options.time_zone),
));

// if the component is static, we display extra diagnostic information
if self.results.is_static() {
if let Some(histogram) = db
.tree()
.subtree(&self.entity_path)
.and_then(|tree| tree.entity.components.get(&self.component_name))
{
if histogram.num_static_messages() > 1 {
ui.label(ctx.re_ui.warning_text(format!(
"Static component value was overridden {} times",
histogram.num_static_messages() - 1,
abey79 marked this conversation as resolved.
Show resolved Hide resolved
)))
.on_hover_text(
"When a static component is logged multiple times, only the last value \
is stored. Previously logged values are overwritten and not \
recoverable.",
);
}

let timeline_message_count = histogram.num_timeline_messages();
if timeline_message_count > 0 {
ui.label(ctx.re_ui.error_text(format!(
"Static component has {} event{} logged on timelines",
timeline_message_count,
if timeline_message_count > 1 { "s" } else { "" }
)))
.on_hover_text(
"Components should be logged either as static or on timelines, but \
never both. Values for static components logged to timelines cannot be \
displayed.",
);
}
}
}
}

// Here we enforce that exactly `max_row` rows are displayed, which means that:
// - For `num_instances == max_row`, then `max_row` rows are displayed.
// - For `num_instances == max_row + 1`, then `max_row-1` rows are displayed and "…2 more"
Expand Down
2 changes: 2 additions & 0 deletions crates/re_data_ui/src/instance_path.rs
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ impl DataUi for InstancePath {
ctx,
ui,
&ComponentPath::new(entity_path.clone(), component_name),
db,
);
}
}
Expand All @@ -89,6 +90,7 @@ impl DataUi for InstancePath {
ctx,
ui,
&ComponentPath::new(entity_path.clone(), component_name),
db,
);

if instance_key.is_splat() {
Expand Down
10 changes: 9 additions & 1 deletion crates/re_data_ui/src/item_ui.rs
Original file line number Diff line number Diff line change
Expand Up @@ -374,12 +374,14 @@ pub fn component_path_button(
ctx: &ViewerContext<'_>,
ui: &mut egui::Ui,
component_path: &ComponentPath,
db: &re_entity_db::EntityDb,
) -> egui::Response {
component_path_button_to(
ctx,
ui,
component_path.component_name.short_name(),
component_path,
db,
)
.on_hover_text(component_path.component_name.full_name()) // we should show the full name somewhere
}
Expand All @@ -390,11 +392,17 @@ pub fn component_path_button_to(
ui: &mut egui::Ui,
text: impl Into<egui::WidgetText>,
component_path: &ComponentPath,
db: &re_entity_db::EntityDb,
) -> egui::Response {
let item = Item::ComponentPath(component_path.clone());
let is_component_static = db.is_component_static(component_path).unwrap_or_default();
let response = ctx.re_ui.selectable_label_with_icon(
ui,
&icons::COMPONENT,
if is_component_static {
&icons::COMPONENT_STATIC
} else {
&icons::COMPONENT
},
text,
ctx.selection().contains_item(&item),
re_ui::LabelStyle::Normal,
Expand Down
21 changes: 17 additions & 4 deletions crates/re_entity_db/src/entity_db.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,9 @@ use re_data_store::{
DataStore, DataStoreConfig, GarbageCollectionOptions, StoreEvent, StoreSubscriber,
};
use re_log_types::{
ApplicationId, DataCell, DataRow, DataTable, DataTableResult, EntityPath, EntityPathHash,
LogMsg, RowId, SetStoreInfo, StoreId, StoreInfo, StoreKind, TimePoint, TimeRange, TimeRangeF,
Timeline,
ApplicationId, ComponentPath, DataCell, DataRow, DataTable, DataTableResult, EntityPath,
EntityPathHash, LogMsg, RowId, SetStoreInfo, StoreId, StoreInfo, StoreKind, TimePoint,
TimeRange, TimeRangeF, Timeline,
};
use re_query2::PromiseResult;
use re_types_core::{components::InstanceKey, Archetype, Loggable};
Expand Down Expand Up @@ -329,7 +329,20 @@ impl EntityDb {

/// Total number of timeless messages for any entity.
pub fn num_timeless_messages(&self) -> u64 {
self.tree.num_timeless_messages_recursive()
self.tree.num_static_messages_recursive()
abey79 marked this conversation as resolved.
Show resolved Hide resolved
}

/// Returns whether a component is static.
pub fn is_component_static(&self, component_path: &ComponentPath) -> Option<bool> {
if let Some(entity_tree) = self.tree().subtree(component_path.entity_path()) {
entity_tree
.entity
.components
.get(&component_path.component_name)
.map(|component_histogram| component_histogram.is_static())
} else {
None
}
}

pub fn num_rows(&self) -> usize {
Expand Down
4 changes: 2 additions & 2 deletions crates/re_entity_db/src/entity_tree.rs
Original file line number Diff line number Diff line change
Expand Up @@ -245,8 +245,8 @@ impl EntityTree {
}

/// Number of timeless messages in this tree, or any child, recursively.
pub fn num_timeless_messages_recursive(&self) -> u64 {
self.subtree.time_histogram.num_timeless_messages()
pub fn num_static_messages_recursive(&self) -> u64 {
self.subtree.time_histogram.num_static_messages()
}

pub fn time_histogram_for_component(
Expand Down
36 changes: 23 additions & 13 deletions crates/re_entity_db/src/time_histogram_per_timeline.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,19 @@ pub struct TimeHistogramPerTimeline {
/// When do we have data? Ignores timeless.
times: BTreeMap<Timeline, TimeHistogram>,

/// Extra book-keeping used to seed any timelines that include timeless msgs.
num_timeless_messages: u64,
/// Extra bookkeeping used to seed any timelines that include timeless msgs.
abey79 marked this conversation as resolved.
Show resolved Hide resolved
num_static_messages: u64,
}

impl TimeHistogramPerTimeline {
#[inline]
pub fn is_empty(&self) -> bool {
self.times.is_empty() && self.num_timeless_messages == 0
self.times.is_empty() && self.num_static_messages == 0
}

#[inline]
pub fn is_static(&self) -> bool {
self.num_static_messages > 0
}

#[inline]
Expand All @@ -47,20 +52,25 @@ impl TimeHistogramPerTimeline {
}

#[inline]
pub fn num_timeless_messages(&self) -> u64 {
self.num_timeless_messages
pub fn num_static_messages(&self) -> u64 {
self.num_static_messages
}

/// Total number of (non-static) message over all timelines.
pub fn num_timeline_messages(&self) -> u64 {
abey79 marked this conversation as resolved.
Show resolved Hide resolved
self.times.values().map(|hist| hist.total_count()).sum()
}

pub fn add(&mut self, times: &[(Timeline, TimeInt)], n: u32) {
if times.is_empty() {
self.num_timeless_messages = self
.num_timeless_messages
self.num_static_messages = self
.num_static_messages
.checked_add(n as u64)
.unwrap_or_else(|| {
re_log::debug!(
current = self.num_timeless_messages,
current = self.num_static_messages,
added = n,
"book keeping overflowed"
"bookkeeping overflowed"
);
u64::MAX
});
Expand All @@ -76,15 +86,15 @@ impl TimeHistogramPerTimeline {

pub fn remove(&mut self, timepoint: &TimePoint, n: u32) {
if timepoint.is_static() {
self.num_timeless_messages = self
.num_timeless_messages
self.num_static_messages = self
.num_static_messages
.checked_sub(n as u64)
.unwrap_or_else(|| {
// TODO(#4355): hitting this on plots demo
re_log::debug!(
current = self.num_timeless_messages,
current = self.num_static_messages,
removed = n,
"book keeping underflowed"
"bookkeeping underflowed"
);
u64::MIN
});
Expand Down
6 changes: 6 additions & 0 deletions crates/re_query_cache2/src/latest_at/results.rs
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,12 @@ impl CachedLatestAtComponentResults {
_ => None,
}
}

/// Returns whether the resolved data is static.
#[inline]
pub fn is_static(&self) -> bool {
self.index.0 == TimeInt::STATIC
}
}

impl SizeBytes for CachedLatestAtComponentResults {
Expand Down
1 change: 1 addition & 0 deletions crates/re_time_panel/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ re_data_store.workspace = true
re_data_ui.workspace = true
re_entity_db.workspace = true
re_format.workspace = true
re_log.workspace = true
re_log_types.workspace = true
re_tracing.workspace = true
re_ui.workspace = true
Expand Down
14 changes: 4 additions & 10 deletions crates/re_time_panel/src/data_density_graph.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ use egui::{epaint::Vertex, lerp, pos2, remap, Color32, NumExt as _, Rect, Shape}

use re_data_ui::item_ui;
use re_entity_db::TimeHistogram;
use re_log_types::{ComponentPath, TimeInt, TimeRange, TimeReal};
use re_log_types::{ComponentPath, TimeRange, TimeReal};
use re_viewer_context::{Item, TimeControl, UiVerbosity, ViewerContext};

use crate::TimePanelItem;
Expand Down Expand Up @@ -368,14 +368,13 @@ fn smooth(density: &[f32]) -> Vec<f32> {

#[allow(clippy::too_many_arguments)]
pub fn data_density_graph_ui(
data_dentity_graph_painter: &mut DataDensityGraphPainter,
data_density_graph_painter: &mut DataDensityGraphPainter,
ctx: &ViewerContext<'_>,
time_ctrl: &mut TimeControl,
db: &re_entity_db::EntityDb,
time_area_response: &egui::Response,
time_area_painter: &egui::Painter,
ui: &egui::Ui,
num_timeless_messages: usize,
time_histogram: &TimeHistogram,
row_rect: Rect,
time_ranges_ui: &TimeRangesUi,
Expand Down Expand Up @@ -435,11 +434,6 @@ pub fn data_density_graph_ui(
}
};

add_data_point(
TimeRange::point(TimeInt::MIN_TIME_PANEL),
num_timeless_messages,
);

let visible_time_range = time_ranges_ui
.time_range_from_x_range((row_rect.left() - MARGIN_X)..=(row_rect.right() + MARGIN_X));

Expand Down Expand Up @@ -479,7 +473,7 @@ pub fn data_density_graph_ui(
density_graph.buckets = smooth(&density_graph.buckets);

density_graph.paint(
data_dentity_graph_painter,
data_density_graph_painter,
row_rect.y_range(),
time_area_painter,
graph_color(ctx, &item.to_item(), ui),
Expand Down Expand Up @@ -559,7 +553,7 @@ fn show_row_ids_tooltip(

if let Some(component_name) = component_name {
let component_path = ComponentPath::new(entity_path.clone(), *component_name);
item_ui::component_path_button(ctx, ui, &component_path);
item_ui::component_path_button(ctx, ui, &component_path, db);
ui.add_space(8.0);
component_path.data_ui(ctx, ui, verbosity, &query, db);
} else {
Expand Down
Loading
Loading