1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178
use re_chunk::EntityPath;
use re_chunk_store::external::re_chunk::Chunk;
use re_data_source::DataSource;
use re_log_types::StoreId;
use re_ui::{UICommand, UICommandSender};
// ----------------------------------------------------------------------------
/// Commands used by internal system components
// TODO(jleibs): Is there a better crate for this?
#[derive(strum_macros::IntoStaticStr)]
pub enum SystemCommand {
/// Make this the active application.
ActivateApp(re_log_types::ApplicationId),
/// Close this app and all its recordings.
CloseApp(re_log_types::ApplicationId),
/// Load some data.
LoadDataSource(DataSource),
/// Clear everything that came from this source, and close the source.
ClearSourceAndItsStores(re_smart_channel::SmartChannelSource),
AddReceiver(re_smart_channel::Receiver<re_log_types::LogMsg>),
/// Reset the `Viewer` to the default state
ResetViewer,
/// Reset the `Blueprint` to the default state
ClearActiveBlueprint,
/// Clear the blueprint and generate a new one
ClearAndGenerateBlueprint,
/// If this is a recording, switch to it.
ActivateRecording(StoreId),
/// Close a recording or blueprint (free its memory).
CloseStore(StoreId),
/// Close all stores and show the welcome screen again.
CloseAllRecordings,
/// Update the blueprint with additional data
///
/// The [`StoreId`] should generally be the currently selected blueprint
/// but is tracked manually to ensure self-consistency if the blueprint
/// is both modified and changed in the same frame.
///
/// Instead of using this directly, consider using
/// [`crate::ViewerContext::save_blueprint_archetype`] or similar.
UpdateBlueprint(StoreId, Vec<Chunk>),
UndoBlueprint {
blueprint_id: StoreId,
},
RedoBlueprint {
blueprint_id: StoreId,
},
/// Drop a specific entity from a store.
///
/// Also drops all recursive children.
///
/// The [`StoreId`] should generally be the currently selected blueprint
/// but is tracked manually to ensure self-consistency if the blueprint
/// is both modified and changed in the same frame.
DropEntity(StoreId, EntityPath),
/// Show a timeline of the blueprint data.
#[cfg(debug_assertions)]
EnableInspectBlueprintTimeline(bool),
/// Set the item selection.
SetSelection(crate::Item),
/// Set the active timeline for the given recording.
SetActiveTimeline {
rec_id: StoreId,
timeline: re_chunk::Timeline,
},
/// Sets the focus to the given item.
///
/// The focused item is cleared out every frame.
/// Focusing is triggered either explicitly by ui-elements saying so
/// or by double-clicking on a button representing an item.
///
/// Unlike item selection, item focusing is not global state.
/// It may however have stateful effects in certain views,
/// e.g. the 3D view may follow the last focused item as it moves,
/// or a frame may be highlighted for a few frames.
///
/// Just like selection highlighting, the exact behavior of focusing is up to the receiving views.
SetFocus(crate::Item),
/// Add a task, run on a background thread, that saves something to disk.
#[cfg(not(target_arch = "wasm32"))]
FileSaver(Box<dyn FnOnce() -> anyhow::Result<std::path::PathBuf> + Send + 'static>),
}
impl std::fmt::Debug for SystemCommand {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
// not all variant contents can be made `Debug`, so we only output the variant name
f.write_str(self.into())
}
}
/// Interface for sending [`SystemCommand`] messages.
pub trait SystemCommandSender {
fn send_system(&self, command: SystemCommand);
}
// ----------------------------------------------------------------------------
/// Sender that queues up the execution of commands.
#[derive(Clone)]
pub struct CommandSender {
system_sender: std::sync::mpsc::Sender<SystemCommand>,
ui_sender: std::sync::mpsc::Sender<UICommand>,
}
/// Receiver for the [`CommandSender`]
pub struct CommandReceiver {
system_receiver: std::sync::mpsc::Receiver<SystemCommand>,
ui_receiver: std::sync::mpsc::Receiver<UICommand>,
}
impl CommandReceiver {
/// Receive a [`SystemCommand`] to be executed if any is queued.
pub fn recv_system(&self) -> Option<SystemCommand> {
// The only way this can fail (other than being empty)
// is if the sender has been dropped.
self.system_receiver.try_recv().ok()
}
/// Receive a [`UICommand`] to be executed if any is queued.
pub fn recv_ui(&self) -> Option<UICommand> {
// The only way this can fail (other than being empty)
// is if the sender has been dropped.
self.ui_receiver.try_recv().ok()
}
}
/// Creates a new command channel.
pub fn command_channel() -> (CommandSender, CommandReceiver) {
let (system_sender, system_receiver) = std::sync::mpsc::channel();
let (ui_sender, ui_receiver) = std::sync::mpsc::channel();
(
CommandSender {
system_sender,
ui_sender,
},
CommandReceiver {
system_receiver,
ui_receiver,
},
)
}
// ----------------------------------------------------------------------------
impl SystemCommandSender for CommandSender {
/// Send a command to be executed.
fn send_system(&self, command: SystemCommand) {
// The only way this can fail is if the receiver has been dropped.
self.system_sender.send(command).ok();
}
}
impl UICommandSender for CommandSender {
/// Send a command to be executed.
fn send_ui(&self, command: UICommand) {
// The only way this can fail is if the receiver has been dropped.
self.ui_sender.send(command).ok();
}
}