use once_cell::sync::OnceCell;
use re_entity_db::InstancePath;
use re_viewer_context::{
ContainerId, Contents, Item, ItemCollection, ItemContext, ViewId, ViewerContext,
};
use re_viewport_blueprint::{ContainerBlueprint, ViewportBlueprint};
mod actions;
pub mod collapse_expand;
mod sub_menu;
use actions::{
add_container::AddContainerAction,
add_entities_to_new_view::AddEntitiesToNewViewAction,
add_view::AddViewAction,
clone_view::CloneViewAction,
collapse_expand_all::CollapseExpandAllAction,
move_contents_to_new_container::MoveContentsToNewContainerAction,
remove::RemoveAction,
show_hide::{HideAction, ShowAction},
};
use sub_menu::SubMenu;
#[derive(Debug, Clone, Copy)]
pub enum SelectionUpdateBehavior {
UseSelection,
OverrideSelection,
Ignore,
}
pub fn context_menu_ui_for_item(
ctx: &ViewerContext<'_>,
viewport_blueprint: &ViewportBlueprint,
item: &Item,
item_response: &egui::Response,
selection_update_behavior: SelectionUpdateBehavior,
) {
context_menu_ui_for_item_with_context_impl(
ctx,
viewport_blueprint,
item,
None,
item_response,
selection_update_behavior,
);
}
pub fn context_menu_ui_for_item_with_context(
ctx: &ViewerContext<'_>,
viewport_blueprint: &ViewportBlueprint,
item: &Item,
item_context: ItemContext,
item_response: &egui::Response,
selection_update_behavior: SelectionUpdateBehavior,
) {
context_menu_ui_for_item_with_context_impl(
ctx,
viewport_blueprint,
item,
Some(item_context),
item_response,
selection_update_behavior,
);
}
fn context_menu_ui_for_item_with_context_impl(
ctx: &ViewerContext<'_>,
viewport_blueprint: &ViewportBlueprint,
item: &Item,
item_context: Option<ItemContext>,
item_response: &egui::Response,
selection_update_behavior: SelectionUpdateBehavior,
) {
item_response.context_menu(|ui| {
if ui.input_mut(|i| i.consume_key(egui::Modifiers::NONE, egui::Key::Escape)) {
ui.close_menu();
return;
}
let mut show_context_menu = |selection: &ItemCollection| {
let context_menu_ctx = ContextMenuContext {
viewer_context: ctx,
viewport_blueprint,
selection,
clicked_item: item,
};
show_context_menu_for_selection(&context_menu_ctx, ui);
};
let item_collection = ItemCollection::from(std::iter::once((item.clone(), item_context)));
match selection_update_behavior {
SelectionUpdateBehavior::UseSelection => {
if !ctx.selection().contains_item(item) {
if item_response.hovered() && item_response.secondary_clicked() {
show_context_menu(&item_collection);
ctx.selection_state().set_selection(item_collection);
} else {
show_context_menu(ctx.selection());
}
} else {
show_context_menu(ctx.selection());
}
}
SelectionUpdateBehavior::OverrideSelection => {
show_context_menu(&item_collection);
if item_response.secondary_clicked() {
ctx.selection_state().set_selection(item_collection);
}
}
SelectionUpdateBehavior::Ignore => {
show_context_menu(&item_collection);
}
};
});
}
fn action_list(
ctx: &ViewerContext<'_>,
) -> &'static Vec<Vec<Box<dyn ContextMenuAction + Sync + Send>>> {
use egui_tiles::ContainerKind;
static CONTEXT_MENU_ACTIONS: OnceCell<Vec<Vec<Box<dyn ContextMenuAction + Sync + Send>>>> =
OnceCell::new();
static_assertions::const_assert_eq!(ContainerKind::ALL.len(), 4);
CONTEXT_MENU_ACTIONS.get_or_init(|| {
vec![
vec![
Box::new(ShowAction),
Box::new(HideAction),
Box::new(RemoveAction),
],
vec![
Box::new(actions::ScreenshotAction::CopyScreenshot),
Box::new(actions::ScreenshotAction::SaveScreenshot),
],
vec![
Box::new(CollapseExpandAllAction::ExpandAll),
Box::new(CollapseExpandAllAction::CollapseAll),
],
vec![Box::new(CloneViewAction)],
vec![
Box::new(SubMenu {
label: "Add container".to_owned(),
actions: vec![
Box::new(AddContainerAction(ContainerKind::Tabs)),
Box::new(AddContainerAction(ContainerKind::Horizontal)),
Box::new(AddContainerAction(ContainerKind::Vertical)),
Box::new(AddContainerAction(ContainerKind::Grid)),
],
}),
Box::new(SubMenu {
label: "Add view".to_owned(),
actions: ctx
.view_class_registry
.iter_registry()
.map(|entry| {
Box::new(AddViewAction {
icon: entry.class.icon(),
id: entry.identifier,
})
as Box<dyn ContextMenuAction + Sync + Send>
})
.collect(),
}),
],
vec![Box::new(SubMenu {
label: "Move to new container".to_owned(),
actions: vec![
Box::new(MoveContentsToNewContainerAction(ContainerKind::Tabs)),
Box::new(MoveContentsToNewContainerAction(ContainerKind::Horizontal)),
Box::new(MoveContentsToNewContainerAction(ContainerKind::Vertical)),
Box::new(MoveContentsToNewContainerAction(ContainerKind::Grid)),
],
})],
vec![Box::new(AddEntitiesToNewViewAction)],
]
})
}
fn show_context_menu_for_selection(ctx: &ContextMenuContext<'_>, ui: &mut egui::Ui) {
ui.style_mut().wrap_mode = Some(egui::TextWrapMode::Extend); let mut should_display_separator = false;
for action_section in action_list(ctx.viewer_context) {
let mut any_action_displayed = false;
for action in action_section {
if !action.supports_selection(ctx) {
continue;
}
any_action_displayed = true;
if should_display_separator {
ui.separator();
should_display_separator = false;
}
let response = action.ui(ctx, ui);
if response.clicked() {
ui.close_menu();
}
}
should_display_separator |= any_action_displayed;
}
if !should_display_separator {
ui.label(egui::RichText::from("No action available for the current selection").italics());
}
}
struct ContextMenuContext<'a> {
viewer_context: &'a ViewerContext<'a>,
viewport_blueprint: &'a ViewportBlueprint,
selection: &'a ItemCollection,
clicked_item: &'a Item,
}
impl<'a> ContextMenuContext<'a> {
pub fn clicked_item_enclosing_container_id_and_position(&self) -> Option<(ContainerId, usize)> {
match self.clicked_item {
Item::View(view_id) | Item::DataResult(view_id, _) => Some(Contents::View(*view_id)),
Item::Container(container_id) => Some(Contents::Container(*container_id)),
_ => None,
}
.and_then(|c: Contents| self.viewport_blueprint.find_parent_and_position_index(&c))
}
pub fn clicked_item_enclosing_container_and_position(
&self,
) -> Option<(&'a ContainerBlueprint, usize)> {
self.clicked_item_enclosing_container_id_and_position()
.and_then(|(container_id, pos)| {
self.viewport_blueprint
.container(&container_id)
.map(|container| (container, pos))
})
}
pub fn egui_context(&self) -> &egui::Context {
self.viewer_context.egui_ctx
}
}
trait ContextMenuAction {
fn supports_selection(&self, ctx: &ContextMenuContext<'_>) -> bool {
if ctx.selection.len() > 1 && !self.supports_multi_selection(ctx) {
return false;
}
ctx.selection
.iter()
.all(|(item, _)| self.supports_item(ctx, item))
}
fn supports_multi_selection(&self, _ctx: &ContextMenuContext<'_>) -> bool {
false
}
fn supports_item(&self, _ctx: &ContextMenuContext<'_>, _item: &Item) -> bool {
false
}
fn ui(&self, ctx: &ContextMenuContext<'_>, ui: &mut egui::Ui) -> egui::Response {
let label = self.label(ctx);
let response = if let Some(icon) = self.icon() {
ui.add(egui::Button::image_and_text(icon.as_image(), label))
} else {
ui.button(label)
};
if response.clicked() {
self.process_selection(ctx);
}
response
}
fn icon(&self) -> Option<&'static re_ui::Icon> {
None
}
fn label(&self, _ctx: &ContextMenuContext<'_>) -> String {
String::new()
}
fn process_selection(&self, ctx: &ContextMenuContext<'_>) {
for (item, _) in ctx.selection.iter() {
match item {
Item::AppId(app_id) => self.process_app_id(ctx, app_id),
Item::DataSource(data_source) => self.process_data_source(ctx, data_source),
Item::StoreId(store_id) => self.process_store_id(ctx, store_id),
Item::ComponentPath(component_path) => {
self.process_component_path(ctx, component_path);
}
Item::View(view_id) => self.process_view(ctx, view_id),
Item::InstancePath(instance_path) => self.process_instance_path(ctx, instance_path),
Item::DataResult(view_id, instance_path) => {
self.process_data_result(ctx, view_id, instance_path);
}
Item::Container(container_id) => self.process_container(ctx, container_id),
}
}
}
fn process_app_id(&self, _ctx: &ContextMenuContext<'_>, _app_id: &re_log_types::ApplicationId) {
}
fn process_data_source(
&self,
_ctx: &ContextMenuContext<'_>,
_data_source: &re_smart_channel::SmartChannelSource,
) {
}
fn process_store_id(&self, _ctx: &ContextMenuContext<'_>, _store_id: &re_log_types::StoreId) {}
fn process_container(&self, _ctx: &ContextMenuContext<'_>, _container_id: &ContainerId) {}
fn process_view(&self, _ctx: &ContextMenuContext<'_>, _view_id: &ViewId) {}
fn process_data_result(
&self,
_ctx: &ContextMenuContext<'_>,
_view_id: &ViewId,
_instance_path: &InstancePath,
) {
}
fn process_instance_path(&self, _ctx: &ContextMenuContext<'_>, _instance_path: &InstancePath) {}
fn process_component_path(
&self,
_ctx: &ContextMenuContext<'_>,
_component_path: &re_log_types::ComponentPath,
) {
}
}