use std::sync::Arc;
use re_chunk_store::LatestAtQuery;
use re_entity_db::EntityDb;
use re_log_types::{StoreId, StoreKind};
use crate::{
blueprint_timeline, command_channel, ApplicationSelectionState, CommandReceiver, CommandSender,
ComponentUiRegistry, ItemCollection, RecordingConfig, StoreContext, SystemCommand,
ViewClassRegistry, ViewerContext,
};
pub struct TestContext {
pub recording_store: EntityDb,
pub blueprint_store: EntityDb,
pub view_class_registry: ViewClassRegistry,
pub selection_state: ApplicationSelectionState,
pub recording_config: RecordingConfig,
pub blueprint_query: LatestAtQuery,
component_ui_registry: ComponentUiRegistry,
command_sender: CommandSender,
command_receiver: CommandReceiver,
}
impl Default for TestContext {
fn default() -> Self {
let recording_store = EntityDb::new(StoreId::random(StoreKind::Recording));
let blueprint_store = EntityDb::new(StoreId::random(StoreKind::Blueprint));
let (command_sender, command_receiver) = command_channel();
let recording_config = RecordingConfig::default();
let blueprint_query = LatestAtQuery::latest(blueprint_timeline());
let component_ui_registry = ComponentUiRegistry::new(Box::new(
|_ctx, _ui, _ui_layout, _query, _db, _entity_path, _row_id, _component| {},
));
Self {
recording_store,
blueprint_store,
view_class_registry: Default::default(),
selection_state: Default::default(),
recording_config,
blueprint_query,
component_ui_registry,
command_sender,
command_receiver,
}
}
}
impl TestContext {
pub fn edit_selection(&mut self, edit_fn: impl FnOnce(&mut ApplicationSelectionState)) {
edit_fn(&mut self.selection_state);
self.selection_state.on_frame_start(|_| true, None);
}
pub fn run(&self, egui_ctx: &egui::Context, func: impl FnOnce(&ViewerContext<'_>)) {
re_ui::apply_style_and_install_loaders(egui_ctx);
let store_context = StoreContext {
app_id: "rerun_test".into(),
blueprint: &self.blueprint_store,
default_blueprint: None,
recording: &self.recording_store,
bundle: &Default::default(),
caches: &Default::default(),
hub: &Default::default(),
};
let drag_and_drop_manager = crate::DragAndDropManager::new(ItemCollection::default());
let ctx = ViewerContext {
app_options: &Default::default(),
cache: &Default::default(),
reflection: &Default::default(),
component_ui_registry: &self.component_ui_registry,
view_class_registry: &self.view_class_registry,
store_context: &store_context,
applicable_entities_per_visualizer: &Default::default(),
indicated_entities_per_visualizer: &Default::default(),
query_results: &Default::default(),
rec_cfg: &self.recording_config,
blueprint_cfg: &Default::default(),
selection_state: &self.selection_state,
blueprint_query: &self.blueprint_query,
egui_ctx,
render_ctx: None,
command_sender: &self.command_sender,
focused_item: &None,
drag_and_drop_manager: &drag_and_drop_manager,
};
func(&ctx);
}
pub fn run_in_egui_central_panel(
&self,
mut func: impl FnMut(&ViewerContext<'_>, &mut egui::Ui),
) {
egui::__run_test_ctx(|ctx| {
egui::CentralPanel::default().show(ctx, |ui| {
let egui_ctx = ui.ctx().clone();
self.run(&egui_ctx, |ctx| {
func(ctx, ui);
});
});
});
}
pub fn handle_system_commands(&mut self) {
while let Some(command) = self.command_receiver.recv_system() {
let mut handled = true;
let command_name = format!("{command:?}");
match command {
SystemCommand::UpdateBlueprint(store_id, chunks) => {
assert_eq!(store_id, self.blueprint_store.store_id());
for chunk in chunks {
self.blueprint_store
.add_chunk(&Arc::new(chunk))
.expect("Updating the blueprint chunk store failed");
}
}
SystemCommand::DropEntity(store_id, entity_path) => {
assert_eq!(store_id, self.blueprint_store.store_id());
self.blueprint_store
.drop_entity_path_recursive(&entity_path);
}
SystemCommand::SetSelection(item) => {
self.selection_state.set_selection(item);
}
SystemCommand::SetActiveTimeline { rec_id, timeline } => {
assert_eq!(rec_id, self.recording_store.store_id());
self.recording_config
.time_ctrl
.write()
.set_timeline(timeline);
}
SystemCommand::SetFocus(_)
| SystemCommand::ActivateApp(_)
| SystemCommand::CloseApp(_)
| SystemCommand::LoadDataSource(_)
| SystemCommand::ClearSourceAndItsStores(_)
| SystemCommand::AddReceiver(_)
| SystemCommand::ResetViewer
| SystemCommand::ClearActiveBlueprint
| SystemCommand::ClearAndGenerateBlueprint
| SystemCommand::ActivateRecording(_)
| SystemCommand::CloseStore(_)
| SystemCommand::UndoBlueprint { .. }
| SystemCommand::RedoBlueprint { .. }
| SystemCommand::CloseAllRecordings => handled = false,
#[cfg(debug_assertions)]
SystemCommand::EnableInspectBlueprintTimeline(_) => handled = false,
#[cfg(not(target_arch = "wasm32"))]
SystemCommand::FileSaver(_) => handled = false,
}
eprintln!(
"{} system command: {command_name:?}",
if handled { "Handled" } else { "Ignored" }
);
}
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::Item;
use re_entity_db::InstancePath;
#[test]
fn test_edit_selection() {
let mut test_context = TestContext::default();
let item = Item::InstancePath(InstancePath::entity_all("/entity/path".into()));
test_context.edit_selection(|selection_state| {
selection_state.set_selection(item.clone());
});
test_context.run_in_egui_central_panel(|ctx, _| {
assert_eq!(
ctx.selection_state.selected_items().single_item(),
Some(&item)
);
});
}
}