use re_viewer::external::{
arrow, eframe, egui, re_chunk_store, re_entity_db, re_log, re_log_types, re_memory, re_types,
};
#[global_allocator]
static GLOBAL: re_memory::AccountingAllocator<mimalloc::MiMalloc> =
re_memory::AccountingAllocator::new(mimalloc::MiMalloc);
fn main() -> Result<(), Box<dyn std::error::Error>> {
let main_thread_token = re_viewer::MainThreadToken::i_promise_i_am_on_the_main_thread();
re_log::setup_logging();
re_crash_handler::install_crash_handlers(re_viewer::build_info());
let rx = re_sdk_comms::serve(
"0.0.0.0",
re_sdk_comms::DEFAULT_SERVER_PORT,
Default::default(),
)?;
let mut native_options = re_viewer::native::eframe_options(None);
native_options.viewport = native_options
.viewport
.with_app_id("rerun_extend_viewer_ui_example");
let startup_options = re_viewer::StartupOptions::default();
let app_env = re_viewer::AppEnvironment::Custom("My Wrapper".to_owned());
let window_title = "My Customized Viewer";
eframe::run_native(
window_title,
native_options,
Box::new(move |cc| {
re_viewer::customize_eframe_and_setup_renderer(cc)?;
let mut rerun_app = re_viewer::App::new(
main_thread_token,
re_viewer::build_info(),
&app_env,
startup_options,
cc,
);
rerun_app.add_receiver(rx);
Ok(Box::new(MyApp { rerun_app }))
}),
)?;
Ok(())
}
struct MyApp {
rerun_app: re_viewer::App,
}
impl eframe::App for MyApp {
fn save(&mut self, storage: &mut dyn eframe::Storage) {
self.rerun_app.save(storage);
}
fn update(&mut self, ctx: &egui::Context, frame: &mut eframe::Frame) {
egui::SidePanel::right("my_side_panel")
.default_width(200.0)
.show(ctx, |ui| {
self.ui(ui);
});
self.rerun_app.update(ctx, frame);
}
}
impl MyApp {
fn ui(&mut self, ui: &mut egui::Ui) {
ui.add_space(4.0);
ui.vertical_centered(|ui| {
ui.strong("My custom panel");
});
ui.separator();
if let Some(entity_db) = self.rerun_app.recording_db() {
entity_db_ui(ui, entity_db);
} else {
ui.label("No log database loaded yet.");
}
}
}
fn entity_db_ui(ui: &mut egui::Ui, entity_db: &re_entity_db::EntityDb) {
if let Some(store_info) = entity_db.store_info() {
ui.label(format!("Application ID: {}", store_info.application_id));
}
let timeline = re_log_types::Timeline::log_time();
ui.separator();
ui.strong("Entities:");
egui::ScrollArea::vertical()
.auto_shrink([false, true])
.show(ui, |ui| {
for entity_path in entity_db.entity_paths() {
ui.collapsing(entity_path.to_string(), |ui| {
entity_ui(ui, entity_db, timeline, entity_path);
});
}
});
}
fn entity_ui(
ui: &mut egui::Ui,
entity_db: &re_entity_db::EntityDb,
timeline: re_log_types::Timeline,
entity_path: &re_log_types::EntityPath,
) {
if let Some(components) = entity_db
.storage_engine()
.store()
.all_components_on_timeline_sorted(&timeline, entity_path)
{
for component in components {
ui.collapsing(component.to_string(), |ui| {
component_ui(ui, entity_db, timeline, entity_path, component);
});
}
}
}
fn component_ui(
ui: &mut egui::Ui,
entity_db: &re_entity_db::EntityDb,
timeline: re_log_types::Timeline,
entity_path: &re_log_types::EntityPath,
component_name: re_types::ComponentName,
) {
let query = re_chunk_store::LatestAtQuery::latest(timeline);
let results =
entity_db
.storage_engine()
.cache()
.latest_at(&query, entity_path, [component_name]);
if let Some(data) = results.component_batch_raw(&component_name) {
egui::ScrollArea::vertical()
.auto_shrink([false, true])
.show(ui, |ui| {
let num_instances = data.len();
for i in 0..num_instances {
ui.label(format_arrow(&*data.slice(i, 1)));
}
});
};
}
fn format_arrow(array: &dyn arrow::array::Array) -> String {
use arrow::util::display::{ArrayFormatter, FormatOptions};
let num_bytes = array.get_buffer_memory_size();
if array.len() == 1 && num_bytes < 256 {
let options = FormatOptions::default();
if let Ok(formatter) = ArrayFormatter::try_new(array, &options) {
return formatter.value(0).to_string();
}
}
format!("{num_bytes} bytes")
}