From 1a7903101deedc8d5911cd7ca64703236f45a3b2 Mon Sep 17 00:00:00 2001 From: Lucas Pickup Date: Thu, 4 Jan 2024 16:22:48 -0700 Subject: [PATCH 1/3] Convert TextLog body to display as immutable TextEdit rather than Label. This allows selecting and copying text from the log body. --- .../src/space_view_class.rs | 69 +++++++++++++++---- .../src/visualizer_system.rs | 5 ++ 2 files changed, 62 insertions(+), 12 deletions(-) diff --git a/crates/re_space_view_text_log/src/space_view_class.rs b/crates/re_space_view_text_log/src/space_view_class.rs index 71a8fe4b16fe..eb3147a748c6 100644 --- a/crates/re_space_view_text_log/src/space_view_class.rs +++ b/crates/re_space_view_text_log/src/space_view_class.rs @@ -1,3 +1,4 @@ +use egui::Widget; use re_entity_db::EntityProperties; use std::collections::BTreeMap; @@ -24,6 +25,8 @@ pub struct TextSpaceViewState { pub filters: ViewTextFilters, monospace: bool, + + selectable: bool, } impl SpaceViewState for TextSpaceViewState { @@ -125,6 +128,7 @@ impl SpaceViewClass for TextSpaceView { .radio_value(ui, &mut state.monospace, false, "Proportional"); ctx.re_ui .radio_value(ui, &mut state.monospace, true, "Monospace"); + ctx.re_ui.checkbox(ui, &mut state.selectable, "Selectable"); }); ui.end_row(); }); @@ -402,16 +406,60 @@ fn table_ui( // body row.col(|ui| { - let mut text = egui::RichText::new(entry.body.as_str()); + if state.selectable { + // Use a &str with TextEdit so body is selectable but immutable. + let entry_text_rows = entry.num_body_lines; + let mut binding = entry.body.as_str(); + let mut text = egui::TextEdit::multiline(&mut binding) + .min_size([0.0, 0.0].into()) + .desired_rows(entry_text_rows) + .desired_width(f32::INFINITY); + // Font information for LayoutJob, based on default_layouter in `egui::TextEdit::show_contents` + let font_id = if state.monospace { + egui::TextStyle::Monospace.into() + } else { + egui::FontSelection::default() + } + .resolve(ui.style()); + let text_color = if let Some(color) = entry.color { + color.into() + } else { + // TODO(lupickup): Understand why TextEdit text color is white instead of grey like the RichText version. + ui.visuals() + .override_text_color + .unwrap_or_else(|| ui.visuals().widgets.inactive.text_color()) + }; + let mut layouter = |ui: &egui::Ui, string: &str, wrap_width: f32| { + let mut layout_job = egui::text::LayoutJob::simple( + string.to_owned(), + font_id.clone(), + text_color, + wrap_width, + ); + layout_job.wrap.max_rows = entry_text_rows; + // Fill to end of body column instead of choosing a full "word" to ellide + layout_job.wrap.break_anywhere = true; + ui.fonts(|f| f.layout_job(layout_job)) + }; + text = text.layouter(&mut layouter); + + let mut response = text.ui(ui); + if response.hovered() { + response = response.on_hover_text(entry.body.as_str()); + } + } else { + let mut text = egui::RichText::new(entry.body.as_str()); - if state.monospace { - text = text.monospace(); - } - if let Some(color) = entry.color { - text = text.color(color); - } + if state.monospace { + text = text.monospace(); + } - ui.label(text); + if let Some(color) = entry.color { + text = text.color(color); + } + + ui.label(text); + } }); }); }); @@ -428,8 +476,5 @@ fn table_ui( } fn calc_row_height(entry: &Entry) -> f32 { - // Simple, fast, ugly, and functional - let num_newlines = entry.body.bytes().filter(|&c| c == b'\n').count(); - let num_rows = 1 + num_newlines; - num_rows as f32 * re_ui::ReUi::table_line_height() + entry.num_body_lines as f32 * re_ui::ReUi::table_line_height() } diff --git a/crates/re_space_view_text_log/src/visualizer_system.rs b/crates/re_space_view_text_log/src/visualizer_system.rs index 7f0610a93b9a..b23b89d98f8e 100644 --- a/crates/re_space_view_text_log/src/visualizer_system.rs +++ b/crates/re_space_view_text_log/src/visualizer_system.rs @@ -26,6 +26,8 @@ pub struct Entry { pub body: Text, + pub num_body_lines: usize, + pub level: Option, } @@ -78,12 +80,15 @@ impl VisualizerSystem for TextLogSystem { let colors = arch_view.iter_optional_component::()?; for (body, level, color) in itertools::izip!(bodies, levels, colors) { + // Simple, fast, ugly, and functional + let num_body_lines = body.bytes().filter(|&c| c == b'\n').count() + 1; self.entries.push(Entry { row_id: arch_view.primary_row_id(), entity_path: data_result.entity_path.clone(), time: time.map(|time| time.as_i64()), color, body, + num_body_lines, level, }); } From 62926002c4c62f257813765672e4ec8ab134e220 Mon Sep 17 00:00:00 2001 From: Lucas Pickup Date: Fri, 5 Jan 2024 11:43:22 -0700 Subject: [PATCH 2/3] Fixup clippy warnings. --- crates/re_space_view_text_log/src/space_view_class.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/crates/re_space_view_text_log/src/space_view_class.rs b/crates/re_space_view_text_log/src/space_view_class.rs index eb3147a748c6..0364e7660235 100644 --- a/crates/re_space_view_text_log/src/space_view_class.rs +++ b/crates/re_space_view_text_log/src/space_view_class.rs @@ -443,9 +443,9 @@ fn table_ui( }; text = text.layouter(&mut layouter); - let mut response = text.ui(ui); + let response = text.ui(ui); if response.hovered() { - response = response.on_hover_text(entry.body.as_str()); + response.on_hover_text(entry.body.as_str()); } } else { let mut text = egui::RichText::new(entry.body.as_str()); From 937db0e5dd98a082dd2656cd237417376f737803 Mon Sep 17 00:00:00 2001 From: Lucas Pickup Date: Fri, 5 Jan 2024 15:05:25 -0700 Subject: [PATCH 3/3] Apparently spelling gray is hard. --- crates/re_space_view_text_log/src/space_view_class.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/re_space_view_text_log/src/space_view_class.rs b/crates/re_space_view_text_log/src/space_view_class.rs index 0364e7660235..f665bbbf11fb 100644 --- a/crates/re_space_view_text_log/src/space_view_class.rs +++ b/crates/re_space_view_text_log/src/space_view_class.rs @@ -424,7 +424,7 @@ fn table_ui( let text_color = if let Some(color) = entry.color { color.into() } else { - // TODO(lupickup): Understand why TextEdit text color is white instead of grey like the RichText version. + // TODO(lupickup): Understand why TextEdit text color is white instead of gray like the RichText version. ui.visuals() .override_text_color .unwrap_or_else(|| ui.visuals().widgets.inactive.text_color())