use egui::{emath::RectTransform, pos2, vec2, Align2, Color32, Pos2, Rect, Shape, Vec2};
use re_math::IsoTransform;
use re_entity_db::EntityPath;
use re_log::ResultExt as _;
use re_renderer::view_builder::{TargetConfiguration, ViewBuilder};
use re_types::{
archetypes::Pinhole,
blueprint::{
archetypes::{Background, NearClipPlane, VisualBounds2D},
components as blueprint_components,
},
components::ViewCoordinates,
};
use re_ui::{ContextExt as _, ModifiersMarkdown, MouseButtonMarkdown};
use re_view::controls::{DRAG_PAN2D_BUTTON, ZOOM_SCROLL_MODIFIER};
use re_viewer_context::{
gpu_bridge, ItemSpaceContext, ViewQuery, ViewSystemExecutionError, ViewerContext,
};
use re_viewport_blueprint::ViewProperty;
use super::{eye::Eye, ui::create_labels};
use crate::{
query_pinhole_legacy, ui::SpatialViewState, view_kind::SpatialViewKind,
visualizers::collect_ui_labels, SpatialView2D,
};
fn ui_from_scene(
ctx: &ViewerContext<'_>,
response: &egui::Response,
view_class: &SpatialView2D,
view_state: &mut SpatialViewState,
bounds_property: &ViewProperty,
) -> RectTransform {
let bounds: blueprint_components::VisualBounds2D = bounds_property
.component_or_fallback(ctx, view_class, view_state)
.ok_or_log_error()
.unwrap_or_default();
view_state.visual_bounds_2d = Some(bounds);
let mut bounds_rect: egui::Rect = bounds.into();
let mut letterboxed_bounds = bounds_rect;
let ui_from_scene = RectTransform::from_to(bounds_rect, response.rect);
let scale_aspect = ui_from_scene.scale().x / ui_from_scene.scale().y;
if scale_aspect < 1.0 {
let add = bounds_rect.height() * (1.0 / scale_aspect - 1.0);
letterboxed_bounds.min.y -= 0.5 * add;
letterboxed_bounds.max.y += 0.5 * add;
} else {
let add = bounds_rect.width() * (scale_aspect - 1.0);
letterboxed_bounds.min.x -= 0.5 * add;
letterboxed_bounds.max.x += 0.5 * add;
}
let ui_from_scene = RectTransform::from_to(letterboxed_bounds, response.rect);
let mut pan_delta_in_ui = response.drag_delta();
if response.hovered() {
pan_delta_in_ui += response.ctx.input(|i| i.smooth_scroll_delta);
}
if pan_delta_in_ui != Vec2::ZERO {
bounds_rect = bounds_rect.translate(-pan_delta_in_ui / ui_from_scene.scale());
}
if response.hovered() {
let zoom_delta = response.ctx.input(|i| i.zoom_delta_2d());
if zoom_delta != Vec2::splat(1.0) {
let zoom_center_in_ui = response
.hover_pos()
.unwrap_or_else(|| response.rect.center());
let zoom_center_in_scene = ui_from_scene
.inverse()
.transform_pos(zoom_center_in_ui)
.to_vec2();
bounds_rect = scale_rect(
bounds_rect.translate(-zoom_center_in_scene),
Vec2::splat(1.0) / zoom_delta,
)
.translate(zoom_center_in_scene);
}
}
let updated_bounds: blueprint_components::VisualBounds2D = bounds_rect.into();
if response.double_clicked() {
bounds_property.reset_blueprint_component::<blueprint_components::VisualBounds2D>(ctx);
} else if bounds != updated_bounds {
bounds_property.save_blueprint_component(ctx, &updated_bounds);
}
view_state.visual_bounds_2d = Some(bounds);
RectTransform::from_to(letterboxed_bounds, response.rect)
}
fn scale_rect(rect: Rect, factor: Vec2) -> Rect {
Rect::from_min_max(
(factor * rect.min.to_vec2()).to_pos2(),
(factor * rect.max.to_vec2()).to_pos2(),
)
}
pub fn help_markdown(egui_ctx: &egui::Context) -> String {
format!(
"# 2D View
Display 2D content in the reference frame defined by the space origin.
## Navigation controls
- Pinch gesture or {zoom_scroll_modifier} + scroll to zoom.
- Click and drag with the {drag_pan2d_button} to pan.
- Double-click to reset the view.",
zoom_scroll_modifier = ModifiersMarkdown(ZOOM_SCROLL_MODIFIER, egui_ctx),
drag_pan2d_button = MouseButtonMarkdown(DRAG_PAN2D_BUTTON),
)
.to_owned()
}
impl SpatialView2D {
pub fn view_2d(
&self,
ctx: &ViewerContext<'_>,
ui: &mut egui::Ui,
state: &mut SpatialViewState,
query: &ViewQuery<'_>,
system_output: re_viewer_context::SystemExecutionOutput,
) -> Result<(), ViewSystemExecutionError> {
re_tracing::profile_function!();
if ui.available_size().min_elem() <= 0.0 {
return Ok(());
}
state.pinhole_at_origin =
query_pinhole_legacy(ctx, &ctx.current_query(), query.space_origin);
let (response, painter) =
ui.allocate_painter(ui.available_size(), egui::Sense::click_and_drag());
let bounds_property = ViewProperty::from_archetype::<VisualBounds2D>(
ctx.blueprint_db(),
ctx.blueprint_query,
query.view_id,
);
let clip_property = ViewProperty::from_archetype::<NearClipPlane>(
ctx.blueprint_db(),
ctx.blueprint_query,
query.view_id,
);
let ui_from_scene = ui_from_scene(ctx, &response, self, state, &bounds_property);
let scene_from_ui = ui_from_scene.inverse();
let near_clip_plane: blueprint_components::NearClipPlane = clip_property
.component_or_fallback(ctx, self, state)
.ok_or_log_error()
.unwrap_or_default();
let eye = Eye {
world_from_rub_view: IsoTransform::IDENTITY,
fov_y: None,
};
let near_clip_plane = f32::max(f32::MIN_POSITIVE, *near_clip_plane.0);
let scene_bounds = *scene_from_ui.to();
let Ok(target_config) = setup_target_config(
&painter,
scene_bounds,
near_clip_plane,
&query.space_origin.to_string(),
query.highlights.any_outlines(),
&state.pinhole_at_origin,
) else {
return Ok(());
};
let (label_shapes, ui_rects) = create_labels(
collect_ui_labels(&system_output.view_systems),
ui_from_scene,
&eye,
ui,
&query.highlights,
SpatialViewKind::TwoD,
);
let Some(render_ctx) = ctx.render_ctx else {
return Err(ViewSystemExecutionError::NoRenderContextError);
};
let mut view_builder = ViewBuilder::new(render_ctx, target_config);
if let Some(pointer_pos_ui) = response.hover_pos() {
let picking_context = crate::picking::PickingContext::new(
pointer_pos_ui,
scene_from_ui,
ui.ctx().pixels_per_point(),
&eye,
);
crate::picking_ui::picking(
ctx,
&picking_context,
ui,
response,
&mut view_builder,
state,
&system_output,
&ui_rects,
query,
SpatialViewKind::TwoD,
)?;
} else {
state.previous_picking_result = None;
}
for draw_data in system_output.draw_data {
view_builder.queue_draw(draw_data);
}
let background = ViewProperty::from_archetype::<Background>(
ctx.blueprint_db(),
ctx.blueprint_query,
query.view_id,
);
let (background_drawable, clear_color) =
crate::configure_background(ctx, &background, render_ctx, self, state)?;
if let Some(background_drawable) = background_drawable {
view_builder.queue_draw(background_drawable);
}
painter.add(gpu_bridge::new_renderer_callback(
view_builder,
painter.clip_rect(),
clear_color,
));
for selected_context in ctx.selection_state().selection_space_contexts() {
painter.extend(show_projections_from_3d_space(
ui,
query.space_origin,
&ui_from_scene,
selected_context,
ui.ctx().selection_stroke().color,
));
}
if let Some(hovered_context) = ctx.selection_state().hovered_space_context() {
painter.extend(show_projections_from_3d_space(
ui,
query.space_origin,
&ui_from_scene,
hovered_context,
ui.ctx().hover_stroke().color,
));
}
crate::ui::paint_loading_spinners(ui, ui_from_scene, &eye, &system_output.view_systems);
painter.extend(label_shapes);
Ok(())
}
}
fn setup_target_config(
egui_painter: &egui::Painter,
scene_bounds: Rect,
near_clip_plane: f32,
space_name: &str,
any_outlines: bool,
scene_pinhole: &Option<Pinhole>,
) -> anyhow::Result<TargetConfiguration> {
let scene_bounds_size = glam::vec2(scene_bounds.width(), scene_bounds.height());
let pinhole;
let resolution;
if let Some(scene_pinhole) = scene_pinhole {
pinhole = scene_pinhole.clone();
resolution = pinhole.resolution().unwrap_or_else(|| {
re_log::warn_once!("Pinhole projection lacks resolution.");
glam::Vec2::splat(1000.0)
});
} else {
let focal_length = 1000.0; let principal_point = glam::Vec2::splat(500.0); resolution = glam::Vec2::splat(1000.0); pinhole = Pinhole {
image_from_camera: glam::Mat3::from_cols(
glam::vec3(focal_length, 0.0, 0.0),
glam::vec3(0.0, focal_length, 0.0),
principal_point.extend(1.0),
)
.into(),
resolution: Some([resolution.x, resolution.y].into()),
camera_xyz: Some(ViewCoordinates::RDF),
image_plane_distance: None,
};
}
let pinhole_rect = Rect::from_min_size(Pos2::ZERO, egui::vec2(resolution.x, resolution.y));
let focal_length = pinhole.focal_length_in_pixels();
let focal_length = 2.0 / (1.0 / focal_length.x() + 1.0 / focal_length.y()); let projection_from_view = re_renderer::view_builder::Projection::Perspective {
vertical_fov: pinhole.fov_y().unwrap_or(Eye::DEFAULT_FOV_Y),
near_plane_distance: near_clip_plane * focal_length / 500.0, aspect_ratio: pinhole
.aspect_ratio()
.unwrap_or(scene_bounds_size.x / scene_bounds_size.y), };
let view_from_world = re_math::IsoTransform::look_at_rh(
pinhole.principal_point().extend(-focal_length),
pinhole.principal_point().extend(0.0),
-glam::Vec3::Y,
)
.ok_or_else(|| anyhow::format_err!("Failed to compute camera transform for 2D view."))?;
let mut viewport_transformation = re_renderer::RectTransform {
region: re_render_rect_from_egui_rect(pinhole_rect),
region_of_interest: re_render_rect_from_egui_rect(scene_bounds),
};
let image_center = 0.5 * resolution;
viewport_transformation.region_of_interest.min += image_center - pinhole.principal_point();
let pixels_per_point = egui_painter.ctx().pixels_per_point();
let resolution_in_pixel =
gpu_bridge::viewport_resolution_in_pixels(egui_painter.clip_rect(), pixels_per_point);
anyhow::ensure!(0 < resolution_in_pixel[0] && 0 < resolution_in_pixel[1]);
Ok({
let name = space_name.into();
TargetConfiguration {
name,
resolution_in_pixel,
view_from_world,
projection_from_view,
viewport_transformation,
pixels_per_point,
outline_config: any_outlines.then(|| re_view::outline_config(egui_painter.ctx())),
blend_with_background: false,
}
})
}
fn re_render_rect_from_egui_rect(rect: egui::Rect) -> re_renderer::RectF32 {
re_renderer::RectF32 {
min: glam::vec2(rect.left(), rect.top()),
extent: glam::vec2(rect.width(), rect.height()),
}
}
fn show_projections_from_3d_space(
ui: &egui::Ui,
space: &EntityPath,
ui_from_scene: &RectTransform,
space_context: &ItemSpaceContext,
circle_fill_color: egui::Color32,
) -> Vec<Shape> {
let mut shapes = Vec::new();
if let ItemSpaceContext::ThreeD {
point_in_space_cameras: target_spaces,
..
} = space_context
{
for (space_2d, pos_2d) in target_spaces {
if space_2d == space {
if let Some(pos_2d) = pos_2d {
let pos_in_ui = ui_from_scene.transform_pos(pos2(pos_2d.x, pos_2d.y));
let radius = 4.0;
shapes.push(Shape::circle_filled(
pos_in_ui,
radius + 2.0,
Color32::BLACK,
));
shapes.push(Shape::circle_filled(pos_in_ui, radius, circle_fill_color));
let text_color = Color32::WHITE;
let text = format!("Depth: {:.3} m", pos_2d.z);
let font_id = egui::TextStyle::Body.resolve(ui.style());
let galley = ui.fonts(|fonts| fonts.layout_no_wrap(text, font_id, text_color));
let rect = Align2::CENTER_TOP.anchor_rect(Rect::from_min_size(
pos_in_ui + vec2(0.0, 5.0),
galley.size(),
));
shapes.push(Shape::rect_filled(
rect,
2.0,
Color32::from_black_alpha(196),
));
shapes.push(Shape::galley(rect.min, galley, text_color));
}
}
}
}
shapes
}